diff --git a/.changeset/agent-manager-tab-add-button.md b/.changeset/agent-manager-tab-add-button.md new file mode 100644 index 0000000000..93098cacf8 --- /dev/null +++ b/.changeset/agent-manager-tab-add-button.md @@ -0,0 +1,5 @@ +--- +"kilo-code": patch +--- + +Keep the Agent Manager new-tab button visible at the end of the tab row, separated by a divider, and show all tab tooltips above the tabs. diff --git a/.changeset/agent-manager-tab-controls.md b/.changeset/agent-manager-tab-controls.md new file mode 100644 index 0000000000..8d9b184dd1 --- /dev/null +++ b/.changeset/agent-manager-tab-controls.md @@ -0,0 +1,5 @@ +--- +"kilo-code": patch +--- + +Move Agent Manager tab creation and search controls to the left side of the tab bar with clearer toolbar separators. diff --git a/.changeset/breezy-models-float.md b/.changeset/breezy-models-float.md new file mode 100644 index 0000000000..84b355eb52 --- /dev/null +++ b/.changeset/breezy-models-float.md @@ -0,0 +1,5 @@ +--- +"@kilocode/kilo-jetbrains": patch +--- + +Improve the JetBrains model picker with sections, favorites, and smart filtering. diff --git a/.changeset/editor-sized-jetbrains-chat.md b/.changeset/editor-sized-jetbrains-chat.md new file mode 100644 index 0000000000..221bd814f6 --- /dev/null +++ b/.changeset/editor-sized-jetbrains-chat.md @@ -0,0 +1,5 @@ +--- +"@kilocode/kilo-jetbrains": patch +--- + +Scale JetBrains chat transcript fonts from the editor font settings. diff --git a/.changeset/favorite-picker-index.md b/.changeset/favorite-picker-index.md new file mode 100644 index 0000000000..602a32bfe0 --- /dev/null +++ b/.changeset/favorite-picker-index.md @@ -0,0 +1,5 @@ +--- +"@kilocode/kilo-jetbrains": patch +--- + +Keep the JetBrains model picker at the same row position when models are favorited or unfavorited. diff --git a/.changeset/hide-empty-task-header-graph.md b/.changeset/hide-empty-task-header-graph.md new file mode 100644 index 0000000000..06609ec09a --- /dev/null +++ b/.changeset/hide-empty-task-header-graph.md @@ -0,0 +1,5 @@ +--- +"kilo-code": patch +--- + +Hide the task header graph area until timeline data is available. diff --git a/.changeset/jetbrains-chat-renderers.md b/.changeset/jetbrains-chat-renderers.md new file mode 100644 index 0000000000..0a099733c1 --- /dev/null +++ b/.changeset/jetbrains-chat-renderers.md @@ -0,0 +1,5 @@ +--- +"@kilocode/kilo-jetbrains": patch +--- + +Match VS Code-style reasoning and tool run rendering in JetBrains chat. diff --git a/.changeset/jetbrains-empty-panel-layout.md b/.changeset/jetbrains-empty-panel-layout.md new file mode 100644 index 0000000000..3f13739eeb --- /dev/null +++ b/.changeset/jetbrains-empty-panel-layout.md @@ -0,0 +1,5 @@ +--- +"@kilocode/kilo-jetbrains": patch +--- + +Center description text and remove the fixed-width cap from the empty session panel so recent sessions can stretch naturally. diff --git a/.changeset/jetbrains-existing-session-scroll.md b/.changeset/jetbrains-existing-session-scroll.md new file mode 100644 index 0000000000..534092cc3f --- /dev/null +++ b/.changeset/jetbrains-existing-session-scroll.md @@ -0,0 +1,5 @@ +--- +"@kilocode/kilo-jetbrains": patch +--- + +Open existing JetBrains sessions scrolled to the latest message after history loads. diff --git a/.changeset/jetbrains-header-tooltips.md b/.changeset/jetbrains-header-tooltips.md new file mode 100644 index 0000000000..eabe4154aa --- /dev/null +++ b/.changeset/jetbrains-header-tooltips.md @@ -0,0 +1,5 @@ +--- +"@kilocode/kilo-jetbrains": patch +--- + +Remember the JetBrains session header metrics expansion preference and show file names in read/write timeline tooltips. diff --git a/.changeset/jetbrains-history-navigation.md b/.changeset/jetbrains-history-navigation.md new file mode 100644 index 0000000000..92f361fc77 --- /dev/null +++ b/.changeset/jetbrains-history-navigation.md @@ -0,0 +1,5 @@ +--- +"@kilocode/kilo-jetbrains": patch +--- + +Improve JetBrains history navigation with Back support and a Show History action in recent sessions. diff --git a/.changeset/jetbrains-history-tabs.md b/.changeset/jetbrains-history-tabs.md new file mode 100644 index 0000000000..859a598e3e --- /dev/null +++ b/.changeset/jetbrains-history-tabs.md @@ -0,0 +1,5 @@ +--- +"@kilocode/kilo-jetbrains": patch +--- + +Improve the JetBrains history panel with native tabs, independent local and cloud search, and date-grouped cloud sessions. diff --git a/.changeset/jetbrains-mode-picker.md b/.changeset/jetbrains-mode-picker.md new file mode 100644 index 0000000000..7d3b6b2011 --- /dev/null +++ b/.changeset/jetbrains-mode-picker.md @@ -0,0 +1,5 @@ +--- +"@kilocode/kilo-jetbrains": patch +--- + +Improve the JetBrains mode picker with descriptions, deprecated badges, and native IntelliJ selection behavior. diff --git a/.changeset/jetbrains-scroll-bottom.md b/.changeset/jetbrains-scroll-bottom.md new file mode 100644 index 0000000000..b98469fac7 --- /dev/null +++ b/.changeset/jetbrains-scroll-bottom.md @@ -0,0 +1,5 @@ +--- +"@kilocode/kilo-jetbrains": patch +--- + +Keep the JetBrains chat transcript pinned to bottom reliably while responses stream. diff --git a/.changeset/jetbrains-scroll-button.md b/.changeset/jetbrains-scroll-button.md new file mode 100644 index 0000000000..31762b7e48 --- /dev/null +++ b/.changeset/jetbrains-scroll-button.md @@ -0,0 +1,5 @@ +--- +"@kilocode/kilo-jetbrains": patch +--- + +Show a scroll-to-bottom button when JetBrains chat sessions are scrolled away from the latest message. diff --git a/.changeset/jetbrains-send-action.md b/.changeset/jetbrains-send-action.md new file mode 100644 index 0000000000..6f6f30a3d4 --- /dev/null +++ b/.changeset/jetbrains-send-action.md @@ -0,0 +1,5 @@ +--- +"@kilocode/kilo-jetbrains": patch +--- + +Support rebinding the JetBrains prompt Send action in the IntelliJ keymap. diff --git a/.changeset/jetbrains-session-header.md b/.changeset/jetbrains-session-header.md new file mode 100644 index 0000000000..9c13aa6c9d --- /dev/null +++ b/.changeset/jetbrains-session-header.md @@ -0,0 +1,5 @@ +--- +"@kilocode/kilo-jetbrains": patch +--- + +Show collapsible session metadata, context usage, token usage, and compaction controls in the JetBrains session header. diff --git a/.changeset/jetbrains-session-history.md b/.changeset/jetbrains-session-history.md new file mode 100644 index 0000000000..8aa827208c --- /dev/null +++ b/.changeset/jetbrains-session-history.md @@ -0,0 +1,5 @@ +--- +"@kilocode/kilo-jetbrains": patch +--- + +Add a native session history panel to the JetBrains plugin. diff --git a/.changeset/jetbrains-session-layout-padding.md b/.changeset/jetbrains-session-layout-padding.md new file mode 100644 index 0000000000..2fbf40c42d --- /dev/null +++ b/.changeset/jetbrains-session-layout-padding.md @@ -0,0 +1,5 @@ +--- +"@kilocode/kilo-jetbrains": patch +--- + +Control JetBrains chat transcript spacing from the session layout. diff --git a/.changeset/jetbrains-session-loading-state.md b/.changeset/jetbrains-session-loading-state.md new file mode 100644 index 0000000000..bc930ebcbc --- /dev/null +++ b/.changeset/jetbrains-session-loading-state.md @@ -0,0 +1,5 @@ +--- +"@kilocode/kilo-jetbrains": patch +--- + +Show loading immediately when opening JetBrains sessions without showing duplicate progress for new sessions. diff --git a/.changeset/jetbrains-session-scroll-layout.md b/.changeset/jetbrains-session-scroll-layout.md new file mode 100644 index 0000000000..ea065c6788 --- /dev/null +++ b/.changeset/jetbrains-session-scroll-layout.md @@ -0,0 +1,5 @@ +--- +"@kilocode/kilo-jetbrains": patch +--- + +Reduce redundant JetBrains chat scroll layout work while keeping bottom-follow behavior. diff --git a/.changeset/jetbrains-step-finish-timeline.md b/.changeset/jetbrains-step-finish-timeline.md new file mode 100644 index 0000000000..6afba10a4b --- /dev/null +++ b/.changeset/jetbrains-step-finish-timeline.md @@ -0,0 +1,5 @@ +--- +"@kilocode/kilo-jetbrains": patch +--- + +Show completed assistant steps in the JetBrains session timeline. diff --git a/.changeset/jetbrains-stop-action.md b/.changeset/jetbrains-stop-action.md new file mode 100644 index 0000000000..34bdae579d --- /dev/null +++ b/.changeset/jetbrains-stop-action.md @@ -0,0 +1,5 @@ +--- +"@kilocode/kilo-jetbrains": patch +--- + +Support stopping active JetBrains sessions from the prompt action button. diff --git a/.changeset/model-picker-natural-size.md b/.changeset/model-picker-natural-size.md new file mode 100644 index 0000000000..5fa0954e27 --- /dev/null +++ b/.changeset/model-picker-natural-size.md @@ -0,0 +1,5 @@ +--- +"@kilocode/kilo-jetbrains": patch +--- + +Size the JetBrains model picker from its rendered model rows instead of fixed popup dimensions. diff --git a/.changeset/quiet-otters-reason.md b/.changeset/quiet-otters-reason.md new file mode 100644 index 0000000000..ad7b31eb35 --- /dev/null +++ b/.changeset/quiet-otters-reason.md @@ -0,0 +1,5 @@ +--- +"@kilocode/kilo-jetbrains": patch +--- + +Support reasoning effort selection and resetting JetBrains chat model overrides to the default model. diff --git a/.changeset/reasoning-picker-align.md b/.changeset/reasoning-picker-align.md new file mode 100644 index 0000000000..4a21b9e976 --- /dev/null +++ b/.changeset/reasoning-picker-align.md @@ -0,0 +1,5 @@ +--- +"@kilocode/kilo-jetbrains": patch +--- + +Align checked and unchecked reasoning picker rows in the JetBrains plugin. diff --git a/.changeset/revert-checkpoints-stay.md b/.changeset/revert-checkpoints-stay.md new file mode 100644 index 0000000000..bb505f02d5 --- /dev/null +++ b/.changeset/revert-checkpoints-stay.md @@ -0,0 +1,5 @@ +--- +"kilo-code": patch +--- + +Keep older revert checkpoints available after reverting a message in the VS Code extension. diff --git a/.changeset/review-plus-contrast.md b/.changeset/review-plus-contrast.md new file mode 100644 index 0000000000..49414a2be1 --- /dev/null +++ b/.changeset/review-plus-contrast.md @@ -0,0 +1,6 @@ +--- +"kilo-code": patch +"@kilocode/kilo-ui": patch +--- + +Improve review comment plus icon contrast in diff viewers. diff --git a/.changeset/stable-shell-streaming.md b/.changeset/stable-shell-streaming.md new file mode 100644 index 0000000000..bdae1f250a --- /dev/null +++ b/.changeset/stable-shell-streaming.md @@ -0,0 +1,5 @@ +--- +"kilo-code": patch +--- + +Keep shell command blocks stable while output streams so pending animations and expanded state do not reset. diff --git a/.changeset/steady-crabs-scroll.md b/.changeset/steady-crabs-scroll.md new file mode 100644 index 0000000000..9688855539 --- /dev/null +++ b/.changeset/steady-crabs-scroll.md @@ -0,0 +1,5 @@ +--- +"@kilocode/kilo-jetbrains": patch +--- + +Preserve JetBrains chat scroll position when reviewing earlier session output. diff --git a/.github/actions/setup-git-committer/action.yml b/.github/actions/setup-git-committer/action.yml index 0fff208f62..0474b670a6 100644 --- a/.github/actions/setup-git-committer/action.yml +++ b/.github/actions/setup-git-committer/action.yml @@ -39,6 +39,15 @@ runs: - name: Clear checkout auth run: | git config --local --unset-all http.https://github.com/.extraheader || true + # kilocode_change start + keys="$(git config --local --name-only --get-regexp '^includeIf\.gitdir:' || true)" + for key in $keys; do + path="$(git config --local --get "$key" || true)" + case "$path" in + *git-credentials-*.config) git config --local --unset-all "$key" || true ;; + esac + done + # kilocode_change end shell: bash - name: Configure git remote diff --git a/.github/workflows/auto-docs.yml b/.github/workflows/auto-docs.yml index 4364ee8748..5f84be47f3 100644 --- a/.github/workflows/auto-docs.yml +++ b/.github/workflows/auto-docs.yml @@ -1,35 +1,113 @@ -name: Trigger webhook for feat PRs +name: Auto-docs dry-run intake +# Trigger on merged feature PRs, or manually with a specific PR number. +# IMPORTANT: This workflow intentionally does NOT check out PR code and does NOT +# execute any code from the PR branch. It only reads PR metadata and forwards it +# to a webhook. This is required for safe use of pull_request_target on a public repo. on: - pull_request: + pull_request_target: types: [closed] + workflow_dispatch: + inputs: + pr_number: + description: "PR number to test" + required: true + +# Minimal permissions: read repo contents and PR metadata only. +permissions: + contents: read + pull-requests: read + +# One run per PR at a time. Does not cancel in progress — let the current run +# finish so we don't drop events when a PR is quickly closed/reopened/closed. +concurrency: + group: auto-docs-${{ github.event.pull_request.number || inputs.pr_number }} + cancel-in-progress: false jobs: - call-webhook: + call-valtown: if: > - github.event.pull_request.merged == true && - (startsWith(github.event.pull_request.title, 'feat:') || - startsWith(github.event.pull_request.title, 'feat(')) + github.event_name == 'workflow_dispatch' || + ( + github.event.pull_request.merged == true && + ( + startsWith(github.event.pull_request.title, 'feat:') || + startsWith(github.event.pull_request.title, 'feat(') + ) + ) + runs-on: ubuntu-latest + timeout-minutes: 5 + steps: - - name: Send webhook safely + - name: Build webhook payload + env: + GH_TOKEN: ${{ github.token }} + EVENT_NAME: ${{ github.event_name }} + REPO: ${{ github.repository }} + PR_NUMBER_FROM_EVENT: ${{ github.event.pull_request.number }} + PR_NUMBER_FROM_INPUT: ${{ inputs.pr_number }} run: | - payload=$(jq -n \ - --arg repo "${GITHUB_REPOSITORY}" \ - --arg pr_number "${{ github.event.pull_request.number }}" \ - --arg title "${{ github.event.pull_request.title }}" \ - --arg body "${{ github.event.pull_request.body }}" \ - --arg author "${{ github.event.pull_request.user.login }}" \ - --arg merged_at "${{ github.event.pull_request.merged_at }}" \ + set -euo pipefail + + PR_NUMBER="${PR_NUMBER_FROM_EVENT:-$PR_NUMBER_FROM_INPUT}" + + if [ -z "$PR_NUMBER" ]; then + echo "Missing PR number" + exit 1 + fi + + gh api "repos/$REPO/pulls/$PR_NUMBER" > pr.json + + jq -n \ + --arg event_name "$EVENT_NAME" \ + --arg repo "$REPO" \ + --arg pr_number "$PR_NUMBER" \ + --slurpfile pr pr.json \ '{ + event_name: $event_name, repo: $repo, pr_number: $pr_number, - title: $title, - description: $body, - author: $author, - merged_at: $merged_at - }' - ) - curl --fail-with-body -X POST "${{ secrets.DOC_WEBHOOK_URL }}" \ + title: $pr[0].title, + description: ($pr[0].body // ""), + author: $pr[0].user.login, + merged_at: ($pr[0].merged_at // ""), + pr_url: $pr[0].html_url, + base_branch: $pr[0].base.ref, + head_branch: $pr[0].head.ref + }' > payload.json + + echo "Payload created:" + jq '{ + event_name, + repo, + pr_number, + title, + author, + merged_at, + pr_url, + base_branch, + head_branch + }' payload.json + + - name: Send webhook to Val Town + env: + DOC_WEBHOOK_URL: ${{ secrets.DOC_WEBHOOK_URL }} + DOC_WEBHOOK_SECRET: ${{ secrets.DOC_WEBHOOK_SECRET }} + run: | + set -euo pipefail + + if [ -z "${DOC_WEBHOOK_URL:-}" ]; then + echo "DOC_WEBHOOK_URL secret is not set" + exit 1 + fi + + if [ -z "${DOC_WEBHOOK_SECRET:-}" ]; then + echo "DOC_WEBHOOK_SECRET secret is not set" + exit 1 + fi + + curl --fail-with-body -sS -X POST "$DOC_WEBHOOK_URL" \ -H "Content-Type: application/json" \ - --data-raw "$payload" + -H "X-Docs-Webhook-Secret: $DOC_WEBHOOK_SECRET" \ + --data-binary @payload.json diff --git a/.github/workflows/beta.yml b/.github/workflows/beta.yml index 5c5ca6159a..90fd102189 100644 --- a/.github/workflows/beta.yml +++ b/.github/workflows/beta.yml @@ -14,8 +14,9 @@ jobs: pull-requests: write steps: - name: Checkout repository - uses: actions/checkout@v4 + uses: actions/checkout@v6 # kilocode_change with: + persist-credentials: false # kilocode_change fetch-depth: 0 - name: Setup Bun diff --git a/.github/workflows/check-kilo-generated-artifacts.yml b/.github/workflows/check-kilo-generated-artifacts.yml new file mode 100644 index 0000000000..1fa1c083f2 --- /dev/null +++ b/.github/workflows/check-kilo-generated-artifacts.yml @@ -0,0 +1,27 @@ +# kilocode_change - new file +name: Check Kilo generated artifacts + +on: + pull_request: + paths: + - ".gitignore" + - ".github/workflows/check-kilo-generated-artifacts.yml" + - ".kilo/**" + - ".kilocode/**" + - "script/check-kilo-generated-artifacts.ts" + workflow_dispatch: + +jobs: + check: + name: Check Kilo generated artifacts + if: github.repository == 'Kilo-Org/kilocode' + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v6 # kilocode_change + with: + ref: ${{ github.event.pull_request.head.sha || github.sha }} + + - uses: oven-sh/setup-bun@v2 + + - name: Run check + run: bun run script/check-kilo-generated-artifacts.ts diff --git a/.github/workflows/check-md-table-padding.yml b/.github/workflows/check-md-table-padding.yml index 51d6706597..18c3c7aa96 100644 --- a/.github/workflows/check-md-table-padding.yml +++ b/.github/workflows/check-md-table-padding.yml @@ -14,7 +14,7 @@ jobs: if: github.repository == 'Kilo-Org/kilocode' runs-on: ubuntu-latest steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v6 # kilocode_change with: ref: ${{ github.event.pull_request.head.sha || github.sha }} diff --git a/.github/workflows/check-opencode-annotations.yml b/.github/workflows/check-opencode-annotations.yml index 1fa6156e4c..dd9f3dfc4c 100644 --- a/.github/workflows/check-opencode-annotations.yml +++ b/.github/workflows/check-opencode-annotations.yml @@ -6,9 +6,6 @@ on: - ".github/**" - "github/**" - "sdks/vscode/**" - - "packages/app/**" - - "packages/desktop/**" - - "packages/desktop-electron/**" - "packages/extensions/**" - "packages/opencode/**" - "packages/script/**" @@ -24,7 +21,7 @@ jobs: if: github.repository == 'clickzetta/czcode' # czcode_change runs-on: ubuntu-latest steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v6 # kilocode_change with: ref: ${{ github.event.pull_request.head.sha || github.sha }} fetch-depth: 0 @@ -34,9 +31,21 @@ jobs: - name: Check kilocode_change annotations in shared upstream files env: BASE_SHA: ${{ github.event.pull_request.base.sha }} + PR_TITLE: ${{ github.event.pull_request.title }} run: | + # czcode_change start - skip annotation check for upstream merge PRs + if [[ "$PR_TITLE" == merge:* || "$PR_TITLE" == "merge: upstream"* ]]; then + echo "Upstream merge PR detected ('$PR_TITLE') — skipping annotation check." + exit 0 + fi + # czcode_change end if [ -n "$BASE_SHA" ]; then bun run script/check-opencode-annotations.ts --base "$BASE_SHA" else echo "No PR base SHA available (workflow_dispatch without PR context) — skipping." fi + + # kilocode_change start + - name: Check workflow allowlist + run: bun run script/check-workflows.ts + # kilocode_change end diff --git a/.github/workflows/close-issues.yml b/.github/workflows/close-issues.yml index 04b6ae7ac8..aba47e131f 100644 --- a/.github/workflows/close-issues.yml +++ b/.github/workflows/close-issues.yml @@ -12,7 +12,7 @@ jobs: contents: read issues: write steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v6 # kilocode_change - uses: oven-sh/setup-bun@v2 with: diff --git a/.github/workflows/compliance-close.yml b/.github/workflows/compliance-close.yml deleted file mode 100644 index d56d3b4d97..0000000000 --- a/.github/workflows/compliance-close.yml +++ /dev/null @@ -1,96 +0,0 @@ -name: compliance-close - -on: - schedule: - # Run every 30 minutes to check for expired compliance windows - - cron: "*/30 * * * *" - workflow_dispatch: - -permissions: - contents: read - issues: write - pull-requests: write - -jobs: - close-non-compliant: - if: github.repository == 'Kilo-Org/kilocode' && false # kilocode_change - disabled: not needed in kilocode repo - runs-on: ubuntu-latest - steps: - - name: Close non-compliant issues and PRs after 2 hours - uses: actions/github-script@v7 - with: - script: | - const { data: items } = await github.rest.issues.listForRepo({ - owner: context.repo.owner, - repo: context.repo.repo, - labels: 'needs:compliance', - state: 'open', - per_page: 100, - }); - - if (items.length === 0) { - core.info('No open issues/PRs with needs:compliance label'); - return; - } - - const now = Date.now(); - const twoHours = 2 * 60 * 60 * 1000; - - for (const item of items) { - const isPR = !!item.pull_request; - const kind = isPR ? 'PR' : 'issue'; - - const { data: comments } = await github.rest.issues.listComments({ - owner: context.repo.owner, - repo: context.repo.repo, - issue_number: item.number, - }); - - const complianceComment = comments.find(c => c.body.includes('')); - if (!complianceComment) continue; - - const commentAge = now - new Date(complianceComment.created_at).getTime(); - if (commentAge < twoHours) { - core.info(`${kind} #${item.number} still within 2-hour window (${Math.round(commentAge / 60000)}m elapsed)`); - continue; - } - - const closeMessage = isPR - ? 'This pull request has been automatically closed because it was not updated to meet our [contributing guidelines](../blob/dev/CONTRIBUTING.md) within the 2-hour window.\n\nFeel free to open a new pull request that follows our guidelines.' - : 'This issue has been automatically closed because it was not updated to meet our [contributing guidelines](../blob/dev/CONTRIBUTING.md) within the 2-hour window.\n\nFeel free to open a new issue that follows our issue templates.'; - - await github.rest.issues.createComment({ - owner: context.repo.owner, - repo: context.repo.repo, - issue_number: item.number, - body: closeMessage, - }); - - try { - await github.rest.issues.removeLabel({ - owner: context.repo.owner, - repo: context.repo.repo, - issue_number: item.number, - name: 'needs:compliance', - }); - } catch (e) {} - - if (isPR) { - await github.rest.pulls.update({ - owner: context.repo.owner, - repo: context.repo.repo, - pull_number: item.number, - state: 'closed', - }); - } else { - await github.rest.issues.update({ - owner: context.repo.owner, - repo: context.repo.repo, - issue_number: item.number, - state: 'closed', - state_reason: 'not_planned', - }); - } - - core.info(`Closed non-compliant ${kind} #${item.number} after 2-hour window`); - } diff --git a/.github/workflows/containers.yml b/.github/workflows/containers.yml index 03bc186787..a261d6f5b3 100644 --- a/.github/workflows/containers.yml +++ b/.github/workflows/containers.yml @@ -26,7 +26,7 @@ jobs: OWNER: ${{ github.repository_owner }} # kilocode_change end - - uses: actions/checkout@v4 + - uses: actions/checkout@v6 # kilocode_change - uses: ./.github/actions/setup-bun diff --git a/.github/workflows/daily-issues-recap.yml b/.github/workflows/daily-issues-recap.yml deleted file mode 100644 index 185de2f5b3..0000000000 --- a/.github/workflows/daily-issues-recap.yml +++ /dev/null @@ -1,173 +0,0 @@ -name: daily-issues-recap - -on: - schedule: - # Run at 6 PM EST (23:00 UTC, or 22:00 UTC during daylight saving) - - cron: "0 23 * * *" - workflow_dispatch: # Allow manual trigger for testing - -jobs: - daily-recap: - # kilocode_change start - disabled for kilo-cli fork (OpenCode Discord integration) - if: false - # kilocode_change end - runs-on: blacksmith-4vcpu-ubuntu-2404 - permissions: - contents: read - issues: read - steps: - - name: Checkout repository - uses: actions/checkout@v4 - with: - fetch-depth: 1 - - - uses: ./.github/actions/setup-bun - - - name: Install opencode - run: curl -fsSL https://opencode.ai/install | bash - - - name: Generate daily issues recap - id: recap - env: - OPENCODE_API_KEY: ${{ secrets.OPENCODE_API_KEY }} - GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} - KILO_PERMISSION: | - { - "bash": { - "*": "deny", - "gh issue*": "allow", - "gh search*": "allow" - }, - "webfetch": "deny", - "edit": "deny", - "write": "deny" - } - run: | - # Get today's date range - TODAY=$(date -u +%Y-%m-%d) - - opencode run -m kilo/anthropic/claude-sonnet-4.5 "Generate a daily issues recap for the OpenCode repository. - - TODAY'S DATE: ${TODAY} - - STEP 1: Gather today's issues - Search for all OPEN issues created today (${TODAY}) using: - gh issue list --repo ${{ github.repository }} --state open --search \"created:${TODAY}\" --json number,title,body,labels,state,comments,createdAt,author --limit 500 - - IMPORTANT: EXCLUDE all issues authored by Anomaly team members. Filter out issues where the author login matches ANY of these: - adamdotdevin, Brendonovich, fwang, Hona, iamdavidhill, jayair, kitlangton, kommander, MrMushrooooom, R44VC0RP, rekram1-node, thdxr - This recap is specifically for COMMUNITY (external) issues only. - - STEP 2: Analyze and categorize - For each issue created today, categorize it: - - **Severity Assessment:** - - CRITICAL: Crashes, data loss, security issues, blocks major functionality - - HIGH: Significant bugs affecting many users, important features broken - - MEDIUM: Bugs with workarounds, minor features broken - - LOW: Minor issues, cosmetic, nice-to-haves - - **Activity Assessment:** - - Note issues with high comment counts or engagement - - Note issues from repeat reporters (check if author has filed before) - - STEP 3: Cross-reference with existing issues - For issues that seem like feature requests or recurring bugs: - - Search for similar older issues to identify patterns - - Note if this is a frequently requested feature - - Identify any issues that are duplicates of long-standing requests - - STEP 4: Generate the recap - Create a structured recap with these sections: - - ===DISCORD_START=== - **Daily Issues Recap - ${TODAY}** - - **Summary Stats** - - Total issues opened today: [count] - - By category: [bugs/features/questions] - - **Critical/High Priority Issues** - [List any CRITICAL or HIGH severity issues with brief descriptions and issue numbers] - - **Most Active/Discussed** - [Issues with significant engagement or from active community members] - - **Trending Topics** - [Patterns noticed - e.g., 'Multiple reports about X', 'Continued interest in Y feature'] - - **Duplicates & Related** - [Issues that relate to existing open issues] - ===DISCORD_END=== - - STEP 5: Format for Discord - Format the recap as a Discord-compatible message: - - Use Discord markdown (**, __, etc.) - - BE EXTREMELY CONCISE - this is an EOD summary, not a detailed report - - Use hyperlinked issue numbers with suppressed embeds: [#1234]() - - Group related issues on single lines where possible - - Add emoji sparingly for critical items only - - HARD LIMIT: Keep under 1800 characters total - - Skip sections that have nothing notable (e.g., if no critical issues, omit that section) - - Prioritize signal over completeness - only surface what matters - - OUTPUT: Output ONLY the content between ===DISCORD_START=== and ===DISCORD_END=== markers. Include the markers so I can extract it." > /tmp/recap_raw.txt - - # Extract only the Discord message between markers - sed -n '/===DISCORD_START===/,/===DISCORD_END===/p' /tmp/recap_raw.txt | grep -v '===DISCORD' > /tmp/recap.txt - - echo "recap_file=/tmp/recap.txt" >> $GITHUB_OUTPUT - - - name: Post to Discord - env: - DISCORD_WEBHOOK_URL: ${{ secrets.DISCORD_ISSUES_WEBHOOK_URL }} - run: | - if [ -z "$DISCORD_WEBHOOK_URL" ]; then - echo "Warning: DISCORD_ISSUES_WEBHOOK_URL secret not set, skipping Discord post" - cat /tmp/recap.txt - exit 0 - fi - - # Read the recap - RECAP_RAW=$(cat /tmp/recap.txt) - RECAP_LENGTH=${#RECAP_RAW} - - echo "Recap length: ${RECAP_LENGTH} chars" - - # Function to post a message to Discord - post_to_discord() { - local msg="$1" - local content=$(echo "$msg" | jq -Rs '.') - curl -s -H "Content-Type: application/json" \ - -X POST \ - -d "{\"content\": ${content}}" \ - "$DISCORD_WEBHOOK_URL" - sleep 1 - } - - # If under limit, send as single message - if [ "$RECAP_LENGTH" -le 1950 ]; then - post_to_discord "$RECAP_RAW" - else - echo "Splitting into multiple messages..." - remaining="$RECAP_RAW" - while [ ${#remaining} -gt 0 ]; do - if [ ${#remaining} -le 1950 ]; then - post_to_discord "$remaining" - break - else - chunk="${remaining:0:1900}" - last_newline=$(echo "$chunk" | grep -bo $'\n' | tail -1 | cut -d: -f1) - if [ -n "$last_newline" ] && [ "$last_newline" -gt 500 ]; then - chunk="${remaining:0:$last_newline}" - remaining="${remaining:$((last_newline+1))}" - else - chunk="${remaining:0:1900}" - remaining="${remaining:1900}" - fi - post_to_discord "$chunk" - fi - done - fi - - echo "Posted daily recap to Discord" diff --git a/.github/workflows/daily-pr-recap.yml b/.github/workflows/daily-pr-recap.yml deleted file mode 100644 index 7049531b5d..0000000000 --- a/.github/workflows/daily-pr-recap.yml +++ /dev/null @@ -1,176 +0,0 @@ -name: daily-pr-recap - -on: - schedule: - # Run at 5pm EST (22:00 UTC, or 21:00 UTC during daylight saving) - - cron: "0 22 * * *" - workflow_dispatch: # Allow manual trigger for testing - -jobs: - pr-recap: - # kilocode_change start - disabled for kilo-cli fork (OpenCode Discord integration) - if: false - # kilocode_change end - runs-on: blacksmith-4vcpu-ubuntu-2404 - permissions: - contents: read - pull-requests: read - steps: - - name: Checkout repository - uses: actions/checkout@v4 - with: - fetch-depth: 1 - - - uses: ./.github/actions/setup-bun - - - name: Install opencode - run: curl -fsSL https://opencode.ai/install | bash - - - name: Generate daily PR recap - id: recap - env: - OPENCODE_API_KEY: ${{ secrets.OPENCODE_API_KEY }} - GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} - KILO_PERMISSION: | - { - "bash": { - "*": "deny", - "gh pr*": "allow", - "gh search*": "allow" - }, - "webfetch": "deny", - "edit": "deny", - "write": "deny" - } - run: | - TODAY=$(date -u +%Y-%m-%d) - - opencode run -m kilo/anthropic/claude-sonnet-4.5 "Generate a daily PR activity recap for the OpenCode repository. - - TODAY'S DATE: ${TODAY} - - STEP 1: Gather PR data - Run these commands to gather PR information. ONLY include OPEN PRs created or updated TODAY (${TODAY}): - - # Open PRs created today - gh pr list --repo ${{ github.repository }} --state open --search \"created:${TODAY}\" --json number,title,author,labels,createdAt,updatedAt,reviewDecision,isDraft,additions,deletions --limit 100 - - # Open PRs with activity today (updated today) - gh pr list --repo ${{ github.repository }} --state open --search \"updated:${TODAY}\" --json number,title,author,labels,createdAt,updatedAt,reviewDecision,isDraft,additions,deletions --limit 100 - - IMPORTANT: EXCLUDE all PRs authored by Anomaly team members. Filter out PRs where the author login matches ANY of these: - adamdotdevin, Brendonovich, fwang, Hona, iamdavidhill, jayair, kitlangton, kommander, MrMushrooooom, R44VC0RP, rekram1-node, thdxr - This recap is specifically for COMMUNITY (external) contributions only. - - - - STEP 2: For high-activity PRs, check comment counts - For promising PRs, run: - gh pr view [NUMBER] --repo ${{ github.repository }} --json comments --jq '[.comments[] | select(.author.login != \"copilot-pull-request-reviewer\" and .author.login != \"github-actions\")] | length' - - IMPORTANT: When counting comments/activity, EXCLUDE these bot accounts: - - copilot-pull-request-reviewer - - github-actions - - STEP 3: Identify what matters (ONLY from today's PRs) - - **Bug Fixes From Today:** - - PRs with 'fix' or 'bug' in title created/updated today - - Small bug fixes (< 100 lines changed) that are easy to review - - Bug fixes from community contributors - - **High Activity Today:** - - PRs with significant human comments today (excluding bots listed above) - - PRs with back-and-forth discussion today - - **Quick Wins:** - - Small PRs (< 50 lines) that are approved or nearly approved - - PRs that just need a final review - - STEP 4: Generate the recap - Create a structured recap: - - ===DISCORD_START=== - **Daily PR Recap - ${TODAY}** - - **New PRs Today** - [PRs opened today - group by type: bug fixes, features, etc.] - - **Active PRs Today** - [PRs with activity/updates today - significant discussion] - - **Quick Wins** - [Small PRs ready to merge] - ===DISCORD_END=== - - STEP 5: Format for Discord - - Use Discord markdown (**, __, etc.) - - BE EXTREMELY CONCISE - surface what we might miss - - Use hyperlinked PR numbers with suppressed embeds: [#1234]() - - Include PR author: [#1234]() (@author) - - For bug fixes, add brief description of what it fixes - - Show line count for quick wins: \"(+15/-3 lines)\" - - HARD LIMIT: Keep under 1800 characters total - - Skip empty sections - - Focus on PRs that need human eyes - - OUTPUT: Output ONLY the content between ===DISCORD_START=== and ===DISCORD_END=== markers. Include the markers so I can extract it." > /tmp/pr_recap_raw.txt - - # Extract only the Discord message between markers - sed -n '/===DISCORD_START===/,/===DISCORD_END===/p' /tmp/pr_recap_raw.txt | grep -v '===DISCORD' > /tmp/pr_recap.txt - - echo "recap_file=/tmp/pr_recap.txt" >> $GITHUB_OUTPUT - - - name: Post to Discord - env: - DISCORD_WEBHOOK_URL: ${{ secrets.DISCORD_ISSUES_WEBHOOK_URL }} - run: | - if [ -z "$DISCORD_WEBHOOK_URL" ]; then - echo "Warning: DISCORD_ISSUES_WEBHOOK_URL secret not set, skipping Discord post" - cat /tmp/pr_recap.txt - exit 0 - fi - - # Read the recap - RECAP_RAW=$(cat /tmp/pr_recap.txt) - RECAP_LENGTH=${#RECAP_RAW} - - echo "Recap length: ${RECAP_LENGTH} chars" - - # Function to post a message to Discord - post_to_discord() { - local msg="$1" - local content=$(echo "$msg" | jq -Rs '.') - curl -s -H "Content-Type: application/json" \ - -X POST \ - -d "{\"content\": ${content}}" \ - "$DISCORD_WEBHOOK_URL" - sleep 1 - } - - # If under limit, send as single message - if [ "$RECAP_LENGTH" -le 1950 ]; then - post_to_discord "$RECAP_RAW" - else - echo "Splitting into multiple messages..." - remaining="$RECAP_RAW" - while [ ${#remaining} -gt 0 ]; do - if [ ${#remaining} -le 1950 ]; then - post_to_discord "$remaining" - break - else - chunk="${remaining:0:1900}" - last_newline=$(echo "$chunk" | grep -bo $'\n' | tail -1 | cut -d: -f1) - if [ -n "$last_newline" ] && [ "$last_newline" -gt 500 ]; then - chunk="${remaining:0:$last_newline}" - remaining="${remaining:$((last_newline+1))}" - else - chunk="${remaining:0:1900}" - remaining="${remaining:1900}" - fi - post_to_discord "$chunk" - fi - done - fi - - echo "Posted daily PR recap to Discord" diff --git a/.github/workflows/disabled/compliance-close.yml.disabled b/.github/workflows/disabled/compliance-close.yml.disabled new file mode 100644 index 0000000000..c3bcf9f686 --- /dev/null +++ b/.github/workflows/disabled/compliance-close.yml.disabled @@ -0,0 +1,95 @@ +name: compliance-close + +on: + schedule: + # Run every 30 minutes to check for expired compliance windows + - cron: "*/30 * * * *" + workflow_dispatch: + +permissions: + contents: read + issues: write + pull-requests: write + +jobs: + close-non-compliant: + runs-on: ubuntu-latest + steps: + - name: Close non-compliant issues and PRs after 2 hours + uses: actions/github-script@v7 + with: + script: | + const { data: items } = await github.rest.issues.listForRepo({ + owner: context.repo.owner, + repo: context.repo.repo, + labels: 'needs:compliance', + state: 'open', + per_page: 100, + }); + + if (items.length === 0) { + core.info('No open issues/PRs with needs:compliance label'); + return; + } + + const now = Date.now(); + const twoHours = 2 * 60 * 60 * 1000; + + for (const item of items) { + const isPR = !!item.pull_request; + const kind = isPR ? 'PR' : 'issue'; + + const { data: comments } = await github.rest.issues.listComments({ + owner: context.repo.owner, + repo: context.repo.repo, + issue_number: item.number, + }); + + const complianceComment = comments.find(c => c.body.includes('')); + if (!complianceComment) continue; + + const commentAge = now - new Date(complianceComment.created_at).getTime(); + if (commentAge < twoHours) { + core.info(`${kind} #${item.number} still within 2-hour window (${Math.round(commentAge / 60000)}m elapsed)`); + continue; + } + + const closeMessage = isPR + ? 'This pull request has been automatically closed because it was not updated to meet our [contributing guidelines](../blob/dev/CONTRIBUTING.md) within the 2-hour window.\n\nFeel free to open a new pull request that follows our guidelines.' + : 'This issue has been automatically closed because it was not updated to meet our [contributing guidelines](../blob/dev/CONTRIBUTING.md) within the 2-hour window.\n\nFeel free to open a new issue that follows our issue templates.'; + + await github.rest.issues.createComment({ + owner: context.repo.owner, + repo: context.repo.repo, + issue_number: item.number, + body: closeMessage, + }); + + try { + await github.rest.issues.removeLabel({ + owner: context.repo.owner, + repo: context.repo.repo, + issue_number: item.number, + name: 'needs:compliance', + }); + } catch (e) {} + + if (isPR) { + await github.rest.pulls.update({ + owner: context.repo.owner, + repo: context.repo.repo, + pull_number: item.number, + state: 'closed', + }); + } else { + await github.rest.issues.update({ + owner: context.repo.owner, + repo: context.repo.repo, + issue_number: item.number, + state: 'closed', + state_reason: 'not_planned', + }); + } + + core.info(`Closed non-compliant ${kind} #${item.number} after 2-hour window`); + } diff --git a/.github/workflows/disabled/daily-issues-recap.yml.disabled b/.github/workflows/disabled/daily-issues-recap.yml.disabled new file mode 100644 index 0000000000..789712ef51 --- /dev/null +++ b/.github/workflows/disabled/daily-issues-recap.yml.disabled @@ -0,0 +1,170 @@ +name: daily-issues-recap + +on: + schedule: + # Run at 6 PM EST (23:00 UTC, or 22:00 UTC during daylight saving) + - cron: "0 23 * * *" + workflow_dispatch: # Allow manual trigger for testing + +jobs: + daily-recap: + runs-on: blacksmith-4vcpu-ubuntu-2404 + permissions: + contents: read + issues: read + steps: + - name: Checkout repository + uses: actions/checkout@v4 + with: + fetch-depth: 1 + + - uses: ./.github/actions/setup-bun + + - name: Install opencode + run: curl -fsSL https://kilo.ai/install | bash + + - name: Generate daily issues recap + id: recap + env: + OPENCODE_API_KEY: ${{ secrets.OPENCODE_API_KEY }} + GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} + KILO_PERMISSION: | + { + "bash": { + "*": "deny", + "gh issue*": "allow", + "gh search*": "allow" + }, + "webfetch": "deny", + "edit": "deny", + "write": "deny" + } + run: | + # Get today's date range + TODAY=$(date -u +%Y-%m-%d) + + opencode run -m opencode/claude-sonnet-4-5 "Generate a daily issues recap for the Kilo repository. + + TODAY'S DATE: ${TODAY} + + STEP 1: Gather today's issues + Search for all OPEN issues created today (${TODAY}) using: + gh issue list --repo ${{ github.repository }} --state open --search \"created:${TODAY}\" --json number,title,body,labels,state,comments,createdAt,author --limit 500 + + IMPORTANT: EXCLUDE all issues authored by Anomaly team members. Filter out issues where the author login matches ANY of these: + adamdotdevin, Brendonovich, fwang, Hona, iamdavidhill, jayair, kitlangton, kommander, MrMushrooooom, R44VC0RP, rekram1-node, thdxr + This recap is specifically for COMMUNITY (external) issues only. + + STEP 2: Analyze and categorize + For each issue created today, categorize it: + + **Severity Assessment:** + - CRITICAL: Crashes, data loss, security issues, blocks major functionality + - HIGH: Significant bugs affecting many users, important features broken + - MEDIUM: Bugs with workarounds, minor features broken + - LOW: Minor issues, cosmetic, nice-to-haves + + **Activity Assessment:** + - Note issues with high comment counts or engagement + - Note issues from repeat reporters (check if author has filed before) + + STEP 3: Cross-reference with existing issues + For issues that seem like feature requests or recurring bugs: + - Search for similar older issues to identify patterns + - Note if this is a frequently requested feature + - Identify any issues that are duplicates of long-standing requests + + STEP 4: Generate the recap + Create a structured recap with these sections: + + ===DISCORD_START=== + **Daily Issues Recap - ${TODAY}** + + **Summary Stats** + - Total issues opened today: [count] + - By category: [bugs/features/questions] + + **Critical/High Priority Issues** + [List any CRITICAL or HIGH severity issues with brief descriptions and issue numbers] + + **Most Active/Discussed** + [Issues with significant engagement or from active community members] + + **Trending Topics** + [Patterns noticed - e.g., 'Multiple reports about X', 'Continued interest in Y feature'] + + **Duplicates & Related** + [Issues that relate to existing open issues] + ===DISCORD_END=== + + STEP 5: Format for Discord + Format the recap as a Discord-compatible message: + - Use Discord markdown (**, __, etc.) + - BE EXTREMELY CONCISE - this is an EOD summary, not a detailed report + - Use hyperlinked issue numbers with suppressed embeds: [#1234]() + - Group related issues on single lines where possible + - Add emoji sparingly for critical items only + - HARD LIMIT: Keep under 1800 characters total + - Skip sections that have nothing notable (e.g., if no critical issues, omit that section) + - Prioritize signal over completeness - only surface what matters + + OUTPUT: Output ONLY the content between ===DISCORD_START=== and ===DISCORD_END=== markers. Include the markers so I can extract it." > /tmp/recap_raw.txt + + # Extract only the Discord message between markers + sed -n '/===DISCORD_START===/,/===DISCORD_END===/p' /tmp/recap_raw.txt | grep -v '===DISCORD' > /tmp/recap.txt + + echo "recap_file=/tmp/recap.txt" >> $GITHUB_OUTPUT + + - name: Post to Discord + env: + DISCORD_WEBHOOK_URL: ${{ secrets.DISCORD_ISSUES_WEBHOOK_URL }} + run: | + if [ -z "$DISCORD_WEBHOOK_URL" ]; then + echo "Warning: DISCORD_ISSUES_WEBHOOK_URL secret not set, skipping Discord post" + cat /tmp/recap.txt + exit 0 + fi + + # Read the recap + RECAP_RAW=$(cat /tmp/recap.txt) + RECAP_LENGTH=${#RECAP_RAW} + + echo "Recap length: ${RECAP_LENGTH} chars" + + # Function to post a message to Discord + post_to_discord() { + local msg="$1" + local content=$(echo "$msg" | jq -Rs '.') + curl -s -H "Content-Type: application/json" \ + -X POST \ + -d "{\"content\": ${content}}" \ + "$DISCORD_WEBHOOK_URL" + sleep 1 + } + + # If under limit, send as single message + if [ "$RECAP_LENGTH" -le 1950 ]; then + post_to_discord "$RECAP_RAW" + else + echo "Splitting into multiple messages..." + remaining="$RECAP_RAW" + while [ ${#remaining} -gt 0 ]; do + if [ ${#remaining} -le 1950 ]; then + post_to_discord "$remaining" + break + else + chunk="${remaining:0:1900}" + last_newline=$(echo "$chunk" | grep -bo $'\n' | tail -1 | cut -d: -f1) + if [ -n "$last_newline" ] && [ "$last_newline" -gt 500 ]; then + chunk="${remaining:0:$last_newline}" + remaining="${remaining:$((last_newline+1))}" + else + chunk="${remaining:0:1900}" + remaining="${remaining:1900}" + fi + post_to_discord "$chunk" + fi + done + fi + + echo "Posted daily recap to Discord" diff --git a/.github/workflows/disabled/daily-pr-recap.yml.disabled b/.github/workflows/disabled/daily-pr-recap.yml.disabled new file mode 100644 index 0000000000..ff026917f8 --- /dev/null +++ b/.github/workflows/disabled/daily-pr-recap.yml.disabled @@ -0,0 +1,173 @@ +name: daily-pr-recap + +on: + schedule: + # Run at 5pm EST (22:00 UTC, or 21:00 UTC during daylight saving) + - cron: "0 22 * * *" + workflow_dispatch: # Allow manual trigger for testing + +jobs: + pr-recap: + runs-on: blacksmith-4vcpu-ubuntu-2404 + permissions: + contents: read + pull-requests: read + steps: + - name: Checkout repository + uses: actions/checkout@v4 + with: + fetch-depth: 1 + + - uses: ./.github/actions/setup-bun + + - name: Install opencode + run: curl -fsSL https://kilo.ai/install | bash + + - name: Generate daily PR recap + id: recap + env: + OPENCODE_API_KEY: ${{ secrets.OPENCODE_API_KEY }} + GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} + KILO_PERMISSION: | + { + "bash": { + "*": "deny", + "gh pr*": "allow", + "gh search*": "allow" + }, + "webfetch": "deny", + "edit": "deny", + "write": "deny" + } + run: | + TODAY=$(date -u +%Y-%m-%d) + + opencode run -m opencode/claude-sonnet-4-5 "Generate a daily PR activity recap for the Kilo repository. + + TODAY'S DATE: ${TODAY} + + STEP 1: Gather PR data + Run these commands to gather PR information. ONLY include OPEN PRs created or updated TODAY (${TODAY}): + + # Open PRs created today + gh pr list --repo ${{ github.repository }} --state open --search \"created:${TODAY}\" --json number,title,author,labels,createdAt,updatedAt,reviewDecision,isDraft,additions,deletions --limit 100 + + # Open PRs with activity today (updated today) + gh pr list --repo ${{ github.repository }} --state open --search \"updated:${TODAY}\" --json number,title,author,labels,createdAt,updatedAt,reviewDecision,isDraft,additions,deletions --limit 100 + + IMPORTANT: EXCLUDE all PRs authored by Anomaly team members. Filter out PRs where the author login matches ANY of these: + adamdotdevin, Brendonovich, fwang, Hona, iamdavidhill, jayair, kitlangton, kommander, MrMushrooooom, R44VC0RP, rekram1-node, thdxr + This recap is specifically for COMMUNITY (external) contributions only. + + + + STEP 2: For high-activity PRs, check comment counts + For promising PRs, run: + gh pr view [NUMBER] --repo ${{ github.repository }} --json comments --jq '[.comments[] | select(.author.login != \"copilot-pull-request-reviewer\" and .author.login != \"github-actions\")] | length' + + IMPORTANT: When counting comments/activity, EXCLUDE these bot accounts: + - copilot-pull-request-reviewer + - github-actions + + STEP 3: Identify what matters (ONLY from today's PRs) + + **Bug Fixes From Today:** + - PRs with 'fix' or 'bug' in title created/updated today + - Small bug fixes (< 100 lines changed) that are easy to review + - Bug fixes from community contributors + + **High Activity Today:** + - PRs with significant human comments today (excluding bots listed above) + - PRs with back-and-forth discussion today + + **Quick Wins:** + - Small PRs (< 50 lines) that are approved or nearly approved + - PRs that just need a final review + + STEP 4: Generate the recap + Create a structured recap: + + ===DISCORD_START=== + **Daily PR Recap - ${TODAY}** + + **New PRs Today** + [PRs opened today - group by type: bug fixes, features, etc.] + + **Active PRs Today** + [PRs with activity/updates today - significant discussion] + + **Quick Wins** + [Small PRs ready to merge] + ===DISCORD_END=== + + STEP 5: Format for Discord + - Use Discord markdown (**, __, etc.) + - BE EXTREMELY CONCISE - surface what we might miss + - Use hyperlinked PR numbers with suppressed embeds: [#1234]() + - Include PR author: [#1234]() (@author) + - For bug fixes, add brief description of what it fixes + - Show line count for quick wins: \"(+15/-3 lines)\" + - HARD LIMIT: Keep under 1800 characters total + - Skip empty sections + - Focus on PRs that need human eyes + + OUTPUT: Output ONLY the content between ===DISCORD_START=== and ===DISCORD_END=== markers. Include the markers so I can extract it." > /tmp/pr_recap_raw.txt + + # Extract only the Discord message between markers + sed -n '/===DISCORD_START===/,/===DISCORD_END===/p' /tmp/pr_recap_raw.txt | grep -v '===DISCORD' > /tmp/pr_recap.txt + + echo "recap_file=/tmp/pr_recap.txt" >> $GITHUB_OUTPUT + + - name: Post to Discord + env: + DISCORD_WEBHOOK_URL: ${{ secrets.DISCORD_ISSUES_WEBHOOK_URL }} + run: | + if [ -z "$DISCORD_WEBHOOK_URL" ]; then + echo "Warning: DISCORD_ISSUES_WEBHOOK_URL secret not set, skipping Discord post" + cat /tmp/pr_recap.txt + exit 0 + fi + + # Read the recap + RECAP_RAW=$(cat /tmp/pr_recap.txt) + RECAP_LENGTH=${#RECAP_RAW} + + echo "Recap length: ${RECAP_LENGTH} chars" + + # Function to post a message to Discord + post_to_discord() { + local msg="$1" + local content=$(echo "$msg" | jq -Rs '.') + curl -s -H "Content-Type: application/json" \ + -X POST \ + -d "{\"content\": ${content}}" \ + "$DISCORD_WEBHOOK_URL" + sleep 1 + } + + # If under limit, send as single message + if [ "$RECAP_LENGTH" -le 1950 ]; then + post_to_discord "$RECAP_RAW" + else + echo "Splitting into multiple messages..." + remaining="$RECAP_RAW" + while [ ${#remaining} -gt 0 ]; do + if [ ${#remaining} -le 1950 ]; then + post_to_discord "$remaining" + break + else + chunk="${remaining:0:1900}" + last_newline=$(echo "$chunk" | grep -bo $'\n' | tail -1 | cut -d: -f1) + if [ -n "$last_newline" ] && [ "$last_newline" -gt 500 ]; then + chunk="${remaining:0:$last_newline}" + remaining="${remaining:$((last_newline+1))}" + else + chunk="${remaining:0:1900}" + remaining="${remaining:1900}" + fi + post_to_discord "$chunk" + fi + done + fi + + echo "Posted daily PR recap to Discord" diff --git a/.github/workflows/disabled/pr-management.yml.disabled b/.github/workflows/disabled/pr-management.yml.disabled new file mode 100644 index 0000000000..1e331bd6f2 --- /dev/null +++ b/.github/workflows/disabled/pr-management.yml.disabled @@ -0,0 +1,95 @@ +name: pr-management + +on: + pull_request_target: + types: [opened] + +jobs: + check-duplicates: + runs-on: blacksmith-4vcpu-ubuntu-2404 + permissions: + contents: read + pull-requests: write + steps: + - name: Checkout repository + uses: actions/checkout@v4 + with: + fetch-depth: 1 + + - name: Check team membership + id: team-check + run: | + LOGIN="${{ github.event.pull_request.user.login }}" + if [ "$LOGIN" = "opencode-agent[bot]" ] || grep -qxF "$LOGIN" .github/TEAM_MEMBERS; then + echo "is_team=true" >> "$GITHUB_OUTPUT" + echo "Skipping: $LOGIN is a team member or bot" + else + echo "is_team=false" >> "$GITHUB_OUTPUT" + fi + + - name: Setup Bun + if: steps.team-check.outputs.is_team != 'true' + uses: ./.github/actions/setup-bun + + - name: Install dependencies + if: steps.team-check.outputs.is_team != 'true' + run: bun install + + - name: Install opencode + if: steps.team-check.outputs.is_team != 'true' + run: curl -fsSL https://kilo.ai/install | bash + + - name: Build prompt + if: steps.team-check.outputs.is_team != 'true' + env: + GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} + PR_NUMBER: ${{ github.event.pull_request.number }} + run: | + { + echo "Check for duplicate PRs related to this new PR:" + echo "" + echo "CURRENT_PR_NUMBER: $PR_NUMBER" + echo "" + echo "Title: $(gh pr view "$PR_NUMBER" --json title --jq .title)" + echo "" + echo "Description:" + gh pr view "$PR_NUMBER" --json body --jq .body + } > pr_info.txt + + - name: Check for duplicate PRs + if: steps.team-check.outputs.is_team != 'true' + env: + OPENCODE_API_KEY: ${{ secrets.OPENCODE_API_KEY }} + GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} + PR_NUMBER: ${{ github.event.pull_request.number }} + run: | + COMMENT=$(bun script/duplicate-pr.ts -f pr_info.txt "Check the attached file for PR details and search for duplicates") + + if [ "$COMMENT" != "No duplicate PRs found" ]; then + gh pr comment "$PR_NUMBER" --body "_The following comment was made by an LLM, it may be inaccurate:_ + + $COMMENT" + fi + + add-contributor-label: + runs-on: ubuntu-latest + permissions: + pull-requests: write + issues: write + steps: + - name: Add Contributor Label + uses: actions/github-script@v8 + with: + script: | + const isPR = !!context.payload.pull_request; + const issueNumber = isPR ? context.payload.pull_request.number : context.payload.issue.number; + const authorAssociation = isPR ? context.payload.pull_request.author_association : context.payload.issue.author_association; + + if (authorAssociation === 'CONTRIBUTOR') { + await github.rest.issues.addLabels({ + owner: context.repo.owner, + repo: context.repo.repo, + issue_number: issueNumber, + labels: ['contributor'] + }); + } diff --git a/.github/workflows/disabled/pr-standards.yml.disabled b/.github/workflows/disabled/pr-standards.yml.disabled new file mode 100644 index 0000000000..1edbd5d061 --- /dev/null +++ b/.github/workflows/disabled/pr-standards.yml.disabled @@ -0,0 +1,351 @@ +name: pr-standards + +on: + pull_request_target: + types: [opened, edited, synchronize] + +jobs: + check-standards: + runs-on: ubuntu-latest + permissions: + contents: read + pull-requests: write + steps: + - name: Check PR standards + uses: actions/github-script@v7 + with: + script: | + const pr = context.payload.pull_request; + const login = pr.user.login; + + // Skip PRs older than Feb 18, 2026 at 6PM EST (Feb 19, 2026 00:00 UTC) + const cutoff = new Date('2026-02-19T00:00:00Z'); + const prCreated = new Date(pr.created_at); + if (prCreated < cutoff) { + console.log(`Skipping: PR #${pr.number} was created before cutoff (${prCreated.toISOString()})`); + return; + } + + // Check if author is a team member or bot + if (login === 'opencode-agent[bot]') return; + const { data: file } = await github.rest.repos.getContent({ + owner: context.repo.owner, + repo: context.repo.repo, + path: '.github/TEAM_MEMBERS', + ref: 'dev' + }); + const members = Buffer.from(file.content, 'base64').toString().split('\n').map(l => l.trim()).filter(Boolean); + if (members.includes(login)) { + console.log(`Skipping: ${login} is a team member`); + return; + } + + const title = pr.title; + + async function addLabel(label) { + await github.rest.issues.addLabels({ + owner: context.repo.owner, + repo: context.repo.repo, + issue_number: pr.number, + labels: [label] + }); + } + + async function removeLabel(label) { + try { + await github.rest.issues.removeLabel({ + owner: context.repo.owner, + repo: context.repo.repo, + issue_number: pr.number, + name: label + }); + } catch (e) { + // Label wasn't present, ignore + } + } + + async function comment(marker, body) { + const markerText = ``; + const { data: comments } = await github.rest.issues.listComments({ + owner: context.repo.owner, + repo: context.repo.repo, + issue_number: pr.number + }); + + const existing = comments.find(c => c.body.includes(markerText)); + if (existing) return; + + await github.rest.issues.createComment({ + owner: context.repo.owner, + repo: context.repo.repo, + issue_number: pr.number, + body: markerText + '\n' + body + }); + } + + // Step 1: Check title format + // Matches: feat:, feat(scope):, feat (scope):, etc. + const titlePattern = /^(feat|fix|docs|chore|refactor|test)\s*(\([a-zA-Z0-9-]+\))?\s*:/; + const hasValidTitle = titlePattern.test(title); + + if (!hasValidTitle) { + await addLabel('needs:title'); + await comment('title', `Hey! Your PR title \`${title}\` doesn't follow conventional commit format. + + Please update it to start with one of: + - \`feat:\` or \`feat(scope):\` new feature + - \`fix:\` or \`fix(scope):\` bug fix + - \`docs:\` or \`docs(scope):\` documentation changes + - \`chore:\` or \`chore(scope):\` maintenance tasks + - \`refactor:\` or \`refactor(scope):\` code refactoring + - \`test:\` or \`test(scope):\` adding or updating tests + + Where \`scope\` is the package name (e.g., \`app\`, \`desktop\`, \`opencode\`). + + See [CONTRIBUTING.md](../blob/dev/CONTRIBUTING.md#pr-titles) for details.`); + return; + } + + await removeLabel('needs:title'); + + // Step 2: Check for linked issue (skip for docs/refactor/feat PRs) + const skipIssueCheck = /^(docs|refactor|feat)\s*(\([a-zA-Z0-9-]+\))?\s*:/.test(title); + if (skipIssueCheck) { + await removeLabel('needs:issue'); + console.log('Skipping issue check for docs/refactor/feat PR'); + return; + } + const query = ` + query($owner: String!, $repo: String!, $number: Int!) { + repository(owner: $owner, name: $repo) { + pullRequest(number: $number) { + closingIssuesReferences(first: 1) { + totalCount + } + } + } + } + `; + + const result = await github.graphql(query, { + owner: context.repo.owner, + repo: context.repo.repo, + number: pr.number + }); + + const linkedIssues = result.repository.pullRequest.closingIssuesReferences.totalCount; + + if (linkedIssues === 0) { + await addLabel('needs:issue'); + await comment('issue', `Thanks for your contribution! + + This PR doesn't have a linked issue. All PRs must reference an existing issue. + + Please: + 1. Open an issue describing the bug/feature (if one doesn't exist) + 2. Add \`Fixes #\` or \`Closes #\` to this PR description + + See [CONTRIBUTING.md](../blob/dev/CONTRIBUTING.md#issue-first-policy) for details.`); + return; + } + + await removeLabel('needs:issue'); + console.log('PR meets all standards'); + + check-compliance: + runs-on: ubuntu-latest + permissions: + contents: read + pull-requests: write + steps: + - name: Check PR template compliance + uses: actions/github-script@v7 + with: + script: | + const pr = context.payload.pull_request; + const login = pr.user.login; + + // Skip PRs older than Feb 18, 2026 at 6PM EST (Feb 19, 2026 00:00 UTC) + const cutoff = new Date('2026-02-19T00:00:00Z'); + const prCreated = new Date(pr.created_at); + if (prCreated < cutoff) { + console.log(`Skipping: PR #${pr.number} was created before cutoff (${prCreated.toISOString()})`); + return; + } + + // Check if author is a team member or bot + if (login === 'opencode-agent[bot]') return; + const { data: file } = await github.rest.repos.getContent({ + owner: context.repo.owner, + repo: context.repo.repo, + path: '.github/TEAM_MEMBERS', + ref: 'dev' + }); + const members = Buffer.from(file.content, 'base64').toString().split('\n').map(l => l.trim()).filter(Boolean); + if (members.includes(login)) { + console.log(`Skipping: ${login} is a team member`); + return; + } + + const body = pr.body || ''; + const title = pr.title; + const isDocsRefactorOrFeat = /^(docs|refactor|feat)\s*(\([a-zA-Z0-9-]+\))?\s*:/.test(title); + + const issues = []; + + // Check: template sections exist + const hasWhatSection = /### What does this PR do\?/.test(body); + const hasTypeSection = /### Type of change/.test(body); + const hasVerifySection = /### How did you verify your code works\?/.test(body); + const hasChecklistSection = /### Checklist/.test(body); + const hasIssueSection = /### Issue for this PR/.test(body); + + if (!hasWhatSection || !hasTypeSection || !hasVerifySection || !hasChecklistSection || !hasIssueSection) { + issues.push('PR description is missing required template sections. Please use the [PR template](../blob/dev/.github/pull_request_template.md).'); + } + + // Check: "What does this PR do?" has real content (not just placeholder text) + if (hasWhatSection) { + const whatMatch = body.match(/### What does this PR do\?\s*\n([\s\S]*?)(?=###|$)/); + const whatContent = whatMatch ? whatMatch[1].trim() : ''; + const placeholder = 'Please provide a description of the issue'; + const onlyPlaceholder = whatContent.includes(placeholder) && whatContent.replace(placeholder, '').replace(/[*\s]/g, '').length < 20; + if (!whatContent || onlyPlaceholder) { + issues.push('"What does this PR do?" section is empty or only contains placeholder text. Please describe your changes.'); + } + } + + // Check: at least one "Type of change" checkbox is checked + if (hasTypeSection) { + const typeMatch = body.match(/### Type of change\s*\n([\s\S]*?)(?=###|$)/); + const typeContent = typeMatch ? typeMatch[1] : ''; + const hasCheckedBox = /- \[x\]/i.test(typeContent); + if (!hasCheckedBox) { + issues.push('No "Type of change" checkbox is checked. Please select at least one.'); + } + } + + // Check: issue reference (skip for docs/refactor/feat) + if (!isDocsRefactorOrFeat && hasIssueSection) { + const issueMatch = body.match(/### Issue for this PR\s*\n([\s\S]*?)(?=###|$)/); + const issueContent = issueMatch ? issueMatch[1].trim() : ''; + const hasIssueRef = /(closes|fixes|resolves)\s+#\d+/i.test(issueContent) || /#\d+/.test(issueContent); + if (!hasIssueRef) { + issues.push('No issue referenced. Please add `Closes #` linking to the relevant issue.'); + } + } + + // Check: "How did you verify" has content + if (hasVerifySection) { + const verifyMatch = body.match(/### How did you verify your code works\?\s*\n([\s\S]*?)(?=###|$)/); + const verifyContent = verifyMatch ? verifyMatch[1].trim() : ''; + if (!verifyContent) { + issues.push('"How did you verify your code works?" section is empty. Please explain how you tested.'); + } + } + + // Check: checklist boxes are checked + if (hasChecklistSection) { + const checklistMatch = body.match(/### Checklist\s*\n([\s\S]*?)(?=###|$)/); + const checklistContent = checklistMatch ? checklistMatch[1] : ''; + const unchecked = (checklistContent.match(/- \[ \]/g) || []).length; + const checked = (checklistContent.match(/- \[x\]/gi) || []).length; + if (checked < 2) { + issues.push('Not all checklist items are checked. Please confirm you have tested locally and have not included unrelated changes.'); + } + } + + // Helper functions + async function addLabel(label) { + await github.rest.issues.addLabels({ + owner: context.repo.owner, + repo: context.repo.repo, + issue_number: pr.number, + labels: [label] + }); + } + + async function removeLabel(label) { + try { + await github.rest.issues.removeLabel({ + owner: context.repo.owner, + repo: context.repo.repo, + issue_number: pr.number, + name: label + }); + } catch (e) {} + } + + const hasComplianceLabel = pr.labels.some(l => l.name === 'needs:compliance'); + + if (issues.length > 0) { + // Non-compliant + if (!hasComplianceLabel) { + await addLabel('needs:compliance'); + } + + const marker = ''; + const { data: comments } = await github.rest.issues.listComments({ + owner: context.repo.owner, + repo: context.repo.repo, + issue_number: pr.number + }); + const existing = comments.find(c => c.body.includes(marker)); + + const body_text = `${marker} + This PR doesn't fully meet our [contributing guidelines](../blob/dev/CONTRIBUTING.md) and [PR template](../blob/dev/.github/pull_request_template.md). + + **What needs to be fixed:** + ${issues.map(i => `- ${i}`).join('\n')} + + Please edit this PR description to address the above within **2 hours**, or it will be automatically closed. + + If you believe this was flagged incorrectly, please let a maintainer know.`; + + if (existing) { + await github.rest.issues.updateComment({ + owner: context.repo.owner, + repo: context.repo.repo, + comment_id: existing.id, + body: body_text + }); + } else { + await github.rest.issues.createComment({ + owner: context.repo.owner, + repo: context.repo.repo, + issue_number: pr.number, + body: body_text + }); + } + + console.log(`PR #${pr.number} is non-compliant: ${issues.join(', ')}`); + } else if (hasComplianceLabel) { + // Was non-compliant, now fixed + await removeLabel('needs:compliance'); + + const { data: comments } = await github.rest.issues.listComments({ + owner: context.repo.owner, + repo: context.repo.repo, + issue_number: pr.number + }); + const marker = ''; + const existing = comments.find(c => c.body.includes(marker)); + if (existing) { + await github.rest.issues.deleteComment({ + owner: context.repo.owner, + repo: context.repo.repo, + comment_id: existing.id + }); + } + + await github.rest.issues.createComment({ + owner: context.repo.owner, + repo: context.repo.repo, + issue_number: pr.number, + body: 'Thanks for updating your PR! It now meets our contributing guidelines. :+1:' + }); + + console.log(`PR #${pr.number} is now compliant, label removed`); + } else { + console.log(`PR #${pr.number} is compliant`); + } diff --git a/.github/workflows/disabled/storybook.yml.disabled b/.github/workflows/disabled/storybook.yml.disabled new file mode 100644 index 0000000000..6d143a8a22 --- /dev/null +++ b/.github/workflows/disabled/storybook.yml.disabled @@ -0,0 +1,38 @@ +name: storybook + +on: + push: + branches: [dev] + paths: + - ".github/workflows/storybook.yml" + - "package.json" + - "bun.lock" + - "packages/storybook/**" + - "packages/ui/**" + pull_request: + branches: [dev] + paths: + - ".github/workflows/storybook.yml" + - "package.json" + - "bun.lock" + - "packages/storybook/**" + - "packages/ui/**" + workflow_dispatch: + +concurrency: + group: ${{ github.workflow }}-${{ github.ref }} + cancel-in-progress: true + +jobs: + build: + name: storybook build + runs-on: blacksmith-4vcpu-ubuntu-2404 + steps: + - name: Checkout repository + uses: actions/checkout@v4 + + - name: Setup Bun + uses: ./.github/actions/setup-bun + + - name: Build Storybook + run: bun --cwd packages/storybook build diff --git a/.github/workflows/docs-build.yml b/.github/workflows/docs-build.yml index b485662e48..316fa74838 100644 --- a/.github/workflows/docs-build.yml +++ b/.github/workflows/docs-build.yml @@ -16,7 +16,7 @@ jobs: runs-on: blacksmith-4vcpu-ubuntu-2404 steps: - name: Checkout repository - uses: actions/checkout@v4 + uses: actions/checkout@v6 # kilocode_change - name: Setup Bun uses: ./.github/actions/setup-bun diff --git a/.github/workflows/docs-check-links.yml b/.github/workflows/docs-check-links.yml index 75b47a6707..4eb00f444a 100644 --- a/.github/workflows/docs-check-links.yml +++ b/.github/workflows/docs-check-links.yml @@ -17,7 +17,7 @@ jobs: if: github.repository == 'Kilo-Org/kilocode' runs-on: ubuntu-latest steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v6 # kilocode_change - name: Link Checker uses: Kilo-Org/lychee-action@8646ba30535128ac92d33dfc9133794bfdd9b411 diff --git a/.github/workflows/duplicate-issues.yml b/.github/workflows/duplicate-issues.yml index dfc5039ef2..cf02d8e0a4 100644 --- a/.github/workflows/duplicate-issues.yml +++ b/.github/workflows/duplicate-issues.yml @@ -13,7 +13,7 @@ jobs: issues: write steps: - name: Checkout repository - uses: actions/checkout@v4 + uses: actions/checkout@v6 # kilocode_change with: fetch-depth: 1 @@ -128,7 +128,7 @@ jobs: issues: write steps: - name: Checkout repository - uses: actions/checkout@v4 + uses: actions/checkout@v6 # kilocode_change with: fetch-depth: 1 diff --git a/.github/workflows/generate.yml b/.github/workflows/generate.yml index 25b03a7bfc..2a3f221c17 100644 --- a/.github/workflows/generate.yml +++ b/.github/workflows/generate.yml @@ -14,7 +14,11 @@ jobs: pull-requests: write steps: - name: Checkout repository - uses: actions/checkout@v4 + uses: actions/checkout@v6 # kilocode_change + # kilocode_change start + with: + persist-credentials: false + # kilocode_change end - name: Setup Bun uses: ./.github/actions/setup-bun diff --git a/.github/workflows/nix-eval.yml b/.github/workflows/nix-eval.yml index dbe5f0bfd0..51c467e1b2 100644 --- a/.github/workflows/nix-eval.yml +++ b/.github/workflows/nix-eval.yml @@ -40,8 +40,6 @@ jobs: SYSTEMS="x86_64-linux aarch64-linux x86_64-darwin aarch64-darwin" PACKAGES="kilo" - # TODO: move 'desktop' to PACKAGES when #11755 is fixed - OPTIONAL_PACKAGES="desktop" echo "" echo "=== Evaluating packages for all systems ===" @@ -61,23 +59,6 @@ jobs: done done - echo "" - echo "=== Evaluating optional packages ===" - for system in $SYSTEMS; do - echo "" - echo "--- $system ---" - for pkg in $OPTIONAL_PACKAGES; do - printf " %s: " "$pkg" - if output=$(nix eval ".#packages.$system.$pkg.drvPath" --raw 2>&1); then - echo "✓" - else - echo "✗" - echo "::warning::Evaluation failed for packages.$system.$pkg" - echo "$output" - fi - done - done - echo "" echo "=== Evaluating devShells for all systems ===" for system in $SYSTEMS; do diff --git a/.github/workflows/nix-hashes.yml b/.github/workflows/nix-hashes.yml index cfa4ee633a..be48bb895e 100644 --- a/.github/workflows/nix-hashes.yml +++ b/.github/workflows/nix-hashes.yml @@ -73,7 +73,7 @@ jobs: echo "Computed hash for ${SYSTEM}: $HASH" - name: Upload hash - uses: actions/upload-artifact@v4 + uses: actions/upload-artifact@v7 # kilocode_change with: name: hash-${{ matrix.system }} path: hash.txt @@ -86,7 +86,7 @@ jobs: steps: - name: Checkout repository - uses: actions/checkout@v4 + uses: actions/checkout@v6 # kilocode_change with: persist-credentials: false fetch-depth: 0 @@ -106,7 +106,7 @@ jobs: git pull --rebase --autostash origin "$GITHUB_REF_NAME" - name: Download hash artifacts - uses: actions/download-artifact@v4 + uses: actions/download-artifact@v8 # kilocode_change with: path: hashes pattern: hash-* diff --git a/.github/workflows/pr-management.yml b/.github/workflows/pr-management.yml deleted file mode 100644 index 7abe0a0ac1..0000000000 --- a/.github/workflows/pr-management.yml +++ /dev/null @@ -1,113 +0,0 @@ -name: pr-management - -on: - pull_request_target: - types: [opened] - -jobs: - # kilocode_change start - check-author: - if: github.repository == 'Kilo-Org/kilocode' && false # kilocode_change - disabled: not needed in kilocode repo - uses: ./.github/workflows/check-org-member.yml - with: - username: ${{ github.event.pull_request.user.login }} - secrets: - kilo-maintainer-app-id: ${{ secrets.KILO_MAINTAINER_APP_ID }} - kilo-maintainer-app-secret: ${{ secrets.KILO_MAINTAINER_APP_SECRET }} - - check-duplicates: - needs: check-author - if: github.repository == 'Kilo-Org/kilocode' && false # kilocode_change - disabled: not needed in kilocode repo - # kilocode_change end - runs-on: blacksmith-4vcpu-ubuntu-2404 - permissions: - contents: read - pull-requests: write - steps: - - name: Checkout repository - uses: actions/checkout@v4 - with: - fetch-depth: 1 - - - name: Check team membership - id: team-check - run: | - LOGIN="${{ github.event.pull_request.user.login }}" - if [ "$LOGIN" = "opencode-agent[bot]" ] || grep -qxF "$LOGIN" .github/TEAM_MEMBERS; then - echo "is_team=true" >> "$GITHUB_OUTPUT" - echo "Skipping: $LOGIN is a team member or bot" - else - echo "is_team=false" >> "$GITHUB_OUTPUT" - fi - - - name: Setup Bun - if: steps.team-check.outputs.is_team != 'true' - uses: ./.github/actions/setup-bun - - - name: Install dependencies - if: steps.team-check.outputs.is_team != 'true' - run: bun install - - # kilocode_change start - - name: Setup Kilo - uses: ./.github/actions/setup-kilo - # kilocode_change end - - - name: Build prompt - if: steps.team-check.outputs.is_team != 'true' - env: - GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} - PR_NUMBER: ${{ github.event.pull_request.number }} - run: | - { - echo "Check for duplicate PRs related to this new PR:" - echo "" - echo "CURRENT_PR_NUMBER: $PR_NUMBER" - echo "" - echo "Title: $(gh pr view "$PR_NUMBER" --json title --jq .title)" - echo "" - echo "Description:" - gh pr view "$PR_NUMBER" --json body --jq .body - } > pr_info.txt - - - name: Check for duplicate PRs - if: steps.team-check.outputs.is_team != 'true' - env: - # kilocode_change start - KILO_API_KEY: ${{ secrets.KILO_API_KEY }} - KILO_ORG_ID: ${{ secrets.KILO_ORG_ID }} - # kilocode_change end - GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} - PR_NUMBER: ${{ github.event.pull_request.number }} - run: | - COMMENT=$(bun script/duplicate-pr.ts -f pr_info.txt "Check the attached file for PR details and search for duplicates") - - if [ "$COMMENT" != "No duplicate PRs found" ]; then - gh pr comment "$PR_NUMBER" --body "_The following comment was made by an LLM, it may be inaccurate:_ - - $COMMENT" - fi - - add-contributor-label: - if: github.repository == 'Kilo-Org/kilocode' && false # kilocode_change - disabled: not needed in kilocode repo - runs-on: blacksmith-2vcpu-ubuntu-2404 # kilocode_change - permissions: - pull-requests: write - issues: write - steps: - - name: Add Contributor Label - uses: actions/github-script@v8 - with: - script: | - const isPR = !!context.payload.pull_request; - const issueNumber = isPR ? context.payload.pull_request.number : context.payload.issue.number; - const authorAssociation = isPR ? context.payload.pull_request.author_association : context.payload.issue.author_association; - - if (authorAssociation === 'CONTRIBUTOR') { - await github.rest.issues.addLabels({ - owner: context.repo.owner, - repo: context.repo.repo, - issue_number: issueNumber, - labels: ['contributor'] - }); - } diff --git a/.github/workflows/pr-standards.yml b/.github/workflows/pr-standards.yml deleted file mode 100644 index 75fb650144..0000000000 --- a/.github/workflows/pr-standards.yml +++ /dev/null @@ -1,340 +0,0 @@ -name: pr-standards - -on: - pull_request_target: - types: [opened, edited, synchronize] - -jobs: - # kilocode_change start - check-author: - if: github.repository == 'Kilo-Org/kilocode' && false # kilocode_change - disabled: not needed in kilocode repo - uses: ./.github/workflows/check-org-member.yml - with: - username: ${{ github.event.pull_request.user.login }} - secrets: - kilo-maintainer-app-id: ${{ secrets.KILO_MAINTAINER_APP_ID }} - kilo-maintainer-app-secret: ${{ secrets.KILO_MAINTAINER_APP_SECRET }} - - check-standards: - needs: check-author - if: github.repository == 'Kilo-Org/kilocode' && false # kilocode_change - disabled: not needed in kilocode repo - # kilocode_change end - runs-on: blacksmith-4vcpu-ubuntu-2404 - permissions: - pull-requests: write - steps: - - name: Check PR standards - uses: actions/github-script@v7 - with: - script: | - const pr = context.payload.pull_request; - const title = pr.title; - - async function addLabel(label) { - await github.rest.issues.addLabels({ - owner: context.repo.owner, - repo: context.repo.repo, - issue_number: pr.number, - labels: [label] - }); - } - - async function removeLabel(label) { - try { - await github.rest.issues.removeLabel({ - owner: context.repo.owner, - repo: context.repo.repo, - issue_number: pr.number, - name: label - }); - } catch (e) { - // Label wasn't present, ignore - } - } - - async function comment(marker, body) { - const markerText = ``; - const { data: comments } = await github.rest.issues.listComments({ - owner: context.repo.owner, - repo: context.repo.repo, - issue_number: pr.number - }); - - const existing = comments.find(c => c.body.includes(markerText)); - if (existing) return; - - await github.rest.issues.createComment({ - owner: context.repo.owner, - repo: context.repo.repo, - issue_number: pr.number, - body: markerText + '\n' + body - }); - } - - // Step 1: Check title format - // Matches: feat:, feat(scope):, feat (scope):, etc. - const titlePattern = /^(feat|fix|docs|chore|refactor|test)\s*(\([a-zA-Z0-9-]+\))?\s*:/; - const hasValidTitle = titlePattern.test(title); - - if (!hasValidTitle) { - await addLabel('needs:title'); - await comment('title', `Hey! Your PR title \`${title}\` doesn't follow conventional commit format. - - Please update it to start with one of: - - \`feat:\` or \`feat(scope):\` new feature - - \`fix:\` or \`fix(scope):\` bug fix - - \`docs:\` or \`docs(scope):\` documentation changes - - \`chore:\` or \`chore(scope):\` maintenance tasks - - \`refactor:\` or \`refactor(scope):\` code refactoring - - \`test:\` or \`test(scope):\` adding or updating tests - - Where \`scope\` is the package name (e.g., \`app\`, \`desktop\`, \`kilo\`). - - See [CONTRIBUTING.md](../blob/main/CONTRIBUTING.md#pr-titles) for details.`); - return; - } - - await removeLabel('needs:title'); - - // Step 2: Check for linked issue (skip for docs/refactor/feat PRs) - const skipIssueCheck = /^(docs|refactor|feat)\s*(\([a-zA-Z0-9-]+\))?\s*:/.test(title); - if (skipIssueCheck) { - await removeLabel('needs:issue'); - console.log('Skipping issue check for docs/refactor/feat PR'); - return; - } - const query = ` - query($owner: String!, $repo: String!, $number: Int!) { - repository(owner: $owner, name: $repo) { - pullRequest(number: $number) { - closingIssuesReferences(first: 1) { - totalCount - } - } - } - } - `; - - const result = await github.graphql(query, { - owner: context.repo.owner, - repo: context.repo.repo, - number: pr.number - }); - - const linkedIssues = result.repository.pullRequest.closingIssuesReferences.totalCount; - - if (linkedIssues === 0) { - await addLabel('needs:issue'); - await comment('issue', `Thanks for your contribution! - - This PR doesn't have a linked issue. All PRs must reference an existing issue. - - Please: - 1. Open an issue describing the bug/feature (if one doesn't exist) - 2. Add \`Fixes #\` or \`Closes #\` to this PR description - - See [CONTRIBUTING.md](../blob/main/CONTRIBUTING.md#issue-first-policy) for details.`); - return; - } - - await removeLabel('needs:issue'); - console.log('PR meets all standards'); - - check-compliance: - if: github.repository == 'Kilo-Org/kilocode' && false # kilocode_change - disabled: not needed in kilocode repo - runs-on: ubuntu-latest - permissions: - contents: read - pull-requests: write - steps: - - name: Check PR template compliance - uses: actions/github-script@v7 - with: - script: | - const pr = context.payload.pull_request; - const login = pr.user.login; - - // Skip PRs older than Feb 18, 2026 at 6PM EST (Feb 19, 2026 00:00 UTC) - const cutoff = new Date('2026-02-19T00:00:00Z'); - const prCreated = new Date(pr.created_at); - if (prCreated < cutoff) { - console.log(`Skipping: PR #${pr.number} was created before cutoff (${prCreated.toISOString()})`); - return; - } - - // Check if author is a team member or bot - if (login === 'opencode-agent[bot]') return; - const { data: file } = await github.rest.repos.getContent({ - owner: context.repo.owner, - repo: context.repo.repo, - path: '.github/TEAM_MEMBERS', - ref: 'dev' - }); - const members = Buffer.from(file.content, 'base64').toString().split('\n').map(l => l.trim()).filter(Boolean); - if (members.includes(login)) { - console.log(`Skipping: ${login} is a team member`); - return; - } - - const body = pr.body || ''; - const title = pr.title; - const isDocsRefactorOrFeat = /^(docs|refactor|feat)\s*(\([a-zA-Z0-9-]+\))?\s*:/.test(title); - - const issues = []; - - // Check: template sections exist - const hasWhatSection = /### What does this PR do\?/.test(body); - const hasTypeSection = /### Type of change/.test(body); - const hasVerifySection = /### How did you verify your code works\?/.test(body); - const hasChecklistSection = /### Checklist/.test(body); - const hasIssueSection = /### Issue for this PR/.test(body); - - if (!hasWhatSection || !hasTypeSection || !hasVerifySection || !hasChecklistSection || !hasIssueSection) { - issues.push('PR description is missing required template sections. Please use the [PR template](../blob/dev/.github/pull_request_template.md).'); - } - - // Check: "What does this PR do?" has real content (not just placeholder text) - if (hasWhatSection) { - const whatMatch = body.match(/### What does this PR do\?\s*\n([\s\S]*?)(?=###|$)/); - const whatContent = whatMatch ? whatMatch[1].trim() : ''; - const placeholder = 'Please provide a description of the issue'; - const onlyPlaceholder = whatContent.includes(placeholder) && whatContent.replace(placeholder, '').replace(/[*\s]/g, '').length < 20; - if (!whatContent || onlyPlaceholder) { - issues.push('"What does this PR do?" section is empty or only contains placeholder text. Please describe your changes.'); - } - } - - // Check: at least one "Type of change" checkbox is checked - if (hasTypeSection) { - const typeMatch = body.match(/### Type of change\s*\n([\s\S]*?)(?=###|$)/); - const typeContent = typeMatch ? typeMatch[1] : ''; - const hasCheckedBox = /- \[x\]/i.test(typeContent); - if (!hasCheckedBox) { - issues.push('No "Type of change" checkbox is checked. Please select at least one.'); - } - } - - // Check: issue reference (skip for docs/refactor/feat) - if (!isDocsRefactorOrFeat && hasIssueSection) { - const issueMatch = body.match(/### Issue for this PR\s*\n([\s\S]*?)(?=###|$)/); - const issueContent = issueMatch ? issueMatch[1].trim() : ''; - const hasIssueRef = /(closes|fixes|resolves)\s+#\d+/i.test(issueContent) || /#\d+/.test(issueContent); - if (!hasIssueRef) { - issues.push('No issue referenced. Please add `Closes #` linking to the relevant issue.'); - } - } - - // Check: "How did you verify" has content - if (hasVerifySection) { - const verifyMatch = body.match(/### How did you verify your code works\?\s*\n([\s\S]*?)(?=###|$)/); - const verifyContent = verifyMatch ? verifyMatch[1].trim() : ''; - if (!verifyContent) { - issues.push('"How did you verify your code works?" section is empty. Please explain how you tested.'); - } - } - - // Check: checklist boxes are checked - if (hasChecklistSection) { - const checklistMatch = body.match(/### Checklist\s*\n([\s\S]*?)(?=###|$)/); - const checklistContent = checklistMatch ? checklistMatch[1] : ''; - const unchecked = (checklistContent.match(/- \[ \]/g) || []).length; - const checked = (checklistContent.match(/- \[x\]/gi) || []).length; - if (checked < 2) { - issues.push('Not all checklist items are checked. Please confirm you have tested locally and have not included unrelated changes.'); - } - } - - // Helper functions - async function addLabel(label) { - await github.rest.issues.addLabels({ - owner: context.repo.owner, - repo: context.repo.repo, - issue_number: pr.number, - labels: [label] - }); - } - - async function removeLabel(label) { - try { - await github.rest.issues.removeLabel({ - owner: context.repo.owner, - repo: context.repo.repo, - issue_number: pr.number, - name: label - }); - } catch (e) {} - } - - const hasComplianceLabel = pr.labels.some(l => l.name === 'needs:compliance'); - - if (issues.length > 0) { - // Non-compliant - if (!hasComplianceLabel) { - await addLabel('needs:compliance'); - } - - const marker = ''; - const { data: comments } = await github.rest.issues.listComments({ - owner: context.repo.owner, - repo: context.repo.repo, - issue_number: pr.number - }); - const existing = comments.find(c => c.body.includes(marker)); - - const body_text = `${marker} - This PR doesn't fully meet our [contributing guidelines](../blob/dev/CONTRIBUTING.md) and [PR template](../blob/dev/.github/pull_request_template.md). - - **What needs to be fixed:** - ${issues.map(i => `- ${i}`).join('\n')} - - Please edit this PR description to address the above within **2 hours**, or it will be automatically closed. - - If you believe this was flagged incorrectly, please let a maintainer know.`; - - if (existing) { - await github.rest.issues.updateComment({ - owner: context.repo.owner, - repo: context.repo.repo, - comment_id: existing.id, - body: body_text - }); - } else { - await github.rest.issues.createComment({ - owner: context.repo.owner, - repo: context.repo.repo, - issue_number: pr.number, - body: body_text - }); - } - - console.log(`PR #${pr.number} is non-compliant: ${issues.join(', ')}`); - } else if (hasComplianceLabel) { - // Was non-compliant, now fixed - await removeLabel('needs:compliance'); - - const { data: comments } = await github.rest.issues.listComments({ - owner: context.repo.owner, - repo: context.repo.repo, - issue_number: pr.number - }); - const marker = ''; - const existing = comments.find(c => c.body.includes(marker)); - if (existing) { - await github.rest.issues.deleteComment({ - owner: context.repo.owner, - repo: context.repo.repo, - comment_id: existing.id - }); - } - - await github.rest.issues.createComment({ - owner: context.repo.owner, - repo: context.repo.repo, - issue_number: pr.number, - body: 'Thanks for updating your PR! It now meets our contributing guidelines. :+1:' - }); - - console.log(`PR #${pr.number} is now compliant, label removed`); - } else { - console.log(`PR #${pr.number} is compliant`); - } diff --git a/.github/workflows/publish-jetbrains.yml b/.github/workflows/publish-jetbrains.yml new file mode 100644 index 0000000000..4db34d33be --- /dev/null +++ b/.github/workflows/publish-jetbrains.yml @@ -0,0 +1,167 @@ +# kilocode_change - new file +name: publish-jetbrains + +on: + push: + tags: + - "jetbrains/*" + +concurrency: + group: publish-jetbrains-${{ github.ref }} + cancel-in-progress: false + +permissions: + contents: write + +env: + FORCE_JAVASCRIPT_ACTIONS_TO_NODE24: true + +jobs: + publish: + if: github.repository == 'Kilo-Org/kilocode' + runs-on: blacksmith-8vcpu-ubuntu-2404 + steps: + - name: Checkout repository + uses: actions/checkout@v4 + + - name: Setup Node + uses: actions/setup-node@v4 + with: + node-version: "24" + + - name: Setup Bun + uses: ./.github/actions/setup-bun + + - name: Setup Java + uses: actions/setup-java@v4 + with: + distribution: temurin + java-version: "21" + + - name: Setup Gradle + uses: gradle/actions/setup-gradle@v4 + + - name: Install build tools + run: | + sudo apt-get update + sudo apt-get install -y patchelf zip + + - name: Validate version tag + id: version + run: | + tag="$GITHUB_REF_NAME" + if [[ "$tag" != jetbrains/v* ]]; then + echo "Unsupported tag '$tag'. Expected jetbrains/v." >&2 + exit 1 + fi + + version="${tag#jetbrains/v}" + if [[ "$version" =~ ^[0-9]+\.[0-9]+\.[0-9]+-rc\.[0-9]+$ ]]; then + { + echo "version=$version" + echo "kind=rc" + echo "marketplace_channel=eap" + echo "cli_channel=rc" + } >> "$GITHUB_OUTPUT" + exit 0 + fi + + if [[ "$version" =~ ^[0-9]+\.[0-9]+\.[0-9]+$ ]]; then + { + echo "version=$version" + echo "kind=stable" + echo "marketplace_channel=default" + echo "cli_channel=latest" + } >> "$GITHUB_OUTPUT" + echo "Stable JetBrains Marketplace publishing is implemented but intentionally disabled; use an rc tag such as jetbrains/v7.0.1-rc.1." >&2 + exit 1 + fi + + echo "Unsupported JetBrains plugin version '$version'. Expected jetbrains/vx.y.z-rc.n or jetbrains/vx.y.z." >&2 + exit 1 + + - name: Validate publishing secrets + run: | + missing=0 + for name in JETBRAINS_MARKETPLACE_TOKEN JETBRAINS_CERTIFICATE_CHAIN JETBRAINS_PRIVATE_KEY JETBRAINS_PRIVATE_KEY_PASSWORD; do + if [[ -z "${!name}" ]]; then + echo "Missing required secret: $name" >&2 + missing=1 + fi + done + exit "$missing" + env: + JETBRAINS_MARKETPLACE_TOKEN: ${{ secrets.JETBRAINS_MARKETPLACE_TOKEN }} + JETBRAINS_CERTIFICATE_CHAIN: ${{ secrets.JETBRAINS_CERTIFICATE_CHAIN }} + JETBRAINS_PRIVATE_KEY: ${{ secrets.JETBRAINS_PRIVATE_KEY }} + JETBRAINS_PRIVATE_KEY_PASSWORD: ${{ secrets.JETBRAINS_PRIVATE_KEY_PASSWORD }} + + - name: Prepare CLI resources + working-directory: packages/kilo-jetbrains + run: bun script/build.ts --production --prepare-cli + env: + KILO_VERSION: ${{ steps.version.outputs.version }} + KILO_CHANNEL: ${{ steps.version.outputs.cli_channel }} + GH_TOKEN: ${{ github.token }} + GH_REPO: ${{ github.repository }} + + - name: Verify plugin + working-directory: packages/kilo-jetbrains + run: ./gradlew verifyPlugin -Pproduction=true -Pkilo.channel="$CHANNEL" + env: + CHANNEL: ${{ steps.version.outputs.marketplace_channel }} + + - name: Publish to JetBrains Marketplace + working-directory: packages/kilo-jetbrains + run: ./gradlew publishPlugin -Pproduction=true -Pkilo.channel="$CHANNEL" + env: + CHANNEL: ${{ steps.version.outputs.marketplace_channel }} + JETBRAINS_MARKETPLACE_TOKEN: ${{ secrets.JETBRAINS_MARKETPLACE_TOKEN }} + JETBRAINS_CERTIFICATE_CHAIN: ${{ secrets.JETBRAINS_CERTIFICATE_CHAIN }} + JETBRAINS_PRIVATE_KEY: ${{ secrets.JETBRAINS_PRIVATE_KEY }} + JETBRAINS_PRIVATE_KEY_PASSWORD: ${{ secrets.JETBRAINS_PRIVATE_KEY_PASSWORD }} + + - name: Resolve plugin archive + id: archive + run: | + mapfile -t signed < <(compgen -G "packages/kilo-jetbrains/build/distributions/*-signed.zip") + if [[ "${#signed[@]}" -eq 1 ]]; then + echo "path=${signed[0]}" >> "$GITHUB_OUTPUT" + exit 0 + fi + if [[ "${#signed[@]}" -gt 1 ]]; then + echo "Expected exactly one signed JetBrains plugin ZIP, found ${#signed[@]}." >&2 + printf '%s\n' "${signed[@]}" >&2 + exit 1 + fi + + mapfile -t files < <(compgen -G "packages/kilo-jetbrains/build/distributions/*.zip") + if [[ "${#files[@]}" -ne 1 ]]; then + echo "Expected exactly one JetBrains plugin ZIP, found ${#files[@]}." >&2 + printf '%s\n' "${files[@]}" >&2 + exit 1 + fi + echo "path=${files[0]}" >> "$GITHUB_OUTPUT" + + - name: Upload to GitHub Release + run: | + tag="$GITHUB_REF_NAME" + title="JetBrains $VERSION" + notes="JetBrains plugin $VERSION." + if gh release view "$tag" --repo "$GITHUB_REPOSITORY" >/dev/null 2>&1; then + gh release upload "$tag" "$ARCHIVE" --clobber --repo "$GITHUB_REPOSITORY" + exit 0 + fi + gh release create "$tag" "$ARCHIVE" --title "$title" --notes "$notes" --prerelease --repo "$GITHUB_REPOSITORY" + env: + GH_TOKEN: ${{ github.token }} + VERSION: ${{ steps.version.outputs.version }} + ARCHIVE: ${{ steps.archive.outputs.path }} + + - name: Upload workflow artifact + if: always() + uses: actions/upload-artifact@v4 + with: + name: kilo-jetbrains-${{ steps.version.outputs.version }} + path: packages/kilo-jetbrains/build/distributions/*.zip + if-no-files-found: ignore diff --git a/.github/workflows/publish.yml b/.github/workflows/publish.yml index 948a36d28b..34f35c22ce 100644 --- a/.github/workflows/publish.yml +++ b/.github/workflows/publish.yml @@ -42,7 +42,7 @@ jobs: runs-on: blacksmith-4vcpu-ubuntu-2404 if: github.repository == 'Kilo-Org/kilocode' steps: - - uses: actions/checkout@v3 + - uses: actions/checkout@v6 # kilocode_change with: fetch-depth: 0 @@ -78,7 +78,7 @@ jobs: runs-on: blacksmith-4vcpu-ubuntu-2404 if: github.repository == 'Kilo-Org/kilocode' steps: - - uses: actions/checkout@v3 + - uses: actions/checkout@v6 # kilocode_change with: fetch-tags: true @@ -123,11 +123,11 @@ jobs: runs-on: blacksmith-4vcpu-ubuntu-2404 if: github.repository == 'Kilo-Org/kilocode' steps: - - uses: actions/checkout@v3 + - uses: actions/checkout@v6 # kilocode_change - uses: ./.github/actions/setup-bun - - uses: actions/setup-node@v4 + - uses: actions/setup-node@v6 # kilocode_change with: node-version: "24" registry-url: "https://registry.npmjs.org" @@ -156,127 +156,11 @@ jobs: KILO_PRE_RELEASE: ${{ inputs.pre_release }} GH_REPO: ${{ github.repository }} - - uses: actions/upload-artifact@v4 + - uses: actions/upload-artifact@v7 # kilocode_change with: name: kilo-vscode path: packages/kilo-vscode/out - # build-tauri: - # needs: - # - build-cli - # - version - # continue-on-error: false - # strategy: - # fail-fast: false - # matrix: - # settings: - # - host: macos-latest - # target: x86_64-apple-darwin - # - host: macos-latest - # target: aarch64-apple-darwin - # - host: blacksmith-4vcpu-windows-2025 - # target: x86_64-pc-windows-msvc - # - host: blacksmith-4vcpu-ubuntu-2404 - # target: x86_64-unknown-linux-gnu - # - host: blacksmith-8vcpu-ubuntu-2404-arm - # target: aarch64-unknown-linux-gnu - # runs-on: ${{ matrix.settings.host }} - # steps: - # - uses: actions/checkout@v3 - # with: - # fetch-tags: true - - # - uses: apple-actions/import-codesign-certs@v2 - # if: ${{ runner.os == 'macOS' }} - # with: - # keychain: build - # p12-file-base64: ${{ secrets.APPLE_CERTIFICATE }} - # p12-password: ${{ secrets.APPLE_CERTIFICATE_PASSWORD }} - - # - name: Verify Certificate - # if: ${{ runner.os == 'macOS' }} - # run: | - # CERT_INFO=$(security find-identity -v -p codesigning build.keychain | grep "Developer ID Application") - # CERT_ID=$(echo "$CERT_INFO" | awk -F'"' '{print $2}') - # echo "CERT_ID=$CERT_ID" >> $GITHUB_ENV - # echo "Certificate imported." - - # - name: Setup Apple API Key - # if: ${{ runner.os == 'macOS' }} - # run: | - # echo "${{ secrets.APPLE_API_KEY_PATH }}" > $RUNNER_TEMP/apple-api-key.p8 - - # - uses: ./.github/actions/setup-bun - - # - name: Cache apt packages - # if: contains(matrix.settings.host, 'ubuntu') - # uses: actions/cache@v4 - # with: - # path: /var/cache/apt/archives - # key: ${{ runner.os }}-${{ matrix.settings.target }}-apt-${{ hashFiles('.github/workflows/publish.yml') }} - # restore-keys: | - # ${{ runner.os }}-${{ matrix.settings.target }}-apt- - - # - name: install dependencies (ubuntu only) - # if: contains(matrix.settings.host, 'ubuntu') - # run: | - # sudo apt-get update - # sudo apt-get install -y libwebkit2gtk-4.1-dev libappindicator3-dev librsvg2-dev patchelf - - # - name: install Rust stable - # uses: dtolnay/rust-toolchain@stable - # with: - # targets: ${{ matrix.settings.target }} - - # - uses: Swatinem/rust-cache@v2 - # with: - # workspaces: packages/desktop/src-tauri - # shared-key: ${{ matrix.settings.target }} - - # - name: Prepare - # run: | - # cd packages/desktop - # bun ./scripts/prepare.ts - # env: - # KILO_VERSION: ${{ needs.version.outputs.version }} - # GITHUB_TOKEN: ${{ steps.committer.outputs.token }} - # RUST_TARGET: ${{ matrix.settings.target }} - # GH_TOKEN: ${{ github.token }} - # GITHUB_RUN_ID: ${{ github.run_id }} - - # # Fixes AppImage build issues, can be removed when https://github.com/tauri-apps/tauri/pull/12491 is released - # - name: Install tauri-cli from portable appimage branch - # if: contains(matrix.settings.host, 'ubuntu') - # run: | - # cargo install tauri-cli --git https://github.com/tauri-apps/tauri --branch feat/truly-portable-appimage --force - # echo "Installed tauri-cli version:" - # cargo tauri --version - - # - name: Build and upload artifacts - # uses: tauri-apps/tauri-action@390cbe447412ced1303d35abe75287949e43437a - # timeout-minutes: 60 - # with: - # projectPath: packages/desktop - # uploadWorkflowArtifacts: true - # tauriScript: ${{ (contains(matrix.settings.host, 'ubuntu') && 'cargo tauri') || '' }} - # args: --target ${{ matrix.settings.target }} --config ./src-tauri/tauri.prod.conf.json --verbose - # updaterJsonPreferNsis: true - # releaseId: ${{ needs.version.outputs.release }} - # tagName: ${{ needs.version.outputs.tag }} - # releaseDraft: true - # releaseAssetNamePattern: opencode-desktop-[platform]-[arch][ext] - # env: - # GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} - # TAURI_BUNDLER_NEW_APPIMAGE_FORMAT: true - # TAURI_SIGNING_PRIVATE_KEY: ${{ secrets.TAURI_SIGNING_PRIVATE_KEY }} - # TAURI_SIGNING_PRIVATE_KEY_PASSWORD: ${{ secrets.TAURI_SIGNING_PRIVATE_KEY_PASSWORD }} - # APPLE_CERTIFICATE: ${{ secrets.APPLE_CERTIFICATE }} - # APPLE_CERTIFICATE_PASSWORD: ${{ secrets.APPLE_CERTIFICATE_PASSWORD }} - # APPLE_SIGNING_IDENTITY: ${{ env.CERT_ID }} - # APPLE_API_ISSUER: ${{ secrets.APPLE_API_ISSUER }} - # APPLE_API_KEY: ${{ secrets.APPLE_API_KEY }} - # APPLE_API_KEY_PATH: ${{ runner.temp }}/apple-api-key.p8 - # kilocode_change start # Run smoke tests against CLI assets uploaded to the draft GitHub release # before publishing the release and package artifacts. @@ -297,10 +181,13 @@ jobs: - build-cli - build-vscode - smoke-test - # - build-tauri runs-on: ubuntu-24.04 steps: - - uses: actions/checkout@v3 + - uses: actions/checkout@v6 # kilocode_change + # kilocode_change start + with: + persist-credentials: false + # kilocode_change end - uses: ./.github/actions/setup-bun @@ -317,7 +204,7 @@ jobs: - name: Set up Docker Buildx uses: docker/setup-buildx-action@v3 - - uses: actions/setup-node@v4 + - uses: actions/setup-node@v6 # kilocode_change with: node-version: "24" registry-url: "https://registry.npmjs.org" @@ -352,7 +239,7 @@ jobs: path: packages/kilo-vscode/out - name: Cache apt packages (AUR) - uses: actions/cache@v4 + uses: actions/cache@v5 # kilocode_change with: path: /var/cache/apt/archives key: ${{ runner.os }}-apt-aur-${{ hashFiles('.github/workflows/publish.yml') }} diff --git a/.github/workflows/smoke-test.yml b/.github/workflows/smoke-test.yml index a041bc7256..b0604eb8eb 100644 --- a/.github/workflows/smoke-test.yml +++ b/.github/workflows/smoke-test.yml @@ -47,7 +47,7 @@ jobs: KILO_ORG_ID: ${{ secrets.KILO_ORG_ID }} steps: - name: Checkout kilo-bench - uses: actions/checkout@v4 + uses: actions/checkout@v6 # kilocode_change with: repository: Kilo-Org/kilo-bench token: ${{ secrets.BENCH_GITHUB_TOKEN }} @@ -96,6 +96,16 @@ jobs: echo "cli_url=$URL" >> "$GITHUB_OUTPUT" echo "::notice::Testing CLI v${VERSION} via asset API: $URL" + # Harbor's default agent-setup timeout is 360s. The hello-world container + # (FROM ubuntu:24.04) needs apt-get update + apt-get install + a NodeSource + # curl|bash + apt install nodejs before the CLI even downloads, and on + # the Blacksmith runners that occasionally pushes past 6 min when an apt + # mirror or NodeSource cdn is slow, killing the run with + # AgentSetupTimeoutError. Doubling the multiplier to 2 gives enough + # headroom for transient mirror/cdn slowness while still finishing well + # under timeout-minutes. Note: --timeout-multiplier scales every timeout + # uniformly (setup, agent, verifier, env build), but they're all upper + # limits so this is harmless. - name: Run smoke test — hello-world env: KILO_CLI_URL: ${{ steps.cli.outputs.cli_url }} @@ -104,7 +114,8 @@ jobs: ./scripts/run_eval.sh \ -m kilo/anthropic/claude-sonnet-4.6 \ -d hello-world \ - --job-name smoke-test-hello-world + --job-name smoke-test-hello-world \ + --timeout-multiplier 2 - name: Run smoke test — log-summary-date-ranges env: @@ -115,18 +126,26 @@ jobs: -m kilo/anthropic/claude-sonnet-4.6 \ -d terminal-bench-sample \ -t "log-summary-date-ranges" \ - --job-name smoke-test-log-summary + --job-name smoke-test-log-summary \ + --timeout-multiplier 2 - name: Validate results run: python3 scripts/validate_smoke_test.py jobs/smoke-test-*/ + # Also upload the agent setup logs (stdout/stderr/return-code from the + # CLI install script) so we can see exactly which step stalls when the + # next AgentSetupTimeoutError happens. These come from the install + # script in kilo-bench, which uses `set -euo pipefail` (no `set -x`) + # and never echoes auth tokens or API keys, so the captured output is + # safe to upload. - name: Upload results if: always() - uses: actions/upload-artifact@v4 + uses: actions/upload-artifact@v7 # kilocode_change with: name: smoke-test-results path: | jobs/smoke-test-*/**/result.json jobs/smoke-test-*/**/trajectory.json + jobs/smoke-test-*/**/agent/setup/*.txt retention-days: 30 if-no-files-found: warn diff --git a/.github/workflows/source-check-links.yml b/.github/workflows/source-check-links.yml index 60df85b4b5..659de3d09b 100644 --- a/.github/workflows/source-check-links.yml +++ b/.github/workflows/source-check-links.yml @@ -16,7 +16,7 @@ jobs: if: github.repository == 'Kilo-Org/kilocode' runs-on: ubuntu-latest steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v6 # kilocode_change - uses: oven-sh/setup-bun@v2 diff --git a/.github/workflows/storybook.yml b/.github/workflows/storybook.yml deleted file mode 100644 index 9cd606324a..0000000000 --- a/.github/workflows/storybook.yml +++ /dev/null @@ -1,39 +0,0 @@ -name: storybook - -on: - push: - branches: [dev] - paths: - - ".github/workflows/storybook.yml" - - "package.json" - - "bun.lock" - - "packages/storybook/**" - - "packages/ui/**" - pull_request: - branches: [dev] - paths: - - ".github/workflows/storybook.yml" - - "package.json" - - "bun.lock" - - "packages/storybook/**" - - "packages/ui/**" - workflow_dispatch: - -concurrency: - group: ${{ github.workflow }}-${{ github.ref }} - cancel-in-progress: true - -jobs: - build: - if: false - name: storybook build - runs-on: blacksmith-4vcpu-ubuntu-2404 - steps: - - name: Checkout repository - uses: actions/checkout@v4 - - - name: Setup Bun - uses: ./.github/actions/setup-bun - - - name: Build Storybook - run: bun --cwd packages/storybook build diff --git a/.github/workflows/test-vscode.yml b/.github/workflows/test-vscode.yml index a3e66fff45..6056958584 100644 --- a/.github/workflows/test-vscode.yml +++ b/.github/workflows/test-vscode.yml @@ -27,7 +27,7 @@ jobs: shell: bash steps: - name: Checkout repository - uses: actions/checkout@v4 + uses: actions/checkout@v6 # kilocode_change with: token: ${{ secrets.GITHUB_TOKEN }} diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml index 2f2dba910d..559d65f0a9 100644 --- a/.github/workflows/test.yml +++ b/.github/workflows/test.yml @@ -28,21 +28,21 @@ jobs: matrix: settings: - name: linux - host: blacksmith-8vcpu-ubuntu-2404 + host: blacksmith-4vcpu-ubuntu-2404 # kilocode_change - name: windows - host: blacksmith-8vcpu-windows-2025 + host: blacksmith-4vcpu-windows-2025 # kilocode_change runs-on: ${{ matrix.settings.host }} defaults: run: shell: bash steps: - name: Checkout repository - uses: actions/checkout@v4 + uses: actions/checkout@v6 # kilocode_change with: token: ${{ secrets.GITHUB_TOKEN }} - name: Setup Node - uses: actions/setup-node@v4 + uses: actions/setup-node@v6 # kilocode_change with: node-version: "24" @@ -55,7 +55,7 @@ jobs: git config --global user.name "kilo-maintainer[bot]" - name: Cache Turbo - uses: actions/cache@v4 + uses: actions/cache@v5 # kilocode_change with: path: node_modules/.cache/turbo key: turbo-${{ runner.os }}-${{ hashFiles('turbo.json', '**/package.json') }}-${{ github.sha }} @@ -80,7 +80,7 @@ jobs: - name: Upload unit artifacts if: always() - uses: actions/upload-artifact@v4 + uses: actions/upload-artifact@v7 # kilocode_change with: name: unit-${{ matrix.settings.name }}-${{ github.run_attempt }} include-hidden-files: true @@ -88,87 +88,6 @@ jobs: retention-days: 7 path: packages/*/.artifacts/unit/junit.xml - e2e: - # kilocode_change - disabled: packages/app is not actively maintained - if: false - name: e2e (${{ matrix.settings.name }}) - strategy: - fail-fast: false - matrix: - settings: - - name: linux - host: blacksmith-4vcpu-ubuntu-2404 - - name: windows - host: blacksmith-4vcpu-windows-2025 - runs-on: ${{ matrix.settings.host }} - env: - PLAYWRIGHT_BROWSERS_PATH: ${{ github.workspace }}/.playwright-browsers - defaults: - run: - shell: bash - steps: - - name: Checkout repository - uses: actions/checkout@v4 - with: - token: ${{ secrets.GITHUB_TOKEN }} - - - name: Setup Node - uses: actions/setup-node@v4 - with: - node-version: "24" - - - name: Setup Bun - uses: ./.github/actions/setup-bun - - - name: Read Playwright version - id: playwright-version - run: | - version=$(node -e 'console.log(require("./package.json").workspaces.catalog["@playwright/test"])') - echo "version=$version" >> "$GITHUB_OUTPUT" - - - name: Cache Playwright browsers - id: playwright-cache - uses: actions/cache@v4 - with: - path: ${{ github.workspace }}/.playwright-browsers - key: ${{ runner.os }}-${{ runner.arch }}-playwright-${{ steps.playwright-version.outputs.version }}-chromium - - - name: Install Playwright system dependencies - if: runner.os == 'Linux' - working-directory: packages/app - run: bunx playwright install-deps chromium - - - name: Install Playwright browsers - if: steps.playwright-cache.outputs.cache-hit != 'true' - working-directory: packages/app - run: bunx playwright install chromium - - - name: Run app e2e tests - run: bun --cwd packages/app test:e2e:local - env: - CI: true - # kilocode_change start - KILO_API_KEY: ${{ secrets.KILO_API_KEY }} - KILO_ORG_ID: ${{ secrets.KILO_ORG_ID }} - KILO_DISABLE_SHARE: "true" - KILO_DISABLE_SESSION_INGEST: "true" - KILO_E2E_REQUIRE_PAID: "true" - # kilocode_change end - PLAYWRIGHT_JUNIT_OUTPUT: e2e/junit-${{ matrix.settings.name }}.xml - timeout-minutes: 30 - - - name: Upload Playwright artifacts - if: always() - uses: actions/upload-artifact@v4 - with: - name: playwright-${{ matrix.settings.name }}-${{ github.run_attempt }} - if-no-files-found: ignore - retention-days: 7 - path: | - packages/app/e2e/junit-*.xml - packages/app/e2e/test-results - packages/app/e2e/playwright-report - required: name: test (linux) runs-on: blacksmith-4vcpu-ubuntu-2404 diff --git a/.github/workflows/triage.yml b/.github/workflows/triage.yml index b767380365..b467817c71 100644 --- a/.github/workflows/triage.yml +++ b/.github/workflows/triage.yml @@ -13,7 +13,7 @@ jobs: issues: write steps: - name: Checkout repository - uses: actions/checkout@v4 + uses: actions/checkout@v6 # kilocode_change with: fetch-depth: 1 diff --git a/.github/workflows/typecheck.yml b/.github/workflows/typecheck.yml index 894f5dac6f..6c7ae42095 100644 --- a/.github/workflows/typecheck.yml +++ b/.github/workflows/typecheck.yml @@ -12,7 +12,7 @@ jobs: runs-on: blacksmith-4vcpu-ubuntu-2404 steps: - name: Checkout repository - uses: actions/checkout@v4 + uses: actions/checkout@v6 # kilocode_change - name: Setup Bun uses: ./.github/actions/setup-bun diff --git a/.github/workflows/visual-regression.yml b/.github/workflows/visual-regression.yml index 4f49c4aa46..d059175adc 100644 --- a/.github/workflows/visual-regression.yml +++ b/.github/workflows/visual-regression.yml @@ -12,7 +12,7 @@ jobs: matched: ${{ steps.filter.outputs.matched }} is_fork: ${{ steps.fork-check.outputs.is_fork }} steps: - - uses: actions/checkout@v4 + - uses: actions/checkout@v6 # kilocode_change - uses: Kilo-Org/paths-filter@master id: filter with: @@ -47,7 +47,7 @@ jobs: steps: - name: Checkout (internal) if: needs.check-paths.outputs.is_fork != 'true' - uses: actions/checkout@v4 + uses: actions/checkout@v6 # kilocode_change with: lfs: true token: ${{ secrets.BOT_PAT }} @@ -55,7 +55,7 @@ jobs: - name: Checkout (fork) if: needs.check-paths.outputs.is_fork == 'true' - uses: actions/checkout@v4 + uses: actions/checkout@v6 # kilocode_change with: lfs: true @@ -77,7 +77,7 @@ jobs: bun-version: latest - name: Cache Bun modules - uses: actions/cache@v4 + uses: actions/cache@v5 # kilocode_change with: path: ~/.bun/install/cache key: bun-${{ hashFiles('bun.lock') }} @@ -87,7 +87,7 @@ jobs: - name: Cache Playwright browsers id: playwright-cache - uses: actions/cache@v4 + uses: actions/cache@v5 # kilocode_change with: path: ~/.cache/ms-playwright key: playwright-${{ hashFiles('packages/kilo-ui/package.json') }} @@ -103,7 +103,7 @@ jobs: - name: Cache Storybook build id: storybook-cache - uses: actions/cache@v4 + uses: actions/cache@v5 # kilocode_change with: path: packages/kilo-ui/storybook-static key: storybook-${{ hashFiles('packages/kilo-ui/src/**', 'packages/kilo-ui/.storybook/**', 'packages/ui/src/**', 'packages/kilo-ui/package.json') }} @@ -195,7 +195,7 @@ jobs: - name: Upload test results on failure if: failure() - uses: actions/upload-artifact@v4 + uses: actions/upload-artifact@v7 # kilocode_change with: name: visual-regression-results path: packages/kilo-ui/test-results/ @@ -211,7 +211,7 @@ jobs: steps: - name: Checkout (internal) if: needs.check-paths.outputs.is_fork != 'true' - uses: actions/checkout@v4 + uses: actions/checkout@v6 # kilocode_change with: lfs: true token: ${{ secrets.BOT_PAT }} @@ -219,7 +219,7 @@ jobs: - name: Checkout (fork) if: needs.check-paths.outputs.is_fork == 'true' - uses: actions/checkout@v4 + uses: actions/checkout@v6 # kilocode_change with: lfs: true @@ -241,7 +241,7 @@ jobs: bun-version: latest - name: Cache Bun modules - uses: actions/cache@v4 + uses: actions/cache@v5 # kilocode_change with: path: ~/.bun/install/cache key: bun-${{ hashFiles('bun.lock') }} @@ -251,7 +251,7 @@ jobs: - name: Cache Playwright browsers id: playwright-cache-vscode - uses: actions/cache@v4 + uses: actions/cache@v5 # kilocode_change with: path: ~/.cache/ms-playwright key: playwright-vscode-${{ hashFiles('packages/kilo-vscode/package.json') }} @@ -267,7 +267,7 @@ jobs: - name: Cache Storybook build id: storybook-cache-vscode - uses: actions/cache@v4 + uses: actions/cache@v5 # kilocode_change with: path: packages/kilo-vscode/storybook-static key: storybook-vscode-${{ hashFiles('packages/kilo-vscode/webview-ui/src/**', 'packages/kilo-vscode/.storybook/**', 'packages/kilo-ui/src/**', 'packages/ui/src/**', 'packages/kilo-vscode/package.json') }} @@ -386,7 +386,7 @@ jobs: - name: Upload test results on failure if: failure() - uses: actions/upload-artifact@v4 + uses: actions/upload-artifact@v7 # kilocode_change with: name: visual-regression-vscode-results path: packages/kilo-vscode/test-results/ diff --git a/.gitignore b/.gitignore index e28de87d26..8add4fe6d4 100644 --- a/.gitignore +++ b/.gitignore @@ -45,8 +45,28 @@ telemetry-id tsconfig.tsbuildinfo # Kilo +.kilo/.gitignore +.kilo/package.json +.kilo/package-lock.json +.kilo/pnpm-lock.yaml +.kilo/bun.lock +.kilo/yarn.lock +.kilo/node_modules .kilo/plans/*upstream-merge-report-*.md +.kilocode/.gitignore +.kilocode/package.json +.kilocode/package-lock.json +.kilocode/pnpm-lock.yaml +.kilocode/bun.lock +.kilocode/yarn.lock +.kilocode/node_modules # Test Artifacts packages/app/.artifacts/ +<<<<<<< HEAD packages/opencode/.artifacts/.env +||||||| 12f7967ca4 +packages/opencode/.artifacts/ +======= +packages/opencode/.artifacts/ +>>>>>>> yunqiqiliang/opencode-v7.3.0 diff --git a/.kilo/agent/upstream-merge.md b/.kilo/agent/upstream-merge.md new file mode 100644 index 0000000000..e81c9cea36 --- /dev/null +++ b/.kilo/agent/upstream-merge.md @@ -0,0 +1,413 @@ +--- +description: Resolve upstream opencode merge conflicts interactively +mode: primary +permission: + read: ask + edit: ask + webfetch: ask + bash: + "*": ask + "git status *": allow + "git log *": allow + "git diff *": allow + "git show *": allow + "git ls-files *": allow + "git ls-tree *": allow + "git grep *": allow + "git hash-object *": allow + "git remote -v *": allow + "git rev-parse *": allow + "git merge-base *": allow + "git show-ref *": allow + "git worktree list": allow + "git branch --show-current": allow + "grep *": allow + "rg *": allow + "head *": allow + "tail *": allow + "cat *": allow + "wc *": allow + "ls *": allow + "pwd *": allow + "diff *": allow + "gh pr view *": allow + "gh run view *": allow + "gh api \"repos/sst/opencode/commits/dev\" *": allow + "axiom *": allow + "bun test *": allow + "bun run typecheck *": allow + "bun run lint *": allow + "bun run script/check-opencode-annotations.ts *": allow + "script/upstream/find-conflict-markers.sh *": allow + "./script/upstream/find-conflict-markers.sh *": allow +--- + +Resolve the manual part of an upstream merge. + +**Do not load the `kilocode-merge-minimizer` skill.** That skill is for +authoring new Kilo changes against shared upstream files; during an upstream +merge it gives the wrong guidance (it nudges toward extracting Kilo logic out +of conflict regions, which is exactly the opposite of what merge resolution +needs). Follow the rules in this agent file instead. + +The user will provide the upstream version (for example `v1.1.50` or `1.1.50`) +in their first message. If they don't, infer it from the current branch name, +from `upstream-merge-report-.md`, or from the newest relevant report +file. + +## Workflow + +### 1. Inspect the current merge state + +- `git status --short` +- `git diff --name-only --diff-filter=U` +- `upstream-merge-report-.md` when present +- `.worktrees/opencode-merge/auto-merge` for the automated merge snapshot when present + +### 2. Read every conflicted file end-to-end before planning + +Use `script/upstream/find-conflict-markers.sh ` to jump to each region, +then read enough surrounding lines to understand the code — not just the +conflict hunk. Specifically check: + +- is this a plain 3-way on a single expression, or a structural refactor? +- does upstream rename/move something that invalidates a HEAD-only declaration? +- does a `kilocode_change` marker in HEAD encode a bug fix, a feature, or a + defensive check? +- is the conflicted block referenced by *non-conflicted* code elsewhere in the + same file (imports, signatures, call sites) that will break if we drop it? + +When HEAD includes a non-obvious Kilo-specific wrapper (e.g. a helper in +`packages/opencode/src/kilocode/`), find out why it exists before deciding to +keep or bypass it: + +```bash +git log --all --oneline -S "" -- packages/opencode/src/kilocode/ +git log --all --oneline -- +``` + +Look at the commit message and any PR reference. "We wrote our own because of +PR #NNNN" is a real constraint; "we wrote our own because of a typo" is not. + +When upstream narrows an externally-visible compatibility list (models, +providers, routes, config keys, file formats), verify the intent from upstream +PRs, issues, release notes, or current docs before dropping entries. Treat +silent list shrinkage during a refactor as suspicious until proven intentional. + +### 3. Write a plan in chat and get approval + +For every conflicted file (and any adjacent file the resolution forces you to +touch — see §6) include: + +- expected resolution kind: `hybrid`, `take-ours`, `take-theirs`, `regenerated`, + `removed`, `renamed`, or `other` +- risk level: `low`, `medium`, or `high` +- one-sentence rationale (what Kilo behaviour is preserved, what upstream + feature is adopted, what is dropped) +- verification commands you expect to run (targeted tests, typecheck) + +Group files by risk level. Ask the user which batch to start with. You can +resolve an entire `low` batch in one pass if the user approves the batch, but +resolve `medium` and `high` files one at a time. + +**Do not resolve a file until the user has approved that file's (or batch's) +strategy.** + +### 4. Before every edit, explain reasoning before showing the diff + +The user needs to review intent, not just the raw change. For each file, in +order: + +1. Show the conflict's surrounding context (10–30 lines around each conflict + region, in chat). +2. Explain what each of the three sides (HEAD, merge-base, upstream) is doing. +3. State which Kilo behaviour must survive and why (reference PR numbers / + `kilocode_change` comments when possible). +4. State the resolution and why it is better than the alternatives. +5. Then apply the edit. The tool will display the diff — the user only has to + verify the diff matches the reasoning. + +Do not lead with the diff. A diff without reasoning forces the user to +reverse-engineer the decision. + +### 5. Apply resolution rules + +Reference worktrees when present: + +- `.worktrees/opencode-merge/opencode` — pristine upstream tree +- `.worktrees/opencode-merge/kilo-main` — Kilo base snapshot +- `.worktrees/opencode-merge/auto-merge` — automated merge snapshot (original + conflict reference) + +Apply in order: + +- prefer upstream code and architecture whenever compatible with Kilo behaviour +- preserve Kilo-specific behaviour marked with `kilocode_change` +- keep `kilocode_change` markers around Kilo-specific code in shared opencode + files +- when upstream refactors a region that HEAD had annotated with + `kilocode_change`, **check whether the marker encodes a bug fix or a feature + delta**. Bug fixes (missing `await`, defensive null-check, error capture) + usually need to be re-applied on top of the upstream refactor. Example from + v1.14.30: `Workspace.isSyncing` was missing an `await` — upstream's Effect + refactor reintroduced the same bug, so we had to port the fix into the new + `Effect.gen` block. +- when a `take-theirs` drops a line that was the target of a Kilo pre-filter, + the upstream line may be actively wrong for Kilo — e.g. an inner `continue` + filter whose condition collides with an outer filter Kilo added. Re-read the + surrounding 20 lines before committing to `take-theirs`. +- if Kilo-specific code must be refactored to fit new upstream architecture, + explain the refactor in the final summary +- if upstream moved the relevant logic to another file, port the Kilo behaviour + there and list both paths in the final summary. Verify the new file already + carries the Kilo-renamed symbols (e.g. `x-kilo-directory`) by diffing against + pristine upstream. +- if upstream extracts shared policy into a helper, move Kilo-specific additions + into the helper when possible instead of keeping a pre-check at the old call + site. The extracted helper should stay the source of truth for all callers. +- if upstream deleted a file, analyse whether the Kilo behaviour should be + ported elsewhere or removed rather than restoring the deleted file +- if tests fail only because upstream intentionally removed behaviour, remove + or update the obsolete tests rather than adding the old file back +- do not modify unrelated files + +When removing code that existed in one side of a conflict, prefer +**commenting it out with `kilocode_change` markers** over deletion when the +surrounding structure (an `if`, a loop) still makes sense. That keeps the +intent visible to the next merger. Example: + +```ts +} else if (input?.scope !== "project" && !Flag.KILO_EXPERIMENTAL_WORKSPACES) { + // kilocode_change start - directory filtering handled by KiloSession.filters above + // if (input?.directory) { + // conditions.push(eq(SessionTable.directory, input.directory)) + // } + // kilocode_change end +} +``` + +Use `TODO:` not `NOTE:` for follow-ups. `TODO` is searchable and implies an +owner will act on it; `NOTE` reads as permanent commentary. + +### 6. Look for adjacent files the conflict forces you to touch + +Upstream restructures sometimes split one file into several (e.g. `permission.ts` +→ `groups/permission.ts` + `handlers/permission.ts`). Only the *renamed* file +shows up in `git diff --diff-filter=U`; the new sibling may need a Kilo feature +ported in too. After resolving the flagged file, check: + +- files that import from the resolved file — do they compile? +- files at paths implied by new imports (e.g. `../middleware/*`, `./handlers/*`) +- `kilocode_change` comments in the *auto-merge* snapshot that didn't end up in + the working tree because the hosting file was renamed + +Add any such files to the plan as `hybrid` or `take-ours` with the same +approval flow. + +### 6.5. Scan auto-merged files for latent bugs + +Files not in `--diff-filter=U` merged without conflict markers but may still +be broken. Check every auto-merged file for: + +- **Duplicate declarations in the same scope.** If both sides added equivalent + code independently, auto-merge keeps both. Grep touched functions for + repeated identifiers before trusting the merge. +- **Duplicate keys in config/manifest files.** If both sides added the same + entry to a shared manifest (dependencies, scripts, workflow lists), the + merged file may have the key twice. This often breaks install/setup before + any test runs — a cheap early win to scan for. +- **Orphaned imports and references.** A rename upstream may leave a Kilo + callsite pointing at a now-missing export. Run full typecheck from the repo + root; references that silently survived the merge surface there. +- **Partial auto-merges.** Upstream may have refactored a region Kilo + deliberately stubbed out (commented blocks, removed fallbacks). If the + auto-merge pulled in references to names that only exist in the removed + path, the file compiles upstream but breaks on Kilo. + +### 7. Verify each resolution before moving on + +- confirm `script/upstream/find-conflict-markers.sh ` prints nothing +- read the final file region (the new shape after edit) and sanity-check imports +- for apparently-unused symbols upstream introduced, `grep` the file and the + rest of the package before deleting — they may be called from non-conflicted + code elsewhere. Example: `isTheme` in `theme.tsx` looked unused at the + resolution site but was called twice further down. +- run the smallest relevant check (single `bun test` file, or `bun run + typecheck` in the touched package) +- summarise the exact resolution, tradeoff, and verification result in chat +- ask the user to approve the resolved file before staging it or resolving the + next one (for `medium` / `high`; `low` batches can be staged together) + +### 8. Run the full checks once everything is resolved + +- `git diff --name-only --diff-filter=U` returns empty +- `bun run typecheck` from `packages/opencode/` (targeted) and from repo root + (catches non-conflicted call-site breakage) +- relevant targeted tests. Tests that hang or time out in an unrelated part of + the graph may be pre-existing — note them, don't block the merge on them +- `bun run script/check-opencode-annotations.ts` if `packages/opencode/` shared + files changed. Note that this tool compares against the merge base via `HEAD` + and will be silent until the merge commit lands +- other CI guards that touched files imply (knip for `kilo-vscode/`, + `check-kilocode-change`, source-links, visual regression) + +### 9. Commit with the standard message + +Per `script/upstream/README.md`: + +```bash +git commit -m "resolve merge conflicts" +``` + +The default `git merge` auto-message (`Merge branch '…' into …`) is also fine, +but `resolve merge conflicts` is the convention for these PRs. + +### 9.5. Handle downstream API renames as separate commits + +Upstream often renames exported APIs. The rename itself auto-merges cleanly in +shared code, but the change cascades into Kilo-only files (kilocode tests, +kilo-specific source, plugins) that still reference the old symbol. Those +files don't appear in `--diff-filter=U` because their own content didn't +conflict. + +Keep the behavioural merge commit focused on resolution decisions. Land the +cascade in one or more follow-up commits: + +- after the merge commit, run full repo typecheck and collect every "cannot + find name" / "property does not exist" error +- bulk-rename with a mechanical transform when the rename is one-to-one +- restructure or parameter-thread when upstream changed semantics, not just + the name (e.g. moved a helper behind a dependency-injected surface, so + callers now need the injected handle) +- split large downstream refactors into their own commits with messages that + name the rename + +Reviewers can then skim the behavioural commit without untangling mechanical +rename noise from merge decisions. + +### 9.6. Handle upstream-added tests that diverge from Kilo + +Upstream sometimes adds tests that encode design contracts Kilo intentionally +breaks. These auto-merge cleanly and then fail. Three resolution patterns: + +- **Rewrite the test** when the test is a contract assertion and Kilo has a + different but equally valid contract. Invert or adjust the assertion with a + `kilocode_change` marker explaining the divergence. +- **Skip the test** when the test relies on patterns that Kilo has replaced + (interception seams that are bypassed by dependency injection, fixture + helpers bound to a removed API, assumptions about serialization shape that + Kilo's extensions break). Mark with `kilocode_change` and a rationale + explaining what would need to change for the test to run. +- **Delete the test** when it covers functionality Kilo deliberately removed + (fallback paths, deprecated endpoints, products Kilo doesn't ship). Note + the deletion in the PR body. + +Never silently delete; always leave a breadcrumb. A future reviewer should be +able to understand why this one upstream test is treated differently. + +### 10. Resync version strings in a separate commit + +Upstream stamps its own version into shared files — notably +`packages/extensions/zed/extension.toml` (version field + 5 Kilo-Org download +URLs), and any `package.json` that upstream bumped in the same release window. +After the merge this leaves parts of the tree pointing at upstream's version +(e.g. `1.14.30`), whose release tag does not exist on Kilo's pipeline, so the +Zed download URLs silently 404. + +Fix this in a dedicated commit *after* `resolve merge conflicts`: + +```bash +bun run script/sync-versions.ts # uses root package.json version +# or, to target an explicit version: +bun run script/sync-versions.ts 7.2.41 +git add -A +git commit -m "chore: resync versions after upstream merge" +``` + +The script rewrites every top-level `"version"` in `package.json` files +(excluding `node_modules`, hidden dirs, and `packages/kilo-jetbrains/` which +tracks its own cadence), plus the Zed extension toml. It is idempotent — rerun +it any time to rebase the version back onto Kilo main (useful during +long-running upstream merges where `main` releases in the meantime). + +Keeping this in its own commit makes reviewers' job easier: the merge commit +only contains behavioural resolutions, and the version resync is a trivial +diff they can skim in one glance. + +### 11. Write the PR body + +Structure the description so reviewers can skim: + +- **Non-trivial merge decisions**: a short section per file (or group of + related files) that required more than a mechanical `take-ours`/`take-theirs`. + Focus on *what Kilo behaviour survived* and *what upstream features were + adopted*. Link to Kilo PRs when a `kilocode_change` encodes a specific fix. +- **Notable auto-merged changes**: new columns, new helper files, renamed + middleware — anything reviewers should eyeball even though git didn't flag + it. +- **What to test**: explicit, scenario-level test steps for each non-trivial + change. Don't list tests; list *user-visible behaviour* so a tester who + doesn't read the diff can exercise it. +- **CI guards to watch**: typecheck, knip, annotation check, visual regression. +- **Follow-ups**: any `TODO:` you left in code, as a bullet list with links. + +## User-approval checkpoints + +Every manual merge decision requires explicit user approval **before applying** +and **again after verification**. Be especially cautious when a decision is +destructive, changes auth, billing, data deletion, public API compatibility, +config schema behaviour, migrations, provider routing, or security posture. + +## Common pitfalls + +- Auto-merged code can reference declarations that still live inside conflict + blocks. +- Related sibling files can need edits even when they are not listed as + unmerged — especially after upstream structural splits. +- `renamed` should be used only when behaviour moves to a different file. +- Function signatures can drift across conflict boundaries (args added, return + types widened). Grep for every call site before finalising. +- Full-repo typecheck is the catch-all for non-conflicted call-site breakage. +- Upstream can reintroduce bugs a Kilo `kilocode_change` had already fixed — + during big refactors check every `kilocode_change` the refactor touched. +- "Take-theirs" on an inner conditional is often wrong when Kilo added an outer + pre-filter whose whole point was to widen what makes it to the inner block. +- Apparently-unused upstream-added declarations may be called from + non-conflicted code elsewhere. Grep before deleting. +- Stricter DOM lib types (upstream TS upgrade) can surface latent casting + issues around `WebSocket.send`, `Headers`, etc. — prefer narrowing the Kilo + type over adding `any` casts. +- Auto-merge can duplicate the same declaration twice in one scope when both + sides added equivalent code independently. Silent for git, caught by + typecheck. Same hazard for duplicated object keys in config/manifest files — + those can break install before any test runs. +- Kilo code may rely on ambient context (async-local storage, globally-set + flags, process env) being populated at a lifecycle moment that upstream + refactors away. If Kilo behaviour reads ambient state during init, forked + work, or event handlers, check the refactor still establishes that state at + the right time. Fix by restoring the ambient state, or by threading the + needed value through explicitly. +- Tests that intercept via process-global or module-global spies can become + no-ops after upstream moves the intercepted code path through dependency + injection. The production code no longer touches the spied symbol. Fixing + the test usually means injecting a mock at the new seam rather than tweaking + the spy. +- When Kilo extends a shared data shape with extra optional fields, different + serialization paths for that shape can diverge on whether missing values are + omitted or emitted as null. Parity tests between two such paths break on + every Kilo addition — audit the encoding assumption before adding fields. +- Rule ordering in allowlist/permission evaluation is usually last-match-wins. + Re-declaring a catch-all rule "for safety" in a later ruleset silently + overrides more specific allow rules from an earlier ruleset. Treat a + redundant catch-all as destructive, not defensive. +- Upstream-added tests can encode a design contract Kilo deliberately breaks. + The test passes upstream because upstream doesn't share Kilo's requirement. + Decide between refactoring Kilo to match the upstream contract or rewriting + the test to assert Kilo's divergent contract — with a `kilocode_change` + marker explaining the divergence. +- CI and local can show different test failures. Tests that read user-local + state (home dir, global config, auth tokens) pass in one environment but + fail in the other. A green local run does not imply green CI. +- Dependency manifests and lockfiles move together. When the merge edits one, + regenerate and commit the other in the same change — otherwise CI breaks on + the follow-up setup step. diff --git a/.kilo/skills/gh-issues/SKILL.md b/.kilo/skills/gh-issues/SKILL.md new file mode 100644 index 0000000000..9a5bfefdf3 --- /dev/null +++ b/.kilo/skills/gh-issues/SKILL.md @@ -0,0 +1,68 @@ +--- +name: gh-issues +description: Use when creating, triaging, or commenting on GitHub issues for the Kilo VS Code extension or JetBrains plugin via `gh`. Covers issue templates, project board assignment, title conventions, and required `gh` scopes. +--- + +# GitHub Issues + +Use this skill whenever you create or manage a GitHub issue with `gh` for either the VS Code extension or the JetBrains plugin. + +## Templates + +The repo defines issue templates in `.github/ISSUE_TEMPLATE/`. Pick the matching template instead of opening a blank issue: + +| Template | When to use | +|---|---| +| `Bug report` (`bug-report.yml`) | Reproducible defects with steps, expected, and actual behavior | +| `Feature Request` (`feature-request.yml`) | New capabilities, enhancements, or behavior changes | +| `Question` (`question.yml`) | Usage or design questions that aren't obviously bugs or feature requests | + +Pass the template title to `gh issue create --template`. + +## Title Conventions + +- Use a plain, descriptive title that reads cleanly as a standalone sentence. +- Do not add platform-specific prefixes such as `[JetBrains]`, `[Jetbrains]`, `[JB]`, `[VS Code]`, `[VSCode]`, or similar. Routing happens through project boards, not the title. + +## Project Boards + +Every new issue must land on the correct project board: + +| Surface | Project | URL | +|---|---|---| +| VS Code extension | `VS Code Extension` | https://github.com/orgs/Kilo-Org/projects/25 | +| JetBrains plugin | `Jetbrains Plugin` | https://github.com/orgs/Kilo-Org/projects/39 | + +Pass the project title to `gh issue create --project`. + +## Recipes + +Create a VS Code extension bug report and add it to the board: + +```bash +gh issue create \ + --template "Bug report" \ + --project "VS Code Extension" \ + --title "Sidebar chat fails to render after reload" \ + --body "..." +``` + +Create a JetBrains feature request: + +```bash +gh issue create \ + --template "Feature Request" \ + --project "Jetbrains Plugin" \ + --title "Support Kotlin Multiplatform target detection" \ + --body "..." +``` + +## Scope Errors + +If `gh` reports a missing scope when assigning a project, refresh the auth token and retry: + +```bash +gh auth refresh -s project +``` + +After the refresh succeeds, re-run the original `gh issue create` command. Do not fall back to creating the issue without the project — the board assignment is required. diff --git a/.kilo/skills/jetbrains-ui-style/SKILL.md b/.kilo/skills/jetbrains-ui-style/SKILL.md deleted file mode 100644 index 5536a2c445..0000000000 --- a/.kilo/skills/jetbrains-ui-style/SKILL.md +++ /dev/null @@ -1,888 +0,0 @@ ---- -name: jetbrains-ui-style -description: Use when creating, modifying, or reviewing Kotlin/Swing UI code for the Kilo JetBrains plugin. Applies to dialogs, settings pages, tool windows, forms, panels, component layout, sizing, spacing, colors, borders, icons, lists, trees, popups, notifications, and IntelliJ platform UI components. ---- - -# JetBrains UI Style - -Use this skill whenever creating, modifying, or reviewing UI code in `packages/kilo-jetbrains`. - -## Scope - -This skill covers Kotlin/Swing UI code for the Kilo JetBrains plugin, including dialogs, settings pages, tool windows, forms, panels, component layout, sizing, spacing, colors, borders, icons, lists, trees, popups, notifications, and IntelliJ Platform UI components. - -This plugin is a split-mode JetBrains plugin. UI code belongs in `frontend` unless there is a specific split-mode reason to place it elsewhere. - -## Primary Rules - -- Prefer Kotlin UI DSL v2 whenever the UI is a dialog, settings page, form, options panel, or structured component layout. -- Use manual Swing only for custom rendering, transcript/chat surfaces, highly dynamic panels, virtualized lists, custom painting, or layouts the DSL cannot express cleanly. -- Keep generated UI code minimal. -- Do not set default Swing properties explicitly. For example, avoid `isOpaque = false` unless the component default differs or there is a documented rendering reason. -- Avoid hardcoded dimensions. Prefer layout semantics, component defaults, DSL sizing helpers, and `JBUI` utilities. -- Do not add decorative helper functions, wrappers, or defensive UI code unless they materially improve clarity or correctness. -- Prefer IntelliJ platform components and theme-aware APIs. -- Do not use hardcoded runtime colors in generated UI examples or implementation code. -- Derive UI colors from semantic IntelliJ theme APIs such as `UIUtil`, `JBUI.CurrentTheme`, `NamedColorUtil`, `JBColor.namedColor`, or `JBColor.lazy`. -- Do not use hardcoded font sizes or font families in generated UI examples or implementation code. -- Use component defaults, `JBFont`, `RelativeFont`, or existing component fonts for typography. -- Put user-visible strings in `*.properties` files. - -## Decision Tree - -- Use Kotlin UI DSL v2 for dialogs, settings pages, forms, options panels, and structured component layouts. -- Use Kotlin UI DSL v2 inside tool windows when rendering form-like controls or structured subpanels. -- Use standard Swing with IntelliJ Platform component replacements for tool-window shells, action-driven UI, custom components, transcript surfaces, and custom renderers. -- Use the Action System for menus and toolbars. -- Do not use Kotlin Compose. -- Do not use JCEF. - -| Need | API | -|---|---| -| Dialogs, settings pages, forms, any layout with components | Preferred: Kotlin UI DSL v2 (`com.intellij.ui.dsl.builder`) | -| Tool window panels, action-driven UI, custom components | Standard Swing with IntelliJ Platform component replacements | -| Menus and toolbars | Action System | - -## Minimal Code Rule - -Avoid explicit defaults and manual layout when the DSL can express the UI: - -```kotlin -// Avoid -val panel = JPanel(BorderLayout()).apply { - isOpaque = false - preferredSize = Dimension(420, 260) - border = EmptyBorder(12, 12, 12, 12) -} -``` - -Prefer DSL and platform spacing: - -```kotlin -panel { - row(KiloBundle.message("settings.name")) { - textField() - .align(AlignX.FILL) - .resizableColumn() - } -} -``` - -Avoid explicit default properties: - -```kotlin -// Avoid unless there is a documented rendering reason -component.isOpaque = false -``` - -Prefer omitting the assignment and relying on the component/platform default. - -Avoid raw fixed sizes: - -```kotlin -// Avoid -component.preferredSize = Dimension(300, 40) -``` - -Prefer semantic sizing: - -```kotlin -textField() - .columns(COLUMNS_MEDIUM) - .align(AlignX.FILL) -``` - -If a fixed size is genuinely required, use `JBUI.size(...)` or `JBUI.scale(...)` and keep the reason obvious from context. - -## Do Not Use Kotlin Compose - -Do not use Kotlin Compose or `intellij.platform.compose` in this plugin. The JetBrains modular template uses Compose for its demo tool window, but Kilo should use standard Swing with IntelliJ Platform components only. Keep all plugin UI in the existing Swing-based stack. - -## Do Not Use JCEF - -Do not use JCEF (`JBCefBrowser`) in this plugin. JCEF does not work in JetBrains remote development (split mode): the frontend process runs on the client machine but JCEF requires a display on the host, making it effectively unusable for remote users. Use standard Swing with IntelliJ Platform components for all UI. - -## Kotlin UI DSL v2 - -Use Kotlin UI DSL v2 as the default way to build UI for dialogs, settings pages, forms, and any layout composed of standard components. It produces correct spacing, label alignment, HiDPI scaling, and accessibility automatically. Only fall back to manual Swing layout when you need a fully custom component, such as a canvas, rich list renderer, transcript surface, or tool-window chrome that the DSL cannot express. - -The top-level builder is `panel { }` and returns `DialogPanel`. Structure is `panel -> row -> cells`. Cell factory methods such as `textField()`, `checkBox()`, and `label()` add components. The DSL lives in `com.intellij.ui.dsl.builder`. - -To explore DSL capabilities interactively: Tools -> Internal Actions -> UI -> Kotlin UI DSL -> UI DSL Showcase. This requires internal mode: `-Didea.is.internal=true`. - -### Basics: Panel, Row, Cell - -Rows occupy full width. The last cell in a row takes remaining space. Rows have a `layout` property. - -```kotlin -panel { - row("Row1 label:") { - textField() - label("Some text") - } - row("Row2:") { - label("This text is aligned with previous row") - } -} -``` - -### Row Layout - -Every row uses one of three layouts. Default is `LABEL_ALIGNED` when a label is provided for the row, `INDEPENDENT` otherwise. - -| Layout | Behavior | -|---|---| -| `LABEL_ALIGNED` | Label column and content columns, aligned across rows | -| `INDEPENDENT` | All cells are independent, no cross-row alignment | -| `PARENT_GRID` | Cells align with the parent grid columns across rows | - -```kotlin -panel { - row("PARENT_GRID:") { - label("Col 1") - label("Col 2") - }.layout(RowLayout.PARENT_GRID) - - row("PARENT_GRID:") { - textField() - textField() - }.layout(RowLayout.PARENT_GRID) - - row("LABEL_ALIGNED default with label:") { - textField() - } - - row { - label("INDEPENDENT default without label:") - textField() - } -} -``` - -### Components Reference - -All cell factory methods available inside `row { }`: - -| Method | Description | -|---|---| -| `checkBox("text")` | Checkbox | -| `threeStateCheckBox("text")` | Three-state checkbox | -| `radioButton("text", value)` | Radio button, must be inside `buttonsGroup {}` | -| `button("text") {}` | Push button | -| `actionButton(action)` | Icon button bound to an `AnAction` | -| `actionsButton(action1, action2, ...)` | Dropdown actions button | -| `segmentedButton(items) { text = it }` | Segmented control | -| `tabbedPaneHeader(items)` | Tab header strip | -| `label("text")` | Static label | -| `text("html")` | Rich text with links, icons, line-width control | -| `link("text") {}` | Focusable clickable link | -| `browserLink("text", "url")` | Opens URL in browser | -| `dropDownLink("default", listOf(...))` | Dropdown link selector | -| `icon(AllIcons.*)` | Icon display | -| `contextHelp("description", "title")` | Help icon with popup | -| `textField()` | Text input | -| `passwordField()` | Password input | -| `textFieldWithBrowseButton()` | Text field and browse dialog | -| `expandableTextField()` | Expandable multi-line text field | -| `extendableTextField()` | Text field with extension icons | -| `intTextField(range)` | Integer input with validation | -| `spinner(intRange)` / `spinner(doubleRange, step)` | Numeric spinner | -| `slider(min, max, minorTick, majorTick)` | Slider, use `.labelTable()` for tick labels | -| `textArea()` | Multi-line text, use `.rows(n)` and `.align(AlignX.FILL)` | -| `comboBox(items)` | Combo box / dropdown | -| `comment("text")` | Gray comment text, standalone | -| `cell(component)` | Wrap any arbitrary Swing component | -| `scrollCell(component)` | Wrap component in a scroll pane | -| `cell()` | Empty placeholder cell for grid alignment | - -Key component examples: - -```kotlin -panel { - var mode = "auto" - - buttonsGroup { - row("Mode:") { - radioButton("Automatic", "auto") - radioButton("Manual", "manual") - } - }.bind({ mode }, { mode = it }) - - row("Slider:") { - slider(0, 10, 1, 5) - .labelTable(mapOf( - 0 to JBLabel("0"), - 5 to JBLabel("5"), - 10 to JBLabel("10"), - )) - } - - row { - label("Text area:") - .align(AlignY.TOP) - .gap(RightGap.SMALL) - textArea() - .rows(5) - .align(AlignX.FILL) - }.layout(RowLayout.PARENT_GRID) -} -``` - -### Component Labels - -Labels for modifiable components must be connected via one of two methods. This ensures correct spacing, mnemonic support, and accessibility. - -- Row label: `row("&Label:") { textField() }`, mnemonic via `&`, label in left column -- Cell label: `textField().label("&Label:", LabelPosition.TOP)`, label attached to cell, optionally on top - -```kotlin -panel { - row("&Row label:") { - textField() - textField() - .label("Cell label at &left:") - } - row { - textField() - .label("Cell label at &top:", LabelPosition.TOP) - } -} -``` - -When a row contains a `checkBox` or `radioButton`, the DSL automatically increases space after the row label per IntelliJ UI Guidelines. - -### Comments - -Three types of comments, each with different placement and semantics: - -| Type | Method | Placement | -|---|---|---| -| Cell comment, bottom | `cell.comment("text")` | Below the cell | -| Cell comment, right | `cell.commentRight("text")` | Right of the cell | -| Cell context help | `cell.contextHelp("text", "title")` | Help icon with popup | -| Row comment | `row.rowComment("text")` | Below the entire row | -| Arbitrary comment | `comment("text")` | Standalone gray text | - -```kotlin -panel { - row { - textField() - .comment("Bottom comment") - textField() - .commentRight("Right comment") - textField() - .contextHelp("Help popup text") - } - - row("Label:") { - textField() - }.rowComment("This comment sits below the whole row") - - row { - comment("Standalone comment, supports links and  icons") - } -} -``` - -Comments support HTML with clickable links, bundled icons via ``, and line width control via `maxLineLength`. Use `MAX_LINE_LENGTH_NO_WRAP` to prevent wrapping. - -### Groups and Structure - -| Method | Grid | Description | -|---|---|---| -| `panel {}` | Own grid | Sub-panel occupying full width | -| `rowsRange {}` | Parent grid | Grouped rows sharing parent grid, useful with `enabledIf` | -| `group("Title") {}` | Own grid | Titled section with vertical spacing before/after | -| `groupRowsRange("Title") {}` | Parent grid | Titled section sharing parent grid alignment | -| `collapsibleGroup("Title") {}` | Own grid | Expandable section, Tab-focusable, supports mnemonics | -| `buttonsGroup("Title") {}` | None | Groups `radioButton` or `checkBox` under a title | -| `separator()` | None | Horizontal separator line | -| Row `panel {}` | Own grid | Sub-panel inside a cell | - -```kotlin -panel { - group("Settings") { - row("Name:") { textField() } - row("Path:") { textFieldWithBrowseButton() } - } - - collapsibleGroup("Advanced") { - row("Timeout:") { intTextField(0..1000) } - } - - var enabled = true - buttonsGroup("Mode:") { - row { radioButton("Automatic", true) } - row { radioButton("Manual", false) } - }.bind({ enabled }, { enabled = it }) - - separator() - - row { - label("Nested panels:") - panel { - row("Sub row 1:") { textField() } - row("Sub row 2:") { textField() } - } - } -} -``` - -### Gaps and Spacing - -- Horizontal gaps: `cell.gap(RightGap.SMALL)` between a label-like checkbox and its related field. Medium gap is the default between cells. -- Two-column layout: `twoColumnsRow({}, {})` or `gap(RightGap.COLUMNS)` with `.layout(RowLayout.PARENT_GRID)`. -- Left indent: `indent {}` for indented sub-content. -- Vertical gaps: `.topGap(TopGap.MEDIUM)` / `.bottomGap(BottomGap.MEDIUM)` on rows to separate unrelated groups. Attach gaps to the related row so hiding rows does not break layout. - -```kotlin -panel { - group("Horizontal Gaps") { - row { - val cb = checkBox("Use mail:") - .gap(RightGap.SMALL) - textField() - .enabledIf(cb.selected) - } - row("Width:") { - textField() - .gap(RightGap.SMALL) - label("pixels") - } - } - - group("Indent") { - row { label("Not indented") } - indent { - row { label("Indented row") } - } - } - - group("Two Columns") { - twoColumnsRow({ - checkBox("First column") - }, { - checkBox("Second column") - }) - } - - group("Vertical Gaps") { - row { checkBox("Option 1") } - row { checkBox("Option 2") } - row { checkBox("Unrelated option") } - .topGap(TopGap.MEDIUM) - } -} -``` - -### Enabled and Visible State - -Bind enabled/visible state to a checkbox or other observable. Works on rows, `indent {}` blocks, `rowsRange {}`, and individual cells. - -```kotlin -panel { - group("Enabled") { - lateinit var cb: Cell - row { cb = checkBox("Enable options") } - indent { - row { checkBox("Option 1") } - row { checkBox("Option 2") } - }.enabledIf(cb.selected) - } - - group("Visible") { - lateinit var cb: Cell - row { cb = checkBox("Show options") } - indent { - row { checkBox("Option 1") } - row { checkBox("Option 2") } - }.visibleIf(cb.selected) - } -} -``` - -### Binding - -Bind component values to model properties. Values are applied on `DialogPanel.apply()`, checked with `.isModified()`, and reverted with `.reset()`. - -| Method | Component | -|---|---| -| `bindSelected(model::prop)` | checkBox | -| `bindText(model::prop)` | textField | -| `bindIntText(model::prop)` | intTextField | -| `bindItem(model::prop.toNullableProperty())` | comboBox | -| `bindValue(model::prop)` | slider | -| `bindIntValue(model::prop)` | spinner | -| `buttonsGroup {}.bind(model::prop)` | radio group | - -```kotlin -enum class Theme { LIGHT, DARK } - -data class Settings( - var name: String = "", - var count: Int = 0, - var enabled: Boolean = false, - var theme: Theme = Theme.LIGHT, -) - -val model = Settings() - -val panel = panel { - row("Name:") { - textField().bindText(model::name) - } - row("Count:") { - intTextField(0..100).bindIntText(model::count) - } - row { - checkBox("Enabled").bindSelected(model::enabled) - } - buttonsGroup("Theme:") { - row { radioButton("Light", Theme.LIGHT) } - row { radioButton("Dark", Theme.DARK) } - }.bind(model::theme) -} - -panel.isModified() -panel.apply() -panel.reset() -``` - -### Validation - -Attach input validation rules to cells. Rules run continuously and display inline error/warning indicators. - -```kotlin -panel { - row("Username:") { - textField() - .columns(COLUMNS_MEDIUM) - .cellValidation { - addInputRule("Must not be empty") { - it.text.isBlank() - } - } - } - row("Port:") { - textField() - .columns(COLUMNS_MEDIUM) - .cellValidation { - addInputRule("Contains non-numeric characters", level = Level.WARNING) { - it.text.contains(Regex("[^0-9]")) - } - } - } -} -``` - -Activate validators by calling `dialogPanel.registerValidators(disposable)` after creating the panel. - -### Tips and Common Patterns - -| Pattern | Usage | -|---|---| -| `.bold()` | Bold text on any cell | -| `.columns(COLUMNS_MEDIUM)` | Set preferred width of textField / comboBox / textArea | -| `.text("initial")` | Set initial text on text components | -| `.resizableColumn()` | Column fills remaining horizontal space | -| `cell()` | Empty placeholder cell for grid alignment | -| `.widthGroup("name")` | Equalize widths across rows, cannot combine with `AlignX.FILL` | -| `.align(AlignX.FILL)` | Stretch component to fill available width | -| `.align(AlignY.TOP)` | Top-align component in its cell | -| `.applyToComponent { }` | Direct access to the underlying Swing component | -| `.selected(true)` | Default-select a radioButton when no bound value matches | -| `.gap(RightGap.COLUMNS)` | Column-level gap for multi-column layouts | - -```kotlin -panel { - row { label("Title").bold() } - - row("Name:") { - textField() - .columns(COLUMNS_MEDIUM) - .resizableColumn() - .align(AlignX.FILL) - } - - row("") { - textField() - }.rowComment("""Use row("") for an empty label column that aligns with labeled rows""") - - row { - text("Comment-colored text") - .applyToComponent { foreground = JBUI.CurrentTheme.ContextHelp.FOREGROUND } - } -} -``` - -## Manual Swing - -Use manual Swing only when Kotlin UI DSL cannot express the UI cleanly. - -Preferred manual Swing rules: - -- Use IntelliJ platform components instead of raw Swing components. -- Use layout managers and component defaults rather than fixed sizes. -- Use `JBUI.Borders.empty(...)`, `JBUI.insets(...)`, `JBUI.size(...)`, and `JBUI.scale(...)` instead of raw AWT/Swing sizing primitives. -- Do not set default properties explicitly. -- Keep custom components small and focused. -- For painting, use theme-aware colors and `JBUI.scale(...)` for pixel values. - -## Platform Spacing and Insets - -Use semantic spacing APIs before inventing numbers. Do not copy fallback values from IntelliJ source into generated UI code; those values are defaults behind theme keys and may change by UI theme, New UI, OS, or IDE version. - -| Need | Preferred source | -|---|---| -| Dialogs, forms, settings | Kotlin UI DSL row/group/indent/gap APIs | -| Component gaps in Kotlin UI DSL | `RightGap.SMALL`, `RightGap.COLUMNS`, `TopGap.MEDIUM`, `BottomGap.MEDIUM`, `.customize(UnscaledGaps(...))` as a last resort | -| Manual Swing empty padding | `JBUI.Borders.empty(...)` | -| Manual Swing insets | `JBUI.insets(...)`, `JBUI.emptyInsets()`, `JBUI.insetsTop(...)`, `JBUI.insetsLeft(...)`, `JBUI.insetsBottom(...)`, `JBUI.insetsRight(...)` | -| Manual Swing dimensions | `JBUI.size(...)`, `JBDimension`, `JBUI.scale(...)` | -| Side separators | `JBUI.Borders.customLineTop(...)`, `customLineBottom(...)`, `customLineLeft(...)`, `customLineRight(...)` | -| Composed borders | `JBUI.Borders.compound(...)`, `JBUI.Borders.merge(...)` | -| Simple `BorderLayout` panels | `JBUI.Panels.simplePanel(...)`, `BorderLayoutPanel` | -| Simple vertical custom Swing groups | `VerticalLayout` | -| Fluent platform panels | `JBPanel.withBorder(...)`, `.andTransparent()`, `.andOpaque()`, `.withBackground(...)` | - -Use `JBUI.CurrentTheme` for context-specific spacing and dimensions: - -| UI area | Preferred source | -|---|---| -| Action list rows | `JBUI.CurrentTheme.ActionsList.cellPadding()` | -| Action icon/text gaps | `JBUI.CurrentTheme.ActionsList.elementIconGap()` | -| Action mnemonic gaps | `JBUI.CurrentTheme.ActionsList.mnemonicIconGap()` / `mnemonicInsets()` | -| Popup selection interior | `JBUI.CurrentTheme.Popup.Selection.innerInsets()` | -| Popup separators | `JBUI.CurrentTheme.Popup.separatorInsets()` / `separatorLabelInsets()` | -| Popup header/search field | `JBUI.CurrentTheme.Popup.headerInsets()`, `searchFieldBorderInsets()`, `searchFieldInputInsets()` | -| Complex popup headers | `JBUI.CurrentTheme.ComplexPopup.headerInsets()` | -| Complex popup text fields | `JBUI.CurrentTheme.ComplexPopup.textFieldBorderInsets()` / `textFieldInputInsets()` | -| Search Everywhere / big popup header | `JBUI.CurrentTheme.BigPopup.headerBorder()` / `headerToolbarInsets()` / `tabInsets()` | -| Popup advertiser/footer | `JBUI.CurrentTheme.Advertiser.border()` | -| Toolbar buttons | `JBUI.CurrentTheme.Toolbar.toolbarButtonInsets()` / `mainToolbarButtonInsets()` | -| Toolbar containers | `JBUI.CurrentTheme.Toolbar.horizontalToolbarInsets()` / `verticalToolbarInsets()` | -| Tool-window headers | `JBUI.CurrentTheme.ToolWindow.headerLabelLeftRightInsets()` / `headerToolbarLeftRightInsets()` / `headerTabLeftRightInsets()` | -| Lists | `JBUI.CurrentTheme.List.rowHeight()` | -| Trees | `JBUI.CurrentTheme.Tree.rowHeight()` | -| VCS log rows | `JBUI.CurrentTheme.VersionControl.Log.rowHeight()` / `verticalPadding()` | -| Help tooltips | `JBUI.CurrentTheme.HelpTooltip.defaultTextBorderInsets()` / `smallTextBorderInsets()` | -| Navigation bar items | `JBUI.CurrentTheme.NavBar.itemInsets()` | - -When using Kotlin UI DSL, prefer DSL semantics over manual padding: - -```kotlin -panel { - group(KiloBundle.message("settings.section")) { - row(KiloBundle.message("settings.name")) { - textField() - .align(AlignX.FILL) - .resizableColumn() - } - indent { - row { checkBox(KiloBundle.message("settings.enabled")) } - } - } -} -``` - -When manual Swing is necessary, get padding from platform utilities: - -```kotlin -val panel = JBUI.Panels.simplePanel(content) - .withBorder(JBUI.Borders.empty(JBUI.CurrentTheme.Popup.headerInsets())) -``` - -For row heights, prefer platform row height APIs over fixed heights: - -```kotlin -component.preferredSize = JBDimension(0, JBUI.CurrentTheme.Tree.rowHeight()) -``` - -## Generic Platform Utilities - -Use existing platform utility classes instead of hand-rolled labels, renderers, borders, colors, fonts, and validation behavior. - -| Need | Preferred API | -|---|---| -| Concise border layout | `BorderLayoutPanel`, `JBUI.Panels.simplePanel(...)` | -| Platform panel helpers | `JBPanel.withBorder(...)`, `.andTransparent()`, `.andOpaque()` | -| Platform label behavior | `JBLabel` | -| Context help | `ContextHelpLabel` | -| Links | `HyperlinkLabel`, `LinkLabel`, Kotlin UI DSL `link(...)` / `browserLink(...)` | -| Error/validation label in legacy UI | `ErrorLabel` only when an inline validation component is required and DSL validation is not available | -| High-performance rich fragments | `SimpleColoredComponent` | -| List renderers | `ColoredListCellRenderer`, Kotlin `listCellRenderer` / `textListCellRenderer` / `LcrRow` | -| Tree renderers | `ColoredTreeCellRenderer` | -| Renderer text styles | `SimpleTextAttributes` | -| Editable list toolbar | `ToolbarDecorator` | -| Platform list | `JBList` | -| Platform tree | `Tree` | - -For renderer text, prefer `SimpleTextAttributes` over direct color/font mutation: - -```kotlin -append(text, SimpleTextAttributes.ERROR_ATTRIBUTES) -append(detail, SimpleTextAttributes.GRAYED_ATTRIBUTES) -append(link, SimpleTextAttributes.LINK_ATTRIBUTES) -``` - -## Tool Windows - -- Register declaratively in module XML via `com.intellij.toolWindow` extension point. -- Implement `ToolWindowFactory.createToolWindowContent()`, called lazily on first click. -- Use `SimpleToolWindowPanel(vertical = true)` as a convenient base for toolbar and content layout. -- Add tabs via `ToolWindow.contentManager`: create content with `ContentFactory.getInstance().createContent(component, title, isLockable)`, then `contentManager.addContent()`. -- For conditional display, implement `ToolWindowFactory.isApplicableAsync(project)`. -- Always use `ToolWindowManager.invokeLater()` instead of `Application.invokeLater()` for tool-window-related EDT tasks. - -## Dialogs - -- Extend `DialogWrapper`. -- Call `init()` from the constructor. -- Override `createCenterPanel()` to return UI content. -- Prefer Kotlin UI DSL v2 for panel contents. -- Override `getPreferredFocusedComponent()` for initial focus. -- Override `getDimensionServiceKey()` for size persistence when useful. -- Show with `showAndGet()` for modal boolean result, or `show()` and then `getExitCode()`. -- For input validation, call `initValidation()` in the constructor and override `doValidate()` to return `null` if valid or `ValidationInfo(message, component)` if invalid. - -## Platform Components - -Always use IntelliJ platform components instead of raw Swing where an equivalent exists. - -| Instead of | Use | Package | -|---|---|---| -| `JLabel` | `JBLabel` | `com.intellij.ui.components` | -| `JTextField` | `JBTextField` | `com.intellij.ui.components` | -| `JTextArea` | `JBTextArea` | `com.intellij.ui.components` | -| `JList` | `JBList` | `com.intellij.ui.components` | -| `JScrollPane` | `JBScrollPane` | `com.intellij.ui.components` | -| `JTable` | `JBTable` | `com.intellij.ui.table` | -| `JTree` | `Tree` | `com.intellij.ui.treeStructure` | -| `JSplitPane` | `JBSplitter` | `com.intellij.ui` | -| `JTabbedPane` | `JBTabs` | `com.intellij.ui.tabs` | -| `JCheckBox` | `JBCheckBox` | `com.intellij.ui.components` | -| Raw runtime colors | `UIUtil`, `JBUI.CurrentTheme`, `NamedColorUtil`, `JBColor.namedColor`, `JBColor.lazy` | `com.intellij.util.ui`, `com.intellij.ui` | -| `EmptyBorder` | `JBUI.Borders.empty()` | `com.intellij.util.ui` | -| Hardcoded pixel sizes | `JBUI.scale(px)` | `com.intellij.util.ui` | - -Inspection `Plugin DevKit | Code | Undesirable class usage` highlights raw Swing usage where a platform replacement exists. - -## Multi-line and Rich Text - -| Need | Component | -|---|---| -| Rich HTML with modern CSS, icons, shortcuts | `JBHtmlPane` (`com.intellij.ui.components.JBHtmlPane`) | -| Simple multi-line label with HTML | `JBLabel` + `XmlStringUtil.wrapInHtml()` | -| Scrollable / wrapping HTML panel | `SwingHelper.createHtmlViewer()` | -| High-performance colored text fragments in trees/lists/tables | `SimpleColoredComponent` | -| Plain-text newline splitting | `MultiLineLabel`, legacy, do not use in new code | - -- Build HTML programmatically with `HtmlChunk` / `HtmlBuilder` (`com.intellij.openapi.util.text.HtmlChunk`). Avoid raw HTML string concatenation because it risks injection and breaks localization. -- For simple wrapping/escaping, use `XmlStringUtil.wrapInHtml(content)`, `XmlStringUtil.wrapInHtmlLines(lines...)`, and `XmlStringUtil.escapeString(text)`. -- For selectable/copyable label text, use `JBLabel.setCopyable(true)`, which switches internally to `JEditorPane` while preserving label appearance. Use `setAllowAutoWrapping(true)` for auto-wrap. -- When creating a `JEditorPane` manually, always use `HTMLEditorKitBuilder` instead of constructing `HTMLEditorKit` directly: `editorPane.setEditorKit(HTMLEditorKitBuilder.simple())` or `.withWordWrapViewFactory().build()`. -- For single-line overflow/ellipsis, use `SwingTextTrimmer`. Do not manually truncate strings. -- Put all user-visible strings in `*.properties` files. HTML markup in values is acceptable. - -## Theme-Derived Colors - -Do not hardcode runtime colors. IntelliJ UI colors must come from the current theme, component state, editor color scheme, or a centralized semantic named color key. - -Prefer semantic helpers for common UI roles: - -| Need | Preferred API | -|---|---| -| Ordinary label text | `UIUtil.getLabelForeground()` | -| Secondary/help text | `UIUtil.getContextHelpForeground()` | -| Info label text | `UIUtil.getLabelInfoForeground()` | -| Success label text | `UIUtil.getLabelSuccessForeground()` | -| Error label text | `UIUtil.getErrorForeground()` | -| Warning label text | `JBUI.CurrentTheme.Label.warningForeground()` | -| Generic label foreground by state | `JBUI.CurrentTheme.Label.foreground(selected)` / `disabledForeground(selected)` | -| Inactive secondary text | `NamedColorUtil.getInactiveTextColor()` | -| Bounds and standard border color | `NamedColorUtil.getBoundsColor()` / `JBColor.border()` | -| Tooltips | `UIUtil.getToolTipForeground()` / `getToolTipBackground()` / `JBUI.CurrentTheme.Tooltip.*` | -| Links | `JBUI.CurrentTheme.Link.Foreground.ENABLED` / `HOVERED` / `PRESSED` / `VISITED` / `DISABLED` / `SECONDARY` | -| List renderer text/background | `UIUtil.getListForeground(selected, focused)` / `UIUtil.getListBackground(selected, focused)` | -| Tree renderer text/background | `UIUtil.getTreeForeground(selected, focused)` / `UIUtil.getTreeBackground(selected, focused)` | -| Non-selected tree content | `UIUtil.getTreeForeground()` / `UIUtil.getTreeBackground()` | - -Prefer `JBUI.CurrentTheme` for component-area colors and borders: - -| Need | Preferred API | -|---|---| -| Popup background | `JBUI.CurrentTheme.Popup.BACKGROUND` | -| Complex popup header | `JBUI.CurrentTheme.ComplexPopup.HEADER_BACKGROUND` | -| Separators | `JBUI.CurrentTheme.CustomFrameDecorations.separatorForeground()` | -| Advertiser/footer surfaces | `JBUI.CurrentTheme.Advertiser.foreground()` / `background()` / `border()` | -| Links | `JBUI.CurrentTheme.Link.Foreground.ENABLED` | -| Tree selection state | `JBUI.CurrentTheme.Tree.foreground(...)` and selection helpers | -| Validation errors | `JBUI.CurrentTheme.Validator.errorBorderColor()` / `errorBackgroundColor()` | -| Validation warnings | `JBUI.CurrentTheme.Validator.warningBorderColor()` / `warningBackgroundColor()` | -| Focus error/warning outlines | `JBUI.CurrentTheme.Focus.errorColor(active)` / `warningColor(active)` | -| Banner-like surfaces | `JBUI.CurrentTheme.Banner.*` | -| Notification tool-window surfaces | `JBUI.CurrentTheme.NotificationError.*`, `NotificationWarning.*`, `NotificationInfo.*` | -| Icon badges | `JBUI.CurrentTheme.IconBadge.*` | - -Use `JBColor.lazy { ... }` for colors that depend on runtime state or the active editor color scheme: - -```kotlin -val bg = JBColor.lazy { EditorColorsManager.getInstance().globalScheme.defaultBackground } -``` - -Use `JBColor.namedColor("Some.Semantic.Key", fallback)` only when defining or consuming a semantic color key. Prefer a fallback from an existing theme API over numeric RGB values in examples. - -For HTML/CSS snippets, compute the color from a theme API and convert it with `ColorUtil.toHtmlColor(...)`. Do not write hardcoded CSS color literals. - -```kotlin -val fg = ColorUtil.toHtmlColor(UIUtil.getContextHelpForeground()) -val html = HtmlChunk.span("color: $fg").addText(text) -``` - -Recommended examples: - -```kotlin -label.applyToComponent { - foreground = UIUtil.getContextHelpForeground() - font = JBFont.medium() -} -``` - -```kotlin -foreground = if (selected) { - UIUtil.getTreeForeground(true, hasFocus) -} else { - UIUtil.getContextHelpForeground() -} -``` - -Avoid inline `Color(...)`, numeric `JBColor(...)`, `Gray.xNN`, `JBColor.GRAY`, and hex color literals in runtime UI code. The exception is a centralized semantic color definition with a named color key when no existing platform key exists. - -## Theme-Derived Fonts - -Do not hardcode font sizes or font families. IntelliJ fonts must come from component defaults, platform typography helpers, or existing component fonts. - -- Prefer component default fonts when no style change is needed. -- Use `JBFont.h1()`, `JBFont.h2()`, `JBFont.h3()`, and `JBFont.h4()` for headings. -- Use `.asBold()`, `.asItalic()`, and `.asPlain()` for style changes on `JBFont` values. -- Use `JBFont.regular()`, `JBFont.medium()`, and `JBFont.small()` for regular and secondary text. -- Use `UIUtil.getLabelFont(UIUtil.FontSize.SMALL)` / `MINI` only when an API expects a raw `Font` and `JBFont` / `RelativeFont` is not a good fit. -- Use `RelativeFont` when adjusting an existing component font relatively. -- Use `RelativeFont.fromResource(...)` for theme-configurable font offsets when matching platform component patterns. -- Use `JBUI.Fonts.smallFont()` only when matching an existing platform component pattern that expects it. -- If editor-like text needs editor typography, derive it from the editor color scheme or existing editor component instead of hardcoding a font. - -For errors, grayed text, shortcuts, and links in renderers, prefer `SimpleTextAttributes.ERROR_ATTRIBUTES`, `GRAYED_ATTRIBUTES`, `SHORTCUT_ATTRIBUTES`, and `LINK_ATTRIBUTES` rather than manually selecting colors and styles. - -Recommended examples: - -```kotlin -label(title).applyToComponent { - font = JBFont.h3().asBold() -} -``` - -```kotlin -RelativeFont.BOLD.install(label) -``` - -Avoid `Font("...")`, raw font sizes, and `deriveFont(14f)` style examples. - -## Borders, Insets, and Spacing - -- Always create borders via `JBUI.Borders.empty(top, left, bottom, right)` and insets via `JBUI.insets()` so they are DPI-aware and auto-update on zoom. -- Use `JBUI.scale(int)` for any pixel dimension to ensure proper HiDPI scaling. -- Prefer Kotlin UI DSL gaps and row layout semantics over manually assigning borders or insets. -- Do not use `EmptyBorder`, raw `Insets`, or raw `Dimension` unless there is no platform alternative and the reason is obvious. - -## Validation and Error UI - -Prefer built-in validation mechanisms over custom error rendering. - -- For Kotlin UI DSL, use `validationOnInput`, `validationOnApply`, `validationInfo`, `errorOnApply`, and `addValidationRule` on cells. -- For DSL dialogs and settings panels, call `DialogPanel.registerValidators(disposable)` when input validation should run continuously. -- Call `DialogPanel.validateAll()` before `apply()` when submitting a DSL panel manually. -- In `DialogWrapper`, use `doValidate()` and return `ValidationInfo(message, component)` for errors. -- Use `ValidationInfo.asWarning()` for warnings and `withOKEnabled()` when a warning should not block submission. -- For hand-built Swing forms, use `ComponentValidator` with `withValidator`, `withFocusValidator`, and `andRegisterOnDocumentListener` instead of custom tooltip/error border logic. -- If custom validation painting is unavoidable, use `JBUI.CurrentTheme.Validator.*` and `JBUI.CurrentTheme.Focus.*` colors. - -Kotlin UI DSL example: - -```kotlin -textField() - .validationOnInput { - if (it.text.isBlank()) error(KiloBundle.message("settings.name.required")) else null - } -``` - -Dialog validation example: - -```kotlin -override fun doValidate(): ValidationInfo? { - if (field.text.isBlank()) return ValidationInfo(KiloBundle.message("settings.name.required"), field) - return null -} -``` - -## Icons - -- Reuse platform icons. Browse at https://intellij-icons.jetbrains.design and access via `AllIcons.*` constants. -- Custom icons belong in `resources/icons/`. -- Load custom icons via `IconLoader.getIcon("/icons/foo.svg", MyClass::class.java)`. -- Organize custom icons in an `icons` package or a `*Icons` object with `@JvmField` on each constant. -- Icon sizing guideline values: actions/nodes = 16x16, tool window = 13x13 classic or 20x20 + 16x16 compact New UI, editor gutter = 12x12 classic or 14x14 New UI. -- Dark variants: `icon.svg` + `icon_dark.svg`. -- HiDPI variants: `icon@2x.svg` + `icon@2x_dark.svg`. -- New UI support: place New UI icons in `expui/`, create `*IconMappings.json`, register via `com.intellij.iconMapper` extension point. -- For SVG asset palette values, follow the IntelliJ icon asset guidelines. Do not copy guideline color literals into runtime Swing UI code. - -## Notifications - -- Declare in module XML: ``. -- Show with `Notification("Kilo Code", "message", NotificationType.INFORMATION).notify(project)`. -- Add actions with `.addAction(NotificationAction.createSimpleExpiring("Label") { ... })`. -- Sticky notifications use `displayType="STICKY_BALLOON"` and `.setSuggestionType(true)`. -- Tool-window-bound notifications use `displayType="TOOL_WINDOW" toolWindowId="Kilo Code"`. -- Prefer non-modal notifications over `Messages.show*()` dialogs. - -## Popups - -- Use `JBPopupFactory.getInstance()` for lightweight floating UI with no chrome and auto-dismiss on focus loss. -- Use `createComponentPopupBuilder(component, focusable)` for arbitrary Swing content. -- Use `createPopupChooserBuilder(list)` for item selection. -- Use `createActionGroupPopup()` for action menus. -- Show with `showInBestPositionFor(editor)`, `showUnderneathOf(component)`, or `showInCenterOf(component)`. - -## Lists and Trees - -- Use `JBList`, not `JList`, for empty text, busy indicator, and tooltip truncation. -- Use `Tree`, not `JTree`, for wide selection painting and auto-scroll on drag-and-drop. -- Use `ColoredListCellRenderer` / `ColoredTreeCellRenderer` for custom renderers. -- Use `append()` for styled text and `setIcon()` for icons. -- Use `ListSpeedSearch(list)` / `TreeSpeedSearch(tree)` for speed search. -- Use `ToolbarDecorator.createDecorator(list).setAddAction { }.setRemoveAction { }.createPanel()` for editable lists with add/remove/reorder toolbar. - -## Before Returning Code - -Review generated UI code and remove: - -- Explicit default property assignments such as unnecessary `isOpaque = false` -- Unnecessary `preferredSize`, `minimumSize`, or `maximumSize` -- Raw `Dimension`, `Insets`, `EmptyBorder`, or `Color` -- Inline runtime colors such as `Color(...)`, numeric `JBColor(...)`, `Gray.xNN`, `JBColor.GRAY`, or hex color literals -- Raw CSS color literals; `ColorUtil.toHtmlColor(themeColor)` is allowed because the source color is theme-derived -- Hardcoded font families, raw font sizes, or numeric-size `deriveFont(...)` calls -- Raw Swing components where IntelliJ replacements exist -- Manual layout code that Kotlin UI DSL can express cleanly -- Hardcoded spacing that should be a DSL gap, row gap, or `JBUI` value -- Extra helpers that do not make the UI clearer or more reusable - -## References - -- IntelliJ Platform UI Guidelines: https://jetbrains.design/intellij/ -- User Interface Components: https://plugins.jetbrains.com/docs/intellij/user-interface-components.html -- UI FAQ: https://plugins.jetbrains.com/docs/intellij/ui-faq.html -- Kotlin UI DSL v2: https://plugins.jetbrains.com/docs/intellij/kotlin-ui-dsl-version-2.html -- Split mode for remote development: https://plugins.jetbrains.com/docs/intellij/split-mode-for-remote-development.html diff --git a/.opencode-version b/.opencode-version new file mode 100644 index 0000000000..6a82e3d1e7 --- /dev/null +++ b/.opencode-version @@ -0,0 +1 @@ +v7.3.0 diff --git a/.opencode/skills/effect/SKILL.md b/.opencode/skills/effect/SKILL.md index 78216ab01c..699d91beeb 100644 --- a/.opencode/skills/effect/SKILL.md +++ b/.opencode/skills/effect/SKILL.md @@ -28,3 +28,18 @@ Use the current Effect v4 / effect-smol source, not memory or older Effect v2/v3 - In tests, prefer the repo's existing Effect test helpers and live tests for filesystem, git, child process, locks, or timing behavior. - Do not introduce `any`, non-null assertions, unchecked casts, or older Effect APIs just to satisfy types. - Do not answer from memory. Verify against `.opencode/references/effect-smol` or nearby code first. +<<<<<<< HEAD +||||||| 12f7967ca4 +- Always use the explore agent with the cloned repository when answering Effect-related questions +- Reference specific files and patterns found in the Effect codebase +- Do not answer from memory - always verify against the source +======= + +## Testing Patterns + +- Use `testEffect(...)` from `packages/opencode/test/lib/effect.ts` for tests that exercise Effect services, layers, runtime context, scoped resources, or platform integrations. +- Use `it.live(...)` for filesystem, git repositories, HTTP servers, sockets, child processes, locks, real time, and other live platform behavior. +- Run tests from package directories such as `packages/opencode`; never run package tests from the repo root. +- Prefer explicit test layers over ad hoc managed runtimes. Keep dependency provisioning visible in the test file. +- Use scoped fixtures and finalizers for resources that must be cleaned up, including temporary directories, flags, databases, fibers, servers, and global state. +>>>>>>> yunqiqiliang/opencode-v7.3.0 diff --git a/AGENTS.md b/AGENTS.md index b893c05650..b7df6cb9c7 100644 --- a/AGENTS.md +++ b/AGENTS.md @@ -11,6 +11,7 @@ Fork chain: **opencode → kilocode → czcode** ## Build and Dev +<<<<<<< HEAD - **Dev**: `~/.bun/bin/bun dev` (runs from root) - **Dev with params**: `~/.bun/bin/bun dev -- help` - **Typecheck**: `~/.bun/bin/bun turbo typecheck` (uses `tsgo`, not `tsc`) @@ -18,6 +19,36 @@ Fork chain: **opencode → kilocode → czcode** - **Single test**: `~/.bun/bin/bun test ./test/tool/tool-define.test.ts` from `packages/opencode/` - **czcode_change check**: `~/.bun/bin/bun run script/check-opencode-annotations.ts` from repo root. CI runs this on PRs touching `packages/opencode/` — every czcode-specific change in shared files must be annotated with `czcode_change` markers. Note: this check may fail on upstream merge PRs (expected — upstream changes don't need czcode markers). Exempt paths (no markers needed): `packages/czcode-lakehouse/`, and czcode-only files (files with `czcode` in the name that don't exist in kilocode upstream). - **Upstream sync**: `~/.bun/bin/bun run script/upstream/list-versions.ts` to see available kilocode versions; `~/.bun/bin/bun run script/upstream/merge.ts v7.x.y` to merge. +||||||| 12f7967ca4 +- **Dev**: `bun run dev` (runs from root) or `bun run --cwd packages/opencode --conditions=browser src/index.ts` +- **Dev with params**: `bun dev -- help` +- **Extension**: `bun run extension` (build + launch VS Code with the extension in dev mode). Pass `--no-build` to skip the build. +- **Typecheck**: `bun turbo typecheck` (uses `tsgo`, not `tsc`) +- **Test**: `bun test` from `packages/opencode/` (NOT from root -- root blocks tests) +- **Single test**: `bun test ./test/tool/tool-define.test.ts` from `packages/opencode/` +- **CLI build artifact size check**: after `bun run script/build.ts --single --skip-install` in `packages/opencode/`, use `du -h dist/*/*/bin/kilo` (scoped package output lives under `dist/@kilocode/`) +- **SDK regen**: After changing server endpoints in `packages/opencode/src/server/`, run `./script/generate.ts` from root to regenerate `packages/sdk/js/` +- **Knip** (unused exports): `bun run knip` from `packages/kilo-vscode/`. CI runs this — all exported types/functions must be imported somewhere. Remove or unexport unused exports before pushing. +- **Source links**: After adding or changing URLs in `packages/kilo-vscode/`, `packages/kilo-vscode/webview-ui/`, or `packages/opencode/src/`, run `bun run script/extract-source-links.ts` from the repo root and commit the updated `packages/kilo-docs/source-links.md`. CI runs this check — the build fails if the file is stale. +- **kilocode_change check**: `bun run check-kilocode-change` from `packages/kilo-vscode/`. CI runs this — `kilocode_change` is a marker for upstream merge conflicts and must not appear in `packages/kilo-vscode/` or `packages/kilo-ui/` (these are entirely Kilo Code additions). Remove the markers before pushing. +- **opencode annotation check**: `bun run script/check-opencode-annotations.ts` from repo root. CI runs this on PRs touching `packages/opencode/` — every Kilo-specific change in shared opencode files must be annotated with `kilocode_change` markers. Exempt paths (no markers needed): `packages/opencode/src/kilocode/`, `packages/opencode/test/kilocode/`, and any path containing `kilocode` in the name. +- **Backend/SDK programmatic testing**: see [TESTING.md](./TESTING.md) for spawning the local main-branch backend (`bun dev serve`) and driving it via `curl` — use this instead of `kilo serve` (prod binary) when testing backend fixes. +======= +- **Dev**: `bun run dev` (runs from root) or `bun run --cwd packages/opencode --conditions=browser src/index.ts` +- **Dev with params**: `bun dev -- help` +- **Extension**: `bun run extension` (build + launch VS Code with the extension in dev mode). Pass `--no-build` to skip the build. +- **Typecheck**: `bun turbo typecheck` (uses `tsgo`, not `tsc`) +- **Test**: `bun test` from `packages/opencode/` (NOT from root -- root blocks tests) +- **Single test**: `bun test ./test/tool/tool-define.test.ts` from `packages/opencode/` +- **CLI build artifact size check**: after `bun run script/build.ts --single --skip-install` in `packages/opencode/`, use `du -h dist/*/*/bin/kilo` (scoped package output lives under `dist/@kilocode/`) +- **SDK regen**: After changing server endpoints in `packages/opencode/src/server/`, run `./script/generate.ts` from root to regenerate `packages/sdk/js/` +- **Knip** (unused exports): `bun run knip` from `packages/kilo-vscode/`. CI runs this — all exported types/functions must be imported somewhere. Remove or unexport unused exports before pushing. +- **Source links**: After adding or changing URLs in `packages/kilo-vscode/`, `packages/kilo-vscode/webview-ui/`, or `packages/opencode/src/`, run `bun run script/extract-source-links.ts` from the repo root and commit the updated `packages/kilo-docs/source-links.md`. CI runs this check — the build fails if the file is stale. +- **kilocode_change check**: `bun run check-kilocode-change` from `packages/kilo-vscode/`. CI runs this — `kilocode_change` is a marker for upstream merge conflicts and must not appear in `packages/kilo-vscode/` or `packages/kilo-ui/` (these are entirely Kilo Code additions). Remove the markers before pushing. +- **opencode annotation check**: `bun run script/check-opencode-annotations.ts` from repo root. CI runs this on PRs touching `packages/opencode/` — every Kilo-specific change in shared opencode files must be annotated with `kilocode_change` markers. Exempt paths (no markers needed): `packages/opencode/src/kilocode/`, `packages/opencode/test/kilocode/`, and any path containing `kilocode` in the name. +- **workflow allowlist**: `bun run script/check-workflows.ts` from repo root. CI runs this as part of the annotations workflow — any `.yml` / `.yaml` file added to or removed from `.github/workflows/` must be reflected in the hardcoded list in `script/check-workflows.ts`. Prevents upstream-merged workflows from silently starting to run in our CI. +- **Backend/SDK programmatic testing**: see [TESTING.md](./TESTING.md) for spawning the local main-branch backend (`bun dev serve`) and driving it via `curl` — use this instead of `kilo serve` (prod binary) when testing backend fixes. +>>>>>>> yunqiqiliang/opencode-v7.3.0 ## Quality Checks @@ -37,10 +68,20 @@ All products are clients of the **CLI** (`packages/opencode/`), which contains t | Product | Package | Description | |---|---|---| +<<<<<<< HEAD | czcode CLI | `packages/opencode/` | Core engine. TUI, `czcode run`, `czcode serve`. Fork of kilocode. | | czcode Lakehouse Plugin | `packages/czcode-lakehouse/` | ClickZetta Lakehouse tools: read_query, write_query, list_objects, describe_object, explain_query, get_context, switch_context. | | czcode TUI Plugins | `packages/opencode/src/kilocode/plugins/czcode-*.tsx` | 10 TUI plugins: connection status, schema browser, VCluster dashboard, role switch, SQL history, sample, count, profile, SingClaw, dotenv loader. | | SingClaw Integration | `packages/opencode/src/kilocode/singclaw/` | Full-screen SingClaw chat via WebSocket RPC. | +||||||| 12f7967ca4 +| Kilo CLI | `packages/opencode/` | Core engine. TUI, `kilo run`, `kilo serve`, `kilo web`. Fork of upstream OpenCode. | +| Kilo VS Code Extension | `packages/kilo-vscode/` | VS Code extension. Bundles the CLI binary, spawns `kilo serve` as a child process. Includes the **Agent Manager** — a multi-session orchestration panel with git worktree isolation. | +| OpenCode Desktop | `packages/desktop/` | Standalone Tauri native app. Bundles CLI as sidecar. Single-session UI. Unrelated to the VS Code extension. Not actively maintained — synced from upstream fork. | +| OpenCode Web | `packages/app/` | Shared SolidJS frontend used by both the desktop app and `kilo web` CLI command. Not actively maintained — synced from upstream fork. | +======= +| Kilo CLI | `packages/opencode/` | Core engine. TUI, `kilo run`, `kilo serve`. Fork of upstream OpenCode. | +| Kilo VS Code Extension | `packages/kilo-vscode/` | VS Code extension. Bundles the CLI binary, spawns `kilo serve` as a child process. Includes the **Agent Manager** — a multi-session orchestration panel with git worktree isolation. | +>>>>>>> yunqiqiliang/opencode-v7.3.0 ## Data Agent Roles @@ -72,17 +113,46 @@ Agent definitions: `packages/opencode/src/kilocode/agent/index.ts` | `/cz_skill-update` | — | Update skills | | `/cz_skill-fix` | — | Fix skill locally | +## Package Instructions + +- When a task primarily touches `packages/kilo-jetbrains/`, read `packages/kilo-jetbrains/AGENTS.md` before planning or editing. +- For JetBrains Kotlin/Swing UI work, also apply `packages/kilo-jetbrains/.kilo/skills/jetbrains-ui-style/SKILL.md`. + ## Monorepo Structure Turborepo + Bun workspaces. The packages you'll work with most: | Package | Name | Purpose | |---|---|---| +<<<<<<< HEAD | `packages/opencode/` | `@kilocode/cli` | Core CLI — agents, tools, sessions, server, TUI. Most work happens here. | | `packages/czcode-lakehouse/` | `@czcode/lakehouse` | Lakehouse plugin — czcode-specific, no annotation markers needed. | | `packages/sdk/js/` | `@kilocode/sdk` | Auto-generated TypeScript SDK. Do not edit `src/gen/` by hand. | | `packages/plugin/` | `@kilocode/plugin` | Plugin/tool interface definitions. | | `packages/util/` | `@opencode-ai/util` | Shared utilities. | +||||||| 12f7967ca4 +| `packages/opencode/` | `@kilocode/cli` | Core CLI -- agents, tools, sessions, server, TUI. This is where most work happens. | +| `packages/sdk/js/` | `@kilocode/sdk` | Auto-generated TypeScript SDK (client for the server API). Do not edit `src/gen/` by hand. | +| `packages/kilo-vscode/` | `kilo-code` | VS Code extension with sidebar chat + Agent Manager. See its own `AGENTS.md` for details. | +| `packages/kilo-gateway/` | `@kilocode/kilo-gateway` | Kilo auth, provider routing, API integration | +| `packages/kilo-telemetry/` | `@kilocode/kilo-telemetry` | PostHog analytics + OpenTelemetry | +| `packages/kilo-i18n/` | `@kilocode/kilo-i18n` | Internationalization / translations | +| `packages/kilo-ui/` | `@kilocode/kilo-ui` | SolidJS component library shared by the extension webview and `packages/app/` | +| `packages/app/` | `@opencode-ai/app` | Shared SolidJS web UI for desktop app and `kilo web` | +| `packages/desktop/` | `@opencode-ai/desktop` | Tauri desktop app shell | +| `packages/util/` | `@opencode-ai/util` | Shared utilities (error, path, retry, slug, etc.) | +| `packages/plugin/` | `@kilocode/plugin` | Plugin/tool interface definitions | +======= +| `packages/opencode/` | `@kilocode/cli` | Core CLI -- agents, tools, sessions, server, TUI. This is where most work happens. | +| `packages/sdk/js/` | `@kilocode/sdk` | Auto-generated TypeScript SDK (client for the server API). Do not edit `src/gen/` by hand. | +| `packages/kilo-vscode/` | `kilo-code` | VS Code extension with sidebar chat + Agent Manager. See its own `AGENTS.md` for details. | +| `packages/kilo-gateway/` | `@kilocode/kilo-gateway` | Kilo auth, provider routing, API integration | +| `packages/kilo-telemetry/` | `@kilocode/kilo-telemetry` | PostHog analytics + OpenTelemetry | +| `packages/kilo-i18n/` | `@kilocode/kilo-i18n` | Internationalization / translations | +| `packages/kilo-ui/` | `@kilocode/kilo-ui` | SolidJS component library shared by the extension webview and docs screenshot stories | +| `packages/util/` | `@opencode-ai/util` | Shared utilities (error, path, retry, slug, etc.) | +| `packages/plugin/` | `@kilocode/plugin` | Plugin/tool interface definitions | +>>>>>>> yunqiqiliang/opencode-v7.3.0 ## Style Guide @@ -119,6 +189,91 @@ Never leave a `catch` block empty. Log it or rethrow. You MUST avoid using `mocks` as much as possible. Tests MUST test actual implementation, do not duplicate logic into a test. +<<<<<<< HEAD +||||||| 12f7967ca4 +## Markdown Tables + +Do not pad markdown table cells for column alignment. Use the compact form with single-space-padded content cells and a minimal separator row: + +``` +| Command | What it runs | +|---|---| +| `kilo serve` | The prod CLI on `$PATH`. | +``` + +Do **not** right-pad cells to line up columns: + +``` +| Command | What it runs | +| ----------------------------- | ------------------------ | +| `kilo serve` | The prod CLI on `$PATH`. | +``` + +Padding makes every content change rewrite the entire table, which blows up diffs on untouched rows. Markdown files are excluded from prettier (see `.prettierignore`) so running the formatter won't re-pad them, and `script/check-md-table-padding.ts` enforces the rule in CI. Run `bun run script/check-md-table-padding.ts --fix` to auto-rewrite padded tables. + +## Commit Conventions + +[Conventional Commits](https://www.conventionalcommits.org/) with scopes matching packages: `vscode`, `cli`, `agent-manager`, `sdk`, `ui`, `i18n`, `kilo-docs`, `gateway`, `telemetry`, `desktop`. Omit scope when spanning multiple packages. + +## Changesets + +User-facing changes (features, fixes, breaking changes) require a changeset file for release notes. Run `bunx changeset add` or manually create `.changeset/.md`. Use `patch` for bug fixes, `minor` for new features, `major` for breaking changes. See `.changeset/README.md` for details. + +Changeset descriptions appear directly in release notes and are read by end users. Keep them concise and feature-oriented — describe **what changed from the user's perspective**, not implementation details. Write in imperative mood (e.g. "Support exporting conversations as markdown" not "Add a new export handler that serializes session messages to .md files"). + +## Pull Requests + +PR descriptions should be 2-3 lines covering **what** changed and **why**. Focus on intent and context a reviewer can't get from the diff — skip file-by-file inventories, test result summaries, and anything obvious from the code itself. + +## GitHub Issues + +- When creating a GitHub issue for the VS Code extension or JetBrains plugin, use the repo's existing issue templates in `.github/ISSUE_TEMPLATE/`. Pick the matching template (`Bug report`, `Feature Request`, or `Question`) instead of opening a blank issue. +- Do not add platform-specific title prefixes such as `[JetBrains]`, `[Jetbrains]`, `[JB]`, `[VS Code]`, `[VSCode]`, or similar. Use a plain, descriptive title. +- Always add VS Code extension issues to the GitHub project `VS Code Extension`: https://github.com/orgs/Kilo-Org/projects/25 +- Always add JetBrains plugin issues to the GitHub project `Jetbrains Plugin`: https://github.com/orgs/Kilo-Org/projects/39 +- When using `gh`, prefer `gh issue create --template "..." --project "..."` with the matching project title. +- If project assignment fails because `gh` is missing the required scope, run `gh auth refresh -s project` and retry. + +======= +## Markdown Tables + +Do not pad markdown table cells for column alignment. Use the compact form with single-space-padded content cells and a minimal separator row: + +``` +| Command | What it runs | +|---|---| +| `kilo serve` | The prod CLI on `$PATH`. | +``` + +Do **not** right-pad cells to line up columns: + +``` +| Command | What it runs | +| ----------------------------- | ------------------------ | +| `kilo serve` | The prod CLI on `$PATH`. | +``` + +Padding makes every content change rewrite the entire table, which blows up diffs on untouched rows. Markdown files are excluded from prettier (see `.prettierignore`) so running the formatter won't re-pad them, and `script/check-md-table-padding.ts` enforces the rule in CI. Run `bun run script/check-md-table-padding.ts --fix` to auto-rewrite padded tables. + +## Commit Conventions + +[Conventional Commits](https://www.conventionalcommits.org/) with scopes matching packages: `vscode`, `cli`, `agent-manager`, `sdk`, `ui`, `i18n`, `kilo-docs`, `gateway`, `telemetry`, `desktop`. Omit scope when spanning multiple packages. + +## Changesets + +User-facing changes (features, fixes, breaking changes) require a changeset file for release notes. Run `bunx changeset add` or manually create `.changeset/.md`. Use `patch` for bug fixes, `minor` for new features, `major` for breaking changes. See `.changeset/README.md` for details. + +Changeset descriptions appear directly in release notes and are read by end users. Keep them concise and feature-oriented — describe **what changed from the user's perspective**, not implementation details. Write in imperative mood (e.g. "Support exporting conversations as markdown" not "Add a new export handler that serializes session messages to .md files"). + +## Pull Requests + +PR descriptions should be 2-3 lines covering **what** changed and **why**. Focus on intent and context a reviewer can't get from the diff — skip file-by-file inventories, test result summaries, and anything obvious from the code itself. + +## GitHub Issues + +When creating or managing GitHub issues for the VS Code extension or JetBrains plugin via `gh`, load `.kilo/skills/gh-issues/SKILL.md`. It covers templates, project boards (`VS Code Extension`, `Jetbrains Plugin`), title conventions, and the `gh auth refresh -s project` recovery path. + +>>>>>>> yunqiqiliang/opencode-v7.3.0 ## Fork Merge Process czcode is a fork of [kilocode](https://github.com/Kilo-Org/kilocode). @@ -143,21 +298,49 @@ We regularly merge upstream changes from kilocode. To minimize merge conflicts: czcode uses **two layers** of change markers corresponding to the fork chain: +<<<<<<< HEAD | Marker | Purpose | Used when | |---|---|---| | `kilocode_change` | Marks kilocode changes relative to opencode | Merging opencode → kilocode (upstream of us) | | `czcode_change` | Marks czcode changes relative to kilocode | Merging kilocode → czcode (our direct upstream) | +||||||| 12f7967ca4 +### Kilocode Change Markers +======= +### Git conflict style -**Rule: any code czcode modifies or adds that kilocode might also change needs a `czcode_change` marker.** This includes files inside `packages/opencode/src/kilocode/` — that directory is shared with kilocode upstream and will be overwritten during merges. +`bun install` sets `merge.conflictStyle=zdiff3` repo-locally via `script/setup-git.ts` (wired into `postinstall`). Conflicts include the common ancestor between `|||||||` and `=======`, which is what `script/upstream/` and `mergiraf` rely on for structural resolution and what makes manual resolution on shared opencode files tractable. If you've overridden it in your user config, the repo-local setting takes precedence — don't override it back. -Mark czcode-specific changes with `czcode_change` comments. +### Kilocode Change Markers +>>>>>>> yunqiqiliang/opencode-v7.3.0 -**Single line:** +<<<<<<< HEAD +**Rule: any code czcode modifies or adds that kilocode might also change needs a `czcode_change` marker.** This includes files inside `packages/opencode/src/kilocode/` — that directory is shared with kilocode upstream and will be overwritten during merges. +Mark czcode-specific changes with `czcode_change` comments. +||||||| 12f7967ca4 +To minimize merge conflicts when syncing with upstream, mark Kilo Code-specific changes in shared code with `kilocode_change` comments. +======= +When editing shared upstream files, mark Kilo-specific lines with `kilocode_change` comments so future merges can find them. The basic forms are: +>>>>>>> yunqiqiliang/opencode-v7.3.0 + +- Single line: `const value = 42 // kilocode_change` +- Multi-line block: wrap with `// kilocode_change start` / `// kilocode_change end` +- New file in a shared path: `// kilocode_change - new file` at the top +- JSX/TSX: use `{/* kilocode_change */}` (and `{/* kilocode_change start */}` / `end`) + +<<<<<<< HEAD ```typescript const value = 42 // czcode_change ``` +||||||| 12f7967ca4 +```typescript +const value = 42 // kilocode_change +``` +======= +Markers are NOT needed in paths that contain `kilocode` in the name (e.g. `packages/opencode/src/kilocode/`, `packages/opencode/test/kilocode/`) — these are entirely Kilo Code additions and won't conflict with upstream. +>>>>>>> yunqiqiliang/opencode-v7.3.0 +<<<<<<< HEAD **Multi-line:** ```typescript @@ -224,3 +407,46 @@ After merging upstream kilocode changes, test these before releasing: - [ ] Copy to clipboard toast auto-dismisses (2 seconds) - [ ] Sidebar shows Lakehouse connection info - [ ] `czcode_change` annotation check passes +||||||| 12f7967ca4 +**Multi-line:** + +```typescript +// kilocode_change start +const foo = 1 +const bar = 2 +// kilocode_change end +``` + +**New files:** + +```typescript +// kilocode_change - new file +``` + + +**JSX/TSX (inside JSX templates):** + + +```tsx +{/* kilocode_change */} +``` + + +```tsx +{/* kilocode_change start */} + +{/* kilocode_change end */} +``` + +#### When markers are NOT needed + +Code in these paths is Kilo Code-specific and does NOT need `kilocode_change` markers: + +- `packages/opencode/src/kilocode/` - All files in this directory +- `packages/opencode/test/kilocode/` - All test files for kilocode +- Any other path containing `kilocode` in filename or directory name + +These paths are entirely Kilo Code additions and won't conflict with upstream. +======= +For decision rules on when to keep changes inline vs. extract Kilo logic, marker placement guidance, and verification commands, load `.kilo/skills/kilocode-merge-minimizer/SKILL.md`. +>>>>>>> yunqiqiliang/opencode-v7.3.0 diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index 55c67bf738..473d18714d 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -61,6 +61,121 @@ czcode serve CZCODE_API_URL=http://localhost:3000 ~/.bun/bin/bun dev ``` +<<<<<<< HEAD +||||||| 12f7967ca4 +This redirects all gateway traffic (auth, model listing, provider routing, profile, etc.) to your local server. The default is `https://api.kilo.ai`. + +There are also optional overrides for other services: + +| Variable | Default | Purpose | +|---|---|---| +| `KILO_API_URL` | `https://api.kilo.ai` | Kilo API (gateway, auth, models, profile) | +| `KILO_SESSION_INGEST_URL` | `https://ingest.kilosessions.ai` | Session export / cloud sync | +| `KILO_MODELS_URL` | `https://models.dev` | Model metadata | + +> **VS Code:** The repo includes a "VSCode - Run Extension (Local Backend)" launch config in `.vscode/launch.json` that sets `KILO_API_URL=http://localhost:3000` automatically. + +## Issue Template Requirements + +If you open an issue through the GitHub web UI, GitHub will guide you through the correct template automatically. + +If you open an issue through `gh issue create`, the API, or another tool that bypasses the web UI, include the equivalent required fields yourself so the issue still matches the template. + +Current required fields by issue type: + +- **Bug report:** include a `Description`. +- **Feature request:** prefix the title with `[FEATURE]:`, include confirmation that the feature has not already been suggested, and add a description of the enhancement. +- **Question:** include the `Question`. + +Recommended fields for bug reports, even when not strictly required by the template: + +- Plugins +- Kilo version +- Steps to reproduce +- Screenshot and/or share link +- Operating System +- Terminal + +## Pull Request Expectations + +- **Issue First Policy:** All PRs must reference an existing issue. +- **UI Changes:** Include screenshots or videos (before/after). +- **Logic Changes:** Explain how you verified it works. +- **PR Titles:** Follow conventional commit standards (`feat:`, `fix:`, `docs:`, etc.). + +## Issue First Policy + +All pull requests must reference an existing issue. + +This helps reviewers understand the problem statement, discussion, and intended scope before reviewing the code change. + +## PR Titles + +Use conventional commit style PR titles such as: + +- `feat: add MCP settings tab` +- `fix: correct Windows path handling` +- `docs: clarify issue template requirements` +- `chore: bump TypeScript to 5.8` +- `refactor: extract diff renderer into a hook` +- `test: cover ServerManager orphan cleanup` + +## Issue and PR Lifecycle + +To keep our backlog manageable, we automatically close inactive issues and PRs after a period of inactivity. This isn't a judgment on quality — older items tend to lose context over time and we'd rather start fresh if they're still relevant. Feel free to reopen or create a new issue/PR if you're still working on something! + +======= +This redirects all gateway traffic (auth, model listing, provider routing, profile, etc.) to your local server. The default is `https://api.kilo.ai`. + +There are also optional overrides for other services: + +| Variable | Default | Purpose | +|---|---|---| +| `KILO_API_URL` | `https://api.kilo.ai` | Kilo API (gateway, auth, models, profile) | +| `KILO_SESSION_INGEST_URL` | `https://ingest.kilosessions.ai` | Session export / cloud sync | +| `KILO_MODELS_URL` | `https://models.dev` | Model metadata | + +> **VS Code:** The repo includes a "VSCode - Run Extension (Local Backend)" launch config in `.vscode/launch.json` that sets `KILO_API_URL=http://localhost:3000` automatically. + +## Issue Template Requirements + +If you open an issue through the GitHub web UI, GitHub will guide you through the correct template automatically. + +If you open an issue through `gh issue create`, the API, or another tool that bypasses the web UI, include the equivalent required fields yourself so the issue still matches the template. Issues that skip required fields may be auto-closed by the compliance bot. + +Current required fields by issue type: + +- **Bug report:** include a `Description`. When you can, also add Plugins, Kilo version, Steps to reproduce, Screenshot and/or share link, Operating System, and Terminal so the report matches the full bug template. +- **Feature request:** use a title prefixed with `[FEATURE]:`, complete the required checkbox confirming you have searched for duplicates, and fill in `Describe the enhancement you want to request`. +- **Question:** include the `Question` field. + +## Pull Request Expectations + +- **UI Changes:** Include screenshots or videos (before/after). +- **Logic Changes:** Explain how you verified it works. + +## Issue First Policy + +All pull requests must reference an existing issue. + +This helps reviewers understand the problem statement, discussion, and intended scope before reviewing the code change. + +## PR Titles + +Use conventional commit style PR titles such as: + +- `feat: add MCP settings tab` +- `fix: correct Windows path handling` +- `docs: clarify issue template requirements` +- `chore: bump TypeScript to 5.8` +- `refactor: extract diff renderer into a hook` +- `test: cover ServerManager orphan cleanup` + +## Issue and PR Lifecycle + +To keep our backlog manageable, we automatically close inactive issues and PRs after a period of inactivity. This isn't a judgment on quality — older items tend to lose context over time and we'd rather start fresh if they're still relevant. Feel free to reopen or create a new issue/PR if you're still working on something! + +>>>>>>> yunqiqiliang/opencode-v7.3.0 ## Style Preferences - **Functions:** Keep logic within a single function unless breaking it out adds clear reuse. diff --git a/README.md b/README.md index 4da85f4068..5e34546cf5 100644 --- a/README.md +++ b/README.md @@ -28,7 +28,17 @@ czcode 覆盖三类场景,通过角色切换在同一个工具里完成: - **Tab 键** — 在输入框按 Tab 循环切换角色 - **`czcode.jsonc`** 中设置 `default_agent` 固定默认角色 +<<<<<<< HEAD --- +||||||| 12f7967ca4 +1. Install the Kilo Code extension from the [VS Code Marketplace](https://marketplace.visualstudio.com/items?itemName=kilocode.Kilo-Code). +2. Create your account to access 500+ cutting-edge AI models including Gemini 3.1 Pro, Claude 4.6 Sonnet & Opus, and GPT-5.4 – with transparent pricing that matches provider rates exactly. +3. Start coding with AI that adapts to your workflow. Watch our quick-start guide to see Kilo in action: +======= +1. Install the Kilo Code extension from the [VS Code Marketplace](https://marketplace.visualstudio.com/items?itemName=kilocode.Kilo-Code). +2. Create your account to access 500+ cutting-edge AI models including GPT-5.5, Claude Opus 4.7, Claude Sonnet 4.6, and Gemini 3.1 Pro Preview, with transparent pricing that matches provider rates exactly. +3. Start coding with AI that adapts to your workflow. Watch our quick-start guide to see Kilo in action: +>>>>>>> yunqiqiliang/opencode-v7.3.0 ### 场景二:Lakehouse 应用开发(Code/Plan 角色 + Lakehouse Skills) @@ -122,8 +132,18 @@ cz-cli setup ### 第四步:配置 AI 模型 +<<<<<<< HEAD czcode 默认使用阿里云 DashScope Qwen 模型。配置 API Key(二选一): +||||||| 12f7967ca4 +### Where did Kilo CLI come from? +======= +## FAQ +
+Where did Kilo CLI come from? +>>>>>>> yunqiqiliang/opencode-v7.3.0 + +<<<<<<< HEAD **方式 A:环境变量(推荐)** ```bash @@ -465,3 +485,10 @@ czcode 与 [cz-cli](https://github.com/clickzetta/cz-cli) 共享连接配置, MIT License czcode 是 [KiloCode](https://github.com/Kilo-Org/kilocode) 的 fork,KiloCode 是 [OpenCode](https://github.com/anomalyco/opencode) 的 fork。感谢两个上游项目的开源贡献。 +||||||| 12f7967ca4 +Kilo CLI is a fork of [OpenCode](https://github.com/anomalyco/opencode), enhanced to work within the Kilo agentic engineering platform. +======= +Kilo CLI is a fork of [OpenCode](https://github.com/anomalyco/opencode), enhanced to work within the Kilo agentic engineering platform. + +
+>>>>>>> yunqiqiliang/opencode-v7.3.0 diff --git a/TESTING.md b/TESTING.md index 589af5420c..68f1489bf4 100644 --- a/TESTING.md +++ b/TESTING.md @@ -1,6 +1,12 @@ # TESTING.md +<<<<<<< HEAD How to spin up the **local main-branch** czcode backend and test it with `curl` / `fetch`. Aimed at a running czcode agent iterating on backend fixes without rebuilding the TUI. +||||||| 12f7967ca4 +How to spin up the **local main-branch** Kilo backend and test it with `curl` / `fetch`. Aimed at a running Kilo CLI agent iterating on backend fixes without rebuilding the VS Code extension, TUI, or desktop app. +======= +How to spin up the **local main-branch** Kilo backend and test it with `curl` / `fetch`. Aimed at a running Kilo CLI agent iterating on backend fixes without rebuilding the VS Code extension or TUI. +>>>>>>> yunqiqiliang/opencode-v7.3.0 All examples use plain shell + `curl`. Writing TypeScript files is a last resort (see Section 8). diff --git a/bun.lock b/bun.lock index 425b07a543..3803fba2ff 100644 --- a/bun.lock +++ b/bun.lock @@ -97,7 +97,7 @@ }, "packages/core": { "name": "@opencode-ai/core", - "version": "7.2.34", + "version": "7.2.49", "bin": { "opencode": "./bin/opencode", }, @@ -230,13 +230,14 @@ }, "packages/kilo-docs": { "name": "@kilocode/kilo-docs", - "version": "7.2.34", + "version": "7.2.49", "dependencies": { "@docsearch/css": "^4", "@docsearch/js": "^4", "@markdoc/markdoc": "^0.5.4", "@markdoc/next.js": "^0.5.0", "@vscode/codicons": "^0.0.44", + "@xyflow/react": "12.10.2", "js-yaml": "^4.1.0", "mermaid": "11.12.3", "next": "^16.1.5", @@ -259,7 +260,7 @@ }, "packages/kilo-gateway": { "name": "@kilocode/kilo-gateway", - "version": "7.2.34", + "version": "7.2.49", "dependencies": { "@ai-sdk/alibaba": "1.0.17", "@ai-sdk/anthropic": "3.0.71", @@ -295,7 +296,7 @@ }, "packages/kilo-i18n": { "name": "@kilocode/kilo-i18n", - "version": "7.2.34", + "version": "7.2.49", "devDependencies": { "@tsconfig/node22": "catalog:", "@types/bun": "catalog:", @@ -305,7 +306,7 @@ }, "packages/kilo-indexing": { "name": "@kilocode/kilo-indexing", - "version": "7.2.34", + "version": "7.2.49", "dependencies": { "@aws-sdk/client-bedrock-runtime": "3.1005.0", "@aws-sdk/credential-provider-ini": "3.972.31", @@ -336,11 +337,11 @@ }, "packages/kilo-jetbrains": { "name": "@kilocode/kilo-jetbrains", - "version": "7.2.34", + "version": "7.2.49", }, "packages/kilo-telemetry": { "name": "@kilocode/kilo-telemetry", - "version": "7.2.34", + "version": "7.2.49", "dependencies": { "@kilocode/kilo-gateway": "workspace:*", "@opentelemetry/api": "1.9.0", @@ -360,8 +361,9 @@ }, "packages/kilo-ui": { "name": "@kilocode/kilo-ui", - "version": "7.2.34", + "version": "7.2.49", "dependencies": { + "@kilocode/sdk": "workspace:*", "@kobalte/core": "0.13.11", "@opencode-ai/core": "workspace:*", "@opencode-ai/shared": "workspace:*", @@ -370,6 +372,7 @@ "@solid-primitives/media": "2.3.3", "@solid-primitives/resize-observer": "2.1.5", "@solid-primitives/rootless": "1.5.2", + "diff": "catalog:", "lucide-solid": "0.576.0", "motion": "12.34.5", "motion-dom": "12.34.3", @@ -396,9 +399,10 @@ }, "packages/kilo-vscode": { "name": "kilo-code", - "version": "7.2.34", + "version": "7.2.49", "dependencies": { "@anthropic-ai/sdk": "^0.39.0", + "@kilocode/kilo-gateway": "workspace:*", "@kilocode/kilo-i18n": "workspace:*", "@kilocode/kilo-indexing": "workspace:*", "@kilocode/kilo-ui": "workspace:*", @@ -414,12 +418,14 @@ "dotenv": "^16.4.7", "fastest-levenshtein": "^1.0.16", "friendly-words": "1.3.1", + "fuzzysort": "3.1.0", "ignore": "^7.0.3", "js-tiktoken": "^1.0.18", "lru-cache": "^11.0.2", + "marked": "catalog:", "openai": "^4.85.4", "quick-lru": "^7.0.0", - "simple-git": "3.35.2", + "simple-git": "3.36.0", "solid-js": "^1.9.11", "stream-chat": "9.38.0", "uri-js": "^4.4.1", @@ -457,7 +463,7 @@ }, "packages/opencode": { "name": "@kilocode/cli", - "version": "7.2.34", + "version": "7.2.49", "bin": { "czcode": "./bin/kilo", "kilo": "./bin/kilo", @@ -530,6 +536,7 @@ "ai-gateway-provider": "3.1.2", "bonjour-service": "1.3.0", "bun-pty": "0.4.8", + "chardet": "2.1.1", "chokidar": "4.0.3", "cli-sound": "1.1.3", "clipboardy": "4.0.0", @@ -562,7 +569,7 @@ "ripgrep": "0.3.1", "rotating-file-stream": "3.2.9", "semver": "^7.6.3", - "simple-git": "3.35.2", + "simple-git": "3.36.0", "solid-js": "catalog:", "stream-chat": "9.38.0", "strip-ansi": "7.1.2", @@ -619,7 +626,7 @@ }, "packages/plugin": { "name": "@kilocode/plugin", - "version": "7.2.34", + "version": "7.2.49", "dependencies": { "@kilocode/sdk": "workspace:*", "effect": "catalog:", @@ -634,8 +641,8 @@ "typescript": "catalog:", }, "peerDependencies": { - "@opentui/core": ">=0.1.105", - "@opentui/solid": ">=0.1.105", + "@opentui/core": ">=0.2.2", + "@opentui/solid": ">=0.2.2", }, "optionalPeers": [ "@opentui/core", @@ -644,7 +651,7 @@ }, "packages/script": { "name": "@opencode-ai/script", - "version": "7.2.34", + "version": "7.2.49", "dependencies": { "semver": "^7.6.3", }, @@ -655,7 +662,7 @@ }, "packages/sdk/js": { "name": "@kilocode/sdk", - "version": "7.2.34", + "version": "7.2.49", "dependencies": { "cross-spawn": "catalog:", }, @@ -694,7 +701,7 @@ }, "packages/storybook": { "name": "@opencode-ai/storybook", - "version": "7.2.34", + "version": "7.2.49", "devDependencies": { "@opencode-ai/ui": "workspace:*", "@solidjs/meta": "catalog:", @@ -717,7 +724,7 @@ }, "packages/ui": { "name": "@opencode-ai/ui", - "version": "7.2.34", + "version": "7.2.49", "dependencies": { "@kilocode/sdk": "workspace:*", "@kobalte/core": "catalog:", @@ -733,7 +740,7 @@ "@solidjs/router": "catalog:", "@typescript/native-preview": "catalog:", "diff": "catalog:", - "dompurify": "3.3.3", + "dompurify": "3.4.2", "fuzzysort": "catalog:", "katex": "0.16.27", "luxon": "catalog:", @@ -768,35 +775,14 @@ }, }, }, - "trustedDependencies": [ - "esbuild", - "tree-sitter-powershell", - "protobufjs", - "electron", - "web-tree-sitter", - "tree-sitter-bash", - ], "patchedDependencies": { "@standard-community/standard-openapi@0.2.9": "patches/@standard-community%2Fstandard-openapi@0.2.9.patch", "@npmcli/agent@4.0.0": "patches/@npmcli%2Fagent@4.0.0.patch", "stream-chat@9.38.0": "patches/stream-chat@9.38.0.patch", }, "overrides": { - "@effect/platform-node-shared": "4.0.0-beta.46", "@types/bun": "catalog:", - "@types/node": "catalog:", - "@xmldom/xmldom": ">=0.8.12", - "defu": "6.1.6", - "diff": "8.0.4", - "dompurify": "3.3.3", - "effect": "catalog:", - "fastify": ">=5.8.3", - "happy-dom": ">=20.8.9", - "lodash": "4.18.1", - "path-to-regexp": ">=8.4.0", - "picomatch": ">=2.3.2", "poe-oauth": "0.0.6", - "smol-toml": ">=1.6.1", }, "catalog": { "@cloudflare/workers-types": "4.20251008.0", @@ -808,8 +794,8 @@ "@npmcli/arborist": "9.4.0", "@octokit/rest": "22.0.0", "@openauthjs/openauth": "0.0.0-20250322224806", - "@opentui/core": "0.1.105", - "@opentui/solid": "0.1.105", + "@opentui/core": "0.2.2", + "@opentui/solid": "0.2.2", "@pierre/diffs": "1.1.0-beta.18", "@playwright/test": "1.59.1", "@solid-primitives/storage": "4.3.3", @@ -828,7 +814,7 @@ "ai": "6.0.168", "cross-spawn": "7.0.6", "diff": "8.0.4", - "dompurify": "3.3.3", + "dompurify": "3.4.2", "drizzle-kit": "1.0.0-beta.19-d95b7a4", "drizzle-orm": "1.0.0-beta.19-d95b7a4", "effect": "4.0.0-beta.57", @@ -884,15 +870,15 @@ "@ai-sdk/cohere": ["@ai-sdk/cohere@3.0.27", "", { "dependencies": { "@ai-sdk/provider": "3.0.8", "@ai-sdk/provider-utils": "4.0.21" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-OqcCq2PiFY1dbK/0Ck45KuvE8jfdxRuuAE9Y5w46dAk6U+9vPOeg1CDcmR+ncqmrYrhRl3nmyDttyDahyjCzAw=="], - "@ai-sdk/deepgram": ["@ai-sdk/deepgram@2.0.32", "", { "dependencies": { "@ai-sdk/provider": "3.0.10", "@ai-sdk/provider-utils": "4.0.26" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-v8Jryq/Wie4202/GR5UR+4DYysbTnLmgQ3k1TC2qC1hcedOfTsb1GFE0ZU4N0+402XAxOhB+pJOYrNI+H7p9+Q=="], + "@ai-sdk/deepgram": ["@ai-sdk/deepgram@2.0.33", "", { "dependencies": { "@ai-sdk/provider": "3.0.10", "@ai-sdk/provider-utils": "4.0.27" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-VscTV68g6sXRY4O1yl72/O8y6+tBDvSQax6bqX06hRKWBGxsJ8Jr3LZsNmZnK9Od5Icx565ijK0QgrlNaN4TdQ=="], "@ai-sdk/deepinfra": ["@ai-sdk/deepinfra@2.0.41", "", { "dependencies": { "@ai-sdk/openai-compatible": "2.0.37", "@ai-sdk/provider": "3.0.8", "@ai-sdk/provider-utils": "4.0.21" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-y6RoOP7DGWmDSiSxrUSt5p18sbz+Ixe5lMVPmdE7x+Tr5rlrzvftyHhjWHfqlAtoYERZTGFbP6tPW1OfQcrb4A=="], - "@ai-sdk/deepseek": ["@ai-sdk/deepseek@2.0.32", "", { "dependencies": { "@ai-sdk/provider": "3.0.10", "@ai-sdk/provider-utils": "4.0.26" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-6CO8QO0GfHD6NEHSBvm+POMpS/GdQyDKezt43Bl8sG6WLIGNN6W/LA4f36+mgHsRJ1WPE8hyWigF+fC1sU+fHg=="], + "@ai-sdk/deepseek": ["@ai-sdk/deepseek@2.0.35", "", { "dependencies": { "@ai-sdk/provider": "3.0.10", "@ai-sdk/provider-utils": "4.0.27" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-9DhYurbAvcurOEGN6u2myYDybrrzGfcrkG8hwmFjwTrePW6KCMggm0YxP7e8RkLYcQKqCEMgFlyEB4BM6EmiKg=="], - "@ai-sdk/elevenlabs": ["@ai-sdk/elevenlabs@2.0.32", "", { "dependencies": { "@ai-sdk/provider": "3.0.10", "@ai-sdk/provider-utils": "4.0.26" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-HvKYMniSjya+REjHF4Vp3cz5OA/hGw4RaUl1cAd/By3N/H3Ct6n9cYGHlX4rmo0TS5rh8t2iwQOEOYXJ9xA0GA=="], + "@ai-sdk/elevenlabs": ["@ai-sdk/elevenlabs@2.0.33", "", { "dependencies": { "@ai-sdk/provider": "3.0.10", "@ai-sdk/provider-utils": "4.0.27" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-EtvsWfGrqx3OhzJdoi82qH+4yzEPPKZr2utyQ+w8cHKoFeg0+8Lou9Z3uixy73WEwz8Z1+AR8QT9fZ64AWGYPA=="], - "@ai-sdk/fireworks": ["@ai-sdk/fireworks@2.0.51", "", { "dependencies": { "@ai-sdk/openai-compatible": "2.0.46", "@ai-sdk/provider": "3.0.10", "@ai-sdk/provider-utils": "4.0.26" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-NmUW7ENxWtBhcTnOZafvQuBOE/rVCTPiI1bdbuXcONSLu+jDy7DuA99ttFb2AC0pGeFhmLftYU2BsFHeDNBgWQ=="], + "@ai-sdk/fireworks": ["@ai-sdk/fireworks@2.0.52", "", { "dependencies": { "@ai-sdk/openai-compatible": "2.0.47", "@ai-sdk/provider": "3.0.10", "@ai-sdk/provider-utils": "4.0.27" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-TwNi+Eqa6Jdoz/IjIR3RvgW/BaHP6H8X2BhiDD8b9d/PvVC5ywIAcC1LLDYj/7guC/4fRjkIyiJMWFpL4Qiyhw=="], "@ai-sdk/gateway": ["@ai-sdk/gateway@3.0.104", "", { "dependencies": { "@ai-sdk/provider": "3.0.8", "@ai-sdk/provider-utils": "4.0.23", "@vercel/oidc": "3.2.0" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-ZKX5n74io8VIRlhIMSLWVlvT3sXC8Z7cZ9GHuWBWZDVi96+62AIsWuLGvMfcBA1STYuSoDrp6rIziZmvrTq0TA=="], @@ -950,79 +936,75 @@ "@aws-sdk/client-s3": ["@aws-sdk/client-s3@3.1025.0", "", { "dependencies": { "@aws-crypto/sha1-browser": "5.2.0", "@aws-crypto/sha256-browser": "5.2.0", "@aws-crypto/sha256-js": "5.2.0", "@aws-sdk/core": "^3.973.26", "@aws-sdk/credential-provider-node": "^3.972.29", "@aws-sdk/middleware-bucket-endpoint": "^3.972.8", "@aws-sdk/middleware-expect-continue": "^3.972.8", "@aws-sdk/middleware-flexible-checksums": "^3.974.6", "@aws-sdk/middleware-host-header": "^3.972.8", "@aws-sdk/middleware-location-constraint": "^3.972.8", "@aws-sdk/middleware-logger": "^3.972.8", "@aws-sdk/middleware-recursion-detection": "^3.972.9", "@aws-sdk/middleware-sdk-s3": "^3.972.27", "@aws-sdk/middleware-ssec": "^3.972.8", "@aws-sdk/middleware-user-agent": "^3.972.28", "@aws-sdk/region-config-resolver": "^3.972.10", "@aws-sdk/signature-v4-multi-region": "^3.996.15", "@aws-sdk/types": "^3.973.6", "@aws-sdk/util-endpoints": "^3.996.5", "@aws-sdk/util-user-agent-browser": "^3.972.8", "@aws-sdk/util-user-agent-node": "^3.973.14", "@smithy/config-resolver": "^4.4.13", "@smithy/core": "^3.23.13", "@smithy/eventstream-serde-browser": "^4.2.12", "@smithy/eventstream-serde-config-resolver": "^4.3.12", "@smithy/eventstream-serde-node": "^4.2.12", "@smithy/fetch-http-handler": "^5.3.15", "@smithy/hash-blob-browser": "^4.2.13", "@smithy/hash-node": "^4.2.12", "@smithy/hash-stream-node": "^4.2.12", "@smithy/invalid-dependency": "^4.2.12", "@smithy/md5-js": "^4.2.12", "@smithy/middleware-content-length": "^4.2.12", "@smithy/middleware-endpoint": "^4.4.28", "@smithy/middleware-retry": "^4.4.46", "@smithy/middleware-serde": "^4.2.16", "@smithy/middleware-stack": "^4.2.12", "@smithy/node-config-provider": "^4.3.12", "@smithy/node-http-handler": "^4.5.1", "@smithy/protocol-http": "^5.3.12", "@smithy/smithy-client": "^4.12.8", "@smithy/types": "^4.13.1", "@smithy/url-parser": "^4.2.12", "@smithy/util-base64": "^4.3.2", "@smithy/util-body-length-browser": "^4.2.2", "@smithy/util-body-length-node": "^4.2.3", "@smithy/util-defaults-mode-browser": "^4.3.44", "@smithy/util-defaults-mode-node": "^4.2.48", "@smithy/util-endpoints": "^3.3.3", "@smithy/util-middleware": "^4.2.12", "@smithy/util-retry": "^4.2.13", "@smithy/util-stream": "^4.5.21", "@smithy/util-utf8": "^4.2.2", "@smithy/util-waiter": "^4.2.14", "tslib": "^2.6.2" } }, "sha512-9Byz2fPnuGRRL8DTTD5bYPl1Iwm+ysLiCMgptffa3lNkVLCiUZc5e5TAaOjk0MvyeXieq+jn35AmQL6cgN2KHQ=="], - "@aws-sdk/core": ["@aws-sdk/core@3.974.8", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@aws-sdk/xml-builder": "^3.972.22", "@smithy/core": "^3.23.17", "@smithy/node-config-provider": "^4.3.14", "@smithy/property-provider": "^4.2.14", "@smithy/protocol-http": "^5.3.14", "@smithy/signature-v4": "^5.3.14", "@smithy/smithy-client": "^4.12.13", "@smithy/types": "^4.14.1", "@smithy/util-base64": "^4.3.2", "@smithy/util-middleware": "^4.2.14", "@smithy/util-retry": "^4.3.6", "@smithy/util-utf8": "^4.2.2", "tslib": "^2.6.2" } }, "sha512-njR2qoG6ZuB0kvAS2FyICsFZJ6gmCcf2X/7JcD14sUvGDm26wiZ5BrA6LOiUxKFEF+IVe7kdroxyE00YlkiYsw=="], + "@aws-sdk/core": ["@aws-sdk/core@3.974.9", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@aws-sdk/xml-builder": "^3.972.23", "@smithy/core": "^3.24.1", "@smithy/signature-v4": "^5.4.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-bXxosFunr+v/kqNb99r1NRkrVBha7CG036fRSpWGbC1A/e363XFQN6wcZMx7MYTdRr1tYwNnkrWX2xc1rT3BCQ=="], - "@aws-sdk/crc64-nvme": ["@aws-sdk/crc64-nvme@3.972.7", "", { "dependencies": { "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-QUagVVBbC8gODCF6e1aV0mE2TXWB9Opz4k8EJFdNrujUVQm5R4AjJa1mpOqzwOuROBzqJU9zawzig7M96L8Ejg=="], + "@aws-sdk/crc64-nvme": ["@aws-sdk/crc64-nvme@3.972.8", "", { "dependencies": { "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-fVfUCL/Xh2zINYMPZvj+iBn6XWouQf0DAnjaWCI9MkmqXzL2Iy5FoQB8O7syFe6gN6AH1ecDDU58T51Ou0kFkA=="], - "@aws-sdk/credential-provider-cognito-identity": ["@aws-sdk/credential-provider-cognito-identity@3.972.31", "", { "dependencies": { "@aws-sdk/nested-clients": "^3.997.6", "@aws-sdk/types": "^3.973.8", "@smithy/property-provider": "^4.2.14", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-W5JtzDp3ejzhOOknXlnt+vJsNN2GZdAcBK+hR7HQ1DCacXqS0UpmnIyihIU7CK0IB+XYWeBaN3bBv4pXavp7Vg=="], + "@aws-sdk/credential-provider-cognito-identity": ["@aws-sdk/credential-provider-cognito-identity@3.972.32", "", { "dependencies": { "@aws-sdk/nested-clients": "^3.997.7", "@aws-sdk/types": "^3.973.8", "@smithy/core": "^3.24.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-u0NdqmO0h1nPoElB66FDuZKGDibn3VTE18DzFw4nH6LHebeIw9r+n2o9ap+fLZA/6H/tmdqw5fpgwYt/bb6eYQ=="], - "@aws-sdk/credential-provider-env": ["@aws-sdk/credential-provider-env@3.972.34", "", { "dependencies": { "@aws-sdk/core": "^3.974.8", "@aws-sdk/types": "^3.973.8", "@smithy/property-provider": "^4.2.14", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-XT0jtf8Fw9JE6ppsQeoNnZRiG+jqRixMT1v1ZR17G60UvVdsQmTG8nbEyHuEPfMxDXEhfdARaM/XiEhca4lGHQ=="], + "@aws-sdk/credential-provider-env": ["@aws-sdk/credential-provider-env@3.972.35", "", { "dependencies": { "@aws-sdk/core": "^3.974.9", "@aws-sdk/types": "^3.973.8", "@smithy/core": "^3.24.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-WkFQ8BedszVomhh/Zzs8WwnE/XBmTqZjoQVB8u/4zH6kZCjouXZpPpb93gD8m0EZmzAl7dxHE/y+yDpuKzNCMw=="], - "@aws-sdk/credential-provider-http": ["@aws-sdk/credential-provider-http@3.972.36", "", { "dependencies": { "@aws-sdk/core": "^3.974.8", "@aws-sdk/types": "^3.973.8", "@smithy/fetch-http-handler": "^5.3.17", "@smithy/node-http-handler": "^4.6.1", "@smithy/property-provider": "^4.2.14", "@smithy/protocol-http": "^5.3.14", "@smithy/smithy-client": "^4.12.13", "@smithy/types": "^4.14.1", "@smithy/util-stream": "^4.5.25", "tslib": "^2.6.2" } }, "sha512-DPoGWfy7J7RKxvbf5kOKIGQkD2ek3dbKgzKIGrnLuvZBz5myU+Im/H6pmc14QcnFbqHMqxvtWSgRDSJW3qXLQg=="], + "@aws-sdk/credential-provider-http": ["@aws-sdk/credential-provider-http@3.972.37", "", { "dependencies": { "@aws-sdk/core": "^3.974.9", "@aws-sdk/types": "^3.973.8", "@smithy/core": "^3.24.1", "@smithy/fetch-http-handler": "^5.4.1", "@smithy/node-http-handler": "^4.7.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-ylx0ZJTU+2eNcvXQ69VNR3TVSYa/ibpvdK717/NxqR9aXRMn2QRWZaiI8aa5yY/fOWZ5mknSmxGaVxxtdwv3EA=="], "@aws-sdk/credential-provider-ini": ["@aws-sdk/credential-provider-ini@3.972.31", "", { "dependencies": { "@aws-sdk/core": "^3.974.1", "@aws-sdk/credential-provider-env": "^3.972.27", "@aws-sdk/credential-provider-http": "^3.972.29", "@aws-sdk/credential-provider-login": "^3.972.31", "@aws-sdk/credential-provider-process": "^3.972.27", "@aws-sdk/credential-provider-sso": "^3.972.31", "@aws-sdk/credential-provider-web-identity": "^3.972.31", "@aws-sdk/nested-clients": "^3.996.21", "@aws-sdk/types": "^3.973.8", "@smithy/credential-provider-imds": "^4.2.14", "@smithy/property-provider": "^4.2.14", "@smithy/shared-ini-file-loader": "^4.4.9", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-PuQ7e8WYzAPpzvFcajxf8c0LqSzakVHVlKw8M0oubk8Kf347YOCCqT1seQrHs5AdZuIh2RD9LX4O+Xa5ImEBfQ=="], - "@aws-sdk/credential-provider-login": ["@aws-sdk/credential-provider-login@3.972.38", "", { "dependencies": { "@aws-sdk/core": "^3.974.8", "@aws-sdk/nested-clients": "^3.997.6", "@aws-sdk/types": "^3.973.8", "@smithy/property-provider": "^4.2.14", "@smithy/protocol-http": "^5.3.14", "@smithy/shared-ini-file-loader": "^4.4.9", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-g1NosS8qe4OF++G2UFCM5ovSkgipC7YYor5KCWatG0UoMSO5YFj9C8muePlyVmOBV/WTI16Jo3/s1NUo/o1Bww=="], + "@aws-sdk/credential-provider-login": ["@aws-sdk/credential-provider-login@3.972.39", "", { "dependencies": { "@aws-sdk/core": "^3.974.9", "@aws-sdk/nested-clients": "^3.997.7", "@aws-sdk/types": "^3.973.8", "@smithy/core": "^3.24.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-1hU0NtC04QbFIuoBuF4aQ2A97GsSE5/A0ZJpDijwexsBREIQ4KPRYl3v/FfKCPBYsaTeGjkOFx5nLhWHY24LOw=="], - "@aws-sdk/credential-provider-node": ["@aws-sdk/credential-provider-node@3.972.39", "", { "dependencies": { "@aws-sdk/credential-provider-env": "^3.972.34", "@aws-sdk/credential-provider-http": "^3.972.36", "@aws-sdk/credential-provider-ini": "^3.972.38", "@aws-sdk/credential-provider-process": "^3.972.34", "@aws-sdk/credential-provider-sso": "^3.972.38", "@aws-sdk/credential-provider-web-identity": "^3.972.38", "@aws-sdk/types": "^3.973.8", "@smithy/credential-provider-imds": "^4.2.14", "@smithy/property-provider": "^4.2.14", "@smithy/shared-ini-file-loader": "^4.4.9", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-HEswDQyxUtadoZ/bJsPPENHg7R0Lzym5LuMksJeHvqhCOpP+rtkDLKI4/ZChH4w3cf5kG8n6bZuI8PzajoiqMg=="], + "@aws-sdk/credential-provider-node": ["@aws-sdk/credential-provider-node@3.972.40", "", { "dependencies": { "@aws-sdk/credential-provider-env": "^3.972.35", "@aws-sdk/credential-provider-http": "^3.972.37", "@aws-sdk/credential-provider-ini": "^3.972.39", "@aws-sdk/credential-provider-process": "^3.972.35", "@aws-sdk/credential-provider-sso": "^3.972.39", "@aws-sdk/credential-provider-web-identity": "^3.972.39", "@aws-sdk/types": "^3.973.8", "@smithy/core": "^3.24.1", "@smithy/credential-provider-imds": "^4.3.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-ZgrQaGkpyTlVSCCsffzijVg+KgftTAWYvI5Otc36J/4jNiHb+7MmBiJIR0a5AHLvifC92PiYHt5pijP0dswd1w=="], - "@aws-sdk/credential-provider-process": ["@aws-sdk/credential-provider-process@3.972.34", "", { "dependencies": { "@aws-sdk/core": "^3.974.8", "@aws-sdk/types": "^3.973.8", "@smithy/property-provider": "^4.2.14", "@smithy/shared-ini-file-loader": "^4.4.9", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-T3IFs4EVmVi1dVN5RciFnklCANSzvrQd/VuHY9ThHSQmYkTogjcGkoJEr+oNUPQZnso52183088NqysMPji1/Q=="], + "@aws-sdk/credential-provider-process": ["@aws-sdk/credential-provider-process@3.972.35", "", { "dependencies": { "@aws-sdk/core": "^3.974.9", "@aws-sdk/types": "^3.973.8", "@smithy/core": "^3.24.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-hNj1rAwZWT1vfz54BwH8FUWxZuqStrM25Q5LEIwn2erHPMRVAjLlpZqEbCEEqS99eEEOhdeetnS0WeNa3iYeEQ=="], - "@aws-sdk/credential-provider-sso": ["@aws-sdk/credential-provider-sso@3.972.38", "", { "dependencies": { "@aws-sdk/core": "^3.974.8", "@aws-sdk/nested-clients": "^3.997.6", "@aws-sdk/token-providers": "3.1041.0", "@aws-sdk/types": "^3.973.8", "@smithy/property-provider": "^4.2.14", "@smithy/shared-ini-file-loader": "^4.4.9", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-5ZxG+t0+3Q3QPh8KEjX6syskhgNf7I0MN7oGioTf6Lm1NTjfP7sIcYGNsthXC2qR8vcD3edNZwCr2ovfSSWuRA=="], + "@aws-sdk/credential-provider-sso": ["@aws-sdk/credential-provider-sso@3.972.39", "", { "dependencies": { "@aws-sdk/core": "^3.974.9", "@aws-sdk/nested-clients": "^3.997.7", "@aws-sdk/token-providers": "3.1046.0", "@aws-sdk/types": "^3.973.8", "@smithy/core": "^3.24.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-mwIPNPldyCZkvHozb6E0X/vuQLN1UCjcA6MwUf1gaO7EwghCmuNZXatq0L3zptKFvPC4Nds7+WFUkifI1XmbSw=="], - "@aws-sdk/credential-provider-web-identity": ["@aws-sdk/credential-provider-web-identity@3.972.38", "", { "dependencies": { "@aws-sdk/core": "^3.974.8", "@aws-sdk/nested-clients": "^3.997.6", "@aws-sdk/types": "^3.973.8", "@smithy/property-provider": "^4.2.14", "@smithy/shared-ini-file-loader": "^4.4.9", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-lYHFF30DGI20jZcYX8cm6Ns0V7f1dDN6g/MBDLTyD/5iw+bXs3yBr2iAiHDkx4RFU5JgsnZvCHYKiRVPRdmOgw=="], + "@aws-sdk/credential-provider-web-identity": ["@aws-sdk/credential-provider-web-identity@3.972.39", "", { "dependencies": { "@aws-sdk/core": "^3.974.9", "@aws-sdk/nested-clients": "^3.997.7", "@aws-sdk/types": "^3.973.8", "@smithy/core": "^3.24.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-b9HT8CnpyPVn1hU14Q7ihjwSPlRzToYmRYJxRd5jNHEZ43lrIhoLaTT8JmfQQt5j5M8rTX1iN1X8mvu0SM1dXA=="], "@aws-sdk/credential-providers": ["@aws-sdk/credential-providers@3.1025.0", "", { "dependencies": { "@aws-sdk/client-cognito-identity": "3.1025.0", "@aws-sdk/core": "^3.973.26", "@aws-sdk/credential-provider-cognito-identity": "^3.972.21", "@aws-sdk/credential-provider-env": "^3.972.24", "@aws-sdk/credential-provider-http": "^3.972.26", "@aws-sdk/credential-provider-ini": "^3.972.28", "@aws-sdk/credential-provider-login": "^3.972.28", "@aws-sdk/credential-provider-node": "^3.972.29", "@aws-sdk/credential-provider-process": "^3.972.24", "@aws-sdk/credential-provider-sso": "^3.972.28", "@aws-sdk/credential-provider-web-identity": "^3.972.28", "@aws-sdk/nested-clients": "^3.996.18", "@aws-sdk/types": "^3.973.6", "@smithy/config-resolver": "^4.4.13", "@smithy/core": "^3.23.13", "@smithy/credential-provider-imds": "^4.2.12", "@smithy/node-config-provider": "^4.3.12", "@smithy/property-provider": "^4.2.12", "@smithy/types": "^4.13.1", "tslib": "^2.6.2" } }, "sha512-hOMHzYetTwnpvfbLN8emaw+nnQrqlEV0I5rTrgRKTAx1anzEvls/rD1IXwOvX8Z+B9mgbK+yNFqO3wQkBghI1g=="], - "@aws-sdk/eventstream-handler-node": ["@aws-sdk/eventstream-handler-node@3.972.14", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@smithy/eventstream-codec": "^4.2.14", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-m4X56gxG76/CKfxNVbOFuYwnAZcHgS6HOH8lgp15HoGHIAVTcZfZrXvcYzJFOMLEJgVn+JHBu6EiNV+xSNXXFg=="], + "@aws-sdk/eventstream-handler-node": ["@aws-sdk/eventstream-handler-node@3.972.15", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@smithy/core": "^3.24.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-Al7z1qKPUIRILnB3Ggd1Tz88wjSkQjDErajR7YY33mquTbeN0gTDtJtclNtyhQMWr7ZRpQk0v5/xop4fjT0sug=="], - "@aws-sdk/middleware-bucket-endpoint": ["@aws-sdk/middleware-bucket-endpoint@3.972.10", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@aws-sdk/util-arn-parser": "^3.972.3", "@smithy/node-config-provider": "^4.3.14", "@smithy/protocol-http": "^5.3.14", "@smithy/types": "^4.14.1", "@smithy/util-config-provider": "^4.2.2", "tslib": "^2.6.2" } }, "sha512-Vbc2frZH7wXlMNd+ZZSXUEs/l1Sv8Jj4zUnIfwrYF5lwaLdXHZ9xx4U3rjUcaye3HRhFVc+E5DbBxpRAbB16BA=="], + "@aws-sdk/middleware-bucket-endpoint": ["@aws-sdk/middleware-bucket-endpoint@3.972.11", "", { "dependencies": { "@aws-sdk/core": "^3.974.9", "@aws-sdk/types": "^3.973.8", "@smithy/core": "^3.24.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-AhVDn+qObNacklqmBABnFa3YfVk08CzksuuecL/x+lo95dZxXuAkqJZLUsAEKQ3EiDd5E9wTUBjh0cSogmKMYA=="], - "@aws-sdk/middleware-eventstream": ["@aws-sdk/middleware-eventstream@3.972.10", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@smithy/protocol-http": "^5.3.14", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-QUqLs7Af1II9X4fCRAu+EGHG3KHyOp4RkuLhRKoA3NuFlh6TL8i+zXBl8w2LUxqm44B/Kom45hgSlwA1SpTsXQ=="], + "@aws-sdk/middleware-eventstream": ["@aws-sdk/middleware-eventstream@3.972.11", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@smithy/core": "^3.24.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-A0Z45YInBwsAabF8jf9hEQjXDuq4gFHNNqxCYuk8iREFZ7hw0NZ6+7FFlYa11gJ+i6y79C4J6giQ7fa1EDRYxw=="], - "@aws-sdk/middleware-expect-continue": ["@aws-sdk/middleware-expect-continue@3.972.10", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@smithy/protocol-http": "^5.3.14", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-2Yn0f1Qiq/DjxYR3wfI3LokXnjOhFM7Ssn4LTdFDIxRMCE6I32MAsVnhPX1cUZsuVA9tiZtwwhlSLAtFGxAZlQ=="], + "@aws-sdk/middleware-expect-continue": ["@aws-sdk/middleware-expect-continue@3.972.11", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@smithy/core": "^3.24.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-xpobcctR1AHSrvkiArgTyLffn78Lt9unPMpa/yic9RKn+bOf/5M55UIM6RaPL5xKzI06/GSsTDywTWvzEAbyyw=="], - "@aws-sdk/middleware-flexible-checksums": ["@aws-sdk/middleware-flexible-checksums@3.974.16", "", { "dependencies": { "@aws-crypto/crc32": "5.2.0", "@aws-crypto/crc32c": "5.2.0", "@aws-crypto/util": "5.2.0", "@aws-sdk/core": "^3.974.8", "@aws-sdk/crc64-nvme": "^3.972.7", "@aws-sdk/types": "^3.973.8", "@smithy/is-array-buffer": "^4.2.2", "@smithy/node-config-provider": "^4.3.14", "@smithy/protocol-http": "^5.3.14", "@smithy/types": "^4.14.1", "@smithy/util-middleware": "^4.2.14", "@smithy/util-stream": "^4.5.25", "@smithy/util-utf8": "^4.2.2", "tslib": "^2.6.2" } }, "sha512-6ru8doI0/XzszqLIPXf0E/V7HhAw1Pu94010XCKYtBUfD0LxF0BuOzrUf8OQGR6j2o6wgKTHUniOmndQycHwCA=="], + "@aws-sdk/middleware-flexible-checksums": ["@aws-sdk/middleware-flexible-checksums@3.974.17", "", { "dependencies": { "@aws-crypto/crc32": "5.2.0", "@aws-crypto/crc32c": "5.2.0", "@aws-crypto/util": "5.2.0", "@aws-sdk/core": "^3.974.9", "@aws-sdk/crc64-nvme": "^3.972.8", "@aws-sdk/types": "^3.973.8", "@smithy/core": "^3.24.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-Js24a6sdH9SU5DI5++nlQJayCuOweiiTjnCcAsY75/JtaXF+xysDQ6nRBYx6pUPNY22viRYmdDTFZDaA9AF46g=="], - "@aws-sdk/middleware-host-header": ["@aws-sdk/middleware-host-header@3.972.10", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@smithy/protocol-http": "^5.3.14", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-IJSsIMeVQ8MMCPbuh1AbltkFhLBLXn7aejzfX5YKT/VLDHn++Dcz8886tXckE+wQssyPUhaXrJhdakO2VilRhg=="], + "@aws-sdk/middleware-host-header": ["@aws-sdk/middleware-host-header@3.972.11", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@smithy/core": "^3.24.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-CBC6+tVYaOJo7QXgN1zJ4Ba2f3/Cpy4eRViYFimXW/O5Mn8hBmgXXzHu4vy4ubT80YWnp8aCFygr7dTOa14yQg=="], "@aws-sdk/middleware-location-constraint": ["@aws-sdk/middleware-location-constraint@3.972.10", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-rI3NZvJcEvjoD0+0PI0iUAwlPw2IlSlhyvgBK/3WkKJQE/YiKFedd9dMN2lVacdNxPNhxL/jzQaKQdrGtQagjQ=="], "@aws-sdk/middleware-logger": ["@aws-sdk/middleware-logger@3.972.10", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-OOuGvvz1Dm20SjZo5oEBePFqxt5nf8AwkNDSyUHvD9/bfNASmstcYxFAHUowy4n6Io7mWUZ04JURZwSBvyQanQ=="], - "@aws-sdk/middleware-recursion-detection": ["@aws-sdk/middleware-recursion-detection@3.972.11", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@aws/lambda-invoke-store": "^0.2.2", "@smithy/protocol-http": "^5.3.14", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-+zz6f79Kj9V5qFK2P+D8Ehjnw4AhphAlCAsPjUqEcInA9umtSSKMrHbSagEeOIsDNuvVrH98bjRHcyQukTrhaQ=="], + "@aws-sdk/middleware-recursion-detection": ["@aws-sdk/middleware-recursion-detection@3.972.12", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@aws/lambda-invoke-store": "^0.2.2", "@smithy/core": "^3.24.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-5eltYxKB4MfdQv7/VhWxRbAVQKow5dz9votRFigTYrWJHMQXwLMltlbk7KFWSZh5NDBySfmjT7Jv/DWfYCmDng=="], - "@aws-sdk/middleware-sdk-s3": ["@aws-sdk/middleware-sdk-s3@3.972.37", "", { "dependencies": { "@aws-sdk/core": "^3.974.8", "@aws-sdk/types": "^3.973.8", "@aws-sdk/util-arn-parser": "^3.972.3", "@smithy/core": "^3.23.17", "@smithy/node-config-provider": "^4.3.14", "@smithy/protocol-http": "^5.3.14", "@smithy/signature-v4": "^5.3.14", "@smithy/smithy-client": "^4.12.13", "@smithy/types": "^4.14.1", "@smithy/util-config-provider": "^4.2.2", "@smithy/util-middleware": "^4.2.14", "@smithy/util-stream": "^4.5.25", "@smithy/util-utf8": "^4.2.2", "tslib": "^2.6.2" } }, "sha512-Km7M+i8DrLArVzrid1gfxeGhYHBd3uxvE77g0s5a52zPSVosxzQBnJ0gwWb6NIp/DOk8gsBMhi7V+cpJG0ndTA=="], + "@aws-sdk/middleware-sdk-s3": ["@aws-sdk/middleware-sdk-s3@3.972.38", "", { "dependencies": { "@aws-sdk/core": "^3.974.9", "@aws-sdk/signature-v4-multi-region": "^3.996.26", "@aws-sdk/types": "^3.973.8", "@smithy/core": "^3.24.1", "@smithy/signature-v4": "^5.4.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-Yuv3urkJtd1/b3kIURzHwihc1SV6n1t+uiXffOD2OpylZ7+4/QnO2W73yhLZzK1Z762BaqwQ3IVRqAHWzNbQ4A=="], "@aws-sdk/middleware-ssec": ["@aws-sdk/middleware-ssec@3.972.10", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-Gli9A0u8EVVb+5bFDGS/QbSVg28w/wpEidg1ggVcSj65BDTdGR6punsOcVjqdiu1i42WHWo51MCvARPIIz9juw=="], - "@aws-sdk/middleware-user-agent": ["@aws-sdk/middleware-user-agent@3.972.38", "", { "dependencies": { "@aws-sdk/core": "^3.974.8", "@aws-sdk/types": "^3.973.8", "@aws-sdk/util-endpoints": "^3.996.8", "@smithy/core": "^3.23.17", "@smithy/protocol-http": "^5.3.14", "@smithy/types": "^4.14.1", "@smithy/util-retry": "^4.3.6", "tslib": "^2.6.2" } }, "sha512-iz+B29TXcAZsJpwB+AwG/TTGA5l/VnmMZ2UxtiySOZjI6gCdmviXPwdgzcmuazMy16rXoPY4mYCGe7zdNKfx5A=="], + "@aws-sdk/middleware-user-agent": ["@aws-sdk/middleware-user-agent@3.972.39", "", { "dependencies": { "@aws-sdk/core": "^3.974.9", "@aws-sdk/types": "^3.973.8", "@aws-sdk/util-endpoints": "^3.996.9", "@smithy/core": "^3.24.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-MlNSvNsSVlMKKWaCzA0GP1nS4Cuq3WCXUN1vmMvd+Ctztib5kmRcpmTtKx9kikN8szAc+gcdp7uqJJervV2nQg=="], - "@aws-sdk/middleware-websocket": ["@aws-sdk/middleware-websocket@3.972.16", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@aws-sdk/util-format-url": "^3.972.10", "@smithy/eventstream-codec": "^4.2.14", "@smithy/eventstream-serde-browser": "^4.2.14", "@smithy/fetch-http-handler": "^5.3.17", "@smithy/protocol-http": "^5.3.14", "@smithy/signature-v4": "^5.3.14", "@smithy/types": "^4.14.1", "@smithy/util-base64": "^4.3.2", "@smithy/util-hex-encoding": "^4.2.2", "@smithy/util-utf8": "^4.2.2", "tslib": "^2.6.2" } }, "sha512-86+S9oCyRVGzoMRpQhxkArp7kD2K75GPmaNevd9B6EyNhWoNvnCZZ3WbgN4j7ZT+jvtvBCGZvI2XHsWZJ+BRIg=="], + "@aws-sdk/middleware-websocket": ["@aws-sdk/middleware-websocket@3.972.17", "", { "dependencies": { "@aws-sdk/core": "^3.974.9", "@aws-sdk/types": "^3.973.8", "@smithy/core": "^3.24.1", "@smithy/fetch-http-handler": "^5.4.1", "@smithy/signature-v4": "^5.4.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-0PRAeIunuJRAweM/YWATe0jCHx4BMzSuwry461/TgcwMc8mNShwTdXiG6eEQBD7u+nNPC8Inp9KXjSB6D7uNYg=="], - "@aws-sdk/nested-clients": ["@aws-sdk/nested-clients@3.997.6", "", { "dependencies": { "@aws-crypto/sha256-browser": "5.2.0", "@aws-crypto/sha256-js": "5.2.0", "@aws-sdk/core": "^3.974.8", "@aws-sdk/middleware-host-header": "^3.972.10", "@aws-sdk/middleware-logger": "^3.972.10", "@aws-sdk/middleware-recursion-detection": "^3.972.11", "@aws-sdk/middleware-user-agent": "^3.972.38", "@aws-sdk/region-config-resolver": "^3.972.13", "@aws-sdk/signature-v4-multi-region": "^3.996.25", "@aws-sdk/types": "^3.973.8", "@aws-sdk/util-endpoints": "^3.996.8", "@aws-sdk/util-user-agent-browser": "^3.972.10", "@aws-sdk/util-user-agent-node": "^3.973.24", "@smithy/config-resolver": "^4.4.17", "@smithy/core": "^3.23.17", "@smithy/fetch-http-handler": "^5.3.17", "@smithy/hash-node": "^4.2.14", "@smithy/invalid-dependency": "^4.2.14", "@smithy/middleware-content-length": "^4.2.14", "@smithy/middleware-endpoint": "^4.4.32", "@smithy/middleware-retry": "^4.5.7", "@smithy/middleware-serde": "^4.2.20", "@smithy/middleware-stack": "^4.2.14", "@smithy/node-config-provider": "^4.3.14", "@smithy/node-http-handler": "^4.6.1", "@smithy/protocol-http": "^5.3.14", "@smithy/smithy-client": "^4.12.13", "@smithy/types": "^4.14.1", "@smithy/url-parser": "^4.2.14", "@smithy/util-base64": "^4.3.2", "@smithy/util-body-length-browser": "^4.2.2", "@smithy/util-body-length-node": "^4.2.3", "@smithy/util-defaults-mode-browser": "^4.3.49", "@smithy/util-defaults-mode-node": "^4.2.54", "@smithy/util-endpoints": "^3.4.2", "@smithy/util-middleware": "^4.2.14", "@smithy/util-retry": "^4.3.6", "@smithy/util-utf8": "^4.2.2", "tslib": "^2.6.2" } }, "sha512-WBDnqatJl+kGObpfmfSxqnXeYTu3Me8wx8WCtvoxX3pfWrrTv8I4WTMSSs7PZqcRcVh8WeUKMgGFjMG+52SR1w=="], + "@aws-sdk/nested-clients": ["@aws-sdk/nested-clients@3.997.7", "", { "dependencies": { "@aws-crypto/sha256-browser": "5.2.0", "@aws-crypto/sha256-js": "5.2.0", "@aws-sdk/core": "^3.974.9", "@aws-sdk/middleware-host-header": "^3.972.11", "@aws-sdk/middleware-logger": "^3.972.10", "@aws-sdk/middleware-recursion-detection": "^3.972.12", "@aws-sdk/middleware-user-agent": "^3.972.39", "@aws-sdk/region-config-resolver": "^3.972.14", "@aws-sdk/signature-v4-multi-region": "^3.996.26", "@aws-sdk/types": "^3.973.8", "@aws-sdk/util-endpoints": "^3.996.9", "@aws-sdk/util-user-agent-browser": "^3.972.11", "@aws-sdk/util-user-agent-node": "^3.973.25", "@smithy/core": "^3.24.1", "@smithy/fetch-http-handler": "^5.4.1", "@smithy/node-http-handler": "^4.7.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-jT2AXOODobQfTYGC2SChMSnZ/voIcRV/LHlY1suyhY1bdgP/voKkhEg8Ci1jiGQ4lBiaso5BEAV3ZWWpPTfmYA=="], - "@aws-sdk/region-config-resolver": ["@aws-sdk/region-config-resolver@3.972.13", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@smithy/config-resolver": "^4.4.17", "@smithy/node-config-provider": "^4.3.14", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-CvJ2ZIjK/jVD/lbOpowBVElJyC1YxLTIJ13yM0AEo0t2v7swOzGjSA6lJGH+DwZXQhcjUjoYwc8bVYCX5MDr1A=="], + "@aws-sdk/region-config-resolver": ["@aws-sdk/region-config-resolver@3.972.14", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@smithy/core": "^3.24.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-VuLXVmm7+lKVxqFcOItPkXhjbJ02iUfxkxheRu41SfWf6/xrZup2A2SwHZos/LeQGu3SBHeqTQht80Uo3ienPA=="], - "@aws-sdk/signature-v4-multi-region": ["@aws-sdk/signature-v4-multi-region@3.996.25", "", { "dependencies": { "@aws-sdk/middleware-sdk-s3": "^3.972.37", "@aws-sdk/types": "^3.973.8", "@smithy/protocol-http": "^5.3.14", "@smithy/signature-v4": "^5.3.14", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-+CMIt3e1VzlklAECmG+DtP1sV8iKq25FuA0OKpnJ4KA0kxUtd7CgClY7/RU6VzJBQwbN4EJ9Ue6plvqx1qGadw=="], + "@aws-sdk/signature-v4-multi-region": ["@aws-sdk/signature-v4-multi-region@3.996.26", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@smithy/core": "^3.24.1", "@smithy/signature-v4": "^5.4.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-2N62veqdMZBCwQUHsbhtnaovOFjOa5Dn3dAD1nRqFTUXR4QmirT3HZnfus/L1DS08Vm5CkoKmL0iMVt6YbqEag=="], "@aws-sdk/token-providers": ["@aws-sdk/token-providers@3.1005.0", "", { "dependencies": { "@aws-sdk/core": "^3.973.19", "@aws-sdk/nested-clients": "^3.996.8", "@aws-sdk/types": "^3.973.5", "@smithy/property-provider": "^4.2.11", "@smithy/shared-ini-file-loader": "^4.4.6", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-vMxd+ivKqSxU9bHx5vmAlFKDAkjGotFU56IOkDa5DaTu1WWwbcse0yFHEm9I537oVvodaiwMl3VBwgHfzQ2rvw=="], "@aws-sdk/types": ["@aws-sdk/types@3.973.8", "", { "dependencies": { "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-gjlAdtHMbtR9X5iIhVUvbVcy55KnznpC6bkDUWW9z915bi0ckdUr5cjf16Kp6xq0bP5HBD2xzgbL9F9Quv5vUw=="], - "@aws-sdk/util-arn-parser": ["@aws-sdk/util-arn-parser@3.972.3", "", { "dependencies": { "tslib": "^2.6.2" } }, "sha512-HzSD8PMFrvgi2Kserxuff5VitNq2sgf3w9qxmskKDiDTThWfVteJxuCS9JXiPIPtmCrp+7N9asfIaVhBFORllA=="], - - "@aws-sdk/util-endpoints": ["@aws-sdk/util-endpoints@3.996.8", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@smithy/types": "^4.14.1", "@smithy/url-parser": "^4.2.14", "@smithy/util-endpoints": "^3.4.2", "tslib": "^2.6.2" } }, "sha512-oOZHcRDihk5iEe5V25NVWg45b3qEA8OpHWVdU/XQh8Zj4heVPAJqWvMphQnU7LkufmUo10EpvFPZuQMiFLJK3g=="], - - "@aws-sdk/util-format-url": ["@aws-sdk/util-format-url@3.972.10", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@smithy/querystring-builder": "^4.2.14", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-DEKiHNJVtNxdyTeQspzY+15Po/kHm6sF0Cs4HV9Q2+lplB63+DrvdeiSoOSdWEWAoO2RcY1veoXVDz2tWxWCgQ=="], + "@aws-sdk/util-endpoints": ["@aws-sdk/util-endpoints@3.996.9", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@smithy/core": "^3.24.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-ibx8Vd73rCTHekNGeXX8cpGWoBKbNAlwKHL3yjSxxttu5QnNDaSAM7/0MFYDjU31/F4lyrPoQcGirT0ew61xcg=="], "@aws-sdk/util-locate-window": ["@aws-sdk/util-locate-window@3.965.5", "", { "dependencies": { "tslib": "^2.6.2" } }, "sha512-WhlJNNINQB+9qtLtZJcpQdgZw3SCDCpXdUJP7cToGwHbCWCnRckGlc6Bx/OhWwIYFNAn+FIydY8SZ0QmVu3xTQ=="], - "@aws-sdk/util-user-agent-browser": ["@aws-sdk/util-user-agent-browser@3.972.10", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@smithy/types": "^4.14.1", "bowser": "^2.11.0", "tslib": "^2.6.2" } }, "sha512-FAzqXvfEssGdSIz8ejatan0bOdx1qefBWKF/gWmVBXIP1HkS7v/wjjaqrAGGKvyihrXTXW00/2/1nTJtxpXz7g=="], + "@aws-sdk/util-user-agent-browser": ["@aws-sdk/util-user-agent-browser@3.972.11", "", { "dependencies": { "@aws-sdk/types": "^3.973.8", "@smithy/types": "^4.14.1", "bowser": "^2.11.0", "tslib": "^2.6.2" } }, "sha512-kq3RS6XQtHMrLFShbkem6h+8fxazB3jEIsbMC6aaSInOciRGE+eGAqTgJ+obO7Euo/pjM8thVqLiLISEH9X9DA=="], - "@aws-sdk/util-user-agent-node": ["@aws-sdk/util-user-agent-node@3.973.24", "", { "dependencies": { "@aws-sdk/middleware-user-agent": "^3.972.38", "@aws-sdk/types": "^3.973.8", "@smithy/node-config-provider": "^4.3.14", "@smithy/types": "^4.14.1", "@smithy/util-config-provider": "^4.2.2", "tslib": "^2.6.2" }, "peerDependencies": { "aws-crt": ">=1.0.0" }, "optionalPeers": ["aws-crt"] }, "sha512-ZWwlkjcIp7cEL8ZfTpTAPNkwx25p7xol0xlKoWVVf22+nsjwmLcHYtTPjIV1cSpmB/b6DaK4cb1fSkvCXHgRdw=="], + "@aws-sdk/util-user-agent-node": ["@aws-sdk/util-user-agent-node@3.973.25", "", { "dependencies": { "@aws-sdk/middleware-user-agent": "^3.972.39", "@aws-sdk/types": "^3.973.8", "@smithy/core": "^3.24.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" }, "peerDependencies": { "aws-crt": ">=1.0.0" }, "optionalPeers": ["aws-crt"] }, "sha512-066hKH/0nvV7x4ofV/iK9kz8r/qNfcR6rzuEOFqI2vQL/fcTTsDAbTw0jmXkyMzANK8ltQdALj19ns3zuOJiUw=="], - "@aws-sdk/xml-builder": ["@aws-sdk/xml-builder@3.972.22", "", { "dependencies": { "@nodable/entities": "2.1.0", "@smithy/types": "^4.14.1", "fast-xml-parser": "5.7.2", "tslib": "^2.6.2" } }, "sha512-PMYKKtJd70IsSG0yHrdAbxBr+ZWBKLvzFZfD3/urxgf6hXVMzuU5M+3MJ5G67RpOmLBu1fAUN65SbWuKUCOlAA=="], + "@aws-sdk/xml-builder": ["@aws-sdk/xml-builder@3.972.23", "", { "dependencies": { "@nodable/entities": "2.1.0", "@smithy/types": "^4.14.1", "fast-xml-parser": "5.7.2", "tslib": "^2.6.2" } }, "sha512-A0YmgYFv+hTI9c17Ntvd2hSehm9bmJfkb+ggADBwVKA8H/3+Jx94SzR2qOB9bAA9WFeDqnfz9PKKQ+D+YAKomA=="], "@aws/lambda-invoke-store": ["@aws/lambda-invoke-store@0.2.4", "", {}, "sha512-iY8yvjE0y651BixKNPgmv1WrQc+GZ142sb0z4gYnChDDY2YqI4P/jsSopBWrKfAt7LOJAkOXt7rC/hms+WclQQ=="], @@ -1062,11 +1044,11 @@ "@azure/logger": ["@azure/logger@1.3.0", "", { "dependencies": { "@typespec/ts-http-runtime": "^0.3.0", "tslib": "^2.6.2" } }, "sha512-fCqPIfOcLE+CGqGPd66c8bZpwAji98tZ4JI9i/mlTNTlsIWslCfpg48s/ypyLxZTump5sypjrKn2/kY7q8oAbA=="], - "@azure/msal-browser": ["@azure/msal-browser@5.9.0", "", { "dependencies": { "@azure/msal-common": "16.5.2" } }, "sha512-CzE+4PefDSJWj26zU7G1bKchlGRRHMBFreG4tAlGuzyI8hAPiYGobaJvZBgZBf6L63iphX7VH+ityL8VgEQz9Q=="], + "@azure/msal-browser": ["@azure/msal-browser@5.10.1", "", { "dependencies": { "@azure/msal-common": "16.6.1" } }, "sha512-hTbvOi9Ko2Jvn+G/fSmjzHf9WbNcf/o3epMtbeGx/pMwMrVAbi6OgCJVeCfsAb8IybSRpaCSc4EDRlYAhgngUQ=="], - "@azure/msal-common": ["@azure/msal-common@16.5.2", "", {}, "sha512-GkDEL6TYo3HgT3UuqakdgE9PZfc1hMki6+Hwgy1uddb/EauvAKfu85vVhuofRSo22D1xTnWt8Ucwfg4vSCVwvA=="], + "@azure/msal-common": ["@azure/msal-common@16.6.1", "", {}, "sha512-VxKdEtUwDuLD0F1hOQP7kye0YadZxFJfv37Em440geEf/w9uggKnHpRrqwZJOdxmPUOdhZ9kyRtKuAJW8wUcRg=="], - "@azure/msal-node": ["@azure/msal-node@5.1.5", "", { "dependencies": { "@azure/msal-common": "16.5.2", "jsonwebtoken": "^9.0.0" } }, "sha512-ObTeMoNPmq19X3z40et9Xvs4ZoWVeJg43PZMRLG5iwVL+2nCtAerG3YTDItqPp1CfXNwmCXBbg8jn1DOx65c3g=="], + "@azure/msal-node": ["@azure/msal-node@5.2.1", "", { "dependencies": { "@azure/msal-common": "16.6.1", "jsonwebtoken": "^9.0.0" } }, "sha512-tmQiQ2HvtzaeLqYGy3BemiPOSGPY4wCy1IW5zDWITKSs/s35WEd7Zij/hCxvUdAOzj6U3qnyaGbYXY91ortFEQ=="], "@azure/storage-blob": ["@azure/storage-blob@12.31.0", "", { "dependencies": { "@azure/abort-controller": "^2.1.2", "@azure/core-auth": "^1.9.0", "@azure/core-client": "^1.9.3", "@azure/core-http-compat": "^2.2.0", "@azure/core-lro": "^2.2.0", "@azure/core-paging": "^1.6.2", "@azure/core-rest-pipeline": "^1.19.1", "@azure/core-tracing": "^1.2.0", "@azure/core-util": "^1.11.0", "@azure/core-xml": "^1.4.5", "@azure/logger": "^1.1.4", "@azure/storage-common": "^12.3.0", "events": "^3.0.0", "tslib": "^2.8.1" } }, "sha512-DBgNv10aCSxopt92DkTDD0o9xScXeBqPKGmR50FPZQaEcH4JLQ+GEOGEDv19V5BMkB7kxr+m4h6il/cCDPvmHg=="], @@ -1178,15 +1160,7 @@ "@changesets/write": ["@changesets/write@0.4.0", "", { "dependencies": { "@changesets/types": "^6.1.0", "fs-extra": "^7.0.1", "human-id": "^4.1.1", "prettier": "^2.7.1" } }, "sha512-CdTLvIOPiCNuH71pyDu3rA+Q0n65cmAbXnwWH84rKGiFumFzkmHNT8KHTMEchcxN+Kl8I54xGUhJ7l3E7X396Q=="], - "@chevrotain/cst-dts-gen": ["@chevrotain/cst-dts-gen@12.0.0", "", { "dependencies": { "@chevrotain/gast": "12.0.0", "@chevrotain/types": "12.0.0" } }, "sha512-fSL4KXjTl7cDgf0B5Rip9Q05BOrYvkJV/RrBTE/bKDN096E4hN/ySpcBK5B24T76dlQ2i32Zc3PAE27jFnFrKg=="], - - "@chevrotain/gast": ["@chevrotain/gast@12.0.0", "", { "dependencies": { "@chevrotain/types": "12.0.0" } }, "sha512-1ne/m3XsIT8aEdrvT33so0GUC+wkctpUPK6zU9IlOyJLUbR0rg4G7ZiApiJbggpgPir9ERy3FRjT6T7lpgetnQ=="], - - "@chevrotain/regexp-to-ast": ["@chevrotain/regexp-to-ast@12.0.0", "", {}, "sha512-p+EW9MaJwgaHguhoqwOtx/FwuGr+DnNn857sXWOi/mClXIkPGl3rn7hGNWvo31HA3vyeQxjqe+H36yZJwYU8cA=="], - - "@chevrotain/types": ["@chevrotain/types@12.0.0", "", {}, "sha512-S+04vjFQKeuYw0/eW3U52LkAHQsB1ASxsPGsLPUyQgrZ2iNNibQrsidruDzjEX2JYfespXMG0eZmXlhA6z7nWA=="], - - "@chevrotain/utils": ["@chevrotain/utils@12.0.0", "", {}, "sha512-lB59uJoaGIfOOL9knQqQRfhl9g7x8/wqFkp13zTdkRu1huG9kg6IJs1O8hqj9rs6h7orGxHJUKb+mX3rPbWGhA=="], + "@chevrotain/types": ["@chevrotain/types@11.1.2", "", {}, "sha512-U+HFai5+zmJCkK86QsaJtoITlboZHBqrVketcO2ROv865xfCMSFpELQoz1GkX5GzME8pTa+3kbKrZHQtI0gdbw=="], "@clack/core": ["@clack/core@1.0.0-alpha.1", "", { "dependencies": { "picocolors": "^1.0.0", "sisteransi": "^1.0.5" } }, "sha512-rFbCU83JnN7l3W1nfgCqqme4ZZvTTgsiKQ6FM0l+r0P+o2eJpExcocBUWUIwnDzL76Aca9VhUdWmB2MbUv+Qyg=="], @@ -1212,7 +1186,7 @@ "@effect/platform-node": ["@effect/platform-node@4.0.0-beta.57", "", { "dependencies": { "@effect/platform-node-shared": "^4.0.0-beta.57", "mime": "^4.1.0", "undici": "^8.0.2" }, "peerDependencies": { "effect": "^4.0.0-beta.57", "ioredis": "^5.7.0" } }, "sha512-la0xxPSAYOsY0d+uVxEBxok3jYB31iPQmIaZZRUj2SNWqcGGHJc6KorKtI8guqSLuv9FGZ255kBWXRbG6hMeeg=="], - "@effect/platform-node-shared": ["@effect/platform-node-shared@4.0.0-beta.46", "", { "dependencies": { "@types/ws": "^8.18.1", "ws": "^8.19.0" }, "peerDependencies": { "effect": "^4.0.0-beta.46" } }, "sha512-Yzci82XbZ1W3tuiownsJawrJZTGeTrTZKLD0uxdBWCBzlVyqDwoSwRwO5qh33DurJj9B7iS8MDf14fpGRBPNGQ=="], + "@effect/platform-node-shared": ["@effect/platform-node-shared@4.0.0-beta.66", "", { "dependencies": { "@types/ws": "^8.18.1", "ws": "^8.20.0" }, "peerDependencies": { "effect": "^4.0.0-beta.66" } }, "sha512-+ymrhBnESv/hmn5SKTe2//IY9Ox/hGPeoogEWhW47ZGyhFI5eMYFxdEUBa+3IAV05rrBzrxON9lynu68n0DM7w=="], "@electron/asar": ["@electron/asar@3.4.1", "", { "dependencies": { "commander": "^5.0.0", "glob": "^7.1.6", "minimatch": "^3.0.4" }, "bin": { "asar": "bin/asar.js" } }, "sha512-i4/rNPRS84t0vSRa2HorerGRXWyF4vThfHesw0dmcWHp+cspK743UanA0suA5Q5y8kzY2y6YKrvbIUn69BCAiA=="], @@ -1230,7 +1204,7 @@ "@electron/windows-sign": ["@electron/windows-sign@1.2.2", "", { "dependencies": { "cross-dirname": "^0.1.0", "debug": "^4.3.4", "fs-extra": "^11.1.1", "minimist": "^1.2.8", "postject": "^1.0.0-alpha.6" }, "bin": { "electron-windows-sign": "bin/electron-windows-sign.js" } }, "sha512-dfZeox66AvdPtb2lD8OsIIQh12Tp0GNCRUDfBHIKGpbmopZto2/A8nSpYYLoedPIHpqkeblZ/k8OV0Gy7PYuyQ=="], - "@emnapi/core": ["@emnapi/core@1.10.0", "", { "dependencies": { "@emnapi/wasi-threads": "1.2.1", "tslib": "^2.4.0" } }, "sha512-yq6OkJ4p82CAfPl0u9mQebQHKPJkY7WrIuk205cTYnYe+k2Z8YBh11FrbRG/H6ihirqcacOgl2BIO8oyMQLeXw=="], + "@emnapi/core": ["@emnapi/core@1.9.2", "", { "dependencies": { "@emnapi/wasi-threads": "1.2.1", "tslib": "^2.4.0" } }, "sha512-UC+ZhH3XtczQYfOlu3lNEkdW/p4dsJ1r/bP7H8+rhao3TTTMO1ATq/4DdIi23XuGoFY+Cz0JmCbdVl0hz9jZcA=="], "@emnapi/runtime": ["@emnapi/runtime@1.10.0", "", { "dependencies": { "tslib": "^2.4.0" } }, "sha512-ewvYlk86xUoGI0zQRNq/mC+16R1QeDlKQy21Ki3oSYXNgLb45GV1P6A0M+/s6nyCuNDqe5VpaY84BzXGwVbwFA=="], @@ -1366,7 +1340,7 @@ "@iconify/types": ["@iconify/types@2.0.0", "", {}, "sha512-+wluvCrRhXrhyOmRDJ3q8mux9JkKy5SJ/v8ol2tu4FVjyYvtEzkc/3pK15ET6RKg4b4w4BmTk1+gsCUhf21Ykg=="], - "@iconify/utils": ["@iconify/utils@3.1.1", "", { "dependencies": { "@antfu/install-pkg": "^1.1.0", "@iconify/types": "^2.0.0", "mlly": "^1.8.2" } }, "sha512-MwzoDtw9rO1x+qfgLTV/IVXsHDBqeYZoMIQC8SfxfYSlaSUG+oWiAcoiB1yajAda6mqblm4/1/w2E8tRu7a7Tw=="], + "@iconify/utils": ["@iconify/utils@3.1.3", "", { "dependencies": { "@antfu/install-pkg": "^1.1.0", "@iconify/types": "^2.0.0", "import-meta-resolve": "^4.2.0" } }, "sha512-LPKOXPn/zV+zis1oOfGWogaXVpqUybF3ZS6SCZIsz8vg0ivVp9+fVqyYB7xq0aiST/VhUQYGO1qo6uoYSiEJqw=="], "@img/colour": ["@img/colour@1.1.0", "", {}, "sha512-Td76q7j57o/tLVdgS746cYARfSyxk8iEfRxewL9h4OMzYhbW4TAcppl0mT4eyqXddh6L/jwoM75mo7ixa/pCeQ=="], @@ -1584,7 +1558,7 @@ "@mdx-js/react": ["@mdx-js/react@3.1.1", "", { "dependencies": { "@types/mdx": "^2.0.0" }, "peerDependencies": { "@types/react": ">=16", "react": ">=16" } }, "sha512-f++rKLQgUVYDAtECQ6fn/is15GkEH9+nZPM3MS0RcxVqoTfawHvDlSCH7JbMhAM6uJ32v3eXLvLmLvjGu7PTQw=="], - "@mermaid-js/parser": ["@mermaid-js/parser@1.1.0", "", { "dependencies": { "langium": "^4.0.0" } }, "sha512-gxK9ZX2+Fex5zu8LhRQoMeMPEHbc73UKZ0FQ54YrQtUxE1VVhMwzeNtKRPAu5aXks4FasbMe4xB4bWrmq6Jlxw=="], + "@mermaid-js/parser": ["@mermaid-js/parser@1.1.1", "", { "dependencies": { "@chevrotain/types": "~11.1.1" } }, "sha512-VuHdsYMK1bT6X2JbcAaWAhugTRvRBRyuZgd+c22swUeI9g/ntaxF7CY7dYarhZovofCbUNO0G7JesfmNtjYOCw=="], "@mixmark-io/domino": ["@mixmark-io/domino@2.2.0", "", {}, "sha512-Y28PR25bHXUg88kCV7nivXrP2Nj2RueZ3/l/jdx6J9f8J4nsEGcgX0Qe6lt7Pa+J79+kPiJU3LguR6O/6zrLOw=="], @@ -1606,23 +1580,23 @@ "@napi-rs/wasm-runtime": ["@napi-rs/wasm-runtime@1.1.4", "", { "dependencies": { "@tybys/wasm-util": "^0.10.1" }, "peerDependencies": { "@emnapi/core": "^1.7.1", "@emnapi/runtime": "^1.7.1" } }, "sha512-3NQNNgA1YSlJb/kMH1ildASP9HW7/7kYnRI2szWJaofaS1hWmbGI4H+d3+22aGzXXN9IJ+n+GiFVcGipJP18ow=="], - "@next/env": ["@next/env@16.2.4", "", {}, "sha512-dKkkOzOSwFYe5RX6y26fZgkSpVAlIOJKQHIiydQcrWH6y/97+RceSOAdjZ14Qa3zLduVUy0TXcn+EiM6t4rPgw=="], + "@next/env": ["@next/env@16.2.6", "", {}, "sha512-gd8HoHN4ufj73WmR3JmVolrpJR47ILK6LouP5xElPglaVxir6e1a7VzvTvDWkOoPXT9rkkTzyCxBu4yeZfZwcw=="], - "@next/swc-darwin-arm64": ["@next/swc-darwin-arm64@16.2.4", "", { "os": "darwin", "cpu": "arm64" }, "sha512-OXTFFox5EKN1Ym08vfrz+OXxmCcEjT4SFMbNRsWZE99dMqt2Kcusl5MqPXcW232RYkMLQTy0hqgAMEsfEd/l2A=="], + "@next/swc-darwin-arm64": ["@next/swc-darwin-arm64@16.2.6", "", { "os": "darwin", "cpu": "arm64" }, "sha512-ZJGkkcNfYgrrMkqOdZ7zoLa1TOy0qpcMfk/z4Mh/FKUz40gVO+HNQWqmLxf67Z5WB64DRp0dhEbyHfel+6sJUg=="], - "@next/swc-darwin-x64": ["@next/swc-darwin-x64@16.2.4", "", { "os": "darwin", "cpu": "x64" }, "sha512-XhpVnUfmYWvD3YrXu55XdcAkQtOnvaI6wtQa8fuF5fGoKoxIUZ0kWPtcOfqJEWngFF/lOS9l3+O9CcownhiQxQ=="], + "@next/swc-darwin-x64": ["@next/swc-darwin-x64@16.2.6", "", { "os": "darwin", "cpu": "x64" }, "sha512-v/YLBHIY132Ced3puBJ7YJKw1lqsCrgcNo2aRJlCEyQrrCeRJlvGlnmxhPxNQI3KE3N1DN5r9TPNPvka3nq5RQ=="], - "@next/swc-linux-arm64-gnu": ["@next/swc-linux-arm64-gnu@16.2.4", "", { "os": "linux", "cpu": "arm64" }, "sha512-Mx/tjlNA3G8kg14QvuGAJ4xBwPk1tUHq56JxZ8CXnZwz1Etz714soCEzGQQzVMz4bEnGPowzkV6Xrp6wAkEWOQ=="], + "@next/swc-linux-arm64-gnu": ["@next/swc-linux-arm64-gnu@16.2.6", "", { "os": "linux", "cpu": "arm64" }, "sha512-RPOvqlYBbcQjkz9VQQDZ2T2bARIjXZV1KFlt+V2Mr6SW/e4I9fcKsaA0hdyf2FHoTlsV2xnBd5Y912rP/1Ce6w=="], - "@next/swc-linux-arm64-musl": ["@next/swc-linux-arm64-musl@16.2.4", "", { "os": "linux", "cpu": "arm64" }, "sha512-iVMMp14514u7Nup2umQS03nT/bN9HurK8ufylC3FZNykrwjtx7V1A7+4kvhbDSCeonTVqV3Txnv0Lu+m2oDXNg=="], + "@next/swc-linux-arm64-musl": ["@next/swc-linux-arm64-musl@16.2.6", "", { "os": "linux", "cpu": "arm64" }, "sha512-URUTu1+dMkxJsPFgm+OeEvq9wf5sujw0EvgYy80TDGHTSLTnIHeqb0Eu8A3sC95IRgjejQL+kC4mw+4yPxiAXA=="], - "@next/swc-linux-x64-gnu": ["@next/swc-linux-x64-gnu@16.2.4", "", { "os": "linux", "cpu": "x64" }, "sha512-EZOvm1aQWgnI/N/xcWOlnS3RQBk0VtVav5Zo7n4p0A7UKyTDx047k8opDbXgBpHl4CulRqRfbw3QrX2w5UOXMQ=="], + "@next/swc-linux-x64-gnu": ["@next/swc-linux-x64-gnu@16.2.6", "", { "os": "linux", "cpu": "x64" }, "sha512-DOj182mPV8G3UkrayLoREM5YEYI+Dk5wv7Ox9xl1fFibAELEsFD0lDPfHIeILlutMMfdyhlzYPELG3peuKaurw=="], - "@next/swc-linux-x64-musl": ["@next/swc-linux-x64-musl@16.2.4", "", { "os": "linux", "cpu": "x64" }, "sha512-h9FxsngCm9cTBf71AR4fGznDEDx1hS7+kSEiIRjq5kO1oXWm07DxVGZjCvk0SGx7TSjlUqhI8oOyz7NfwAdPoA=="], + "@next/swc-linux-x64-musl": ["@next/swc-linux-x64-musl@16.2.6", "", { "os": "linux", "cpu": "x64" }, "sha512-HKQ5SP/V/ub73UvF7n/zeJlxk2kLmtL7Wzrg4WfmkjmNos5onJ2tKu7yZOPdL18A6Svfn3max29ym+ry7NkK4g=="], - "@next/swc-win32-arm64-msvc": ["@next/swc-win32-arm64-msvc@16.2.4", "", { "os": "win32", "cpu": "arm64" }, "sha512-3NdJV5OXMSOeJYijX+bjaLge3mJBlh4ybydbT4GFoB/2hAojWHtMhl3CYlYoMrjPuodp0nzFVi4Tj2+WaMg+Ow=="], + "@next/swc-win32-arm64-msvc": ["@next/swc-win32-arm64-msvc@16.2.6", "", { "os": "win32", "cpu": "arm64" }, "sha512-LZXpTlPyS5v7HhSmnvsLGP3iIYgYOBnc8r8ArlT55sGHV89bR2HlDdBjWQ+PY6SJMmk8TuVGFuxalnP3k/0Dwg=="], - "@next/swc-win32-x64-msvc": ["@next/swc-win32-x64-msvc@16.2.4", "", { "os": "win32", "cpu": "x64" }, "sha512-kMVGgsqhO5YTYODD9IPGGhA6iprWidQckK3LmPeW08PIFENRmgfb4MjXHO+p//d+ts2rpjvK5gXWzXSMrPl9cw=="], + "@next/swc-win32-x64-msvc": ["@next/swc-win32-x64-msvc@16.2.6", "", { "os": "win32", "cpu": "x64" }, "sha512-F0+4i0h9J6C4eE3EAPWsoCk7UW/dbzOjyzxY0qnDUOYFu6FFmdZ6l97/XdV3/Nz3VYyO7UWjyEJUXkGqcoXfMA=="], "@nodable/entities": ["@nodable/entities@2.1.0", "", {}, "sha512-nyT7T3nbMyBI/lvr6L5TyWbFJAI9FTgVRakNoBqCD+PmID8DzFrrNdLLtHMwMszOtqZa8PAOV24ZqDnQrhQINA=="], @@ -1702,11 +1676,11 @@ "@opencode-ai/desktop-electron": ["@opencode-ai/desktop-electron@workspace:packages/desktop-electron"], - "@opencode-ai/plugin": ["@opencode-ai/plugin@1.14.35", "", { "dependencies": { "@opencode-ai/sdk": "1.14.35", "effect": "4.0.0-beta.59", "zod": "4.1.8" }, "peerDependencies": { "@opentui/core": ">=0.2.2", "@opentui/solid": ">=0.2.2" }, "optionalPeers": ["@opentui/core", "@opentui/solid"] }, "sha512-wbGsQZAYKGe5N+SnaZ1dU0S8d5WDauB8e5CEtOxqsXaQh1lbTUkvkO1PJ92TizcqfwnvaiDH1ND3uTcx5D2ryA=="], + "@opencode-ai/plugin": ["@opencode-ai/plugin@1.14.50", "", { "dependencies": { "@opencode-ai/sdk": "1.14.50", "effect": "4.0.0-beta.65", "zod": "4.1.8" }, "peerDependencies": { "@opentui/core": ">=0.2.9", "@opentui/keymap": ">=0.2.9", "@opentui/solid": ">=0.2.9" }, "optionalPeers": ["@opentui/core", "@opentui/keymap", "@opentui/solid"] }, "sha512-2D4k6r8IFaAajEmezOZ3UKmRRGqEw2YzqotH5zLGT434yKtabduEsQgXa0fWlASut4FVT3JURhq5LqeBUV/k4g=="], "@opencode-ai/script": ["@opencode-ai/script@workspace:packages/script"], - "@opencode-ai/sdk": ["@opencode-ai/sdk@1.14.35", "", { "dependencies": { "cross-spawn": "7.0.6" } }, "sha512-yEeH+JGYEOmUL/fSrS2kWyR/5Qr/7q2+Ck2uecOA7JplgI76XXVtASkecXl+bHRwQSMM2j/f5eTvRUYkEqvGPA=="], + "@opencode-ai/sdk": ["@opencode-ai/sdk@1.14.50", "", { "dependencies": { "cross-spawn": "7.0.6" } }, "sha512-IrTKTFviR4Tj+u0BI8h8XgXIvEpxwkkHqBj6E0aEc4DBErpS3qh2Lkp1xt0OdtCYiLE3wZ1bAIiHOiO66H/7TA=="], "@opencode-ai/shared": ["@opencode-ai/shared@workspace:packages/shared"], @@ -1744,21 +1718,21 @@ "@opentelemetry/semantic-conventions": ["@opentelemetry/semantic-conventions@1.40.0", "", {}, "sha512-cifvXDhcqMwwTlTK04GBNeIe7yyo28Mfby85QXFe1Yk8nmi36Ab/5UQwptOx84SsoGNRg+EVSjwzfSZMy6pmlw=="], - "@opentui/core": ["@opentui/core@0.1.105", "", { "dependencies": { "bun-ffi-structs": "0.1.2", "diff": "8.0.2", "jimp": "1.6.0", "marked": "17.0.1", "yoga-layout": "3.2.1" }, "optionalDependencies": { "@dimforge/rapier2d-simd-compat": "^0.17.3", "@opentui/core-darwin-arm64": "0.1.105", "@opentui/core-darwin-x64": "0.1.105", "@opentui/core-linux-arm64": "0.1.105", "@opentui/core-linux-x64": "0.1.105", "@opentui/core-win32-arm64": "0.1.105", "@opentui/core-win32-x64": "0.1.105", "bun-webgpu": "0.1.5", "planck": "^1.4.2", "three": "0.177.0" }, "peerDependencies": { "web-tree-sitter": "0.25.10" } }, "sha512-vllSOOCW6VIThV/96GRLJ1IxIBuR+ci6FDvnPIAG4s7SJ/FW6zAkqDn1xrtBwwk/lM3QWjLqy8BZc+zwWvveJA=="], + "@opentui/core": ["@opentui/core@0.2.2", "", { "dependencies": { "bun-ffi-structs": "0.2.2", "diff": "9.0.0", "marked": "17.0.1", "string-width": "7.2.0", "strip-ansi": "7.1.2", "yoga-layout": "3.2.1" }, "optionalDependencies": { "@opentui/core-darwin-arm64": "0.2.2", "@opentui/core-darwin-x64": "0.2.2", "@opentui/core-linux-arm64": "0.2.2", "@opentui/core-linux-x64": "0.2.2", "@opentui/core-win32-arm64": "0.2.2", "@opentui/core-win32-x64": "0.2.2" }, "peerDependencies": { "web-tree-sitter": "0.25.10" } }, "sha512-wxg1CD58SVrowu+WgbhZNi3UP/wWxPio2Kj2IeTjomoIE+6EXLxR8eCCxHYVuQUd9E4fknrKkY5HmiSsp6oPow=="], - "@opentui/core-darwin-arm64": ["@opentui/core-darwin-arm64@0.1.105", "", { "os": "darwin", "cpu": "arm64" }, "sha512-1pIL7aer9amwj8EpYoMNtvavKetIe+nX8uBRmYsMQb+KvJoUAZUqENfRW+qHE5WrsOyxx8/QoyXTHw15GG5iLQ=="], + "@opentui/core-darwin-arm64": ["@opentui/core-darwin-arm64@0.2.2", "", { "os": "darwin", "cpu": "arm64" }, "sha512-tY5n3ZRQx+b0kyhQJJLsyJMeZ+0w4FV37YZc/Qqv3qvOqE9kZPw/7adR77FYwWDm/7fax94mLMrR8Y5bKUkDmw=="], - "@opentui/core-darwin-x64": ["@opentui/core-darwin-x64@0.1.105", "", { "os": "darwin", "cpu": "x64" }, "sha512-hLIRSWlK3gY2NRXJGWiTBiMYSmRDjOYFZF6WtUVXhY2SL3sp08dhmr/6dmAVH+3pKCsCipLEsrrcQX6SAihCTA=="], + "@opentui/core-darwin-x64": ["@opentui/core-darwin-x64@0.2.2", "", { "os": "darwin", "cpu": "x64" }, "sha512-W/R7OnqY30FXcTG0tiP2JkQFmgtYbIte5afQ5PC12TliRoee1RqG3iCG6kY1jxW+3Vg6jge88uiSjUEDpeV2gA=="], - "@opentui/core-linux-arm64": ["@opentui/core-linux-arm64@0.1.105", "", { "os": "linux", "cpu": "arm64" }, "sha512-jlRKfPkozTZEkHEePuCWYcTIUtPm+ieInAwGVqGmjbvqjxdVv1/W/Dt6LEZ/9jpRiOPd+FjXAfLe6wa/XWHr+w=="], + "@opentui/core-linux-arm64": ["@opentui/core-linux-arm64@0.2.2", "", { "os": "linux", "cpu": "arm64" }, "sha512-1pzTYFEZauYuw6AGycw2TYGtAlZVGjuUtSdxH1fP51kBPS3oVWduUY2j7GKREz3SU5NulvO2Wc6HWsm3feMqwQ=="], - "@opentui/core-linux-x64": ["@opentui/core-linux-x64@0.1.105", "", { "os": "linux", "cpu": "x64" }, "sha512-kfWS1WMg6qHShmxZX9s1tZc/8JcXw6uyy2UtyTbJdRFExtXGH37oKHi8QK8iPL2ExCx4z7zqVnVJfO3X/Wh7lA=="], + "@opentui/core-linux-x64": ["@opentui/core-linux-x64@0.2.2", "", { "os": "linux", "cpu": "x64" }, "sha512-ucVwUtUYeOYGVFPBLbPoxzbrPdhD0PDyKNQ2X4n1AJ9jlQX4gqBZRcXMEF8hiXDjFxsikZwef7De0ciCcWvAMg=="], - "@opentui/core-win32-arm64": ["@opentui/core-win32-arm64@0.1.105", "", { "os": "win32", "cpu": "arm64" }, "sha512-UFx6A8OpBVbGWK6OAw4GqAqKZgIITJfSOd35pG9yDVKQouHN2OGc2HeeXrH2A4h42p40Xl6IfcqqfllkpC13Dg=="], + "@opentui/core-win32-arm64": ["@opentui/core-win32-arm64@0.2.2", "", { "os": "win32", "cpu": "arm64" }, "sha512-MPhYdJNdxmC5Bqsq6sis/+VkjRgkEjm+bQ1Tl++NSKLuiTU32Re0ImcZlgHbe+LZtZoGMZHVSgZlkGd3oYXO2g=="], - "@opentui/core-win32-x64": ["@opentui/core-win32-x64@0.1.105", "", { "os": "win32", "cpu": "x64" }, "sha512-f9FqqUmxehwhF+cgyazm0YT0v0BYTTCPzd6eztqhl74N3x/kC+jOOz2rdJDC/tTBo1JVsF64KupOnhIs6/Cogg=="], + "@opentui/core-win32-x64": ["@opentui/core-win32-x64@0.2.2", "", { "os": "win32", "cpu": "x64" }, "sha512-19BroLfn2h0RDYfJS5o96Fc8kYCDhRBcseIXtHIkoKIsKMxx62KiDLo/byVye6rp+yQRRB7Xkd2uWqsbdiWo9w=="], - "@opentui/solid": ["@opentui/solid@0.1.105", "", { "dependencies": { "@babel/core": "7.28.0", "@babel/preset-typescript": "7.27.1", "@opentui/core": "0.1.105", "babel-plugin-module-resolver": "5.0.2", "babel-preset-solid": "1.9.10", "entities": "7.0.1", "s-js": "^0.4.9" }, "peerDependencies": { "solid-js": "1.9.11" } }, "sha512-uxnaMP802sCI487pv/Hk9xdFdIj9mkg3eNliAqbqR0Shmd4phcjKEZvPRpijjmI99j4s9nul71jzF3h1oz31Nw=="], + "@opentui/solid": ["@opentui/solid@0.2.2", "", { "dependencies": { "@babel/core": "7.28.0", "@babel/preset-typescript": "7.27.1", "@opentui/core": "0.2.2", "babel-plugin-module-resolver": "5.0.2", "babel-preset-solid": "1.9.12", "entities": "7.0.1", "s-js": "^0.4.9" }, "peerDependencies": { "solid-js": "1.9.12" } }, "sha512-ZBVfCoVAhcUGQWPAWOTdzuVldMaRkuPpCu4U1VZCqmIw9DtbCuiVr0WnDocDxKhJLbTu8bl3qEWtVCf6lTSi3w=="], "@oslojs/asn1": ["@oslojs/asn1@1.0.0", "", { "dependencies": { "@oslojs/binary": "1.0.0" } }, "sha512-zw/wn0sj0j0QKbIXfIlnEcTviaCzYOY3V5rAyjR6YtOByFtJiT574+8p9Wlach0lZH9fddD4yb9laEAIl4vXQA=="], @@ -1770,6 +1744,48 @@ "@oslojs/jwt": ["@oslojs/jwt@0.2.0", "", { "dependencies": { "@oslojs/encoding": "0.4.1" } }, "sha512-bLE7BtHrURedCn4Mco3ma9L4Y1GR2SMBuIvjWr7rmQ4/W/4Jy70TIAgZ+0nIlk0xHz1vNP8x8DCns45Sb2XRbg=="], + "@oxc-parser/binding-android-arm-eabi": ["@oxc-parser/binding-android-arm-eabi@0.127.0", "", { "os": "android", "cpu": "arm" }, "sha512-0LC7ye4hvqbIKxAzThzvswgHLFu2AURKzYLeSVvLdu2TBOYWQDmHnTqPLeA597BcUCxiLqLsS4CJ5uoI5WYWCQ=="], + + "@oxc-parser/binding-android-arm64": ["@oxc-parser/binding-android-arm64@0.127.0", "", { "os": "android", "cpu": "arm64" }, "sha512-b5jtVTH6AU5CJXHNdj7Jj9IEiR9yVjjnwHzPJhGyHGPdcsZSzBCkS9GBbV33niRMvKthDwQRFRJfI4a+k4PvYg=="], + + "@oxc-parser/binding-darwin-arm64": ["@oxc-parser/binding-darwin-arm64@0.127.0", "", { "os": "darwin", "cpu": "arm64" }, "sha512-obCE8B7ISKkJidjlhv9xRGJPOSDG2Yu6PRga9Ruaz35uintHxbp1Ki/Yc71wx4rj3Edrm0a1kzG1TAwit0wFpg=="], + + "@oxc-parser/binding-darwin-x64": ["@oxc-parser/binding-darwin-x64@0.127.0", "", { "os": "darwin", "cpu": "x64" }, "sha512-JL6Xb5IwPQT8rUzlpsX7E+AgfcdNklXNPFp8pjCQQ5MQOQo5rtEB2ui+3Hgg9Sn7Y9Egj6YOLLiHhLpdAe12Aw=="], + + "@oxc-parser/binding-freebsd-x64": ["@oxc-parser/binding-freebsd-x64@0.127.0", "", { "os": "freebsd", "cpu": "x64" }, "sha512-SDQ/3MQFw58fqQz3Z1PhSKFF3JoCF4gmlNjziDm8X02tTahCw0qJbd7FGPDKw1i4VTBZene9JPyC3mHtSvi+wA=="], + + "@oxc-parser/binding-linux-arm-gnueabihf": ["@oxc-parser/binding-linux-arm-gnueabihf@0.127.0", "", { "os": "linux", "cpu": "arm" }, "sha512-Av+D1MIqzV0YMGPT9we2SIZaMKD7Cxs4CvXSx/yxaWHewZjYEjScpOf5igc8IILASViw4WTnjlwUdI1KzVtDHQ=="], + + "@oxc-parser/binding-linux-arm-musleabihf": ["@oxc-parser/binding-linux-arm-musleabihf@0.127.0", "", { "os": "linux", "cpu": "arm" }, "sha512-Cs2fdJ8cPpFdeebj6p4dag8A4+56hPvZ0AhQQzlaLswGz1tz7bXt1nETLeorrM9+AMcWFFkqxcXwDGfTVidY8g=="], + + "@oxc-parser/binding-linux-arm64-gnu": ["@oxc-parser/binding-linux-arm64-gnu@0.127.0", "", { "os": "linux", "cpu": "arm64" }, "sha512-qdOfTcT6SY8gsJrrV92uyEUyjqMGPpIB5JZUG6QN5dukYd+7/j0kX6MwK1DgQj39jtUYixxPiaRUiEN1+0CXgQ=="], + + "@oxc-parser/binding-linux-arm64-musl": ["@oxc-parser/binding-linux-arm64-musl@0.127.0", "", { "os": "linux", "cpu": "arm64" }, "sha512-EoTCZneNFU/P2qrpEM+RHmQwt+CvDkyGESG6qhr7KaegXLZwePfbrkCDfAk8/rhxbDUVGsZILX+2tqPzFtoFWA=="], + + "@oxc-parser/binding-linux-ppc64-gnu": ["@oxc-parser/binding-linux-ppc64-gnu@0.127.0", "", { "os": "linux", "cpu": "ppc64" }, "sha512-zALjmZYgxFLHjXeudcDF0xFGNydTAtkAeXAr2EuC17ywCyFxcmQra4w0BMde0Yi/re4Bi4iwEoEXtYN7l6eBLQ=="], + + "@oxc-parser/binding-linux-riscv64-gnu": ["@oxc-parser/binding-linux-riscv64-gnu@0.127.0", "", { "os": "linux", "cpu": "none" }, "sha512-fPP8M6zQLS7Jz7o9d5ArUSuAuSK3e+WCYVrCpdzeCOejidtZExJ9tjhDrAd3HEPqARBCPmdpqxESPFqy44vkBQ=="], + + "@oxc-parser/binding-linux-riscv64-musl": ["@oxc-parser/binding-linux-riscv64-musl@0.127.0", "", { "os": "linux", "cpu": "none" }, "sha512-7IcC4Ao02oGpfnjt+X/oF4U2mllo2qoSkw5xxiXNKL9MCTsTiAC6616beOuehdxGcnz1bRoPC1RQ2f1GQDdN+g=="], + + "@oxc-parser/binding-linux-s390x-gnu": ["@oxc-parser/binding-linux-s390x-gnu@0.127.0", "", { "os": "linux", "cpu": "s390x" }, "sha512-pbXIhiNFHoqWeqDNLiJ9JkpHz1IM9k4DXa66x+1GTWMG7iLxtkXgE53iiuKSXwmk3zIYmaPVfBvgcAhS583K4Q=="], + + "@oxc-parser/binding-linux-x64-gnu": ["@oxc-parser/binding-linux-x64-gnu@0.127.0", "", { "os": "linux", "cpu": "x64" }, "sha512-MYCguB9RvBvlSd6gbuNI7QwiLoCCAlGnlRJFPrzLI6U1/9wkC/WK6LtBAUln55H1Ctqw45PWmqrobKoMhsYQzQ=="], + + "@oxc-parser/binding-linux-x64-musl": ["@oxc-parser/binding-linux-x64-musl@0.127.0", "", { "os": "linux", "cpu": "x64" }, "sha512-5eY0B/bxf1xIUxb4NOTvOI3KWtBQfPWYyKAzgcrCt0mDibSZygVpO1Pz8bkeiSZ5Jj9+M09dkggG3H8I5d0Uyg=="], + + "@oxc-parser/binding-openharmony-arm64": ["@oxc-parser/binding-openharmony-arm64@0.127.0", "", { "os": "none", "cpu": "arm64" }, "sha512-Gld0ajrFTUXNtdw20fVBuTQx66FA75nIVg+//pPfR3sXkuABB4mTBhl3r9JNzrJpgW//qiwxf0nWXUWGJSL3UQ=="], + + "@oxc-parser/binding-wasm32-wasi": ["@oxc-parser/binding-wasm32-wasi@0.127.0", "", { "dependencies": { "@emnapi/core": "1.9.2", "@emnapi/runtime": "1.9.2", "@napi-rs/wasm-runtime": "^1.1.4" }, "cpu": "none" }, "sha512-T6KVD7rhLzFlwGRXMnxUFfkCZD8FHnb968wVXW1mXzgRFc5RNXOBY2mPPDZ77x5Ln76ltLMgtPg0cOkU1NSrEQ=="], + + "@oxc-parser/binding-win32-arm64-msvc": ["@oxc-parser/binding-win32-arm64-msvc@0.127.0", "", { "os": "win32", "cpu": "arm64" }, "sha512-Ujvw4X+LD1CCGULcsQcvb4YNVoBGqt+JHgNNzGGaCImELiZLk477ifUH53gIbE7EKd933NdTi25JWEr9K2HwXw=="], + + "@oxc-parser/binding-win32-ia32-msvc": ["@oxc-parser/binding-win32-ia32-msvc@0.127.0", "", { "os": "win32", "cpu": "ia32" }, "sha512-0cwxKO7KHQQQfo4Uf4B2SQrhgm+cJaP9OvFFhx52Tkg4bezsacu83GB2/In5bC415Ueeym+kXdnge/57rbSfTw=="], + + "@oxc-parser/binding-win32-x64-msvc": ["@oxc-parser/binding-win32-x64-msvc@0.127.0", "", { "os": "win32", "cpu": "x64" }, "sha512-rOrnSQSCbhI2kowr9XxE7m9a8oQXnBHjnS6j95LxxAnEZ0+Fz20WlRXG4ondQb+ejjt2KOsa65sE6++L6kUd+w=="], + + "@oxc-project/types": ["@oxc-project/types@0.127.0", "", {}, "sha512-aIYXQBo4lCbO4z0R3FHeucQHpF46l2LbMdxRvqvuRuW2OxdnSkcng5B8+K12spgLDj93rtN3+J2Vac/TIO+ciQ=="], + "@oxc-resolver/binding-android-arm-eabi": ["@oxc-resolver/binding-android-arm-eabi@11.19.1", "", { "os": "android", "cpu": "arm" }, "sha512-aUs47y+xyXHUKlbhqHUjBABjvycq6YSD7bpxSW7vplUmdzAlJ93yXY6ZR0c1o1x5A/QKbENCvs3+NlY8IpIVzg=="], "@oxc-resolver/binding-android-arm64": ["@oxc-resolver/binding-android-arm64@11.19.1", "", { "os": "android", "cpu": "arm64" }, "sha512-oolbkRX+m7Pq2LNjr/kKgYeC7bRDMVTWPgxBGMjSpZi/+UskVo4jsMU3MLheZV55jL6c3rNelPl4oD60ggYmqA=="], @@ -1898,9 +1914,9 @@ "@playwright/test": ["@playwright/test@1.57.0", "", { "dependencies": { "playwright": "1.57.0" }, "bin": { "playwright": "cli.js" } }, "sha512-6TyEnHgd6SArQO8UO2OMTxshln3QMWBtPGrOCgs3wVEmQmwyuNtB10IZMfmYDE0riwNR1cu4q+pPcxMVtaG3TA=="], - "@posthog/core": ["@posthog/core@1.28.2", "", { "dependencies": { "@posthog/types": "1.372.8" } }, "sha512-Dii3pxDheG1WioztvV/MqHQMnb16jSFrl2HkDR1T5yf5/R7lPqJCFYt+eXkEdp/oAv/PD+1joyG6Ap/2quFmtQ=="], + "@posthog/core": ["@posthog/core@1.29.1", "", { "dependencies": { "@posthog/types": "1.373.4" } }, "sha512-q+/t/DZALr50YTE0dFgfGSS9EgwcyAlqsn+JS61wLkwdcDM5yu/YTDM8oMKmJupsyjSZlVkDuHZAMd4ab7AxzQ=="], - "@posthog/types": ["@posthog/types@1.372.8", "", {}, "sha512-ALpfCnWsMSM9Cw/6kyLPVpd81ZReEdZwmDxOi+DTJuIo7wDxBiu2cAsjOuA6D/AL22v7HOJrHsmBAPAWqS5X7Q=="], + "@posthog/types": ["@posthog/types@1.373.4", "", {}, "sha512-n+0AbGRYYsbi+CQXQi2rF1lwTSyASlaogcw4YSkzB5KeMa4Y6nhNb7+TTnu9aVor+BycsQYCa2OsBrMMbaTekw=="], "@protobuf-ts/plugin": ["@protobuf-ts/plugin@2.11.1", "", { "dependencies": { "@bufbuild/protobuf": "^2.4.0", "@bufbuild/protoplugin": "^2.4.0", "@protobuf-ts/protoc": "^2.11.1", "@protobuf-ts/runtime": "^2.11.1", "@protobuf-ts/runtime-rpc": "^2.11.1", "typescript": "^3.9" }, "bin": { "protoc-gen-ts": "bin/protoc-gen-ts", "protoc-gen-dump": "bin/protoc-gen-dump" } }, "sha512-HyuprDcw0bEEJqkOWe1rnXUP0gwYLij8YhPuZyZk6cJbIgc/Q0IFgoHQxOXNIXAcXM4Sbehh6kjVnCzasElw1A=="], @@ -2044,105 +2060,85 @@ "@sindresorhus/merge-streams": ["@sindresorhus/merge-streams@2.3.0", "", {}, "sha512-LtoMMhxAlorcGhmFYI+LhPgbPZCkgP6ra1YL604EeF6U98pLlQ3iWIGMdWSC+vWmPBWBNgmDBAhnAobLROJmwg=="], - "@smithy/chunked-blob-reader": ["@smithy/chunked-blob-reader@5.2.2", "", { "dependencies": { "tslib": "^2.6.2" } }, "sha512-St+kVicSyayWQca+I1rGitaOEH6uKgE8IUWoYnnEX26SWdWQcL6LvMSD19Lg+vYHKdT9B2Zuu7rd3i6Wnyb/iw=="], - - "@smithy/chunked-blob-reader-native": ["@smithy/chunked-blob-reader-native@4.2.3", "", { "dependencies": { "@smithy/util-base64": "^4.3.2", "tslib": "^2.6.2" } }, "sha512-jA5k5Udn7Y5717L86h4EIv06wIr3xn8GM1qHRi/Nf31annXcXHJjBKvgztnbn2TxH3xWrPBfgwHsOwZf0UmQWw=="], - - "@smithy/config-resolver": ["@smithy/config-resolver@4.4.17", "", { "dependencies": { "@smithy/node-config-provider": "^4.3.14", "@smithy/types": "^4.14.1", "@smithy/util-config-provider": "^4.2.2", "@smithy/util-endpoints": "^3.4.2", "@smithy/util-middleware": "^4.2.14", "tslib": "^2.6.2" } }, "sha512-TzDZcAnhTyAHbXVxWZo7/tEcrIeFq20IBk8So3OLOetWpR8EwY/yEqBMBFaJMeyEiREDq4NfEl+qO3OAUD+vbQ=="], - - "@smithy/core": ["@smithy/core@3.23.17", "", { "dependencies": { "@smithy/protocol-http": "^5.3.14", "@smithy/types": "^4.14.1", "@smithy/url-parser": "^4.2.14", "@smithy/util-base64": "^4.3.2", "@smithy/util-body-length-browser": "^4.2.2", "@smithy/util-middleware": "^4.2.14", "@smithy/util-stream": "^4.5.25", "@smithy/util-utf8": "^4.2.2", "@smithy/uuid": "^1.1.2", "tslib": "^2.6.2" } }, "sha512-x7BlLbUFL8NWCGjMF9C+1N5cVCxcPa7g6Tv9B4A2luWx3be3oU8hQ96wIwxe/s7OhIzvoJH73HAUSg5JXVlEtQ=="], - - "@smithy/credential-provider-imds": ["@smithy/credential-provider-imds@4.2.14", "", { "dependencies": { "@smithy/node-config-provider": "^4.3.14", "@smithy/property-provider": "^4.2.14", "@smithy/types": "^4.14.1", "@smithy/url-parser": "^4.2.14", "tslib": "^2.6.2" } }, "sha512-Au28zBN48ZAoXdooGUHemuVBrkE+Ie6RPmGNIAJsFqj33Vhb6xAgRifUydZ2aY+M+KaMAETAlKk5NC5h1G7wpg=="], - - "@smithy/eventstream-codec": ["@smithy/eventstream-codec@4.2.14", "", { "dependencies": { "@aws-crypto/crc32": "5.2.0", "@smithy/types": "^4.14.1", "@smithy/util-hex-encoding": "^4.2.2", "tslib": "^2.6.2" } }, "sha512-erZq0nOIpzfeZdCyzZjdJb4nVSKLUmSkaQUVkRGQTXs30gyUGeKnrYEg+Xe1W5gE3aReS7IgsvANwVPxSzY6Pw=="], + "@smithy/config-resolver": ["@smithy/config-resolver@4.5.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-Gxr1czgeGUKgUJBf3fUSvpb1d+EjEl17f5s8qAzn36QIWmVzNPZQb3C9Rdtfya1yu2qUSAhDGoHUvHI/GJjbBQ=="], - "@smithy/eventstream-serde-browser": ["@smithy/eventstream-serde-browser@4.2.14", "", { "dependencies": { "@smithy/eventstream-serde-universal": "^4.2.14", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-8IelTCtTctWRbb+0Dcy+C0aICh1qa0qWXqgjcXDmMuCvPJRnv26hiDZoAau2ILOniki65mCPKqOQs/BaWvO4CQ=="], + "@smithy/core": ["@smithy/core@3.24.2", "", { "dependencies": { "@aws-crypto/crc32": "5.2.0", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-IKS7qX59fAGCYBmt5JChcDswQDupZqT2Yn2ZBA3UgTlsjRNNkQzZobbn95xoAAdtTyJmBiJB3Y02qR3rgy3Zog=="], - "@smithy/eventstream-serde-config-resolver": ["@smithy/eventstream-serde-config-resolver@4.3.14", "", { "dependencies": { "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-sqHiHpYRYo3FJlaIxD1J8PhbcmJAm7IuM16mVnwSkCToD7g00IBZzKuiLNMGmftULmEUX6/UAz8/NN5uMP8bVA=="], + "@smithy/credential-provider-imds": ["@smithy/credential-provider-imds@4.3.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-iYr9ekBjmZ+FwkiHEopqGscBbl78X62cq3p5Dd0eC+gNd7fybNZFQQdDuOQjTVmFymleuA8YRWZnuXWZ8B3kKA=="], - "@smithy/eventstream-serde-node": ["@smithy/eventstream-serde-node@4.2.14", "", { "dependencies": { "@smithy/eventstream-serde-universal": "^4.2.14", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-Ht/8BuGlKfFTy0H3+8eEu0vdpwGztCnaLLXtpXNdQqiR7Hj4vFScU3T436vRAjATglOIPjJXronY+1WxxNLSiw=="], + "@smithy/eventstream-codec": ["@smithy/eventstream-codec@4.3.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-o5vbgeviqo25wICV4Bgh3lC+iyFITJLj9b8sdLPBPIb3WUJFLVaMOpsinN3nwdfjRSdw57q7C7hZi2axc62Ohg=="], - "@smithy/eventstream-serde-universal": ["@smithy/eventstream-serde-universal@4.2.14", "", { "dependencies": { "@smithy/eventstream-codec": "^4.2.14", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-lWyt4T2XQZUZgK3tQ3Wn0w3XBvZsK/vjTuJl6bXbnGZBHH0ZUSONTYiK9TgjTTzU54xQr3DRFwpjmhp0oLm3gg=="], + "@smithy/eventstream-serde-browser": ["@smithy/eventstream-serde-browser@4.3.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-wq3p8g8pyciXSTmysvOvpGeMQaPssgJqyJdfKquwAgpgRW5cKcXb4c0V/K48Lsn24oYK/cox/vHtj/eaGm0PEg=="], - "@smithy/fetch-http-handler": ["@smithy/fetch-http-handler@5.3.17", "", { "dependencies": { "@smithy/protocol-http": "^5.3.14", "@smithy/querystring-builder": "^4.2.14", "@smithy/types": "^4.14.1", "@smithy/util-base64": "^4.3.2", "tslib": "^2.6.2" } }, "sha512-bXOvQzaSm6MnmLaWA1elgfQcAtN4UP3vXqV97bHuoOrHQOJiLT3ds6o9eo5bqd0TJfRFpzdGnDQdW3FACiAVdw=="], + "@smithy/eventstream-serde-config-resolver": ["@smithy/eventstream-serde-config-resolver@4.4.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-/Z4Rn+d/HzU96Ciy2bPDZq2KdlRM18ZhGSiy1lSUZPCpGQxXzGQCIItsk3vMcioBHn8E5t48LEXpTL0rOogrSA=="], - "@smithy/hash-blob-browser": ["@smithy/hash-blob-browser@4.2.15", "", { "dependencies": { "@smithy/chunked-blob-reader": "^5.2.2", "@smithy/chunked-blob-reader-native": "^4.2.3", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-0PJ4Al3fg2nM4qKrAIxyNcApgqHAXcBkN8FeizOz69z0rb26uZ6lMESYtxegaTlXB5Hj84JfwMPavMrwDMjucA=="], + "@smithy/eventstream-serde-node": ["@smithy/eventstream-serde-node@4.3.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-D4b+U+tOm9DVB8sk0/iDTVYLtBbid9T4+kX25k3rLie1SNqsaKNPTkx6dTWuB4TGNczKixeTCB4RGiV7KGO4Jw=="], - "@smithy/hash-node": ["@smithy/hash-node@4.2.14", "", { "dependencies": { "@smithy/types": "^4.14.1", "@smithy/util-buffer-from": "^4.2.2", "@smithy/util-utf8": "^4.2.2", "tslib": "^2.6.2" } }, "sha512-8ZBDY2DD4wr+GGjTpPtiglEsqr0lUP+KHqgZcWczFf6qeZ/YRjMIOoQWVQlmwu7EtxKTd8YXD8lblmYcpBIA1g=="], + "@smithy/fetch-http-handler": ["@smithy/fetch-http-handler@5.4.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-3wF40g8OOCA5BnwQUvwtzZqYBbWWftDjpAlWIUo6Yld3ZzJaMAKqg7MWQBPjE8oLaqvZQUE7tVGlZPsae6A4bQ=="], - "@smithy/hash-stream-node": ["@smithy/hash-stream-node@4.2.14", "", { "dependencies": { "@smithy/types": "^4.14.1", "@smithy/util-utf8": "^4.2.2", "tslib": "^2.6.2" } }, "sha512-tw4GANWkZPb6+BdD4Fgucqzey2+r73Z/GRo9zklsCdwrnxxumUV83ZIaBDdudV4Ylazw3EPTiJZhpX42105ruQ=="], + "@smithy/hash-blob-browser": ["@smithy/hash-blob-browser@4.3.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-p/X2kAc11zKkSZYW4bQb2CuWB0rZyXz52UAOtDpMpj8gXlVE7K9NU3sqh3gpTsPFS/2qvMoDS/w1a839k+815w=="], - "@smithy/invalid-dependency": ["@smithy/invalid-dependency@4.2.14", "", { "dependencies": { "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-c21qJiTSb25xvvOp+H2TNZzPCngrvl5vIPqPB8zQ/DmJF4QWXO19x1dWfMJZ6wZuuWUPPm0gV8C0cU3+ifcWuw=="], + "@smithy/hash-node": ["@smithy/hash-node@4.3.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-27ImyEVgDZ2ZQz1rnQMSlw9rm7x3244oZVIFI1WKi3vNtbxQ75XU2jA9BQuH8eb91zBLlyGjWQ/L/QGvmJfEHA=="], - "@smithy/is-array-buffer": ["@smithy/is-array-buffer@4.2.2", "", { "dependencies": { "tslib": "^2.6.2" } }, "sha512-n6rQ4N8Jj4YTQO3YFrlgZuwKodf4zUFs7EJIWH86pSCWBaAtAGBFfCM7Wx6D2bBJ2xqFNxGBSrUWswT3M0VJow=="], + "@smithy/hash-stream-node": ["@smithy/hash-stream-node@4.3.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-eUIHSr4RWqMQVtsnZ4p4tNDFuyCuOFCO3cfkvMHLbTRUIbCAq/7XEeGor7h0o0KYMGLeBztOKLQ0WQyc0j/8gA=="], - "@smithy/md5-js": ["@smithy/md5-js@4.2.14", "", { "dependencies": { "@smithy/types": "^4.14.1", "@smithy/util-utf8": "^4.2.2", "tslib": "^2.6.2" } }, "sha512-V2v0vx+h0iUSNG1Alt+GNBMSLGCrl9iVsdd+Ap67HPM9PN479x12V8LkuMoKImNZxn3MXeuyUjls+/7ZACZghA=="], + "@smithy/invalid-dependency": ["@smithy/invalid-dependency@4.3.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-bP0RYUtgINyMsUEzTnjnwJ/NX9Aa8y7bj0NbqwrMMqT6vjpp7H9SqGuKsn0+wD+Fu4GXcxUex1mH3k0t6WsatQ=="], - "@smithy/middleware-content-length": ["@smithy/middleware-content-length@4.2.14", "", { "dependencies": { "@smithy/protocol-http": "^5.3.14", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-xhHq7fX4/3lv5NHxLUk3OeEvl0xZ+Ek3qIbWaCL4f9JwgDZEclPBElljaZCAItdGPQl/kSM4LPMOpy1MYgprpw=="], + "@smithy/is-array-buffer": ["@smithy/is-array-buffer@2.2.0", "", { "dependencies": { "tslib": "^2.6.2" } }, "sha512-GGP3O9QFD24uGeAXYUjwSTXARoqpZykHadOmA8G5vfJPK0/DC67qa//0qvqrJzL1xc8WQWX7/yc7fwudjPHPhA=="], - "@smithy/middleware-endpoint": ["@smithy/middleware-endpoint@4.4.32", "", { "dependencies": { "@smithy/core": "^3.23.17", "@smithy/middleware-serde": "^4.2.20", "@smithy/node-config-provider": "^4.3.14", "@smithy/shared-ini-file-loader": "^4.4.9", "@smithy/types": "^4.14.1", "@smithy/url-parser": "^4.2.14", "@smithy/util-middleware": "^4.2.14", "tslib": "^2.6.2" } }, "sha512-ZZkgyjnJppiZbIm6Qbx92pbXYi1uzenIvGhBSCDlc7NwuAkiqSgS75j1czAD25ZLs2FjMjYy1q7gyRVWG6JA0Q=="], + "@smithy/md5-js": ["@smithy/md5-js@4.3.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-i7ES+52XcEuvIYaZrSePh8YHoLjI43hC3HO/KLO2osWEaUdX8kh4YpE6/iyfVud1M1vB3zEGHvwmsmqxDhw/Xw=="], - "@smithy/middleware-retry": ["@smithy/middleware-retry@4.5.7", "", { "dependencies": { "@smithy/core": "^3.23.17", "@smithy/node-config-provider": "^4.3.14", "@smithy/protocol-http": "^5.3.14", "@smithy/service-error-classification": "^4.3.1", "@smithy/smithy-client": "^4.12.13", "@smithy/types": "^4.14.1", "@smithy/util-middleware": "^4.2.14", "@smithy/util-retry": "^4.3.6", "@smithy/uuid": "^1.1.2", "tslib": "^2.6.2" } }, "sha512-bRt6ZImqVSeTk39Nm81K20ObIiAZ3WefY7G6+iz/0tZjs4dgRRjvRX2sgsH+zi6iDCRR/aQvQofLKxxz4rPBZg=="], + "@smithy/middleware-content-length": ["@smithy/middleware-content-length@4.3.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-Zq4MFXu6Cl/00FKXcc6mkio0xzd6D9Q8+AmtBOC6vHpN6Fz1MButZ0zfL46WxhJH/JgKjv5xl4Qo8HZsr6sDBg=="], - "@smithy/middleware-serde": ["@smithy/middleware-serde@4.2.20", "", { "dependencies": { "@smithy/core": "^3.23.17", "@smithy/protocol-http": "^5.3.14", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-Lx9JMO9vArPtiChE3wbEZ5akMIDQpWQtlu90lhACQmNOXcGXRbaDywMHDzuDZ2OkZzP+9wQfZi3YJT9F67zTQQ=="], + "@smithy/middleware-endpoint": ["@smithy/middleware-endpoint@4.5.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-1aiE05F2Er+ZYvGfrfI0+JRNeFY4M+hSjUp0SIKyq98k9iC0Lo71XwLbKWcFgFBZuhg5n6i+MQzRVmeCDlWOjw=="], - "@smithy/middleware-stack": ["@smithy/middleware-stack@4.2.14", "", { "dependencies": { "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-2dvkUKLuFdKsCRmOE4Mn63co0Djtsm+JMh0bYZQupN1pJwMeE8FmQmRLLzzEMN0dnNi7CDCYYH8F0EVwWiPBeA=="], + "@smithy/middleware-retry": ["@smithy/middleware-retry@4.6.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-ovt2p0LV3sSRpt8EBajI6imGMz/kq8F73P0eSOq6bhtvcOZM3xODFOjk6Lz2KvKfe7dVlxpNIiOYYyVe8ofv0A=="], - "@smithy/node-config-provider": ["@smithy/node-config-provider@4.3.14", "", { "dependencies": { "@smithy/property-provider": "^4.2.14", "@smithy/shared-ini-file-loader": "^4.4.9", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-S+gFjyo/weSVL0P1b9Ts8C/CwIfNCgUPikk3sl6QVsfE/uUuO+QsF+NsE/JkpvWqqyz1wg7HFdiaZuj5CoBMRg=="], + "@smithy/middleware-serde": ["@smithy/middleware-serde@4.3.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-hM3b9/JgMjohYHcgh5780ymND7RWGvYuyjNDtQL6P/DawtdbUu5m4oon4FHas+rmjdQ2DHee3cmAetTgU6keJw=="], - "@smithy/node-http-handler": ["@smithy/node-http-handler@4.6.1", "", { "dependencies": { "@smithy/protocol-http": "^5.3.14", "@smithy/querystring-builder": "^4.2.14", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-iB+orM4x3xrr57X3YaXazfKnntl0LHlZB1kcXSGzMV1Tt0+YwEjGlbjk/44qEGtBzXAz6yFDzkYTKSV6Pj2HUg=="], + "@smithy/middleware-stack": ["@smithy/middleware-stack@4.3.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-WthvBnmmksOBbW2I8CRc35QUJn/whgXucpW/hNmE70znXG16/yuC7LGtbDYTr7ak+LNZKzBGmaQR1uDNpeBd1g=="], - "@smithy/property-provider": ["@smithy/property-provider@4.2.14", "", { "dependencies": { "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-WuM31CgfsnQ/10i7NYr0PyxqknD72Y5uMfUMVSniPjbEPceiTErb4eIqJQ+pdxNEAUEWrewrGjIRjVbVHsxZiQ=="], + "@smithy/node-config-provider": ["@smithy/node-config-provider@4.4.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-LKup5BaU51fQM1YI/T64SJnn561tUPCfbpStnEygz5u1AqSAOALYY4e8imzz2EuMjcUtaYhOBtMazaN3TRXTgw=="], - "@smithy/protocol-http": ["@smithy/protocol-http@5.3.14", "", { "dependencies": { "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-dN5F8kHx8RNU0r+pCwNmFZyz6ChjMkzShy/zup6MtkRmmix4vZzJdW+di7x//b1LiynIev88FM18ie+wwPcQtQ=="], + "@smithy/node-http-handler": ["@smithy/node-http-handler@4.7.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-EdksTZ8UXYxGUgQ4mpIKrHoaj9WVGsp66TpZuixLAz1Jex8YDLnS4RH9ktGED5aOpN0OJlEtrsC9IGt76go1eA=="], - "@smithy/querystring-builder": ["@smithy/querystring-builder@4.2.14", "", { "dependencies": { "@smithy/types": "^4.14.1", "@smithy/util-uri-escape": "^4.2.2", "tslib": "^2.6.2" } }, "sha512-XYA5Z0IqTeF+5XDdh4BBmSA0HvbgVZIyv4cmOoUheDNR57K1HgBp9ukUMx3Cr3XpDHHpLBnexPE3LAtDsZkj2A=="], + "@smithy/property-provider": ["@smithy/property-provider@4.3.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-utUCslUrmJxKqSEidPhE1yfBgox78yhZQcHfdU4qvh79Rhi0nNKq6xNG4J/IwPD+Pm9y7a5FdXOgJxmHU5CGoA=="], - "@smithy/querystring-parser": ["@smithy/querystring-parser@4.2.14", "", { "dependencies": { "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-hr+YyqBD23GVvRxGGrcc/oOeNlK3PzT5Fu4dzrDXxzS1LpFiuL2PQQqKPs87M79aW7ziMs+nvB3qdw77SqE7Lw=="], + "@smithy/protocol-http": ["@smithy/protocol-http@5.4.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-rUvFCkbaHglm1zgOJ3QKFcD8jx/e68OUme3n+kAEiefAcGHK46UtmIT0/cuVLuiuSZzTqo8ts0Ju5hy9wqMGZA=="], - "@smithy/service-error-classification": ["@smithy/service-error-classification@4.3.1", "", { "dependencies": { "@smithy/types": "^4.14.1" } }, "sha512-aUQuDGh760ts/8MU+APjIZhlLPKhIIfqyzZaJikLEIMrdxFvxuLYD0WxWzaYWpmLbQlXDe9p7EWM3HsBe0K6Gw=="], + "@smithy/shared-ini-file-loader": ["@smithy/shared-ini-file-loader@4.5.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-4Cyy2IrFCgZBdw8G5eqB0G5FQ3HwH9FRYMJXGAMdo7hVIguVwQtLvEMmveIBlHf6hKQBd116M9IQRCA/7sxrpg=="], - "@smithy/shared-ini-file-loader": ["@smithy/shared-ini-file-loader@4.4.9", "", { "dependencies": { "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-495/V2I15SHgedSJoDPD23JuSfKAp726ZI1V0wtjB07Wh7q/0tri/0e0DLefZCHgxZonrGKt/OCTpAtP1wE1kQ=="], + "@smithy/signature-v4": ["@smithy/signature-v4@5.4.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-1km1OjdLRFuITWpCPofjFqzZ+tbeWuB72ZhcYjbjkCxZ21tTPfIs4GUxRrelMyKMLxLghGD58RENnXorU/O8cw=="], - "@smithy/signature-v4": ["@smithy/signature-v4@5.3.14", "", { "dependencies": { "@smithy/is-array-buffer": "^4.2.2", "@smithy/protocol-http": "^5.3.14", "@smithy/types": "^4.14.1", "@smithy/util-hex-encoding": "^4.2.2", "@smithy/util-middleware": "^4.2.14", "@smithy/util-uri-escape": "^4.2.2", "@smithy/util-utf8": "^4.2.2", "tslib": "^2.6.2" } }, "sha512-1D9Y/nmlVjCeSivCbhZ7hgEpmHyY1h0GvpSZt3l0xcD9JjmjVC1CHOozS6+Gh+/ldMH8JuJ6cujObQqfayAVFA=="], - - "@smithy/smithy-client": ["@smithy/smithy-client@4.12.13", "", { "dependencies": { "@smithy/core": "^3.23.17", "@smithy/middleware-endpoint": "^4.4.32", "@smithy/middleware-stack": "^4.2.14", "@smithy/protocol-http": "^5.3.14", "@smithy/types": "^4.14.1", "@smithy/util-stream": "^4.5.25", "tslib": "^2.6.2" } }, "sha512-y/Pcj1V9+qG98gyu1gvftHB7rDpdh+7kIBIggs55yGm3JdtBV8GT8IFF3a1qxZ79QnaJHX9GXzvBG6tAd+czJA=="], + "@smithy/smithy-client": ["@smithy/smithy-client@4.13.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-lK+Ssl8FzZHvdiPwB6qWLlPV6ih8FCr2BbRV+6/7QWabtMoiTbMTiGrrKsfKu6fcYzt+5akGAY7Xqna+EySq5g=="], "@smithy/types": ["@smithy/types@4.14.1", "", { "dependencies": { "tslib": "^2.6.2" } }, "sha512-59b5HtSVrVR/eYNei3BUj3DCPKD/G7EtDDe7OEJE7i7FtQFugYo6MxbotS8mVJkLNVf8gYaAlEBwwtJ9HzhWSg=="], - "@smithy/url-parser": ["@smithy/url-parser@4.2.14", "", { "dependencies": { "@smithy/querystring-parser": "^4.2.14", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-p06BiBigJ8bTA3MgnOfCtDUWnAMY0YfedO/GRpmc7p+wg3KW8vbXy1xwSu5ASy0wV7rRYtlfZOIKH4XqfhjSQQ=="], - - "@smithy/util-base64": ["@smithy/util-base64@4.3.2", "", { "dependencies": { "@smithy/util-buffer-from": "^4.2.2", "@smithy/util-utf8": "^4.2.2", "tslib": "^2.6.2" } }, "sha512-XRH6b0H/5A3SgblmMa5ErXQ2XKhfbQB+Fm/oyLZ2O2kCUrwgg55bU0RekmzAhuwOjA9qdN5VU2BprOvGGUkOOQ=="], - - "@smithy/util-body-length-browser": ["@smithy/util-body-length-browser@4.2.2", "", { "dependencies": { "tslib": "^2.6.2" } }, "sha512-JKCrLNOup3OOgmzeaKQwi4ZCTWlYR5H4Gm1r2uTMVBXoemo1UEghk5vtMi1xSu2ymgKVGW631e2fp9/R610ZjQ=="], - - "@smithy/util-body-length-node": ["@smithy/util-body-length-node@4.2.3", "", { "dependencies": { "tslib": "^2.6.2" } }, "sha512-ZkJGvqBzMHVHE7r/hcuCxlTY8pQr1kMtdsVPs7ex4mMU+EAbcXppfo5NmyxMYi2XU49eqaz56j2gsk4dHHPG/g=="], - - "@smithy/util-buffer-from": ["@smithy/util-buffer-from@4.2.2", "", { "dependencies": { "@smithy/is-array-buffer": "^4.2.2", "tslib": "^2.6.2" } }, "sha512-FDXD7cvUoFWwN6vtQfEta540Y/YBe5JneK3SoZg9bThSoOAC/eGeYEua6RkBgKjGa/sz6Y+DuBZj3+YEY21y4Q=="], + "@smithy/url-parser": ["@smithy/url-parser@4.3.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-ADEFjhX7Iv+cWrwkK+31tqoUzICzYAjB8GvpGDMoCQ71CA519Qd1ZRg1gDveYe20WQJxwW/ls4F0fSMZZTZnQQ=="], - "@smithy/util-config-provider": ["@smithy/util-config-provider@4.2.2", "", { "dependencies": { "tslib": "^2.6.2" } }, "sha512-dWU03V3XUprJwaUIFVv4iOnS1FC9HnMHDfUrlNDSh4315v0cWyaIErP8KiqGVbf5z+JupoVpNM7ZB3jFiTejvQ=="], + "@smithy/util-base64": ["@smithy/util-base64@4.4.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-+dxoGuQMmtP/m3ApPgliWQOTasVP9AHaKWCvczdiJlYk0SmTWrXMKq3lyiooeqMXWLuptcptQPiUYPitGzJjQQ=="], - "@smithy/util-defaults-mode-browser": ["@smithy/util-defaults-mode-browser@4.3.49", "", { "dependencies": { "@smithy/property-provider": "^4.2.14", "@smithy/smithy-client": "^4.12.13", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-a5bNrdiONYB/qE2BuKegvUMd/+ZDwdg4vsNuuSzYE8qs2EYAdK9CynL+Rzn29PbPiUqoz/cbpRbcLzD5lEevHw=="], + "@smithy/util-body-length-browser": ["@smithy/util-body-length-browser@4.3.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-69ETVcLni5dcIneXl7/Kc9Km/MqxOB4rGtr0zhuiVAz6mEr4QLo0P+c9bHZZzqrL5Tizd3fbPOVYpUOW1w4+MQ=="], - "@smithy/util-defaults-mode-node": ["@smithy/util-defaults-mode-node@4.2.54", "", { "dependencies": { "@smithy/config-resolver": "^4.4.17", "@smithy/credential-provider-imds": "^4.2.14", "@smithy/node-config-provider": "^4.3.14", "@smithy/property-provider": "^4.2.14", "@smithy/smithy-client": "^4.12.13", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-g1cvrJvOnzeJgEdf7AE4luI7gp6L8weE0y9a9wQUSGtjb8QRHDbCJYuE4Sy0SD9N8RrnNPFsPltAz/OSoBR9Zw=="], + "@smithy/util-body-length-node": ["@smithy/util-body-length-node@4.3.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-P6E2/3h8FZgMadSoUx/xZALw8pwDBQux4W/AYu1TwM/icy/P2G0LXjhLkBclgaR6AJr6xXjaT5p+vivgIQ+wYQ=="], - "@smithy/util-endpoints": ["@smithy/util-endpoints@3.4.2", "", { "dependencies": { "@smithy/node-config-provider": "^4.3.14", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-a55Tr+3OKld4TTtnT+RhKOQHyPxm3j/xL4OR83WBUhLJaKDS9dnJ7arRMOp3t31dcLhApwG9bgvrRXBHlLdIkg=="], + "@smithy/util-buffer-from": ["@smithy/util-buffer-from@2.2.0", "", { "dependencies": { "@smithy/is-array-buffer": "^2.2.0", "tslib": "^2.6.2" } }, "sha512-IJdWBbTcMQ6DA0gdNhh/BwrLkDR+ADW5Kr1aZmd4k3DIF6ezMV4R2NIAmT08wQJ3yUK82thHWmC/TnK/wpMMIA=="], - "@smithy/util-hex-encoding": ["@smithy/util-hex-encoding@4.2.2", "", { "dependencies": { "tslib": "^2.6.2" } }, "sha512-Qcz3W5vuHK4sLQdyT93k/rfrUwdJ8/HZ+nMUOyGdpeGA1Wxt65zYwi3oEl9kOM+RswvYq90fzkNDahPS8K0OIg=="], + "@smithy/util-defaults-mode-browser": ["@smithy/util-defaults-mode-browser@4.4.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-e9TZwwffgUFLgafwzyx4wNnxtgbipHG515xHmurkmjtuyJyCRkAn+Tbd4+iF/fnoCyeJXsZAzPX/OSo2nW9XOQ=="], - "@smithy/util-middleware": ["@smithy/util-middleware@4.2.14", "", { "dependencies": { "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-1Su2vj9RYNDEv/V+2E+jXkkwGsgR7dc4sfHn9Z7ruzQHJIEni9zzw5CauvRXlFJfmgcqYP8fWa0dkh2Q2YaQyw=="], + "@smithy/util-defaults-mode-node": ["@smithy/util-defaults-mode-node@4.3.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-RsmNY73PM8iO8iRreeXxE3MwY/kRRvBlbJpfm3VKMJc6MQ7vN+LulWj+VrDh+Wf5jCLdQyP43OrTSEH8d/lpCA=="], - "@smithy/util-retry": ["@smithy/util-retry@4.3.8", "", { "dependencies": { "@smithy/service-error-classification": "^4.3.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-LUIxbTBi+OpvXpg91poGA6BdyoleMDLnfXjVDqyi2RvZmTveY5loE/FgYUBCR5LU2BThW2SoZRh8dTIIy38IPw=="], + "@smithy/util-endpoints": ["@smithy/util-endpoints@3.5.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-3qGB71aqXTJnNqNlOh3R9u+9nVbR3qgd7BhtBj1Na/fgD/KAJ/+nkUHt/CGmWyEa+P1Jl361hRO2zwlMvuKJGA=="], - "@smithy/util-stream": ["@smithy/util-stream@4.5.25", "", { "dependencies": { "@smithy/fetch-http-handler": "^5.3.17", "@smithy/node-http-handler": "^4.6.1", "@smithy/types": "^4.14.1", "@smithy/util-base64": "^4.3.2", "@smithy/util-buffer-from": "^4.2.2", "@smithy/util-hex-encoding": "^4.2.2", "@smithy/util-utf8": "^4.2.2", "tslib": "^2.6.2" } }, "sha512-/PFpG4k8Ze8Ei+mMKj3oiPICYekthuzePZMgZbCqMiXIHHf4n2aZ4Ps0aSRShycFTGuj/J6XldmC0x0DwednIA=="], + "@smithy/util-middleware": ["@smithy/util-middleware@4.3.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-0GFlqDcWjnJT+TsAj91L4iTPvpR7VySjsALxtGROZaIfR03LLM69nmJDr3V/GjhZfyv/DWlFEnAWyaoKkmby1g=="], - "@smithy/util-uri-escape": ["@smithy/util-uri-escape@4.2.2", "", { "dependencies": { "tslib": "^2.6.2" } }, "sha512-2kAStBlvq+lTXHyAZYfJRb/DfS3rsinLiwb+69SstC9Vb0s9vNWkRwpnj918Pfi85mzi42sOqdV72OLxWAISnw=="], + "@smithy/util-retry": ["@smithy/util-retry@4.4.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-CwOWVLoMWyeHxaFM9a6jpq47yFKx2awaa8oFzQ6e+c5qlIATVc8MX0aN2PGuTgCaTZw//BDgHSjdAFaSbdbJRg=="], - "@smithy/util-utf8": ["@smithy/util-utf8@4.2.2", "", { "dependencies": { "@smithy/util-buffer-from": "^4.2.2", "tslib": "^2.6.2" } }, "sha512-75MeYpjdWRe8M5E3AW0O4Cx3UadweS+cwdXjwYGBW5h/gxxnbeZ877sLPX/ZJA9GVTlL/qG0dXP29JWFCD1Ayw=="], + "@smithy/util-stream": ["@smithy/util-stream@4.6.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-b5kyVsNM3pU36cJGyxwAQajGxs4JkNuaKKNufDu7oASWmzKG5gtXEdVK1rvz99O2JV1B85Pp9bAcN7fXWbN6Aw=="], - "@smithy/util-waiter": ["@smithy/util-waiter@4.3.0", "", { "dependencies": { "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-JyjYmLAfS+pdxF92o4yLgEoy0zhayKTw73FU1aofLWwLcJw7iSqIY2exGmMTrl/lmZugP5p/zxdFSippJDfKWA=="], + "@smithy/util-utf8": ["@smithy/util-utf8@4.3.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-Bswgu9zDwZRp5HBQKIaW6QrKrbwQ9RuOyAg2adsIjJTHA2SmE0XXBk+mYP82Ay6I3XzkY5lM7K9R0XZDFzJ3tw=="], - "@smithy/uuid": ["@smithy/uuid@1.1.2", "", { "dependencies": { "tslib": "^2.6.2" } }, "sha512-O/IEdcCUKkubz60tFbGA7ceITTAJsty+lBjNoorP4Z6XRqaFb/OjQjZODophEcuq68nKm6/0r+6/lLQ+XVpk8g=="], + "@smithy/util-waiter": ["@smithy/util-waiter@4.4.2", "", { "dependencies": { "@smithy/core": "^3.24.2", "tslib": "^2.6.2" } }, "sha512-2lAaVh9CAMVi0yI5nGX8nFeZed2v3CstXqX9sqYcIo2OjABSziqk/o/xekYsNHI4nwLzXeTazcm+eN7DpdSbLA=="], "@socket.io/component-emitter": ["@socket.io/component-emitter@3.1.2", "", {}, "sha512-9BCxFwvbGg/RsZK9tjXd8s4UcwR0MWeFQ1XEKIQVVvAGJyINdrqKMcTRyLoK8Rse1GjzLV9cwjWV1olXRWEXVA=="], @@ -2204,21 +2200,21 @@ "@storybook/addon-docs": ["@storybook/addon-docs@10.2.10", "", { "dependencies": { "@mdx-js/react": "^3.0.0", "@storybook/csf-plugin": "10.2.10", "@storybook/icons": "^2.0.1", "@storybook/react-dom-shim": "10.2.10", "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "react-dom": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "ts-dedent": "^2.0.0" }, "peerDependencies": { "storybook": "^10.2.10" } }, "sha512-2wIYtdvZIzPbQ5194M5Igpy8faNbQ135nuO5ZaZ2VuttqGr+IJcGnDP42zYwbAsGs28G8ohpkbSgIzVyJWUhPQ=="], - "@storybook/addon-links": ["@storybook/addon-links@10.3.6", "", { "dependencies": { "@storybook/global": "^5.0.0" }, "peerDependencies": { "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "storybook": "^10.3.6" }, "optionalPeers": ["react"] }, "sha512-tv9Xd68qRGBAvEubaxNo3FuFq4GwuMiBriD+gLGuFK0+/u3cnkuA264aoR1v6YCH3sT3er3+MBimuyKM3jLDxg=="], + "@storybook/addon-links": ["@storybook/addon-links@10.4.0", "", { "dependencies": { "@storybook/global": "^5.0.0" }, "peerDependencies": { "@types/react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "storybook": "^10.4.0" }, "optionalPeers": ["@types/react", "react"] }, "sha512-+NE1NGDoZD7U5XBEuIJvmh/fxjaVxfTxAYMWHcpwb6Qqx9Ew7gYVou5pKpiweW1wjbh+xScIVg0nPw+WyBCsyg=="], - "@storybook/addon-onboarding": ["@storybook/addon-onboarding@10.3.6", "", { "peerDependencies": { "storybook": "^10.3.6" } }, "sha512-Tys9eOFzCkBygfDWVRa2hpTQI5Y1HQt9ybkJIHdR16GASQ3fhyXVULHVRmGQLEGN+3n84JGvlU8CN9S/OBB1IQ=="], + "@storybook/addon-onboarding": ["@storybook/addon-onboarding@10.4.0", "", { "peerDependencies": { "storybook": "^10.4.0" } }, "sha512-y7YNj8uI+l8A/Fa05lfvsSxI1Tjz7pT03WHSY+LnA0uZHWkw/mP4PXD0mEp6E/lSby4cviwpcg30wafaQe7TbA=="], "@storybook/addon-themes": ["@storybook/addon-themes@10.2.10", "", { "dependencies": { "ts-dedent": "^2.0.0" }, "peerDependencies": { "storybook": "^10.2.10" } }, "sha512-j7ixCgzpWeTU7K4BkNHtEg3NdmRg9YW7ynvv0OjD3vaz4+FUVWOq7PPwb3SktLS1tOl4UA13IpApD8nSpBiY6A=="], - "@storybook/addon-vitest": ["@storybook/addon-vitest@10.3.6", "", { "dependencies": { "@storybook/global": "^5.0.0", "@storybook/icons": "^2.0.1" }, "peerDependencies": { "@vitest/browser": "^3.0.0 || ^4.0.0", "@vitest/browser-playwright": "^4.0.0", "@vitest/runner": "^3.0.0 || ^4.0.0", "storybook": "^10.3.6", "vitest": "^3.0.0 || ^4.0.0" }, "optionalPeers": ["@vitest/browser", "@vitest/browser-playwright", "@vitest/runner", "vitest"] }, "sha512-HXj7RrPJY+xzoNjL+xZu2oLw1fI5BA87Noh1NAXMPuECHR5R5fuRM/tTsJuIGXHFMO06FjSi/rekDIfCj1fL4w=="], + "@storybook/addon-vitest": ["@storybook/addon-vitest@10.4.0", "", { "dependencies": { "@storybook/global": "^5.0.0", "@storybook/icons": "^2.0.2" }, "peerDependencies": { "@vitest/browser": "^3.0.0 || ^4.0.0", "@vitest/browser-playwright": "^4.0.0", "@vitest/runner": "^3.0.0 || ^4.0.0", "storybook": "^10.4.0", "vitest": "^3.0.0 || ^4.0.0" }, "optionalPeers": ["@vitest/browser", "@vitest/browser-playwright", "@vitest/runner", "vitest"] }, "sha512-fJ3aW7EgkcZXB2k8G2VSIs/NrijCR2P+ekOKmJ23EUEjaELpKvM5PMWBOQUsPpRe+P35LcDftKVAKlo0PBEvfw=="], - "@storybook/builder-vite": ["@storybook/builder-vite@10.3.6", "", { "dependencies": { "@storybook/csf-plugin": "10.3.6", "ts-dedent": "^2.0.0" }, "peerDependencies": { "storybook": "^10.3.6", "vite": "^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0" } }, "sha512-gpvR/sE4BcrFtmQZ+Ker7zD23oQzoVeqD9nF6cK6yzY+Q0svJXyX2EPmFG4y+EwygD5/vNzDpP84gGMut8VRwg=="], + "@storybook/builder-vite": ["@storybook/builder-vite@10.4.0", "", { "dependencies": { "@storybook/csf-plugin": "10.4.0", "ts-dedent": "^2.0.0" }, "peerDependencies": { "storybook": "^10.4.0", "vite": "^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0" } }, "sha512-RCq8uzvTc0vhK2aN0y2Z48DJ9Q7oKXh8A5pdU3YAmkgMcX/+Vi3Ju1nmueLrGIO+tKwYGpYS/ccUtscNt92rCw=="], "@storybook/csf-plugin": ["@storybook/csf-plugin@10.2.10", "", { "dependencies": { "unplugin": "^2.3.5" }, "peerDependencies": { "esbuild": "*", "rollup": "*", "storybook": "^10.2.10", "vite": "*", "webpack": "*" }, "optionalPeers": ["esbuild", "rollup", "vite", "webpack"] }, "sha512-aFvgaNDAnKMjuyhPK5ialT22pPqMN0XfPBNPeeNVPYztngkdKBa8WFqF/umDd47HxAjebq+vn6uId1xHyOHH3g=="], "@storybook/global": ["@storybook/global@5.0.0", "", {}, "sha512-FcOqPAXACP0I3oJ/ws6/rrPT9WGhu915Cg8D02a9YxLo0DE9zI+a9A5gRGvmQ09fiWPukqI8ZAEoQEdWUKMQdQ=="], - "@storybook/icons": ["@storybook/icons@2.0.1", "", { "peerDependencies": { "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "react-dom": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" } }, "sha512-/smVjw88yK3CKsiuR71vNgWQ9+NuY2L+e8X7IMrFjexjm6ZR8ULrV2DRkTA61aV6ryefslzHEGDInGpnNeIocg=="], + "@storybook/icons": ["@storybook/icons@2.0.2", "", { "peerDependencies": { "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "react-dom": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" } }, "sha512-KZBCpXsshAIjczYNXR/rlxEtCUX/eAbpFNwKi8bcOomrLA4t/SyPz5RF+lVPO2oZBUE4sAkt43mfJUevQDSEEw=="], "@storybook/react-dom-shim": ["@storybook/react-dom-shim@10.2.10", "", { "peerDependencies": { "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "react-dom": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "storybook": "^10.2.10" } }, "sha512-TmBrhyLHn8B8rvDHKk5uW5BqzO1M1T+fqFNWg88NIAJOoyX4Uc90FIJjDuN1OJmWKGwB5vLmPwaKBYsTe1yS+w=="], @@ -2226,35 +2222,35 @@ "@szmarczak/http-timer": ["@szmarczak/http-timer@4.0.6", "", { "dependencies": { "defer-to-connect": "^2.0.0" } }, "sha512-4BAffykYOgO+5nzBWYwE3W90sBgLJoUPRWWcL8wlyiM8IB8ipJz3UMJ9KXQd1RKQXpKp8Tutn80HZtWsu2u76w=="], - "@tailwindcss/node": ["@tailwindcss/node@4.2.4", "", { "dependencies": { "@jridgewell/remapping": "^2.3.5", "enhanced-resolve": "^5.19.0", "jiti": "^2.6.1", "lightningcss": "1.32.0", "magic-string": "^0.30.21", "source-map-js": "^1.2.1", "tailwindcss": "4.2.4" } }, "sha512-Ai7+yQPxz3ddrDQzFfBKdHEVBg0w3Zl83jnjuwxnZOsnH9pGn93QHQtpU0p/8rYWxvbFZHneni6p1BSLK4DkGA=="], + "@tailwindcss/node": ["@tailwindcss/node@4.3.0", "", { "dependencies": { "@jridgewell/remapping": "^2.3.5", "enhanced-resolve": "^5.21.0", "jiti": "^2.6.1", "lightningcss": "1.32.0", "magic-string": "^0.30.21", "source-map-js": "^1.2.1", "tailwindcss": "4.3.0" } }, "sha512-aFb4gUhFOgdh9AXo4IzBEOzBkkAxm9VigwDJnMIYv3lcfXCJVesNfbEaBl4BNgVRyid92AmdviqwBUBRKSeY3g=="], - "@tailwindcss/oxide": ["@tailwindcss/oxide@4.2.4", "", { "optionalDependencies": { "@tailwindcss/oxide-android-arm64": "4.2.4", "@tailwindcss/oxide-darwin-arm64": "4.2.4", "@tailwindcss/oxide-darwin-x64": "4.2.4", "@tailwindcss/oxide-freebsd-x64": "4.2.4", "@tailwindcss/oxide-linux-arm-gnueabihf": "4.2.4", "@tailwindcss/oxide-linux-arm64-gnu": "4.2.4", "@tailwindcss/oxide-linux-arm64-musl": "4.2.4", "@tailwindcss/oxide-linux-x64-gnu": "4.2.4", "@tailwindcss/oxide-linux-x64-musl": "4.2.4", "@tailwindcss/oxide-wasm32-wasi": "4.2.4", "@tailwindcss/oxide-win32-arm64-msvc": "4.2.4", "@tailwindcss/oxide-win32-x64-msvc": "4.2.4" } }, "sha512-9El/iI069DKDSXwTvB9J4BwdO5JhRrOweGaK25taBAvBXyXqJAX+Jqdvs8r8gKpsI/1m0LeJLyQYTf/WLrBT1Q=="], + "@tailwindcss/oxide": ["@tailwindcss/oxide@4.3.0", "", { "optionalDependencies": { "@tailwindcss/oxide-android-arm64": "4.3.0", "@tailwindcss/oxide-darwin-arm64": "4.3.0", "@tailwindcss/oxide-darwin-x64": "4.3.0", "@tailwindcss/oxide-freebsd-x64": "4.3.0", "@tailwindcss/oxide-linux-arm-gnueabihf": "4.3.0", "@tailwindcss/oxide-linux-arm64-gnu": "4.3.0", "@tailwindcss/oxide-linux-arm64-musl": "4.3.0", "@tailwindcss/oxide-linux-x64-gnu": "4.3.0", "@tailwindcss/oxide-linux-x64-musl": "4.3.0", "@tailwindcss/oxide-wasm32-wasi": "4.3.0", "@tailwindcss/oxide-win32-arm64-msvc": "4.3.0", "@tailwindcss/oxide-win32-x64-msvc": "4.3.0" } }, "sha512-F7HZGBeN9I0/AuuJS5PwcD8xayx5ri5GhjYUDBEVYUkexyA/giwbDNjRVrxSezE3T250OU2K/wp/ltWx3UOefg=="], - "@tailwindcss/oxide-android-arm64": ["@tailwindcss/oxide-android-arm64@4.2.4", "", { "os": "android", "cpu": "arm64" }, "sha512-e7MOr1SAn9U8KlZzPi1ZXGZHeC5anY36qjNwmZv9pOJ8E4Q6jmD1vyEHkQFmNOIN7twGPEMXRHmitN4zCMN03g=="], + "@tailwindcss/oxide-android-arm64": ["@tailwindcss/oxide-android-arm64@4.3.0", "", { "os": "android", "cpu": "arm64" }, "sha512-TJPiq67tKlLuObP6RkwvVGDoxCMBVtDgKkLfa/uyj7/FyxvQwHS+UOnVrXXgbEsfUaMgiVvC4KbJnRr26ho4Ng=="], - "@tailwindcss/oxide-darwin-arm64": ["@tailwindcss/oxide-darwin-arm64@4.2.4", "", { "os": "darwin", "cpu": "arm64" }, "sha512-tSC/Kbqpz/5/o/C2sG7QvOxAKqyd10bq+ypZNf+9Fi2TvbVbv1zNpcEptcsU7DPROaSbVgUXmrzKhurFvo5eDg=="], + "@tailwindcss/oxide-darwin-arm64": ["@tailwindcss/oxide-darwin-arm64@4.3.0", "", { "os": "darwin", "cpu": "arm64" }, "sha512-oMN/WZRb+SO37BmUElEgeEWuU8E/HXRkiODxJxLe1UTHVXLrdVSgfaJV7pSlhRGMSOiXLuxTIjfsF3wYvz8cgQ=="], - "@tailwindcss/oxide-darwin-x64": ["@tailwindcss/oxide-darwin-x64@4.2.4", "", { "os": "darwin", "cpu": "x64" }, "sha512-yPyUXn3yO/ufR6+Kzv0t4fCg2qNr90jxXc5QqBpjlPNd0NqyDXcmQb/6weunH/MEDXW5dhyEi+agTDiqa3WsGg=="], + "@tailwindcss/oxide-darwin-x64": ["@tailwindcss/oxide-darwin-x64@4.3.0", "", { "os": "darwin", "cpu": "x64" }, "sha512-N6CUmu4a6bKVADfw77p+iw6Yd9Q3OBhe0veaDX+QazfuVYlQsHfDgxBrsjQ/IW+zywL8mTrNd0SdJT/zgtvMdA=="], - "@tailwindcss/oxide-freebsd-x64": ["@tailwindcss/oxide-freebsd-x64@4.2.4", "", { "os": "freebsd", "cpu": "x64" }, "sha512-BoMIB4vMQtZsXdGLVc2z+P9DbETkiopogfWZKbWwM8b/1Vinbs4YcUwo+kM/KeLkX3Ygrf4/PsRndKaYhS8Eiw=="], + "@tailwindcss/oxide-freebsd-x64": ["@tailwindcss/oxide-freebsd-x64@4.3.0", "", { "os": "freebsd", "cpu": "x64" }, "sha512-zDL5hBkQdH5C6MpqbK3gQAgP80tsMwSI26vjOzjJtNCMUo0lFgOItzHKBIupOZNQxt3ouPH7RPhvNhiTfCe5CQ=="], - "@tailwindcss/oxide-linux-arm-gnueabihf": ["@tailwindcss/oxide-linux-arm-gnueabihf@4.2.4", "", { "os": "linux", "cpu": "arm" }, "sha512-7pIHBLTHYRAlS7V22JNuTh33yLH4VElwKtB3bwchK/UaKUPpQ0lPQiOWcbm4V3WP2I6fNIJ23vABIvoy2izdwA=="], + "@tailwindcss/oxide-linux-arm-gnueabihf": ["@tailwindcss/oxide-linux-arm-gnueabihf@4.3.0", "", { "os": "linux", "cpu": "arm" }, "sha512-R06HdNi7A7OEoMsf6d4tjZ71RCWnZQPHj2mnotSFURjNLdBC+cIgXQ7l81CqeoiQftjf6OOblxXMInMgN2VzMA=="], - "@tailwindcss/oxide-linux-arm64-gnu": ["@tailwindcss/oxide-linux-arm64-gnu@4.2.4", "", { "os": "linux", "cpu": "arm64" }, "sha512-+E4wxJ0ZGOzSH325reXTWB48l42i93kQqMvDyz5gqfRzRZ7faNhnmvlV4EPGJU3QJM/3Ab5jhJ5pCRUsKn6OQw=="], + "@tailwindcss/oxide-linux-arm64-gnu": ["@tailwindcss/oxide-linux-arm64-gnu@4.3.0", "", { "os": "linux", "cpu": "arm64" }, "sha512-qTJHELX8jetjhRQHCLilkVLmybpzNQAtaI/gaoVoidn/ufbNDbAo8KlK2J+yPoc8wQxvDxCmh/5lr8nC1+lTbg=="], - "@tailwindcss/oxide-linux-arm64-musl": ["@tailwindcss/oxide-linux-arm64-musl@4.2.4", "", { "os": "linux", "cpu": "arm64" }, "sha512-bBADEGAbo4ASnppIziaQJelekCxdMaxisrk+fB7Thit72IBnALp9K6ffA2G4ruj90G9XRS2VQ6q2bCKbfFV82g=="], + "@tailwindcss/oxide-linux-arm64-musl": ["@tailwindcss/oxide-linux-arm64-musl@4.3.0", "", { "os": "linux", "cpu": "arm64" }, "sha512-Z6sukiQsngnWO+l39X4pPbiWT81IC+PLKF+PHxIlyZbGNb9MODfYlXEVlFvej5BOZInWX01kVyzeLvHsXhfczQ=="], - "@tailwindcss/oxide-linux-x64-gnu": ["@tailwindcss/oxide-linux-x64-gnu@4.2.4", "", { "os": "linux", "cpu": "x64" }, "sha512-7Mx25E4WTfnht0TVRTyC00j3i0M+EeFe7wguMDTlX4mRxafznw0CA8WJkFjWYH5BlgELd1kSjuU2JiPnNZbJDA=="], + "@tailwindcss/oxide-linux-x64-gnu": ["@tailwindcss/oxide-linux-x64-gnu@4.3.0", "", { "os": "linux", "cpu": "x64" }, "sha512-DRNdQRpSGzRGfARVuVkxvM8Q12nh19l4BF/G7zGA1oe+9wcC6saFBHTISrpIcKzhiXtSrlSrluCfvMuledoCTQ=="], - "@tailwindcss/oxide-linux-x64-musl": ["@tailwindcss/oxide-linux-x64-musl@4.2.4", "", { "os": "linux", "cpu": "x64" }, "sha512-2wwJRF7nyhOR0hhHoChc04xngV3iS+akccHTGtz965FwF0up4b2lOdo6kI1EbDaEXKgvcrFBYcYQQ/rrnWFVfA=="], + "@tailwindcss/oxide-linux-x64-musl": ["@tailwindcss/oxide-linux-x64-musl@4.3.0", "", { "os": "linux", "cpu": "x64" }, "sha512-Z0IADbDo8bh6I7h2IQMx601AdXBLfFpEdUotft86evd/8ZPflZe9COPO8Q1vw+pfLWIUo9zN/JGZvwuAJqduqg=="], - "@tailwindcss/oxide-wasm32-wasi": ["@tailwindcss/oxide-wasm32-wasi@4.2.4", "", { "dependencies": { "@emnapi/core": "^1.8.1", "@emnapi/runtime": "^1.8.1", "@emnapi/wasi-threads": "^1.1.0", "@napi-rs/wasm-runtime": "^1.1.1", "@tybys/wasm-util": "^0.10.1", "tslib": "^2.8.1" }, "cpu": "none" }, "sha512-FQsqApeor8Fo6gUEklzmaa9994orJZZDBAlQpK2Mq+DslRKFJeD6AjHpBQ0kZFQohVr8o85PPh8eOy86VlSCmw=="], + "@tailwindcss/oxide-wasm32-wasi": ["@tailwindcss/oxide-wasm32-wasi@4.3.0", "", { "dependencies": { "@emnapi/core": "^1.10.0", "@emnapi/runtime": "^1.10.0", "@emnapi/wasi-threads": "^1.2.1", "@napi-rs/wasm-runtime": "^1.1.4", "@tybys/wasm-util": "^0.10.1", "tslib": "^2.8.1" }, "cpu": "none" }, "sha512-HNZGOUxEmElksYR7S6sC5jTeNGpobAsy9u7Gu0AskJ8/20FR9GqebUyB+HBcU/ax6BHuiuJi+Oda4B+YX6H1yA=="], - "@tailwindcss/oxide-win32-arm64-msvc": ["@tailwindcss/oxide-win32-arm64-msvc@4.2.4", "", { "os": "win32", "cpu": "arm64" }, "sha512-L9BXqxC4ToVgwMFqj3pmZRqyHEztulpUJzCxUtLjobMCzTPsGt1Fa9enKbOpY2iIyVtaHNeNvAK8ERP/64sqGQ=="], + "@tailwindcss/oxide-win32-arm64-msvc": ["@tailwindcss/oxide-win32-arm64-msvc@4.3.0", "", { "os": "win32", "cpu": "arm64" }, "sha512-Pe+RPVTi1T+qymuuRpcdvwSVZjnll/f7n8gBxMMh3xLTctMDKqpdfGimbMyioqtLhUYZxdJ9wGNhV7MKHvgZsQ=="], - "@tailwindcss/oxide-win32-x64-msvc": ["@tailwindcss/oxide-win32-x64-msvc@4.2.4", "", { "os": "win32", "cpu": "x64" }, "sha512-ESlKG0EpVJQwRjXDDa9rLvhEAh0mhP1sF7sap9dNZT0yyl9SAG6T7gdP09EH0vIv0UNTlo6jPWyujD6559fZvw=="], + "@tailwindcss/oxide-win32-x64-msvc": ["@tailwindcss/oxide-win32-x64-msvc@4.3.0", "", { "os": "win32", "cpu": "x64" }, "sha512-Mvrf2kXW/yeW/OTezZlCGOirXRcUuLIBx/5Y12BaPM7wJoryG6dfS/NJL8aBPqtTEx/Vm4T4vKzFUcKDT+TKUA=="], - "@tailwindcss/postcss": ["@tailwindcss/postcss@4.2.4", "", { "dependencies": { "@alloc/quick-lru": "^5.2.0", "@tailwindcss/node": "4.2.4", "@tailwindcss/oxide": "4.2.4", "postcss": "^8.5.6", "tailwindcss": "4.2.4" } }, "sha512-wgAVj6nUWAolAu8YFvzT2cTBIElWHkjZwFYovF+xsqKsW2ADxM/X2opxj5NsF/qVccAOjRNe8X2IdPzMsWyHTg=="], + "@tailwindcss/postcss": ["@tailwindcss/postcss@4.3.0", "", { "dependencies": { "@alloc/quick-lru": "^5.2.0", "@tailwindcss/node": "4.3.0", "@tailwindcss/oxide": "4.3.0", "postcss": "^8.5.10", "tailwindcss": "4.3.0" } }, "sha512-Jm05Tjx+9yCLGv5qw1c+84Psds8MnyrEQYCB+FFk2lgGiUjlRqdxke4mVTuYrj2xnVZqKim2Apr5ySuQRYAw/w=="], "@tailwindcss/vite": ["@tailwindcss/vite@4.1.11", "", { "dependencies": { "@tailwindcss/node": "4.1.11", "@tailwindcss/oxide": "4.1.11", "tailwindcss": "4.1.11" }, "peerDependencies": { "vite": "^5.2.0 || ^6 || ^7" } }, "sha512-RHYhrR3hku0MJFRV+fN2gNbDNEh3dwKvY8XJvTxCSXeMOsCRSr+uKvDWQcbizrHgjML6ZmTE5OwMrl5wKcujCw=="], @@ -2264,29 +2260,29 @@ "@tauri-apps/api": ["@tauri-apps/api@2.11.0", "", {}, "sha512-7CinYODhky9lmO23xHnUFv0Xt43fbtWMyxZcLcRBlFkcgXKuEirBvHpmtJ89YMhyeGcq20Wuc47Fa4XjyniywA=="], - "@tauri-apps/cli": ["@tauri-apps/cli@2.11.0", "", { "optionalDependencies": { "@tauri-apps/cli-darwin-arm64": "2.11.0", "@tauri-apps/cli-darwin-x64": "2.11.0", "@tauri-apps/cli-linux-arm-gnueabihf": "2.11.0", "@tauri-apps/cli-linux-arm64-gnu": "2.11.0", "@tauri-apps/cli-linux-arm64-musl": "2.11.0", "@tauri-apps/cli-linux-riscv64-gnu": "2.11.0", "@tauri-apps/cli-linux-x64-gnu": "2.11.0", "@tauri-apps/cli-linux-x64-musl": "2.11.0", "@tauri-apps/cli-win32-arm64-msvc": "2.11.0", "@tauri-apps/cli-win32-ia32-msvc": "2.11.0", "@tauri-apps/cli-win32-x64-msvc": "2.11.0" }, "bin": { "tauri": "tauri.js" } }, "sha512-W5Wbuqsb2pHFPTj4TaRNKTj5rwXhDShPiLSY9T18y4ouSR/NNCptAEFxFsBtyNRgL6Vs1a/q9LzfqqYzEwC+Jw=="], + "@tauri-apps/cli": ["@tauri-apps/cli@2.11.1", "", { "optionalDependencies": { "@tauri-apps/cli-darwin-arm64": "2.11.1", "@tauri-apps/cli-darwin-x64": "2.11.1", "@tauri-apps/cli-linux-arm-gnueabihf": "2.11.1", "@tauri-apps/cli-linux-arm64-gnu": "2.11.1", "@tauri-apps/cli-linux-arm64-musl": "2.11.1", "@tauri-apps/cli-linux-riscv64-gnu": "2.11.1", "@tauri-apps/cli-linux-x64-gnu": "2.11.1", "@tauri-apps/cli-linux-x64-musl": "2.11.1", "@tauri-apps/cli-win32-arm64-msvc": "2.11.1", "@tauri-apps/cli-win32-ia32-msvc": "2.11.1", "@tauri-apps/cli-win32-x64-msvc": "2.11.1" }, "bin": { "tauri": "tauri.js" } }, "sha512-rpEbaJ/HzNb6fwsquwoAbq29/Vt4gADhS423A8fdkwL4edJ0wZmoB8ar7O6JPDL834MUKOCm/rrJ7c9oAaEaYQ=="], - "@tauri-apps/cli-darwin-arm64": ["@tauri-apps/cli-darwin-arm64@2.11.0", "", { "os": "darwin", "cpu": "arm64" }, "sha512-UfMeDNlgIP252rm/KSTuu8yHatPua5TjtUEUf+jyIzVwBNcIl7Ywkdpfj+e5jVVg3EfCTp+4gwuL1dNpgF8clg=="], + "@tauri-apps/cli-darwin-arm64": ["@tauri-apps/cli-darwin-arm64@2.11.1", "", { "os": "darwin", "cpu": "arm64" }, "sha512-6eEKMBXsQPCuM1EmvrjT2+aBuxWQuFdKdW8pzNuNQtpq45nEEpBlD5gr8pUeAyOU1DQKlkFaEc/MPBxb/Pfjtg=="], - "@tauri-apps/cli-darwin-x64": ["@tauri-apps/cli-darwin-x64@2.11.0", "", { "os": "darwin", "cpu": "x64" }, "sha512-lY1+aPlgyMN7vgjtCdQ3+WODfZkebAcxnrCrO0HjqDpKSXieDkrJbimqeaoM4RwhTSrCLRHfVYiYrfE5E131tg=="], + "@tauri-apps/cli-darwin-x64": ["@tauri-apps/cli-darwin-x64@2.11.1", "", { "os": "darwin", "cpu": "x64" }, "sha512-LQUO7exfRWjWALNhetph5guWpMeHphRpokOLk0OIbTTExaNwJNFu3I4vb+CCM/4G/QGoZe/5XikZOJdNEFP1ig=="], - "@tauri-apps/cli-linux-arm-gnueabihf": ["@tauri-apps/cli-linux-arm-gnueabihf@2.11.0", "", { "os": "linux", "cpu": "arm" }, "sha512-5uCP0AusgN3NrKC8EpkuJwjek1k8pEffBdugJSpXPey/QGbPEb8vZ542n/giJ2mZPjMSllDkdhG2QIDpBY4PpQ=="], + "@tauri-apps/cli-linux-arm-gnueabihf": ["@tauri-apps/cli-linux-arm-gnueabihf@2.11.1", "", { "os": "linux", "cpu": "arm" }, "sha512-5i/awiBCRRhOUG8yjn0fMHXIWD5Ez8eEk5LtvOxyQrKuJkRaZDvnbIjZbE183blAwkoA4xN3aO/prJiqscl02Q=="], - "@tauri-apps/cli-linux-arm64-gnu": ["@tauri-apps/cli-linux-arm64-gnu@2.11.0", "", { "os": "linux", "cpu": "arm64" }, "sha512-loDPqtRHMSbIcrH2VBd4GgHoQlF7jJnrZj7MxA2lj1cixS/jEgMAPFqj83U6Wvjete4HfYplbE/gCpSFifA9jw=="], + "@tauri-apps/cli-linux-arm64-gnu": ["@tauri-apps/cli-linux-arm64-gnu@2.11.1", "", { "os": "linux", "cpu": "arm64" }, "sha512-9LrwDw3S9Fygtw/Q6WDhOP+3svJRGAsejeE+GKrc0eO1ThMVhwi2LL6hw4dlKw93IfS7VY1G19sWGxJ/NcU4nA=="], - "@tauri-apps/cli-linux-arm64-musl": ["@tauri-apps/cli-linux-arm64-musl@2.11.0", "", { "os": "linux", "cpu": "arm64" }, "sha512-DtSE8ZBlB9H+L+eHkfZ3myt00EVEyAB3e41juEHoE2qT88fgVlJvyrwa9SZYc/xTwCS9TnmK+R84tpg+ZsAg7Q=="], + "@tauri-apps/cli-linux-arm64-musl": ["@tauri-apps/cli-linux-arm64-musl@2.11.1", "", { "os": "linux", "cpu": "arm64" }, "sha512-mNA5dbbqPqDUdTIwdUYYuhO2GvIe9UnB2r0VU2njxBOS3Opbx4gKNC5yP0Iu4rYmEmqdlwry9VzGZQ3wq9dyFg=="], - "@tauri-apps/cli-linux-riscv64-gnu": ["@tauri-apps/cli-linux-riscv64-gnu@2.11.0", "", { "os": "linux", "cpu": "none" }, "sha512-5QdgS4LD+kntClI1aj2JmwjW38LosNXxwCe8viIHEwqYIWuMPdNEIau6/cLogI38Yzx9DnfCPRfEWLyI+5li8Q=="], + "@tauri-apps/cli-linux-riscv64-gnu": ["@tauri-apps/cli-linux-riscv64-gnu@2.11.1", "", { "os": "linux", "cpu": "none" }, "sha512-fZj3Gwq+6fUs305T5WQiD5iSGJw+j/4w/HGmk4sHDAcy+rp9zU5eaxB7nOyz5/I/nkNAuKPqfp6uIbiUBXkBCw=="], - "@tauri-apps/cli-linux-x64-gnu": ["@tauri-apps/cli-linux-x64-gnu@2.11.0", "", { "os": "linux", "cpu": "x64" }, "sha512-5UynPXo3Zq9khjVdAbD+YogeLltdVUeOah2ioSIM3tu6H7wY9vMy6rgGJhv9r5R8ZXmk9GttMippdqYJWrnLnA=="], + "@tauri-apps/cli-linux-x64-gnu": ["@tauri-apps/cli-linux-x64-gnu@2.11.1", "", { "os": "linux", "cpu": "x64" }, "sha512-XFxGxOvHM7jjeD6ozCKdGfhzJ7lERYDGZl1/Kb4fsvchaJsfLJ981TlyTG8Qy/gFq+f5GitH3bfrX9JAkjPEyw=="], - "@tauri-apps/cli-linux-x64-musl": ["@tauri-apps/cli-linux-x64-musl@2.11.0", "", { "os": "linux", "cpu": "x64" }, "sha512-CNz7fHbApz1Zyhhq73jtGn9JqgNEV/lIWnTnUo6h6ujw+mHsTmkLszvJSM8W6JBaDjNpTTFr/RSNoVL5FMwcTg=="], + "@tauri-apps/cli-linux-x64-musl": ["@tauri-apps/cli-linux-x64-musl@2.11.1", "", { "os": "linux", "cpu": "x64" }, "sha512-d5C2/Zm+68v7R9wTuTCjRQEVrWjcdMkJBZ1+rXse+QdMMlTB9+u9PDNDLw9PQflWxYLaYZ7tjxxL9Nb9II6PbA=="], - "@tauri-apps/cli-win32-arm64-msvc": ["@tauri-apps/cli-win32-arm64-msvc@2.11.0", "", { "os": "win32", "cpu": "arm64" }, "sha512-K+br+VXZ+Xx0n/9FdWohpW5Ugq+2FQUpJScqcPl1hTxXfh3fgjYgt4qA2NgrjlJo+zZPNrmUMl+NLvm0ufEqBQ=="], + "@tauri-apps/cli-win32-arm64-msvc": ["@tauri-apps/cli-win32-arm64-msvc@2.11.1", "", { "os": "win32", "cpu": "arm64" }, "sha512-YdeVWFAR1pTXzUU6NLstPq4G6OLxuDrXCXEBdmBH+5EZIDXUx0D2kJlz3+YjpazkKvAzYpgziTsyRagls0OfRQ=="], - "@tauri-apps/cli-win32-ia32-msvc": ["@tauri-apps/cli-win32-ia32-msvc@2.11.0", "", { "os": "win32", "cpu": "ia32" }, "sha512-OFV+s3MLZnd75zl0ZAFU5riMpGK4waUEA8ZDuijDsnkU0btz/gHhqh5jVlOn8thyvgdtT3Xyoxqo099MMifH3g=="], + "@tauri-apps/cli-win32-ia32-msvc": ["@tauri-apps/cli-win32-ia32-msvc@2.11.1", "", { "os": "win32", "cpu": "ia32" }, "sha512-VBGkuH0eB9K9LLSMv361Gzr5Ou72sCS4+ztpmkWEQ+wd/amhcYOsf3X6qn1RJZDzIhiOYHJEOysZUC3baD01rA=="], - "@tauri-apps/cli-win32-x64-msvc": ["@tauri-apps/cli-win32-x64-msvc@2.11.0", "", { "os": "win32", "cpu": "x64" }, "sha512-AeDTWBd2cOZ6TX133BWsoo+LutG9o0JRcgjMsIfLE13ZugpgCMv/2dJbUiBGeRvbPOGin5A3aYmsArPVV6ZSHQ=="], + "@tauri-apps/cli-win32-x64-msvc": ["@tauri-apps/cli-win32-x64-msvc@2.11.1", "", { "os": "win32", "cpu": "x64" }, "sha512-b3ORhIAKgp9ZYY+zBt7b7r0kLU2kjvyGF0+MS2SBym3emsweGPybEqocJcmtMuxyBhkOKHP4CiuEJEDuAlTx6A=="], "@tauri-apps/plugin-clipboard-manager": ["@tauri-apps/plugin-clipboard-manager@2.3.2", "", { "dependencies": { "@tauri-apps/api": "^2.8.0" } }, "sha512-CUlb5Hqi2oZbcZf4VUyUH53XWPPdtpw43EUpCza5HWZJwxEoDowFzNUDt1tRUXA8Uq+XPn17Ysfptip33sG4eQ=="], @@ -2320,15 +2316,15 @@ "@testing-library/user-event": ["@testing-library/user-event@14.6.1", "", { "peerDependencies": { "@testing-library/dom": ">=7.21.4" } }, "sha512-vq7fv0rnt+QTXgPxr5Hjc210p6YKq2kmdziLgnsZGgLJ9e6VAShx1pACLuRjd/AS/sr7phAR58OIIpf0LlmQNw=="], - "@textlint/ast-node-types": ["@textlint/ast-node-types@15.6.0", "", {}, "sha512-CxZHFbYAU7J0A4izz31wV2ZZfySR6aVj2OSR6/3tppZm7VV6hM7nA7sutsLwIiBL/v4lsB1RM79l4Dc/VrH4qw=="], + "@textlint/ast-node-types": ["@textlint/ast-node-types@15.7.0", "", {}, "sha512-wZILFXsRf2gGK9k0Fr69GUdfuZV9CrtByga8Qkw0CLyKBBfZXdNQlJF2XdZ2Ju9ggrTbAWehGo0RjCsAHSBWtA=="], - "@textlint/linter-formatter": ["@textlint/linter-formatter@15.6.0", "", { "dependencies": { "@azu/format-text": "^1.0.2", "@azu/style-format": "^1.0.1", "@textlint/module-interop": "15.6.0", "@textlint/resolver": "15.6.0", "@textlint/types": "15.6.0", "chalk": "^4.1.2", "debug": "^4.4.3", "js-yaml": "^4.1.1", "lodash": "^4.18.1", "pluralize": "^2.0.0", "string-width": "^4.2.3", "strip-ansi": "^6.0.1", "table": "^6.9.0", "text-table": "^0.2.0" } }, "sha512-IwHRhjwxs0a5t1eNAoKAdV224CDca38LyopPofXpwO/d0J75wBvzf/cBHXNl4TMsLKhYGtR83UprcLEKj/gZsA=="], + "@textlint/linter-formatter": ["@textlint/linter-formatter@15.7.0", "", { "dependencies": { "@azu/format-text": "^1.0.2", "@azu/style-format": "^1.0.1", "@textlint/module-interop": "15.7.0", "@textlint/resolver": "15.7.0", "@textlint/types": "15.7.0", "chalk": "^4.1.2", "debug": "^4.4.3", "js-yaml": "^4.1.1", "lodash": "^4.18.1", "pluralize": "^2.0.0", "string-width": "^4.2.3", "strip-ansi": "^6.0.1", "table": "^6.9.0", "text-table": "^0.2.0" } }, "sha512-wrpDID1bfBt5XIUXtgw8NmGIuRFansWAtX1FLHv/zLRt3sgxCFnyon88SbhPOPITEgIlfFcsESElCSLzWySubQ=="], - "@textlint/module-interop": ["@textlint/module-interop@15.6.0", "", {}, "sha512-MHY6pJx9i5kOlrvUSK51887tYZjHAV2qnr6unBm7LtBLGDFo93utdYqHyWep8r9QLsilQdeijWtufJI46z4v4w=="], + "@textlint/module-interop": ["@textlint/module-interop@15.7.0", "", {}, "sha512-+VdoeGFH66OasijhoO7D3OSrqTfJNAIH2OoQHEc5StlO0dqDO2JfZStCbuBPP/ZbpJvuYdoERBP+nKqTAT24vA=="], - "@textlint/resolver": ["@textlint/resolver@15.6.0", "", {}, "sha512-T1l2Gd3455pwtm0cTewhX/LLy3bL9z6/Fu/am+jj+jjGfXVoknYkjfkZEKrjHlA7xzay0EfUKnu//teYemLeZw=="], + "@textlint/resolver": ["@textlint/resolver@15.7.0", "", {}, "sha512-f94t/8ZR97uhOu2KvBujMGGtfdoJQZLjDNN7+7PNLaTjCtGn+XrqKjSP9lzXgBhUbKSygRN8AlrCMx9S70FHKw=="], - "@textlint/types": ["@textlint/types@15.6.0", "", { "dependencies": { "@textlint/ast-node-types": "15.6.0" } }, "sha512-CvgYb1PiqF4BGyoZebGWzAJCZ4ChJAZ9gtWjpQIMKE4Xe2KlSwDA8m8MsiZIV321f5Ibx38BMjC1Z/2ZYP2GQg=="], + "@textlint/types": ["@textlint/types@15.7.0", "", { "dependencies": { "@textlint/ast-node-types": "15.7.0" } }, "sha512-soItNoFZ8Ua4WCgWwOaTEEyTkX7bjArwVxhN1F2UySeqPsP8QeRiKNUjAIfWQFHddTY6UYR1sHaEh+O0sQPv0Q=="], "@thisbeyond/solid-dnd": ["@thisbeyond/solid-dnd@0.7.5", "", { "peerDependencies": { "solid-js": "^1.5" } }, "sha512-DfI5ff+yYGpK9M21LhYwIPlbP2msKxN2ARwuu6GF8tT1GgNVDTI8VCQvH4TJFoVApP9d44izmAcTh/iTCH2UUw=="], @@ -2438,7 +2434,7 @@ "@types/diff": ["@types/diff@6.0.0", "", {}, "sha512-dhVCYGv3ZSbzmQaBSagrv1WJ6rXCdkyTcDyoNu1MD8JohI7pR7k8wdZEm+mvdxRKXyHVwckFzWU1vJc+Z29MlA=="], - "@types/estree": ["@types/estree@1.0.8", "", {}, "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w=="], + "@types/estree": ["@types/estree@1.0.9", "", {}, "sha512-GhdPgy1el4/ImP05X05Uw4cw2/M93BCUmnEvWZNStlCzEKME4Fkk+YpoA5OiHNQmoS7Cafb8Xa3Pya8m1Qrzeg=="], "@types/fs-extra": ["@types/fs-extra@9.0.13", "", { "dependencies": { "@types/node": "*" } }, "sha512-nEnwB++1u5lVDM2UI4c1+5R+FYaKfaAzS4OococimjVm3nQw3TuzH5UNsocrcTBbhnerblyHj4A49qXbIiZdpA=="], @@ -2478,7 +2474,7 @@ "@types/mssql": ["@types/mssql@9.1.11", "", { "dependencies": { "@types/node": "*", "tarn": "^3.0.1", "tedious": "*" } }, "sha512-vcujgrDbDezCxNDO4KY6gjwduLYOKfrexpRUwhoysRvcXZ3+IgZ/PMYFDgh8c3cQIxZ6skAwYo+H6ibMrBWPjQ=="], - "@types/node": ["@types/node@22.13.9", "", { "dependencies": { "undici-types": "~6.20.0" } }, "sha512-acBjXdRJ3A6Pb3tqnw9HZmyR3Fiol3aGxRCK1x3d+6CDAMjl7I649wpSd+yNURCjbOUGu9tqtLKnTGxmK6CyGw=="], + "@types/node": ["@types/node@25.7.0", "", { "dependencies": { "undici-types": "~7.21.0" } }, "sha512-z+pdZyxE+RTQE9AcboAZCb4otwcrvgHD+GlBpPgn0emDVt0ohrTMhAwlr2Wd9nZ+nihhYFxO2pThz3C5qSu2Eg=="], "@types/node-fetch": ["@types/node-fetch@2.6.13", "", { "dependencies": { "@types/node": "*", "form-data": "^4.0.4" } }, "sha512-QGpRVpzSaUs30JBSGPjOg4Uveu384erbHBoT1zeONvyCfwQxIkUshLAOqN/k9EjGviPRmWTTe6aH2qySWKTVSw=="], @@ -2528,7 +2524,7 @@ "@types/verror": ["@types/verror@1.10.11", "", {}, "sha512-RlDm9K7+o5stv0Co8i8ZRGxDbrTxhJtgjqjFyVh/tXQyl/rYtTKlnTvZ88oSTeYREWurwx20Js4kTuKCsFkUtg=="], - "@types/vscode": ["@types/vscode@1.118.0", "", {}, "sha512-Ah6eTlqDcwIMELEVwQMO++rJAFBRz/oLluLD/vWdYrH1KuI9kfpaM+7pg0OvvascgcJy+ghLCERAYouM4QbzGw=="], + "@types/vscode": ["@types/vscode@1.120.0", "", {}, "sha512-feaT4Rst+FkTch5zz/ZbNCxoIvo55YU80Be2kiL7OJcod4+CUYf2lUBPdIJzozNnSEMq1VRTGrWEcCGFB3fBmA=="], "@types/whatwg-mimetype": ["@types/whatwg-mimetype@3.0.2", "", {}, "sha512-c2AKvDT8ToxLIOUlN51gTiHXflsfIFisS4pO7pDPoKouJCESkhZnEy623gwP9laCy5lnLDAw1vAzu2vM2YLOrA=="], @@ -2542,25 +2538,25 @@ "@types/yauzl": ["@types/yauzl@2.10.3", "", { "dependencies": { "@types/node": "*" } }, "sha512-oJoftv0LSuaDZE3Le4DbKX+KS9G36NzOeSap90UIK0yMA/NhKJhqlSGtNDORNRaIbQfzjXDrQa0ytJ6mNRGz/Q=="], - "@typescript-eslint/eslint-plugin": ["@typescript-eslint/eslint-plugin@8.59.2", "", { "dependencies": { "@eslint-community/regexpp": "^4.12.2", "@typescript-eslint/scope-manager": "8.59.2", "@typescript-eslint/type-utils": "8.59.2", "@typescript-eslint/utils": "8.59.2", "@typescript-eslint/visitor-keys": "8.59.2", "ignore": "^7.0.5", "natural-compare": "^1.4.0", "ts-api-utils": "^2.5.0" }, "peerDependencies": { "@typescript-eslint/parser": "^8.59.2", "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", "typescript": ">=4.8.4 <6.1.0" } }, "sha512-j/bwmkBvHUtPNxzuWe5z6BEk3q54YRyGlBXkSsmfoih7zNrBvl5A9A98anlp/7JbyZcWIJ8KXo/3Tq/DjFLtuQ=="], + "@typescript-eslint/eslint-plugin": ["@typescript-eslint/eslint-plugin@8.59.3", "", { "dependencies": { "@eslint-community/regexpp": "^4.12.2", "@typescript-eslint/scope-manager": "8.59.3", "@typescript-eslint/type-utils": "8.59.3", "@typescript-eslint/utils": "8.59.3", "@typescript-eslint/visitor-keys": "8.59.3", "ignore": "^7.0.5", "natural-compare": "^1.4.0", "ts-api-utils": "^2.5.0" }, "peerDependencies": { "@typescript-eslint/parser": "^8.59.3", "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", "typescript": ">=4.8.4 <6.1.0" } }, "sha512-PwFvSKsXGShKGW6n5bZOhGHEcCZXM8HofLK9fNsEwZXzFRjoY+XT1Vsf1zgyXdwTr0ZYz1/2tkZ0DBTT9jZjhw=="], - "@typescript-eslint/parser": ["@typescript-eslint/parser@8.59.2", "", { "dependencies": { "@typescript-eslint/scope-manager": "8.59.2", "@typescript-eslint/types": "8.59.2", "@typescript-eslint/typescript-estree": "8.59.2", "@typescript-eslint/visitor-keys": "8.59.2", "debug": "^4.4.3" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", "typescript": ">=4.8.4 <6.1.0" } }, "sha512-plR3pp6D+SSUn1HM7xvSkx12/DhoHInI2YF35KAcVFNZvlC0gtrWqx7Qq1oH2Ssgi0vlFRCTbP+DZc7B9+TtsQ=="], + "@typescript-eslint/parser": ["@typescript-eslint/parser@8.59.3", "", { "dependencies": { "@typescript-eslint/scope-manager": "8.59.3", "@typescript-eslint/types": "8.59.3", "@typescript-eslint/typescript-estree": "8.59.3", "@typescript-eslint/visitor-keys": "8.59.3", "debug": "^4.4.3" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", "typescript": ">=4.8.4 <6.1.0" } }, "sha512-HPwA+hVkfcriajbNvTmZv4VRauibay+cWArYUYq7u7W7PmGShMxbPxLvrwDme55a6d5alG3nrYfhyJ/G28XlLg=="], - "@typescript-eslint/project-service": ["@typescript-eslint/project-service@8.59.2", "", { "dependencies": { "@typescript-eslint/tsconfig-utils": "^8.59.2", "@typescript-eslint/types": "^8.59.2", "debug": "^4.4.3" }, "peerDependencies": { "typescript": ">=4.8.4 <6.1.0" } }, "sha512-+2hqvEkeyf/0FBor67duF0Ll7Ot8jyKzDQOSrxazF/danillRq2DwR9dLptsXpoZQqxE1UisSmoZewrlPas9Vw=="], + "@typescript-eslint/project-service": ["@typescript-eslint/project-service@8.59.3", "", { "dependencies": { "@typescript-eslint/tsconfig-utils": "^8.59.3", "@typescript-eslint/types": "^8.59.3", "debug": "^4.4.3" }, "peerDependencies": { "typescript": ">=4.8.4 <6.1.0" } }, "sha512-ECiUWa/KYRGDFUqTNehaRgzDshnJfkTABJxVemHk4ko22gcr0ukloKjWvyQ64g8YCV/UI47kN1dbmjf/GaQYng=="], - "@typescript-eslint/scope-manager": ["@typescript-eslint/scope-manager@8.59.2", "", { "dependencies": { "@typescript-eslint/types": "8.59.2", "@typescript-eslint/visitor-keys": "8.59.2" } }, "sha512-JzfyEpEtOU89CcFSwyNS3mu4MLvLSXqnmX05+aKBDM+TdR5jzcGOEBwxwGNxrEQ7p/z6kK2WyioCGBf2zZBnvg=="], + "@typescript-eslint/scope-manager": ["@typescript-eslint/scope-manager@8.59.3", "", { "dependencies": { "@typescript-eslint/types": "8.59.3", "@typescript-eslint/visitor-keys": "8.59.3" } }, "sha512-t2LvZnoEfzKtnPjgeEu41xw5gxq9mQVfYy4OoZ4Vlt0sk3JwxmhCca/AR7DwOiHrjWgjAj6as4AhRLKSDfvZIA=="], - "@typescript-eslint/tsconfig-utils": ["@typescript-eslint/tsconfig-utils@8.59.2", "", { "peerDependencies": { "typescript": ">=4.8.4 <6.1.0" } }, "sha512-BKK4alN7oi4C/zv4VqHQ+uRU+lTa6JGIZ7s1juw7b3RHo9OfKB+bKX3u0iVZetdsUCBBkSbdWbarJbmN0fTeSw=="], + "@typescript-eslint/tsconfig-utils": ["@typescript-eslint/tsconfig-utils@8.59.3", "", { "peerDependencies": { "typescript": ">=4.8.4 <6.1.0" } }, "sha512-PcIJHjmaREXLgIAIzLnSY9VucEzz8FKXsRgFa1DmdGCK/5tJpW03TKJF01Q6VZd1lLdz2sIKPWaDUZN9dp//dw=="], - "@typescript-eslint/type-utils": ["@typescript-eslint/type-utils@8.59.2", "", { "dependencies": { "@typescript-eslint/types": "8.59.2", "@typescript-eslint/typescript-estree": "8.59.2", "@typescript-eslint/utils": "8.59.2", "debug": "^4.4.3", "ts-api-utils": "^2.5.0" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", "typescript": ">=4.8.4 <6.1.0" } }, "sha512-nhqaj1nmTdVVl/BP5omXNRGO38jn5iosis2vbdmupF2txCf8ylWT8lx+JlvMYYVqzGVKtjojUFoQ3JRWK+mfzQ=="], + "@typescript-eslint/type-utils": ["@typescript-eslint/type-utils@8.59.3", "", { "dependencies": { "@typescript-eslint/types": "8.59.3", "@typescript-eslint/typescript-estree": "8.59.3", "@typescript-eslint/utils": "8.59.3", "debug": "^4.4.3", "ts-api-utils": "^2.5.0" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", "typescript": ">=4.8.4 <6.1.0" } }, "sha512-g71d8QD8UaiHGvrJwyIS1hCX5r63w6Jll+4VEYhEAHXTDIqX1JgxhTAbEHtKntL9kuc4jRo7/GWw5xfCepSccQ=="], - "@typescript-eslint/types": ["@typescript-eslint/types@8.59.2", "", {}, "sha512-e82GVOE8Ps3E++Egvb6Y3Dw0S10u8NkQ9KXmtRhCWJJ8kDhOJTvtMAWnFL16kB1583goCWXsr0NieKCZMs2/0Q=="], + "@typescript-eslint/types": ["@typescript-eslint/types@8.59.3", "", {}, "sha512-ePFoH0g4ludssdRFqqDxQePCxU4WQyRa9+XVwjm7yLn0FKhMeoetC+qBEEI1Eyb1pGSDveTIT09Bvw2WhlGayg=="], - "@typescript-eslint/typescript-estree": ["@typescript-eslint/typescript-estree@8.59.2", "", { "dependencies": { "@typescript-eslint/project-service": "8.59.2", "@typescript-eslint/tsconfig-utils": "8.59.2", "@typescript-eslint/types": "8.59.2", "@typescript-eslint/visitor-keys": "8.59.2", "debug": "^4.4.3", "minimatch": "^10.2.2", "semver": "^7.7.3", "tinyglobby": "^0.2.15", "ts-api-utils": "^2.5.0" }, "peerDependencies": { "typescript": ">=4.8.4 <6.1.0" } }, "sha512-o0XPGNwcWw+FIwStOWn+BwBuEmL6QXP0rsvAFg7ET1dey1Nr6Wb1ac8p5HEsK0ygO/6mUxlk+YWQD9xcb/nnXg=="], + "@typescript-eslint/typescript-estree": ["@typescript-eslint/typescript-estree@8.59.3", "", { "dependencies": { "@typescript-eslint/project-service": "8.59.3", "@typescript-eslint/tsconfig-utils": "8.59.3", "@typescript-eslint/types": "8.59.3", "@typescript-eslint/visitor-keys": "8.59.3", "debug": "^4.4.3", "minimatch": "^10.2.2", "semver": "^7.7.3", "tinyglobby": "^0.2.15", "ts-api-utils": "^2.5.0" }, "peerDependencies": { "typescript": ">=4.8.4 <6.1.0" } }, "sha512-CbRjVRAf7Lr9Kr8RopKcbY45p2VfmmHrm0ygOCYFi7oU8q19m0Fs/6iHS7kNOmwpp+ob07ZVcAqlxUod9lYdmg=="], - "@typescript-eslint/utils": ["@typescript-eslint/utils@8.59.2", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.9.1", "@typescript-eslint/scope-manager": "8.59.2", "@typescript-eslint/types": "8.59.2", "@typescript-eslint/typescript-estree": "8.59.2" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", "typescript": ">=4.8.4 <6.1.0" } }, "sha512-Juw3EinkXqjaffxz6roowvV7GZT/kET5vSKKZT6upl5TXdWkLkYmNPXwDDL2Vkt2DPn0nODIS4egC/0AGxKo/Q=="], + "@typescript-eslint/utils": ["@typescript-eslint/utils@8.59.3", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.9.1", "@typescript-eslint/scope-manager": "8.59.3", "@typescript-eslint/types": "8.59.3", "@typescript-eslint/typescript-estree": "8.59.3" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", "typescript": ">=4.8.4 <6.1.0" } }, "sha512-JAvT14goBzRzzzZyqq3P9BLArIxTtQURUtFgQ/V7FO+eU+Gg6ES+5ymOPP1wRxXcxAYeivCk4uS3jCKWI1K8Zg=="], - "@typescript-eslint/visitor-keys": ["@typescript-eslint/visitor-keys@8.59.2", "", { "dependencies": { "@typescript-eslint/types": "8.59.2", "eslint-visitor-keys": "^5.0.0" } }, "sha512-NwjLUnGy8/Zfx23fl50tRC8rYaYnM52xNRYFAXvmiil9yh1+K6aRVQMnzW6gQB/1DLgWt977lYQn7C+wtgXZiA=="], + "@typescript-eslint/visitor-keys": ["@typescript-eslint/visitor-keys@8.59.3", "", { "dependencies": { "@typescript-eslint/types": "8.59.3", "eslint-visitor-keys": "^5.0.0" } }, "sha512-f1UQF7ggd42YiwI5wGrRaPsa+P0CINBlrkLPmGfpq/u/I/oVtecoEIfFR9ag/oa1sLOsRNZ6xehf6qMZhQGBDg=="], "@typescript/native-preview": ["@typescript/native-preview@7.0.0-dev.20260316.1", "", { "optionalDependencies": { "@typescript/native-preview-darwin-arm64": "7.0.0-dev.20260316.1", "@typescript/native-preview-darwin-x64": "7.0.0-dev.20260316.1", "@typescript/native-preview-linux-arm": "7.0.0-dev.20260316.1", "@typescript/native-preview-linux-arm64": "7.0.0-dev.20260316.1", "@typescript/native-preview-linux-x64": "7.0.0-dev.20260316.1", "@typescript/native-preview-win32-arm64": "7.0.0-dev.20260316.1", "@typescript/native-preview-win32-x64": "7.0.0-dev.20260316.1" }, "bin": { "tsgo": "bin/tsgo.js" } }, "sha512-s+QGNx+3zxTZBuZw3oNOFlHqpbmg0cTgBd/b6SRZ5mo3vFChkhflYqRW2IvTvU9a3PPX3bQAkQ/gWbDZCmNC3Q=="], @@ -2582,7 +2578,7 @@ "@typespec/ts-http-runtime": ["@typespec/ts-http-runtime@0.3.5", "", { "dependencies": { "http-proxy-agent": "^7.0.0", "https-proxy-agent": "^7.0.0", "tslib": "^2.6.2" } }, "sha512-yURCknZhvywvQItHMMmFSo+fq5arCUIyz/CVk7jD89MSai7dkaX8ufjCWp3NttLojoTVbcE72ri+be/TnEbMHw=="], - "@ungap/structured-clone": ["@ungap/structured-clone@1.3.0", "", {}, "sha512-WmoN8qaIAo7WTYWbAZuG8PYEhn5fkz7dZrqTBZ7dtt//lL2Gwms1IcnQ5yHqjDfX8Ft5j4YzDM23f87zBfDe9g=="], + "@ungap/structured-clone": ["@ungap/structured-clone@1.3.1", "", {}, "sha512-mUFwbeTqrVgDQxFveS+df2yfap6iuP20NAKAsBt5jDEoOTDew+zwLAOilHCeQJOVSvmgCX4ogqIrA0mnyr08yQ=="], "@upsetjs/venn.js": ["@upsetjs/venn.js@2.0.0", "", { "optionalDependencies": { "d3-selection": "^3.0.0", "d3-transition": "^3.0.1" } }, "sha512-WbBhLrooyePuQ1VZxrJjtLvTc4NVfpOyKx0sKqioq9bX1C1m7Jgykkn8gLrtwumBioXIqam8DLxp88Adbue6Hw=="], @@ -2606,7 +2602,31 @@ "@vscode/codicons": ["@vscode/codicons@0.0.44", "", {}, "sha512-F7qPRumUK3EHjNdopfICLGRf3iNPoZQt+McTHAn4AlOWPB3W2kL4H0S7uqEqbyZ6rCxaeDjpAn3MCUnwTu/VJQ=="], - "@vscode/ripgrep": ["@vscode/ripgrep@1.17.1", "", { "dependencies": { "https-proxy-agent": "^7.0.2", "proxy-from-env": "^1.1.0", "yauzl": "^2.9.2" } }, "sha512-xTs7DGyAO3IsJYOCTBP8LnTvPiYVKEuyv8s0xyJDBXfs8rhBfqnZPvb6xDT+RnwWzcXqW27xLS/aGrkjX7lNWw=="], + "@vscode/ripgrep": ["@vscode/ripgrep@1.18.0", "", { "optionalDependencies": { "@vscode/ripgrep-darwin-arm64": "1.18.0", "@vscode/ripgrep-darwin-x64": "1.18.0", "@vscode/ripgrep-linux-arm": "1.18.0", "@vscode/ripgrep-linux-arm64": "1.18.0", "@vscode/ripgrep-linux-ia32": "1.18.0", "@vscode/ripgrep-linux-ppc64": "1.18.0", "@vscode/ripgrep-linux-riscv64": "1.18.0", "@vscode/ripgrep-linux-s390x": "1.18.0", "@vscode/ripgrep-linux-x64": "1.18.0", "@vscode/ripgrep-win32-arm64": "1.18.0", "@vscode/ripgrep-win32-ia32": "1.18.0", "@vscode/ripgrep-win32-x64": "1.18.0" } }, "sha512-ns5lWe44tSfbTMbVUsyB+I1819PVSw4AdpgK0RNkzfWfwy6+3IUNSxwSrfTno1/oWaS/hERNz+XLWVyga2aJBQ=="], + + "@vscode/ripgrep-darwin-arm64": ["@vscode/ripgrep-darwin-arm64@1.18.0", "", { "os": "darwin", "cpu": "arm64" }, "sha512-r3ktHSvbFycQNF6sl7sNDPocpsI7J+mEzh1IaZFkY0spm3k2Z9t8hPAeOK7+p0l6p6/swkQC14XWX01low+94Q=="], + + "@vscode/ripgrep-darwin-x64": ["@vscode/ripgrep-darwin-x64@1.18.0", "", { "os": "darwin", "cpu": "x64" }, "sha512-25b4gWbL138dGuQU244ebCKKc0q05ULBMoFSz9oAEUHNeqK/lOJViDS7DRvbDazzAzSEdan391Znks/R5mkaTQ=="], + + "@vscode/ripgrep-linux-arm": ["@vscode/ripgrep-linux-arm@1.18.0", "", { "os": "linux", "cpu": "arm" }, "sha512-GDAvufNDHu8zqLEmXstalQF0Wh6wQvdsBi/Vg3Yi3CK4a8XoFXqqXVEHEZ9xQz3t0NfoSEc9JbvK9DDS6FxyxQ=="], + + "@vscode/ripgrep-linux-arm64": ["@vscode/ripgrep-linux-arm64@1.18.0", "", { "os": "linux", "cpu": "arm64" }, "sha512-lQ/5zTG++U0E3IhVgS4EPTTn/U4okncaRMM5GOFfOYZywS4nuD31GhkHbNYlDk5CuDC68+hYJ0/eQeyCKJDA+g=="], + + "@vscode/ripgrep-linux-ia32": ["@vscode/ripgrep-linux-ia32@1.18.0", "", { "os": "linux", "cpu": "ia32" }, "sha512-YWLkSUtFd4Jh5EepIhA9RJSfv3uMAVMo+2rBIGHPBnvgLrZciIs2cDKei1/p6Wc/aCzUoHyMAg2R6tw4ZCBKGg=="], + + "@vscode/ripgrep-linux-ppc64": ["@vscode/ripgrep-linux-ppc64@1.18.0", "", { "os": "linux", "cpu": "ppc64" }, "sha512-quXVY8fwQ8O/lvU1yrSqSl3jlUzysRSb+AfUfCL/tRtphxsKlFvPAejryZ6vg4Bgvn8XL74xb4qMCDmWgYrT5w=="], + + "@vscode/ripgrep-linux-riscv64": ["@vscode/ripgrep-linux-riscv64@1.18.0", "", { "os": "linux", "cpu": "none" }, "sha512-f5kBQBrWfQt8Q7OhSORuNDei5dkYagBj3y4jImSUXGMy8B/Ke7SltSRcUtjPv166FAFfHCAmWuZp3+cWnX2/Vw=="], + + "@vscode/ripgrep-linux-s390x": ["@vscode/ripgrep-linux-s390x@1.18.0", "", { "os": "linux", "cpu": "s390x" }, "sha512-rTOcJFGGcl2c07RUOWUo4U1ndnemKhY6A9hnMB18uk7jSgJc0d/QLBGWMWpumdtoJtpizn/wIv5mXIisJukusQ=="], + + "@vscode/ripgrep-linux-x64": ["@vscode/ripgrep-linux-x64@1.18.0", "", { "os": "linux", "cpu": "x64" }, "sha512-mQ3bVrUpnD2vs7QT0vX90Lt0cnUq467uFtEktIdsJJmW296RoSULRGqWgzG1AKxyBpNDD6l4ZO4qKf6SgyC23Q=="], + + "@vscode/ripgrep-win32-arm64": ["@vscode/ripgrep-win32-arm64@1.18.0", "", { "os": "win32", "cpu": "arm64" }, "sha512-vfTIjq1OHnzUjxZcHVQAMbnggp8dpGf+0QKFOZHwWPqFwXxQC8eCWM+5NUdoJ6yrElCeMzoUTXoK/LdZaniB+Q=="], + + "@vscode/ripgrep-win32-ia32": ["@vscode/ripgrep-win32-ia32@1.18.0", "", { "os": "win32", "cpu": "ia32" }, "sha512-//rfAE+BOw5AC2EMmepmiE36jUuevtQYNQqqlw1s3m9FlRxjxEut97RkRPHAu9BG4mSojatZx+kXZXNdyI9caQ=="], + + "@vscode/ripgrep-win32-x64": ["@vscode/ripgrep-win32-x64@1.18.0", "", { "os": "win32", "cpu": "x64" }, "sha512-KNPvtElldqILHdnAetujPaowkNbpqJy3ssIGGN6F6Kve9Qi+nNLI2DN01O83JjCEVQbCzl8Ov3QZ9Eov3BR8Dg=="], "@vscode/test-cli": ["@vscode/test-cli@0.0.12", "", { "dependencies": { "@types/mocha": "^10.0.10", "c8": "^10.1.3", "chokidar": "^3.6.0", "enhanced-resolve": "^5.18.3", "glob": "^10.3.10", "minimatch": "^9.0.3", "mocha": "^11.7.4", "supports-color": "^10.2.2", "yargs": "^17.7.2" }, "bin": { "vscode-test": "out/bin.mjs" } }, "sha512-iYN0fDg29+a2Xelle/Y56Xvv7Nc8Thzq4VwpzAF/SIE6918rDicqfsQxV6w1ttr2+SOm+10laGuY9FG2ptEKsQ=="], @@ -2636,9 +2656,9 @@ "@webcontainer/env": ["@webcontainer/env@1.1.1", "", {}, "sha512-6aN99yL695Hi9SuIk1oC88l9o0gmxL1nGWWQ/kNy81HigJ0FoaoTXpytCj6ItzgyCEwA9kF1wixsTuv5cjsgng=="], - "@webgpu/types": ["@webgpu/types@0.1.69", "", {}, "sha512-RPmm6kgRbI8e98zSD3RVACvnuktIja5+yLgDAkTmxLr90BEwdTXRQWNLF3ETTTyH/8mKhznZuN5AveXYFEsMGQ=="], + "@webgpu/types": ["@webgpu/types@0.1.70", "", {}, "sha512-LFiNHHKMvmAEvwVew3JLJmTdShhbdwRFSImUshGhE2mGE8ybQzIo63l5uRp+YKnNx+8Qno8Kf6gN+DKMreIJCA=="], - "@xmldom/xmldom": ["@xmldom/xmldom@0.9.10", "", {}, "sha512-A9gOqLdi6cV4ibazAjcQufGj0B1y/vDqYrcuP6d/6x8P27gRS8643Dj9o1dEKtB6O7fwxb2FgBmJS2mX7gpvdw=="], + "@xmldom/xmldom": ["@xmldom/xmldom@0.8.13", "", {}, "sha512-KRYzxepc14G/CEpEGc3Yn+JKaAeT63smlDr+vjB8jRfgTBBI9wRj/nkQEO+ucV8p8I9bfKLWp37uHgFrbntPvw=="], "@xterm/addon-clipboard": ["@xterm/addon-clipboard@0.2.0", "", { "dependencies": { "js-base64": "^3.7.5" } }, "sha512-Dl31BCtBhLaUEECUbEiVcCLvLBbaeGYdT7NofB8OJkGTD3MWgBsaLjXvfGAD4tQNHhm6mbKyYkR7XD8kiZsdNg=="], @@ -2650,6 +2670,10 @@ "@xterm/xterm": ["@xterm/xterm@6.0.0", "", {}, "sha512-TQwDdQGtwwDt+2cgKDLn0IRaSxYu1tSUjgKarSDkUM0ZNiSRXFpjxEsvc/Zgc5kq5omJ+V0a8/kIM2WD3sMOYg=="], + "@xyflow/react": ["@xyflow/react@12.10.2", "", { "dependencies": { "@xyflow/system": "0.0.76", "classcat": "^5.0.3", "zustand": "^4.4.0" }, "peerDependencies": { "react": ">=17", "react-dom": ">=17" } }, "sha512-CgIi6HwlcHXwlkTpr0fxLv/0sRVNZ8IdwKLzzeCscaYBwpvfcH1QFOCeaTCuEn1FQEs/B8CjnTSjhs8udgmBgQ=="], + + "@xyflow/system": ["@xyflow/system@0.0.76", "", { "dependencies": { "@types/d3-drag": "^3.0.7", "@types/d3-interpolate": "^3.0.4", "@types/d3-selection": "^3.0.10", "@types/d3-transition": "^3.0.8", "@types/d3-zoom": "^3.0.8", "d3-drag": "^3.0.0", "d3-interpolate": "^3.0.1", "d3-selection": "^3.0.0", "d3-zoom": "^3.0.0" } }, "sha512-hvwvnRS1B3REwVDlWexsq7YQaPZeG3/mKo1jv38UmnpWmxihp14bW6VtEOuHEwJX2FvzFw8k77LyKSk/wiZVNA=="], + "@zip.js/zip.js": ["@zip.js/zip.js@2.7.62", "", {}, "sha512-OaLvZ8j4gCkLn048ypkZu29KX30r8/OfFF2w4Jo5WXFr+J04J+lzJ5TKZBVgFXhlvSkqNFQdfnY1Q8TMTCyBVA=="], "abbrev": ["abbrev@4.0.0", "", {}, "sha512-a1wflyaL0tHtJSmLSOVybYhy22vRih4eduhhrkcjgrWGnRfrZtovJ2FRjxuTtkkj47O/baf0R86QU5OuYpz8fA=="], @@ -2762,7 +2786,7 @@ "axe-core": ["axe-core@4.11.4", "", {}, "sha512-KunSNx+TVpkAw/6ULfhnx+HWRecjqZGTOyquAoWHYLRSdK1tB5Ihce1ZW+UY3fj33bYAFWPu7W/GRSmmrCGuxA=="], - "axios": ["axios@1.16.0", "", { "dependencies": { "follow-redirects": "^1.16.0", "form-data": "^4.0.5", "proxy-from-env": "^2.1.0" } }, "sha512-6hp5CwvTPlN2A31g5dxnwAX0orzM7pmCRDLnZSX772mv8WDqICwFjowHuPs04Mc8deIld1+ejhtaMn5vp6b+1w=="], + "axios": ["axios@1.16.1", "", { "dependencies": { "follow-redirects": "^1.16.0", "form-data": "^4.0.5", "https-proxy-agent": "^5.0.1", "proxy-from-env": "^2.1.0" } }, "sha512-caYkukvroVPO8KrzuJEb50Hm07KwfBZPEC3VeFHTsqWHvKTsy54hjJz9BS/cdaypROE2rH6xvm9mHX4fgWkr3A=="], "azure-devops-node-api": ["azure-devops-node-api@12.5.0", "", { "dependencies": { "tunnel": "0.0.6", "typed-rest-client": "^1.8.4" } }, "sha512-R5eFskGvOm3U/GzeAuxRkUsAl0hrAwGgWn6zAd2KrZmrEhWZVqLew4OOupbQlXUuojUzpGtq62SmdhJ06N88og=="], @@ -2772,11 +2796,11 @@ "babel-plugin-module-resolver": ["babel-plugin-module-resolver@5.0.2", "", { "dependencies": { "find-babel-config": "^2.1.1", "glob": "^9.3.3", "pkg-up": "^3.1.0", "reselect": "^4.1.7", "resolve": "^1.22.8" } }, "sha512-9KtaCazHee2xc0ibfqsDeamwDps6FZNo5S0Q81dUqEuFzVwPhcT4J5jOqIVvgCA3Q/wO9hKYxN/Ds3tIsp5ygg=="], - "babel-preset-solid": ["babel-preset-solid@1.9.10", "", { "dependencies": { "babel-plugin-jsx-dom-expressions": "^0.40.3" }, "peerDependencies": { "@babel/core": "^7.0.0", "solid-js": "^1.9.10" }, "optionalPeers": ["solid-js"] }, "sha512-HCelrgua/Y+kqO8RyL04JBWS/cVdrtUv/h45GntgQY+cJl4eBcKkCDV3TdMjtKx1nXwRaR9QXslM/Npm1dxdZQ=="], + "babel-preset-solid": ["babel-preset-solid@1.9.12", "", { "dependencies": { "babel-plugin-jsx-dom-expressions": "^0.40.6" }, "peerDependencies": { "@babel/core": "^7.0.0", "solid-js": "^1.9.12" }, "optionalPeers": ["solid-js"] }, "sha512-LLqnuKVDlKpyBlMPcH6qEvs/wmS9a+NczppxJ3ryS/c0O5IiSFOIBQi9GzyiGDSbcJpx4Gr87jyFTos1MyEuWg=="], "balanced-match": ["balanced-match@4.0.4", "", {}, "sha512-BLrgEcRTwX2o6gGxGOCNyMvGSp35YofuYzw9h1IMTRmKqttAZZVU67bdb9Pr2vUHA8+j3i2tJfjO6C6+4myGTA=="], - "bare-events": ["bare-events@2.8.2", "", { "peerDependencies": { "bare-abort-controller": "*" }, "optionalPeers": ["bare-abort-controller"] }, "sha512-riJjyv1/mHLIPX4RwiK+oW9/4c3TEUeORHKefKAKnZ5kyslbN+HXowtbaVEqt4IMUB7OXlfixcs6gsFeo/jhiQ=="], + "bare-events": ["bare-events@2.8.3", "", { "peerDependencies": { "bare-abort-controller": "*" }, "optionalPeers": ["bare-abort-controller"] }, "sha512-HdUm8EMQBLaJvGUdidNNbqpA1kYkwNcb+MYxkxCLAPJGQzlv9J0C24h8V65Z4c5GLd/JEALDvpFCQgpLJqc0zw=="], "bare-fs": ["bare-fs@4.7.1", "", { "dependencies": { "bare-events": "^2.5.4", "bare-path": "^3.0.0", "bare-stream": "^2.6.4", "bare-url": "^2.2.2", "fast-fifo": "^1.3.2" }, "peerDependencies": { "bare-buffer": "*" }, "optionalPeers": ["bare-buffer"] }, "sha512-WDRsyVN52eAx/lBamKD6uyw8H4228h/x0sGGGegOamM2cd7Pag88GfMQalobXI+HaEUxpCkbKQUDOQqt9wawRw=="], @@ -2786,11 +2810,11 @@ "bare-stream": ["bare-stream@2.13.1", "", { "dependencies": { "streamx": "^2.25.0", "teex": "^1.0.1" }, "peerDependencies": { "bare-abort-controller": "*", "bare-buffer": "*", "bare-events": "*" }, "optionalPeers": ["bare-abort-controller", "bare-buffer", "bare-events"] }, "sha512-Vp0cnjYyrEC4whYTymQ+YZi6pBpfiICZO3cfRG8sy67ZNWe951urv1x4eW1BKNngw3U+3fPYb5JQvHbCtxH7Ow=="], - "bare-url": ["bare-url@2.4.2", "", { "dependencies": { "bare-path": "^3.0.0" } }, "sha512-/9a2j4ac6ckpmAHvod/ob7x439OAHst/drc2Clnq+reRYd/ovddwcF4LfoxHyNk5AuGBnPg+HqFjmE/Zpq6v0A=="], + "bare-url": ["bare-url@2.4.3", "", { "dependencies": { "bare-path": "^3.0.0" } }, "sha512-Kccpc7ACfXaxfeInfqKcZtW4pT5YBn1mesc4sCsun6sRwtbJ4h+sNOaksUpYEJUKfN65YWC6Bw2OJEFiKxq8nQ=="], "base64-js": ["base64-js@1.5.1", "", {}, "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA=="], - "baseline-browser-mapping": ["baseline-browser-mapping@2.10.27", "", { "bin": { "baseline-browser-mapping": "dist/cli.cjs" } }, "sha512-zEs/ufmZoUd7WftKpKyXaT6RFxpQ5Qm9xytKRHvJfxFV9DFJkZph9RvJ1LcOUi0Z1ZVijMte65JbILeV+8QQEA=="], + "baseline-browser-mapping": ["baseline-browser-mapping@2.10.29", "", { "bin": { "baseline-browser-mapping": "dist/cli.cjs" } }, "sha512-Asa2krT+XTPZINCS+2QcyS8WTkObE77RwkydwF7h6DmnKqbvlalz93m/dnphUyCa6SWSP51VgtEUf2FN+gelFQ=="], "before-after-hook": ["before-after-hook@2.2.3", "", {}, "sha512-NzUnlZexiaH/46WDhANlyR2bXRopNg4F/zuSA3OpZnllCUgRaOF2znDioDWrmbNVsuZk6l9pMquQB38cfBZwkQ=="], @@ -2826,7 +2850,7 @@ "bowser": ["bowser@2.14.1", "", {}, "sha512-tzPjzCxygAKWFOJP011oxFHs57HzIhOEracIgAePE4pqB3LikALKnSzUyU4MGs9/iCEUuHlAJTjTc5M+u7YEGg=="], - "brace-expansion": ["brace-expansion@5.0.5", "", { "dependencies": { "balanced-match": "^4.0.2" } }, "sha512-VZznLgtwhn+Mact9tfiwx64fA9erHH/MCXEUfB/0bX/6Fz6ny5EGTXYltMocqg4xFAQZtnO3DHWWXi8RiuN7cQ=="], + "brace-expansion": ["brace-expansion@5.0.6", "", { "dependencies": { "balanced-match": "^4.0.2" } }, "sha512-kLpxurY4Z4r9sgMsyG0Z9uzsBlgiU/EFKhj/h91/8yHu0edo7XuixOIH3VcJ8kkxs6/jPzoI6U9Vj3WqbMQ94g=="], "braces": ["braces@3.0.3", "", { "dependencies": { "fill-range": "^7.1.1" } }, "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA=="], @@ -2848,13 +2872,13 @@ "builder-util-runtime": ["builder-util-runtime@9.5.1", "", { "dependencies": { "debug": "^4.3.4", "sax": "^1.2.4" } }, "sha512-qt41tMfgHTllhResqM5DcnHyDIWNgzHvuY2jDcYP9iaGpkWxTUzV6GQjDeLnlR1/DtdlcsWQbA7sByMpmJFTLQ=="], - "bun-ffi-structs": ["bun-ffi-structs@0.1.2", "", { "peerDependencies": { "typescript": "^5" } }, "sha512-Lh1oQAYHDcnesJauieA4UNkWGXY9hYck7OA5IaRwE3Bp6K2F2pJSNYqq+hIy7P3uOvo3km3oxS8304g5gDMl/w=="], + "bun-ffi-structs": ["bun-ffi-structs@0.2.2", "", { "peerDependencies": { "typescript": "^5" } }, "sha512-N/ZWtyN0piZlrXQT7TO0V+q952orYqkfhXRXM1Hcbb+R3QSiBH4vLnib187Mrs1H7pWIYECAmPeapGYDOMCl+w=="], "bun-pty": ["bun-pty@0.4.8", "", {}, "sha512-rO70Mrbr13+jxHHHu2YBkk2pNqrJE5cJn29WE++PUr+GFA0hq/VgtQPZANJ8dJo6d7XImvBk37Innt8GM7O28w=="], "bun-types": ["bun-types@1.3.12", "", { "dependencies": { "@types/node": "*" } }, "sha512-HqOLj5PoFajAQciOMRiIZGNoKxDJSr6qigAttOX40vJuSp6DN/CxWp9s3C1Xwm4oH7ybueITwiaOcWXoYVoRkA=="], - "bun-webgpu": ["bun-webgpu@0.1.5", "", { "dependencies": { "@webgpu/types": "^0.1.60" }, "optionalDependencies": { "bun-webgpu-darwin-arm64": "^0.1.5", "bun-webgpu-darwin-x64": "^0.1.5", "bun-webgpu-linux-x64": "^0.1.5", "bun-webgpu-win32-x64": "^0.1.5" } }, "sha512-91/K6S5whZKX7CWAm9AylhyKrLGRz6BUiiPiM/kXadSnD4rffljCD/q9cNFftm5YXhx4MvLqw33yEilxogJvwA=="], + "bun-webgpu": ["bun-webgpu@0.1.4", "", { "dependencies": { "@webgpu/types": "^0.1.60" }, "optionalDependencies": { "bun-webgpu-darwin-arm64": "^0.1.4", "bun-webgpu-darwin-x64": "^0.1.4", "bun-webgpu-linux-x64": "^0.1.4", "bun-webgpu-win32-x64": "^0.1.4" } }, "sha512-Kw+HoXl1PMWJTh9wvh63SSRofTA8vYBFCw0XEP1V1fFdQEDhI8Sgf73sdndE/oDpN/7CMx0Yv/q8FCvO39ROMQ=="], "bun-webgpu-darwin-arm64": ["bun-webgpu-darwin-arm64@0.1.7", "", { "os": "darwin", "cpu": "arm64" }, "sha512-mRrFFyHzPWjsTRidAZBRcu808CPQBOUL0P6b4nxLhp+XHcV/mbUHERZMgW9s58tsojQfSdzschiQa8q+JCgRWA=="], @@ -2890,7 +2914,7 @@ "camelcase": ["camelcase@6.3.0", "", {}, "sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA=="], - "caniuse-lite": ["caniuse-lite@1.0.30001791", "", {}, "sha512-yk0l/YSrOnFZk3UROpDLQD9+kC1l4meK/wed583AXrzoarMGJcbRi2Q4RaUYbKxYAsZ8sWmaSa/DsLmdBeI1vQ=="], + "caniuse-lite": ["caniuse-lite@1.0.30001792", "", {}, "sha512-hVLMUZFgR4JJ6ACt1uEESvQN1/dBVqPAKY0hgrV70eN3391K6juAfTjKZLKvOMsx8PxA7gsY1/tLMMTcfFLLpw=="], "cbor": ["cbor@8.1.0", "", { "dependencies": { "nofilter": "^3.1.0" } }, "sha512-DwGjNW9omn6EwP70aXsn7FQJx5kO12tX0bZkaTjzdVFM6/7nhA4t0EENocKGx6D2Bch9PE2KzCUf5SceBdeijg=="], @@ -2916,10 +2940,6 @@ "cheerio-select": ["cheerio-select@2.1.0", "", { "dependencies": { "boolbase": "^1.0.0", "css-select": "^5.1.0", "css-what": "^6.1.0", "domelementtype": "^2.3.0", "domhandler": "^5.0.3", "domutils": "^3.0.1" } }, "sha512-9v9kG0LvzrlcungtnJtpGNxY+fzECQKhK4EGJX2vByejiMX84MFNQw4UxPJl3bFbTMw+Dfs37XaIkCwTZfLh4g=="], - "chevrotain": ["chevrotain@12.0.0", "", { "dependencies": { "@chevrotain/cst-dts-gen": "12.0.0", "@chevrotain/gast": "12.0.0", "@chevrotain/regexp-to-ast": "12.0.0", "@chevrotain/types": "12.0.0", "@chevrotain/utils": "12.0.0" } }, "sha512-csJvb+6kEiQaqo1woTdSAuOWdN0WTLIydkKrBnS+V5gZz0oqBrp4kQ35519QgK6TpBThiG3V1vNSHlIkv4AglQ=="], - - "chevrotain-allstar": ["chevrotain-allstar@0.4.3", "", { "dependencies": { "lodash-es": "^4.18.1" }, "peerDependencies": { "chevrotain": "^12.0.0" } }, "sha512-2X4mkroolSMKqW+H22pyPMUVDqYZzPhephTmg/NODKb1IGYPHfxfhcW0EjS7wcPJNbze2i4vBWT7zT5FKF2lrQ=="], - "chokidar": ["chokidar@4.0.3", "", { "dependencies": { "readdirp": "^4.0.1" } }, "sha512-Qgzu8kfBvo+cA4962jnP1KkS6Dop5NS6g7R5LFYJr4b8Ub94PPQXUksCw9PvXoeXPRRddRNC5C1JQUR2SMGtnA=="], "chownr": ["chownr@3.0.0", "", {}, "sha512-+IxzY9BZOQd/XuYPRmrvEVjF/nqj5kgT4kEq7VofrDoM1MxoRjEWkrCC3EtLi59TVawxTAn+orJwFQcrqEN1+g=="], @@ -2934,6 +2954,8 @@ "citty": ["citty@0.1.6", "", { "dependencies": { "consola": "^3.2.3" } }, "sha512-tskPPKEs8D2KPafUypv2gxwJP8h/OaJmC82QQGGDQcHvXX43xF2VDACcJVmZ0EuSxkpO9Kc4MlrA3q0+FG58AQ=="], + "classcat": ["classcat@5.0.5", "", {}, "sha512-JhZUT7JFcQy/EzW605k/ktHtncoo9vnyW/2GspNYwFlN1C/WmjuV/xtS04e9SOkL2sTdw0VAZ2UGCcQ9lR6p6w=="], + "clean-git-ref": ["clean-git-ref@2.0.1", "", {}, "sha512-bLSptAy2P0s6hU4PzuIMKmMJJSE6gLXGH1cntDu7bWJUksvuM+7ReOK61mozULErYvP6a15rnYl0zFDef+pyPw=="], "clean-stack": ["clean-stack@4.2.0", "", { "dependencies": { "escape-string-regexp": "5.0.0" } }, "sha512-LYv6XPxoyODi36Dp976riBtSY27VmFo+MKqEU9QCCWyTrdEPDog+RWA7xQWHi6Vbp61j5c4cdzzX1NidnwtUWg=="], @@ -3154,7 +3176,7 @@ "define-properties": ["define-properties@1.2.1", "", { "dependencies": { "define-data-property": "^1.0.1", "has-property-descriptors": "^1.0.0", "object-keys": "^1.1.1" } }, "sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg=="], - "defu": ["defu@6.1.6", "", {}, "sha512-f8mefEW4WIVg4LckePx3mALjQSPQgFlg9U8yaPdlsbdYcHQyj9n2zL2LJEA52smeYxOvmd/nB7TpMtHGMTHcug=="], + "defu": ["defu@6.1.7", "", {}, "sha512-7z22QmUWiQ/2d0KkdYmANbRUVABpZ9SNYyH5vx6PZ+nE5bcC0l7uFvEfHlyld/HcGBFTL536ClDt3DEcSlEJAQ=="], "delaunator": ["delaunator@5.1.0", "", { "dependencies": { "robust-predicates": "^3.0.2" } }, "sha512-AGrQ4QSgssa1NGmWmLPqN5NY2KajF5MqxetNEO+o0n3ZwZZeTmt7bBnvzHWrmkZFxGgr4HdyFgelzgi06otLuQ=="], @@ -3204,7 +3226,7 @@ "domhandler": ["domhandler@5.0.3", "", { "dependencies": { "domelementtype": "^2.3.0" } }, "sha512-cgwlv/1iFQiFnU96XXgROh8xTeetsnJiDsTc7TYCLFd9+/WNkIqPTxiM/8pSd8VIrhXGTf1Ny1q1hquVqDJB5w=="], - "dompurify": ["dompurify@3.3.3", "", { "optionalDependencies": { "@types/trusted-types": "^2.0.7" } }, "sha512-Oj6pzI2+RqBfFG+qOaOLbFXLQ90ARpcGG6UePL82bJLtdsa6CYJD7nmiU8MW9nQNOtCHV3lZ/Bzq1X0QYbBZCA=="], + "dompurify": ["dompurify@3.4.2", "", { "optionalDependencies": { "@types/trusted-types": "^2.0.7" } }, "sha512-lHeS9SA/IKeIFFyYciHBr2n0v1VMPlSj843HdLOwjb2OxNwdq9Xykxqhk+FE42MzAdHvInbAolSE4mhahPpjXA=="], "domutils": ["domutils@3.2.2", "", { "dependencies": { "dom-serializer": "^2.0.0", "domelementtype": "^2.3.0", "domhandler": "^5.0.3" } }, "sha512-6kZKyUajlDuqlHKVX1w7gyslj9MPIXzIFiz/rGu35uC1wMi+kMhQwGhl4lt9unC9Vb9INnY9Z3/ZA3+FhASLaw=="], @@ -3244,13 +3266,13 @@ "electron-is-dev": ["electron-is-dev@3.0.1", "", {}, "sha512-8TjjAh8Ec51hUi3o4TaU0mD3GMTOESi866oRNavj9A3IQJ7pmv+MJVmdZBFGw4GFT36X7bkqnuDNYvkQgvyI8Q=="], - "electron-log": ["electron-log@5.4.3", "", {}, "sha512-sOUsM3LjZdugatazSQ/XTyNcw8dfvH1SYhXWiJyfYodAAKOZdHs0txPiLDXFzOZbhXgAgshQkshH2ccq0feyLQ=="], + "electron-log": ["electron-log@5.4.4", "", {}, "sha512-istWgaXjBfURBSS8LWVW9C3jsc6+ac+tY1lXrQEOTp0lVj+a4OlO1Tmqb36GgnEUDv92DGC9VI1HNXwJinWpgA=="], "electron-publish": ["electron-publish@26.8.1", "", { "dependencies": { "@types/fs-extra": "^9.0.11", "builder-util": "26.8.1", "builder-util-runtime": "9.5.1", "chalk": "^4.1.2", "form-data": "^4.0.5", "fs-extra": "^10.1.0", "lazy-val": "^1.0.5", "mime": "^2.5.2" } }, "sha512-q+jrSTIh/Cv4eGZa7oVR+grEJo/FoLMYBAnSL5GCtqwUpr1T+VgKB/dn1pnzxIxqD8S/jP1yilT9VrwCqINR4w=="], "electron-store": ["electron-store@10.1.0", "", { "dependencies": { "conf": "^14.0.0", "type-fest": "^4.41.0" } }, "sha512-oL8bRy7pVCLpwhmXy05Rh/L6O93+k9t6dqSw0+MckIc3OmCTZm6Mp04Q4f/J0rtu84Ky6ywkR8ivtGOmrq+16w=="], - "electron-to-chromium": ["electron-to-chromium@1.5.349", "", {}, "sha512-QsWVGyRuY07Aqb234QytTfwd5d9AJlfNIQ5wIOl1L+PZDzI9d9+Fn0FRale/QYlFxt/bUnB0/nLd1jFPGxGK1A=="], + "electron-to-chromium": ["electron-to-chromium@1.5.355", "", {}, "sha512-LUPZhKzZPYSPme1jEYohpkA+ybYCJztr1quAdBd7E7h3+VOBVcKkwwtBJu41nrjawrRzfb8mtMfzWozoaK0ZIQ=="], "electron-updater": ["electron-updater@6.8.3", "", { "dependencies": { "builder-util-runtime": "9.5.1", "fs-extra": "^10.1.0", "js-yaml": "^4.1.0", "lazy-val": "^1.0.5", "lodash.escaperegexp": "^4.1.2", "lodash.isequal": "^4.5.0", "semver": "~7.7.3", "tiny-typed-emitter": "^2.1.0" } }, "sha512-Z6sgw3jgbikWKXei1ENdqFOxBP0WlXg3TtKfz0rgw2vIZFJUyI4pD7ZN7jrkm7EoMK+tcm/qTnPUdqfZukBlBQ=="], @@ -3274,7 +3296,7 @@ "engine.io-parser": ["engine.io-parser@5.2.3", "", {}, "sha512-HqD3yTBfnBxIrbnM1DoD6Pcq8NECnh8d4As1Qgh0z5Gg3jRRIqijury0CL3ghu/edArpUYiYqQiDUQBIs4np3Q=="], - "enhanced-resolve": ["enhanced-resolve@5.21.0", "", { "dependencies": { "graceful-fs": "^4.2.4", "tapable": "^2.3.3" } }, "sha512-otxSQPw4lkOZWkHpB3zaEQs6gWYEsmX4xQF68ElXC/TWvGxGMSGOvoNbaLXm6/cS/fSfHtsEdw90y20PCd+sCA=="], + "enhanced-resolve": ["enhanced-resolve@5.21.3", "", { "dependencies": { "graceful-fs": "^4.2.4", "tapable": "^2.3.3" } }, "sha512-QyL119InA+XXEkNLNTPCXPugSvOfhwv0JOlGNzvxs0hZaiHLNvXSpudUWsOlsXGWJh8G6ckCScEkVHfX3kw/2Q=="], "enquirer": ["enquirer@2.4.1", "", { "dependencies": { "ansi-colors": "^4.1.1", "strip-ansi": "^6.0.1" } }, "sha512-rRqJg/6gd538VHvR3PSrdRBb/1Vy2YfzHqzvbhGIQpDRKIa4FgV/54b5Q1xYSxOOwKvjXweS26E0Q+nAMwp2pQ=="], @@ -3356,7 +3378,7 @@ "express": ["express@5.2.1", "", { "dependencies": { "accepts": "^2.0.0", "body-parser": "^2.2.1", "content-disposition": "^1.0.0", "content-type": "^1.0.5", "cookie": "^0.7.1", "cookie-signature": "^1.2.1", "debug": "^4.4.0", "depd": "^2.0.0", "encodeurl": "^2.0.0", "escape-html": "^1.0.3", "etag": "^1.8.1", "finalhandler": "^2.1.0", "fresh": "^2.0.0", "http-errors": "^2.0.0", "merge-descriptors": "^2.0.0", "mime-types": "^3.0.0", "on-finished": "^2.4.1", "once": "^1.4.0", "parseurl": "^1.3.3", "proxy-addr": "^2.0.7", "qs": "^6.14.0", "range-parser": "^1.2.1", "router": "^2.2.0", "send": "^1.1.0", "serve-static": "^2.2.0", "statuses": "^2.0.1", "type-is": "^2.0.1", "vary": "^1.1.2" } }, "sha512-hIS4idWWai69NezIdRt2xFVofaF4j+6INOpJlVOLDO8zXGpUVEVzIYk12UUi2JzjEzWL3IOAxcTubgz9Po0yXw=="], - "express-rate-limit": ["express-rate-limit@8.5.0", "", { "dependencies": { "ip-address": "10.1.0" }, "peerDependencies": { "express": ">= 4.11" } }, "sha512-XKhFohWaSBdVJNTi5TaHziqnPkv04I9UQV6q1Wy7Ui6GGQZVW12ojDFwqer14EvCXxjvPG0CyWXx7cAXpALB4Q=="], + "express-rate-limit": ["express-rate-limit@8.5.2", "", { "dependencies": { "ip-address": "^10.2.0" }, "peerDependencies": { "express": ">= 4.11" } }, "sha512-5Kb34ipNX694DH48vN9irak1Qx30nb0PLYHXfJgw4YEjiC3ZEmZJhwOp+VfiCYwFzvFTdB9QkArYS5kXa2cx2A=="], "exsolve": ["exsolve@1.0.8", "", {}, "sha512-LmDxfWXwcTArk8fUEnOfSZpHOJ6zOMUJKOtFLFqJLoKJetuQG874Uc7/Kki7zFLzYybmZhp1M7+98pfMqeX8yA=="], @@ -3374,7 +3396,7 @@ "extsprintf": ["extsprintf@1.4.1", "", {}, "sha512-Wrk35e8ydCKDj/ArClo1VrPVmN8zph5V4AtHwIuHhvMXsKf73UT3BOD+azBIW+3wOJ4FhEH7zyaJCFvChjYvMA=="], - "fast-check": ["fast-check@4.7.0", "", { "dependencies": { "pure-rand": "^8.0.0" } }, "sha512-NsZRtqvSSoCP0HbNjUD+r1JH8zqZalyp6gLY9e7OYs7NK9b6AHOs2baBFeBG7bVNsuoukh89x2Yg3rPsul8ziQ=="], + "fast-check": ["fast-check@4.8.0", "", { "dependencies": { "pure-rand": "^8.0.0" } }, "sha512-GOJ158CUMnN6cSahsv4+ExARvIDuzzinFjkp0E9WtiBa5zcVeLozVkWaE4IzFcc+Y48Wp1EDlUZsXRyAztQcSg=="], "fast-content-type-parse": ["fast-content-type-parse@3.0.0", "", {}, "sha512-ZvLdcY8P+N8mGQJahJV5G4U88CSvT1rP8ApL6uETe88MBXrBHAkZlSEySdUlyztF7ccb+Znos3TFqaepHxdhBg=="], @@ -3390,15 +3412,15 @@ "fast-json-stable-stringify": ["fast-json-stable-stringify@2.1.0", "", {}, "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw=="], - "fast-json-stringify": ["fast-json-stringify@6.3.0", "", { "dependencies": { "@fastify/merge-json-schemas": "^0.2.0", "ajv": "^8.12.0", "ajv-formats": "^3.0.1", "fast-uri": "^3.0.0", "json-schema-ref-resolver": "^3.0.0", "rfdc": "^1.2.0" } }, "sha512-oRCntNDY/329HJPlmdNLIdogNtt6Vyjb1WuT01Soss3slIdyUp8kAcDU3saQTOquEK8KFVfwIIF7FebxUAu+yA=="], + "fast-json-stringify": ["fast-json-stringify@6.4.0", "", { "dependencies": { "@fastify/merge-json-schemas": "^0.2.0", "ajv": "^8.12.0", "ajv-formats": "^3.0.1", "fast-uri": "^3.0.0", "json-schema-ref-resolver": "^3.0.0", "rfdc": "^1.2.0" } }, "sha512-ibRCQ0GZKJIQ+P3Et1h0LhPgp3PMTYk0MH8O+kW3lNYsvmaQww5Nn3f1jf73Q0jR1Yz3a1CDP4/NZD3vOajWJQ=="], "fast-levenshtein": ["fast-levenshtein@2.0.6", "", {}, "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw=="], "fast-querystring": ["fast-querystring@1.1.2", "", { "dependencies": { "fast-decode-uri-component": "^1.0.1" } }, "sha512-g6KuKWmFXc0fID8WWH0jit4g0AGBoJhCkJMb1RmbsSEUNvQ+ZC8D6CUZ+GtF8nMzSPXnhiePyyqqipzNNEnHjg=="], - "fast-uri": ["fast-uri@3.1.1", "", {}, "sha512-h2r7rcm6Ee/J8o0LD5djLuFVcfbZxhvho4vvsbeV0aMvXjUgqv4YpxpkEx0d68l6+IleVfLAdVEfhR7QNMkGHQ=="], + "fast-uri": ["fast-uri@3.1.2", "", {}, "sha512-rVjf7ArG3LTk+FS6Yw81V1DLuZl1bRbNrev6Tmd/9RaroeeRRJhAt7jg/6YFxbvAQXUCavSoZhPPj6oOx+5KjQ=="], - "fast-xml-builder": ["fast-xml-builder@1.1.8", "", { "dependencies": { "path-expression-matcher": "^1.1.3" } }, "sha512-sDVBc2gg8pSKvcbE8rBmOyjSGQf0AdsbqvHeIOv3D/uYNoV4eCReQXyDF8Pdv8+m1FHazACypSz2hR7O2S1LLw=="], + "fast-xml-builder": ["fast-xml-builder@1.2.0", "", { "dependencies": { "path-expression-matcher": "^1.5.0", "xml-naming": "^0.1.0" } }, "sha512-00aAWieqff+ZJhsXA4g1g7M8k+7AYoMUUHF+/zFb5U6Uv/P0Vl4QZo84/IcufzYalLuEj9928bXN9PbbFzMF0Q=="], "fast-xml-parser": ["fast-xml-parser@5.7.2", "", { "dependencies": { "@nodable/entities": "^2.1.0", "fast-xml-builder": "^1.1.5", "path-expression-matcher": "^1.5.0", "strnum": "^2.2.3" }, "bin": { "fxparser": "src/cli/cli.js" } }, "sha512-P7oW7tLbYnhOLQk/Gv7cZgzgMPP/XN03K02/Jy6Y/NHzyIAIpxuZIM/YqAkfiXFPxA2CTm7NtCijK9EDu09u2w=="], @@ -3436,7 +3458,7 @@ "find-exec": ["find-exec@1.0.3", "", { "dependencies": { "shell-quote": "^1.8.1" } }, "sha512-gnG38zW90mS8hm5smNcrBnakPEt+cGJoiMkJwCU0IYnEb0H2NQk0NIljhNW+48oniCriFek/PH6QXbwsJo/qug=="], - "find-my-way": ["find-my-way@9.5.0", "", { "dependencies": { "fast-deep-equal": "^3.1.3", "fast-querystring": "^1.0.0", "safe-regex2": "^5.0.0" } }, "sha512-VW2RfnmscZO5KgBY5XVyKREMW5nMZcxDy+buTOsL+zIPnBlbKm+00sgzoQzq1EVh4aALZLfKdwv6atBGcjvjrQ=="], + "find-my-way": ["find-my-way@9.6.0", "", { "dependencies": { "fast-deep-equal": "^3.1.3", "fast-querystring": "^1.0.0", "safe-regex2": "^5.0.0" } }, "sha512-Zf4Xve4RymLl7NgaavNebZ01joJ8MfVerOG43wy7SHLO+r+K0C6d/SE0BiR7AV5V1VOCFlOP7ecdo+I4qmiHrQ=="], "find-my-way-ts": ["find-my-way-ts@0.1.6", "", {}, "sha512-a85L9ZoXtNAey3Y6Z+eBWW658kO/MwR7zIafkIUPUMf3isZG0NCs2pjW2wtjxAKuJPxMAsHUIP4ZPGv0o5gyTA=="], @@ -3502,7 +3524,7 @@ "get-caller-file": ["get-caller-file@2.0.5", "", {}, "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg=="], - "get-east-asian-width": ["get-east-asian-width@1.5.0", "", {}, "sha512-CQ+bEO+Tva/qlmw24dCejulK5pMzVnUOFOijVogd3KQs07HnRIgp8TGipvCCRT06xeYEbpbgwaCxglFyiuIcmA=="], + "get-east-asian-width": ["get-east-asian-width@1.6.0", "", {}, "sha512-QRbvDIbx6YklUe6RxeTeleMR0yv3cYH6PsPZHcnVn7xv7zO1BHN8r0XETu8n6Ye3Q+ahtSarc3WgtNWmehIBfA=="], "get-intrinsic": ["get-intrinsic@1.3.0", "", { "dependencies": { "call-bind-apply-helpers": "^1.0.2", "es-define-property": "^1.0.1", "es-errors": "^1.3.0", "es-object-atoms": "^1.1.1", "function-bind": "^1.1.2", "get-proto": "^1.0.1", "gopd": "^1.2.0", "has-symbols": "^1.1.0", "hasown": "^2.0.2", "math-intrinsics": "^1.1.0" } }, "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ=="], @@ -3544,7 +3566,7 @@ "graceful-fs": ["graceful-fs@4.2.11", "", {}, "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ=="], - "graphql": ["graphql@16.13.2", "", {}, "sha512-5bJ+nf/UCpAjHM8i06fl7eLyVC9iuNAjm9qzkiu2ZGhM0VscSvS6WDPfAwkdkBuoXGM9FJSbKl6wylMwP9Ktig=="], + "graphql": ["graphql@16.14.0", "", {}, "sha512-BBvQ/406p+4CZbTpCbVPSxfzrZrbnuWSP1ELYgyS6B+hNeKzgrdB4JczCa5VZUBQrDa9hUngm0KnexY6pJRN5Q=="], "graphql-request": ["graphql-request@6.1.0", "", { "dependencies": { "@graphql-typed-document-node/core": "^3.2.0", "cross-fetch": "^3.1.5" }, "peerDependencies": { "graphql": "14 - 16" } }, "sha512-p+XPfS4q7aIpKVcgmnZKhMNqhltk20hfXtkaIkTfjjmiKMJ5xrt5c743cL03y/K7y1rg3WrIC49xGiEQ4mxdNw=="], @@ -3626,6 +3648,8 @@ "import-fresh": ["import-fresh@3.3.1", "", { "dependencies": { "parent-module": "^1.0.0", "resolve-from": "^4.0.0" } }, "sha512-TR3KfrTZTYLPB6jUjfx6MF9WcWrHL9su5TObK4ZkYgBdWKPOFoSoQIdEuTuR82pmtxH2spWG9h6etwfr1pLBqQ=="], + "import-meta-resolve": ["import-meta-resolve@4.2.0", "", {}, "sha512-Iqv2fzaTQN28s/FwZAoFq0ZSs/7hMAHJVX+w8PZl3cY19Pxk6jFFalxQoIfW2826i/fDLXv8IiEZRIT0lDuWcg=="], + "imurmurhash": ["imurmurhash@0.1.4", "", {}, "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA=="], "indent-string": ["indent-string@5.0.0", "", {}, "sha512-m6FAo/spmsW2Ab2fU35JTYwtOKa2yAwXSwgjSv1TJzh4Mh7mC3lzAOVLBprb72XsTrgkEIsl7YrFNAiDiRhIGg=="], @@ -3642,7 +3666,7 @@ "ioredis": ["ioredis@5.10.1", "", { "dependencies": { "@ioredis/commands": "1.5.1", "cluster-key-slot": "^1.1.0", "debug": "^4.3.4", "denque": "^2.1.0", "lodash.defaults": "^4.2.0", "lodash.isarguments": "^3.1.0", "redis-errors": "^1.2.0", "redis-parser": "^3.0.0", "standard-as-callback": "^2.1.0" } }, "sha512-HuEDBTI70aYdx1v6U97SbNx9F1+svQKBDo30o0b9fw055LMepzpOOd0Ccg9Q6tbqmBSJaMuY0fB7yw9/vjBYCA=="], - "ip-address": ["ip-address@10.1.0", "", {}, "sha512-XXADHxXmvT9+CRxhXg56LJovE+bmWnEWB78LB83VZTprKTmaC5QfruXocxzTZ2Kl0DNwKuBdlIhjL8LeY8Sf8Q=="], + "ip-address": ["ip-address@10.2.0", "", {}, "sha512-/+S6j4E9AHvW9SWMSEY9Xfy66O5PWvVEJ08O0y5JGyEKQpojb0K0GKpz/v5HJ/G0vi3D2sjGK78119oXZeE0qA=="], "ipaddr.js": ["ipaddr.js@2.4.0", "", {}, "sha512-9VGk3HGanVE6JoZXHiCpnGy5X0jYDnN4EA4lntFPj+1vIWlFhIylq2CrrCOJH9EAhc5CYhq18F2Av2tgoAPsYQ=="], @@ -3654,7 +3678,7 @@ "is-callable": ["is-callable@1.2.7", "", {}, "sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA=="], - "is-core-module": ["is-core-module@2.16.1", "", { "dependencies": { "hasown": "^2.0.2" } }, "sha512-UfoeMA6fIJ8wTYFEUjelnaGI67v6+N7qXJEvQuIGa99l4xsCruSYOVSQ0uPANn4dAzm8lkYPaKLrrijLq7x23w=="], + "is-core-module": ["is-core-module@2.16.2", "", { "dependencies": { "hasown": "^2.0.3" } }, "sha512-evOr8xfXKxE6qSR0hSXL2r3sd7ALj8+7jQEUvPYcm5sgZFdJ+AYzT6yNmJenvIYQBgIGwfwz08sL8zoL7yq2BA=="], "is-docker": ["is-docker@3.0.0", "", { "bin": { "is-docker": "cli.js" } }, "sha512-eljcgEDlEns/7AXFosB5K/2nCM4P7FQPkGc/DWLy5rmFEWvZayGrik1d9/QIY5nJ4f9YsVvBkA6kJpHn9rISdQ=="], @@ -3728,7 +3752,7 @@ "jimp": ["jimp@1.6.0", "", { "dependencies": { "@jimp/core": "1.6.0", "@jimp/diff": "1.6.0", "@jimp/js-bmp": "1.6.0", "@jimp/js-gif": "1.6.0", "@jimp/js-jpeg": "1.6.0", "@jimp/js-png": "1.6.0", "@jimp/js-tiff": "1.6.0", "@jimp/plugin-blit": "1.6.0", "@jimp/plugin-blur": "1.6.0", "@jimp/plugin-circle": "1.6.0", "@jimp/plugin-color": "1.6.0", "@jimp/plugin-contain": "1.6.0", "@jimp/plugin-cover": "1.6.0", "@jimp/plugin-crop": "1.6.0", "@jimp/plugin-displace": "1.6.0", "@jimp/plugin-dither": "1.6.0", "@jimp/plugin-fisheye": "1.6.0", "@jimp/plugin-flip": "1.6.0", "@jimp/plugin-hash": "1.6.0", "@jimp/plugin-mask": "1.6.0", "@jimp/plugin-print": "1.6.0", "@jimp/plugin-quantize": "1.6.0", "@jimp/plugin-resize": "1.6.0", "@jimp/plugin-rotate": "1.6.0", "@jimp/plugin-threshold": "1.6.0", "@jimp/types": "1.6.0", "@jimp/utils": "1.6.0" } }, "sha512-YcwCHw1kiqEeI5xRpDlPPBGL2EOpBKLwO4yIBJcXWHPj5PnA5urGq0jbyhM5KoNpypQ6VboSoxc9D8HyfvngSg=="], - "jiti": ["jiti@2.6.1", "", { "bin": { "jiti": "lib/jiti-cli.mjs" } }, "sha512-ekilCSN1jwRvIbgeg/57YFh8qQDNbwDb9xT/qu2DAHbFFZUicIl4ygVaAvzveMhMVr3LnpSKTNnwt8PoOfmKhQ=="], + "jiti": ["jiti@2.7.0", "", { "bin": { "jiti": "lib/jiti-cli.mjs" } }, "sha512-AC/7JofJvZGrrneWNaEnJeOLUx+JlGt7tNa0wZiRPT4MY1wmfKjt2+6O2p2uz2+skll8OZZmJMNqeke7kKbNgQ=="], "jmespath": ["jmespath@0.16.0", "", {}, "sha512-9FzQjJ7MATs1tSpnco1K6ayiYE3figslrXA72G2HQ/n76RzvYlofyi5QM+iX4YRs/pu3yzxlVQSST23+dMDknw=="], @@ -3818,8 +3842,6 @@ "kubernetes-types": ["kubernetes-types@1.30.0", "", {}, "sha512-Dew1okvhM/SQcIa2rcgujNndZwU8VnSapDgdxlYoB84ZlpAD43U6KLAFqYo17ykSFGHNPrg0qry0bP+GJd9v7Q=="], - "langium": ["langium@4.2.3", "", { "dependencies": { "@chevrotain/regexp-to-ast": "~12.0.0", "chevrotain": "~12.0.0", "chevrotain-allstar": "~0.4.3", "vscode-languageserver": "~9.0.1", "vscode-languageserver-textdocument": "~1.0.11", "vscode-uri": "~3.1.0" } }, "sha512-sOPIi4hISFnY7twwV97ca1TsxpBtXq0URu/LL1AvxwccPG/RIBBlKS7a/f/EL6w8lTNaS0EFs/F+IdSOaqYpng=="], - "layout-base": ["layout-base@1.0.2", "", {}, "sha512-8h2oVEZNktL4BH2JCOI90iD1yXwL6iNW7KcCKT2QZgQJR2vbqDsldCTPRU9NifTCqHZci57XvQQ15YTu+sTYPg=="], "lazy-val": ["lazy-val@1.0.5", "", {}, "sha512-0/BnGCCfyUMkBpeDgWihanIAF9JmZhHBgUhEqzvf+adhNGLoP6TaiI5oF8oyb3I45P+PcnrqihSf01M0l0G5+Q=="], @@ -3860,7 +3882,7 @@ "linkify-it": ["linkify-it@5.0.0", "", { "dependencies": { "uc.micro": "^2.0.0" } }, "sha512-5aHCbzQRADcdP+ATqnDuhhJ/MRIqDkZX5pyjFHRRysS8vZ5AbqGEoFIb6pYHPZ+L/OC2Lc+xT8uHVVR5CAK/wQ=="], - "linkifyjs": ["linkifyjs@4.3.2", "", {}, "sha512-NT1CJtq3hHIreOianA8aSXn6Cw0JzYOuDQbOrSPe7gqFnCpKP++MQe3ODgO3oh2GJFORkAAdqredOa60z63GbA=="], + "linkifyjs": ["linkifyjs@4.3.3", "", {}, "sha512-P8aEP5U/D1/IlTY2OeYsErdwh9bGuLE30NcXtKEjgdHcahveQoQwM2yZNsioQHsWFz0P7KKudisbrzCgR0sDHg=="], "load-json-file": ["load-json-file@7.0.1", "", {}, "sha512-Gnxj3ev3mB5TkVBGad0JM6dmLiQL+o0t23JPBZ9sd+yvSLk05mFoqKBw5N8gbbkU4TNXyqCgIrl/VM17OgUIgQ=="], @@ -4014,8 +4036,6 @@ "mkdirp-classic": ["mkdirp-classic@0.5.3", "", {}, "sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A=="], - "mlly": ["mlly@1.8.2", "", { "dependencies": { "acorn": "^8.16.0", "pathe": "^2.0.3", "pkg-types": "^1.3.1", "ufo": "^1.6.3" } }, "sha512-d+ObxMQFmbt10sretNDytwt85VrbkhhUA/JBGm1MPaWJ65Cl4wOgLaB1NYvJSZ0Ef03MMEU/0xpPMXUIQ29UfA=="], - "mocha": ["mocha@11.7.5", "", { "dependencies": { "browser-stdout": "^1.3.1", "chokidar": "^4.0.1", "debug": "^4.3.5", "diff": "^7.0.0", "escape-string-regexp": "^4.0.0", "find-up": "^5.0.0", "glob": "^10.4.5", "he": "^1.2.0", "is-path-inside": "^3.0.3", "js-yaml": "^4.1.0", "log-symbols": "^4.1.0", "minimatch": "^9.0.5", "ms": "^2.1.3", "picocolors": "^1.1.1", "serialize-javascript": "^6.0.2", "strip-json-comments": "^3.1.1", "supports-color": "^8.1.1", "workerpool": "^9.2.0", "yargs": "^17.7.2", "yargs-parser": "^21.1.1", "yargs-unparser": "^2.0.0" }, "bin": { "mocha": "bin/mocha.js", "_mocha": "bin/_mocha" } }, "sha512-mTT6RgopEYABzXWFx+GcJ+ZQ32kp4fMf0xvpZIIfSq9Z8lC/++MtcCnQ9t5FP2veYEP95FIYSvW+U9fV4xrlig=="], "morphdom": ["morphdom@2.7.8", "", {}, "sha512-D/fR4xgGUyVRbdMGU6Nejea1RFzYxYtyurG4Fbv2Fi/daKlWKuXGLOdXtl+3eIwL110cI2hz1ZojGICjjFLgTg=="], @@ -4054,9 +4074,9 @@ "negotiator": ["negotiator@1.0.0", "", {}, "sha512-8Ofs/AUQh8MaEcrlq5xOX0CQ9ypTF5dl78mjlMNfOK08fzpgTHQRQPBxcPlEtIw0yRpws+Zo/3r+5WRby7u3Gg=="], - "next": ["next@16.2.4", "", { "dependencies": { "@next/env": "16.2.4", "@swc/helpers": "0.5.15", "baseline-browser-mapping": "^2.9.19", "caniuse-lite": "^1.0.30001579", "postcss": "8.4.31", "styled-jsx": "5.1.6" }, "optionalDependencies": { "@next/swc-darwin-arm64": "16.2.4", "@next/swc-darwin-x64": "16.2.4", "@next/swc-linux-arm64-gnu": "16.2.4", "@next/swc-linux-arm64-musl": "16.2.4", "@next/swc-linux-x64-gnu": "16.2.4", "@next/swc-linux-x64-musl": "16.2.4", "@next/swc-win32-arm64-msvc": "16.2.4", "@next/swc-win32-x64-msvc": "16.2.4", "sharp": "^0.34.5" }, "peerDependencies": { "@opentelemetry/api": "^1.1.0", "@playwright/test": "^1.51.1", "babel-plugin-react-compiler": "*", "react": "^18.2.0 || 19.0.0-rc-de68d2f4-20241204 || ^19.0.0", "react-dom": "^18.2.0 || 19.0.0-rc-de68d2f4-20241204 || ^19.0.0", "sass": "^1.3.0" }, "optionalPeers": ["@opentelemetry/api", "@playwright/test", "babel-plugin-react-compiler", "sass"], "bin": { "next": "dist/bin/next" } }, "sha512-kPvz56wF5frc+FxlHI5qnklCzbq53HTwORaWBGdT0vNoKh1Aya9XC8aPauH4NJxqtzbWsS5mAbctm4cr+EkQ2Q=="], + "next": ["next@16.2.6", "", { "dependencies": { "@next/env": "16.2.6", "@swc/helpers": "0.5.15", "baseline-browser-mapping": "^2.9.19", "caniuse-lite": "^1.0.30001579", "postcss": "8.4.31", "styled-jsx": "5.1.6" }, "optionalDependencies": { "@next/swc-darwin-arm64": "16.2.6", "@next/swc-darwin-x64": "16.2.6", "@next/swc-linux-arm64-gnu": "16.2.6", "@next/swc-linux-arm64-musl": "16.2.6", "@next/swc-linux-x64-gnu": "16.2.6", "@next/swc-linux-x64-musl": "16.2.6", "@next/swc-win32-arm64-msvc": "16.2.6", "@next/swc-win32-x64-msvc": "16.2.6", "sharp": "^0.34.5" }, "peerDependencies": { "@opentelemetry/api": "^1.1.0", "@playwright/test": "^1.51.1", "babel-plugin-react-compiler": "*", "react": "^18.2.0 || 19.0.0-rc-de68d2f4-20241204 || ^19.0.0", "react-dom": "^18.2.0 || 19.0.0-rc-de68d2f4-20241204 || ^19.0.0", "sass": "^1.3.0" }, "optionalPeers": ["@opentelemetry/api", "@playwright/test", "babel-plugin-react-compiler", "sass"], "bin": { "next": "dist/bin/next" } }, "sha512-qOVgKJg1+At15NpeUP+eJgCHvTCgXsogweq87Ri/Ix7PkqQHg4sdaXmSFqKlgaIXE4kW0g25LE68W87UANlHtw=="], - "node-abi": ["node-abi@4.29.0", "", { "dependencies": { "semver": "^7.6.3" } }, "sha512-bGc7hHz6lrdpMqH3XqfiHc5PKzEhjgUj6OLpTXynkLi9JZKyMByI/tdpm4Liu6O2BjtE1lakBWXjOQS1EnSQLQ=="], + "node-abi": ["node-abi@4.31.0", "", { "dependencies": { "semver": "^7.6.3" } }, "sha512-Erq5w/t3syw3s4sDsUaX4QttIdBPsGKTT1DTRsCkTonGggczhlDKm/wDX3o+HPJpQ41EjXCbcmXf0tgr5YZJXw=="], "node-addon-api": ["node-addon-api@7.1.1", "", {}, "sha512-5m3bsyrjFWE1xf7nz7YXdN4udnVtXK6/Yfgn5qnahL6bCkf2yKt4k3nuTKAtT4r3IG8JNR2ncsIMdZuAzJjHQQ=="], @@ -4076,7 +4096,7 @@ "node-html-parser": ["node-html-parser@7.1.0", "", { "dependencies": { "css-select": "^5.1.0", "he": "1.2.0" } }, "sha512-iJo8b2uYGT40Y8BTyy5ufL6IVbN8rbm/1QK2xffXU/1a/v3AAa0d1YAoqBNYqaS4R/HajkWIpIfdE6KcyFh1AQ=="], - "node-releases": ["node-releases@2.0.38", "", {}, "sha512-3qT/88Y3FbH/Kx4szpQQ4HzUbVrHPKTLVpVocKiLfoYvw9XSGOX2FmD2d6DrXbVYyAQTF2HeF6My8jmzx7/CRw=="], + "node-releases": ["node-releases@2.0.44", "", {}, "sha512-5WUyunoPMsvvEhS8AxHtRzP+oA8UCkJ7YRxatWKjngndhDGLiqEVAQKWjFAiAiuL8zMRGzGSJxFnLetoa43qGQ=="], "node-sarif-builder": ["node-sarif-builder@3.4.0", "", { "dependencies": { "@types/sarif": "^2.1.7", "fs-extra": "^11.1.1" } }, "sha512-tGnJW6OKRii9u/b2WiUViTJS+h7Apxx17qsMUjsUeNDiMMX5ZFf8F8Fcz7PAQ6omvOxHZtvDTmOYKJQwmfpjeg=="], @@ -4158,6 +4178,8 @@ "outdent": ["outdent@0.5.0", "", {}, "sha512-/jHxFIzoMXdqPzTaCpFzAAWhpkSjZPF4Vsn6jAfNpmbH/ymsmd7Qc6VE9BGn0L6YMj6uwpQLxCECpus4ukKS9Q=="], + "oxc-parser": ["oxc-parser@0.127.0", "", { "dependencies": { "@oxc-project/types": "^0.127.0" }, "optionalDependencies": { "@oxc-parser/binding-android-arm-eabi": "0.127.0", "@oxc-parser/binding-android-arm64": "0.127.0", "@oxc-parser/binding-darwin-arm64": "0.127.0", "@oxc-parser/binding-darwin-x64": "0.127.0", "@oxc-parser/binding-freebsd-x64": "0.127.0", "@oxc-parser/binding-linux-arm-gnueabihf": "0.127.0", "@oxc-parser/binding-linux-arm-musleabihf": "0.127.0", "@oxc-parser/binding-linux-arm64-gnu": "0.127.0", "@oxc-parser/binding-linux-arm64-musl": "0.127.0", "@oxc-parser/binding-linux-ppc64-gnu": "0.127.0", "@oxc-parser/binding-linux-riscv64-gnu": "0.127.0", "@oxc-parser/binding-linux-riscv64-musl": "0.127.0", "@oxc-parser/binding-linux-s390x-gnu": "0.127.0", "@oxc-parser/binding-linux-x64-gnu": "0.127.0", "@oxc-parser/binding-linux-x64-musl": "0.127.0", "@oxc-parser/binding-openharmony-arm64": "0.127.0", "@oxc-parser/binding-wasm32-wasi": "0.127.0", "@oxc-parser/binding-win32-arm64-msvc": "0.127.0", "@oxc-parser/binding-win32-ia32-msvc": "0.127.0", "@oxc-parser/binding-win32-x64-msvc": "0.127.0" } }, "sha512-bkgD4qHlN7WxLdX8bLXdaU54TtQtAIg/ZBAfm0aje/mo3MRDo3P0hZSgr4U7O3xfX+fQmR5AP04JS/TGcZLcFA=="], + "oxc-resolver": ["oxc-resolver@11.19.1", "", { "optionalDependencies": { "@oxc-resolver/binding-android-arm-eabi": "11.19.1", "@oxc-resolver/binding-android-arm64": "11.19.1", "@oxc-resolver/binding-darwin-arm64": "11.19.1", "@oxc-resolver/binding-darwin-x64": "11.19.1", "@oxc-resolver/binding-freebsd-x64": "11.19.1", "@oxc-resolver/binding-linux-arm-gnueabihf": "11.19.1", "@oxc-resolver/binding-linux-arm-musleabihf": "11.19.1", "@oxc-resolver/binding-linux-arm64-gnu": "11.19.1", "@oxc-resolver/binding-linux-arm64-musl": "11.19.1", "@oxc-resolver/binding-linux-ppc64-gnu": "11.19.1", "@oxc-resolver/binding-linux-riscv64-gnu": "11.19.1", "@oxc-resolver/binding-linux-riscv64-musl": "11.19.1", "@oxc-resolver/binding-linux-s390x-gnu": "11.19.1", "@oxc-resolver/binding-linux-x64-gnu": "11.19.1", "@oxc-resolver/binding-linux-x64-musl": "11.19.1", "@oxc-resolver/binding-openharmony-arm64": "11.19.1", "@oxc-resolver/binding-wasm32-wasi": "11.19.1", "@oxc-resolver/binding-win32-arm64-msvc": "11.19.1", "@oxc-resolver/binding-win32-ia32-msvc": "11.19.1", "@oxc-resolver/binding-win32-x64-msvc": "11.19.1" } }, "sha512-qE/CIg/spwrTBFt5aKmwe3ifeDdLfA2NESN30E42X/lII5ClF8V7Wt6WIJhcGZjp0/Q+nQ+9vgxGk//xZNX2hg=="], "oxlint": ["oxlint@1.60.0", "", { "optionalDependencies": { "@oxlint/binding-android-arm-eabi": "1.60.0", "@oxlint/binding-android-arm64": "1.60.0", "@oxlint/binding-darwin-arm64": "1.60.0", "@oxlint/binding-darwin-x64": "1.60.0", "@oxlint/binding-freebsd-x64": "1.60.0", "@oxlint/binding-linux-arm-gnueabihf": "1.60.0", "@oxlint/binding-linux-arm-musleabihf": "1.60.0", "@oxlint/binding-linux-arm64-gnu": "1.60.0", "@oxlint/binding-linux-arm64-musl": "1.60.0", "@oxlint/binding-linux-ppc64-gnu": "1.60.0", "@oxlint/binding-linux-riscv64-gnu": "1.60.0", "@oxlint/binding-linux-riscv64-musl": "1.60.0", "@oxlint/binding-linux-s390x-gnu": "1.60.0", "@oxlint/binding-linux-x64-gnu": "1.60.0", "@oxlint/binding-linux-x64-musl": "1.60.0", "@oxlint/binding-openharmony-arm64": "1.60.0", "@oxlint/binding-win32-arm64-msvc": "1.60.0", "@oxlint/binding-win32-ia32-msvc": "1.60.0", "@oxlint/binding-win32-x64-msvc": "1.60.0" }, "peerDependencies": { "oxlint-tsgolint": ">=0.18.0" }, "optionalPeers": ["oxlint-tsgolint"], "bin": { "oxlint": "bin/oxlint" } }, "sha512-tnRzTWiWJ9pg3ftRWnD0+Oqh78L6ZSwcEudvCZaER0PIqiAnNyXj5N1dPwjmNpDalkKS9m/WMLN1CTPUBPmsgw=="], @@ -4232,7 +4254,7 @@ "path-scurry": ["path-scurry@2.0.2", "", { "dependencies": { "lru-cache": "^11.0.0", "minipass": "^7.1.2" } }, "sha512-3O/iVVsJAPsOnpwWIeD+d6z/7PmqApyQePUtCndjatj/9I5LylHvt5qluFaBT3I5h3r1ejfR056c+FCv+NnNXg=="], - "path-to-regexp": ["path-to-regexp@8.4.2", "", {}, "sha512-qRcuIdP69NPm4qbACK+aDogI5CBDMi1jKe0ry5rSQJz8JVLsC7jV8XpiJjGRLLol3N+R5ihGYcrPLTno6pAdBA=="], + "path-to-regexp": ["path-to-regexp@0.1.13", "", {}, "sha512-A/AGNMFN3c8bOlvV9RreMdrv7jsmF9XIfDeCd87+I8RNg6s78BhJxMu69NEMHBSJFxKidViTEdruRwEk/WIKqA=="], "path-type": ["path-type@6.0.0", "", {}, "sha512-Vj7sf++t5pBD637NSfkxpHSMfWaeig5+DKWLhcqIYx6mWQz5hdJTGDVMQiJcw1ZYkhs7AazKDGpRVji1LJCZUQ=="], @@ -4266,7 +4288,7 @@ "pkg-conf": ["pkg-conf@4.0.0", "", { "dependencies": { "find-up": "^6.0.0", "load-json-file": "^7.0.0" } }, "sha512-7dmgi4UY4qk+4mj5Cd8v/GExPo0K+SlY+hulOSdfZ/T6jVH6//y7NtzZo5WrfhDBxuQ0jCa7fLZmNaNh7EWL/w=="], - "pkg-types": ["pkg-types@1.3.1", "", { "dependencies": { "confbox": "^0.1.8", "mlly": "^1.7.4", "pathe": "^2.0.1" } }, "sha512-/Jm5M4RvtBFVkKWRu2BLUTNP8/M2a+UwuAX+ae4770q1qVGtfjG+WTCupoZixokjmHiry8uI+dlY8KXYV5HVVQ=="], + "pkg-types": ["pkg-types@2.3.1", "", { "dependencies": { "confbox": "^0.2.4", "exsolve": "^1.0.8", "pathe": "^2.0.3" } }, "sha512-y+ichcgc2LrADuhLNAx8DFjVfgz91pRxfZdI3UDhxHvcVEZsenLO+7XaU5vOp0u/7V/wZ+plyuQxtrDlZJ+yeg=="], "pkg-up": ["pkg-up@3.1.0", "", { "dependencies": { "find-up": "^3.0.0" } }, "sha512-nDywThFk1i4BQK4twPQ6TA4RT8bDY96yeuCVBWL3ePARCiEKDRSrNGbFIgUJpLp+XeIR65v8ra7WuJOFUBtkMA=="], @@ -4298,7 +4320,7 @@ "postcss-value-parser": ["postcss-value-parser@4.2.0", "", {}, "sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ=="], - "posthog-js": ["posthog-js@1.372.8", "", { "dependencies": { "@opentelemetry/api": "^1.9.0", "@opentelemetry/api-logs": "^0.208.0", "@opentelemetry/exporter-logs-otlp-http": "^0.208.0", "@opentelemetry/resources": "^2.2.0", "@opentelemetry/sdk-logs": "^0.208.0", "@posthog/core": "1.28.2", "@posthog/types": "1.372.8", "core-js": "^3.38.1", "dompurify": "^3.3.2", "fflate": "^0.4.8", "preact": "^10.28.2", "query-selector-shadow-dom": "^1.0.1", "web-vitals": "^5.1.0" } }, "sha512-NAFWVScFGnU5jgGNLkrY/UnGH82uULs9RsJ/f26LkLn1l0MOZfq+/z+2RaOOQQX4NHruIVuxqz3J/50bgRG68A=="], + "posthog-js": ["posthog-js@1.373.4", "", { "dependencies": { "@opentelemetry/api": "^1.9.0", "@opentelemetry/api-logs": "^0.208.0", "@opentelemetry/exporter-logs-otlp-http": "^0.208.0", "@opentelemetry/resources": "^2.2.0", "@opentelemetry/sdk-logs": "^0.208.0", "@posthog/core": "1.29.1", "@posthog/types": "1.373.4", "core-js": "^3.38.1", "dompurify": "^3.3.2", "fflate": "^0.4.8", "preact": "^10.28.2", "query-selector-shadow-dom": "^1.0.1", "web-vitals": "^5.1.0" } }, "sha512-6DueUS5KOGZlKsQlelLURmtp9PA52rdCxt/IvuoYKAErlkh2LczogMopUIpeqbwfOnGDQEf8gmDh4z/IFeU4CQ=="], "posthog-node": ["posthog-node@4.4.0", "", { "dependencies": { "axios": "^1.7.4" } }, "sha512-4FEFBuc4FDFkTEiCSIo7DFV0jpprlqfR/SzjVgg7E2rRvfxTW4J47IvCwSCo5C9lBeV0ujDubY/YArHG37aJIw=="], @@ -4342,11 +4364,11 @@ "property-information": ["property-information@7.1.0", "", {}, "sha512-TwEZ+X+yCJmYfL7TPUOcvBZ4QfoT5YenQiJuX//0th53DE6w0xxLEtfK3iyryQFddXuvkIk51EEgrJQ0WJkOmQ=="], - "protobufjs": ["protobufjs@7.5.6", "", { "dependencies": { "@protobufjs/aspromise": "^1.1.2", "@protobufjs/base64": "^1.1.2", "@protobufjs/codegen": "^2.0.5", "@protobufjs/eventemitter": "^1.1.0", "@protobufjs/fetch": "^1.1.0", "@protobufjs/float": "^1.0.2", "@protobufjs/inquire": "^1.1.1", "@protobufjs/path": "^1.1.2", "@protobufjs/pool": "^1.1.0", "@protobufjs/utf8": "^1.1.1", "@types/node": ">=13.7.0", "long": "^5.0.0" } }, "sha512-M71sTMB146U3u0di3yup8iM+zv8yPRNQVr1KK4tyBitl3qFvEGucq/rGDRShD2rsJhtN02RJaJ7j5X5hmy8SJg=="], + "protobufjs": ["protobufjs@7.5.8", "", { "dependencies": { "@protobufjs/aspromise": "^1.1.2", "@protobufjs/base64": "^1.1.2", "@protobufjs/codegen": "^2.0.5", "@protobufjs/eventemitter": "^1.1.0", "@protobufjs/fetch": "^1.1.0", "@protobufjs/float": "^1.0.2", "@protobufjs/inquire": "^1.1.1", "@protobufjs/path": "^1.1.2", "@protobufjs/pool": "^1.1.0", "@protobufjs/utf8": "^1.1.1", "@types/node": ">=13.7.0", "long": "^5.0.0" } }, "sha512-dvpCIeLPbXZS/Ete7yLaO7RenOdken2NHKykBXbsaGxZT0UTltcarBciw+A78SRQs9iMAAVpsYA+l8b1hTePIA=="], "proxy-addr": ["proxy-addr@2.0.7", "", { "dependencies": { "forwarded": "0.2.0", "ipaddr.js": "1.9.1" } }, "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg=="], - "proxy-from-env": ["proxy-from-env@1.1.0", "", {}, "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg=="], + "proxy-from-env": ["proxy-from-env@2.1.0", "", {}, "sha512-cJ+oHTW1VAEa8cJslgmUZrc+sjRKgAKl3Zyse6+PV38hZe/V6Z14TbCuXcan9F9ghlz4QrFr2c92TNF82UkYHA=="], "pump": ["pump@3.0.4", "", { "dependencies": { "end-of-stream": "^1.1.0", "once": "^1.3.1" } }, "sha512-VS7sjc6KR7e1ukRFhQSY5LM2uBWAUPiOPa/A3mkKmiMwSmRFUITt0xuj+/lesgnCv+dPIEYlkzrcyXgquIHMcA=="], @@ -4386,11 +4408,11 @@ "rc9": ["rc9@2.1.2", "", { "dependencies": { "defu": "^6.1.4", "destr": "^2.0.3" } }, "sha512-btXCnMmRIBINM2LDZoEmOogIZU7Qe7zn4BpomSKZ/ykbLObuBdvG+mFq11DL6fjH1DRwHhrlgtYWG96bJiC7Cg=="], - "react": ["react@19.2.5", "", {}, "sha512-llUJLzz1zTUBrskt2pwZgLq59AemifIftw4aB7JxOqf1HY2FDaGDxgwpAPVzHU1kdWabH7FauP4i1oEeer2WCA=="], + "react": ["react@19.2.6", "", {}, "sha512-sfWGGfavi0xr8Pg0sVsyHMAOziVYKgPLNrS7ig+ivMNb3wbCBw3KxtflsGBAwD3gYQlE/AEZsTLgToRrSCjb0Q=="], "react-docgen-typescript": ["react-docgen-typescript@2.4.0", "", { "peerDependencies": { "typescript": ">= 4.3.x" } }, "sha512-ZtAp5XTO5HRzQctjPU0ybY0RRCQO19X/8fxn3w7y2VVTUbGHDKULPTL4ky3vB05euSgG5NpALhEhDPvQ56wvXg=="], - "react-dom": ["react-dom@19.2.5", "", { "dependencies": { "scheduler": "^0.27.0" }, "peerDependencies": { "react": "^19.2.5" } }, "sha512-J5bAZz+DXMMwW/wV3xzKke59Af6CHY7G4uYLN1OvBcKEsWOs4pQExj86BBKamxl/Ik5bx9whOrvBlSDfWzgSag=="], + "react-dom": ["react-dom@19.2.6", "", { "dependencies": { "scheduler": "^0.27.0" }, "peerDependencies": { "react": "^19.2.6" } }, "sha512-0prMI+hvBbPjsWnxDLxlCGyM8PN6UuWjEUCYmZhO67xIV9Xasa/r/vDnq+Xyq4Lo27g8QSbO5YzARu0D1Sps3g=="], "react-is": ["react-is@17.0.2", "", {}, "sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w=="], @@ -4512,7 +4534,7 @@ "secure-json-parse": ["secure-json-parse@4.1.0", "", {}, "sha512-l4KnYfEyqYJxDwlNVyRfO2E4NTHfMKAWdUuA8J0yve2Dz/E/PdBepY03RvyJpssIpRFwJoCD55wA+mEDs6ByWA=="], - "semver": ["semver@7.7.4", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-vFKC2IEtQnVhpT78h1Yp8wzwrf8CM+MzKMHGJZfBtzhZNycRFnXsHk6E5TxIkkMsgNS7mdX3AGB7x2QM2di4lA=="], + "semver": ["semver@7.8.0", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-AcM7dV/5ul4EekoQ29Agm5vri8JNqRyj39o0qpX6vDF2GZrtutZl5RwgD1XnZjiTAfncsJhMI48QQH3sN87YNA=="], "semver-compare": ["semver-compare@1.0.0", "", {}, "sha512-YM3/ITh2MJ5MtzaM429anh+x2jiLVjqILF4m4oyQB18W7Ggea7BfqdH/wGMK7dDiMghv/6WG7znWMwUDzJiXow=="], @@ -4522,9 +4544,9 @@ "serialize-javascript": ["serialize-javascript@6.0.2", "", { "dependencies": { "randombytes": "^2.1.0" } }, "sha512-Saa1xPByTTq2gdeFZYLLo+RFE35NHZkAbqZeWNd3BpzppeVisAqpDjcp8dyf6uIvEqJRd46jemmyA4iFIeVk8g=="], - "seroval": ["seroval@1.5.3", "", {}, "sha512-BXe0x4buEeYiIKaRUnth1WqCILQ3k4O67KP/B4pC3pVz0Mv2c96ngA9QDREUYxWY1sb2RZVRqwI9RcpVMyHCVw=="], + "seroval": ["seroval@1.5.4", "", {}, "sha512-46uFvgrXTVxZcUorgSSRZ4y+ieqLLQRMlG4bnCZKW3qI6BZm7Rg4ntMW4p1mILEEBZWrFlcpp0AyIIlM6jD9iw=="], - "seroval-plugins": ["seroval-plugins@1.5.3", "", { "peerDependencies": { "seroval": "^1.0" } }, "sha512-LhVh4KjjkKmCxOUjoaUwtqbDjyMfnA535yEmmGDuwZcIYtw8ns6tZmeszNTECeUg/3sJpnEjsz/KhQrcPXPw1Q=="], + "seroval-plugins": ["seroval-plugins@1.5.4", "", { "peerDependencies": { "seroval": "^1.0" } }, "sha512-S0xQPhUTefAhNvNWFg0c1J8qJArHt5KdtJ/cFAofo06KD1MVSeFWyl4iiu+ApDIuw0WhjpOfCdgConOfAnLgkw=="], "serve-static": ["serve-static@2.2.1", "", { "dependencies": { "encodeurl": "^2.0.0", "escape-html": "^1.0.3", "parseurl": "^1.3.3", "send": "^1.2.0" } }, "sha512-xRXBn0pPqQTVQiC8wyQrKs2MOlX24zQ0POGaj0kultvoOCstBQM5yvOhAVSUwOMjQtTvsPWoNCHfPGwaaQJhTw=="], @@ -4568,7 +4590,7 @@ "simple-get": ["simple-get@4.0.1", "", { "dependencies": { "decompress-response": "^6.0.0", "once": "^1.3.1", "simple-concat": "^1.0.0" } }, "sha512-brv7p5WgH0jmQJr1ZDDfKDOSeWWg+OVypG99A/5vYGPqJ6pxiaHLy8nxtFjBA7oMa01ebA9gfh1uMCFqOuXxvA=="], - "simple-git": ["simple-git@3.35.2", "", { "dependencies": { "@kwsites/file-exists": "^1.1.1", "@kwsites/promise-deferred": "^1.1.1", "@simple-git/args-pathspec": "^1.0.2", "@simple-git/argv-parser": "^1.0.3", "debug": "^4.4.0" } }, "sha512-ZMjl06lzTm1EScxEGuM6+mEX+NQd14h/B3x0vWU+YOXAMF8sicyi1K4cjTfj5is+35ChJEHDl1EjypzYFWH2FA=="], + "simple-git": ["simple-git@3.36.0", "", { "dependencies": { "@kwsites/file-exists": "^1.1.1", "@kwsites/promise-deferred": "^1.1.1", "@simple-git/args-pathspec": "^1.0.3", "@simple-git/argv-parser": "^1.1.0", "debug": "^4.4.0" } }, "sha512-cGQjLjK8bxJw4QuYT7gxHw3/IouVESbhahSsHrX97MzCL1gu2u7oy38W6L2ZIGECEfIBG4BabsWDPjBxJENv9Q=="], "simple-update-notifier": ["simple-update-notifier@2.0.0", "", { "dependencies": { "semver": "^7.5.3" } }, "sha512-a2B9Y0KlNXl9u/vsW6sTIu9vGEpfKu2wRV6l1H3XEas/0gUIzGzBoP/IouTcUQbm9JWZLH3COxyn03TYlFax6w=="], @@ -4588,7 +4610,7 @@ "socket.io-parser": ["socket.io-parser@4.2.6", "", { "dependencies": { "@socket.io/component-emitter": "~3.1.0", "debug": "~4.4.1" } }, "sha512-asJqbVBDsBCJx0pTqw3WfesSY0iRX+2xzWEWzrpcH7L6fLzrhyF8WPI8UaeM4YCuDfpwA/cgsdugMsmtz8EJeg=="], - "socks": ["socks@2.8.8", "", { "dependencies": { "ip-address": "^10.1.1", "smart-buffer": "^4.2.0" } }, "sha512-NlGELfPrgX2f1TAAcz0WawlLn+0r3FyhhCRpFFK2CemXenPYvzMWWZINv3eDNo9ucdwme7oCHRY0Jnbs4aIkog=="], + "socks": ["socks@2.8.9", "", { "dependencies": { "ip-address": "^10.1.1", "smart-buffer": "^4.2.0" } }, "sha512-LJhUYUvItdQ0LkJTmPeaEObWXAqFyfmP85x0tch/ez9cahmhlBBLbIqDFnvBnUJGagb0JbIQrkBs1wJ+yRYpEw=="], "socks-proxy-agent": ["socks-proxy-agent@8.0.5", "", { "dependencies": { "agent-base": "^7.1.2", "debug": "^4.3.4", "socks": "^2.8.3" } }, "sha512-HehCEsotFqbPW9sJ8WVYB6UbmIMv7kUUORIF2Nncq4VQvBfNBLibW9YZR5dlYCSUhwcD628pRllm7n+E+YTzJw=="], @@ -4698,7 +4720,7 @@ "strip-literal": ["strip-literal@3.1.0", "", { "dependencies": { "js-tokens": "^9.0.1" } }, "sha512-8r3mkIM/2+PpjHoOtiAW8Rg3jJLHaV7xPwG+YRGrv6FP0wwk/toTpATxWYOW0BKdWwl82VT2tFYi5DlROa0Mxg=="], - "strnum": ["strnum@2.2.3", "", {}, "sha512-oKx6RUCuHfT3oyVjtnrmn19H1SiCqgJSg+54XqURKp5aCMbrXrhLjRN9TjuwMjiYstZ0MzDrHqkGZ5dFTKd+zg=="], + "strnum": ["strnum@2.3.0", "", {}, "sha512-ums3KNd42PGyx5xaoVTO1mjU1bH3NpY4vsrVlnv9PNGqQj8wd7rJ6nEypLrJ7z5vxK5RP0yMLo6J/Gsm62DI5Q=="], "strtok3": ["strtok3@6.3.0", "", { "dependencies": { "@tokenizer/token": "^0.3.0", "peek-readable": "^4.1.0" } }, "sha512-fZtbhtvI9I48xDSywd/somNqgUHl2L2cstmXCCif0itOf96jeW18MBSyrLuNicYQVkvpOxkZtkzujiTJ9LW5Jw=="], @@ -4730,11 +4752,11 @@ "table-layout": ["table-layout@4.1.1", "", { "dependencies": { "array-back": "^6.2.2", "wordwrapjs": "^5.1.0" } }, "sha512-iK5/YhZxq5GO5z8wb0bY1317uDF3Zjpha0QFFLA8/trAoiLbQD0HUbMesEaxyzUgDxi2QlcbM8IvqOlEjgoXBA=="], - "tailwindcss": ["tailwindcss@4.2.4", "", {}, "sha512-HhKppgO81FQof5m6TEnuBWCZGgfRAWbaeOaGT00KOy/Pf/j6oUihdvBpA7ltCeAvZpFhW3j0PTclkxsd4IXYDA=="], + "tailwindcss": ["tailwindcss@4.3.0", "", {}, "sha512-y6nxMGB1nMW9R6k96e5gdIFzcfL/gTJRNaqGes1YvkLnPVXzWgbqFF2yLC0T8G774n24cx3Pe8XrKoniCOAH+Q=="], "tapable": ["tapable@2.3.3", "", {}, "sha512-uxc/zpqFg6x7C8vOE7lh6Lbda8eEL9zmVm/PLeTPBRhh1xCgdWaQ+J1CUieGpIfm2HdtsUpRv+HshiasBMcc6A=="], - "tar": ["tar@7.5.13", "", { "dependencies": { "@isaacs/fs-minipass": "^4.0.0", "chownr": "^3.0.0", "minipass": "^7.1.2", "minizlib": "^3.1.0", "yallist": "^5.0.0" } }, "sha512-tOG/7GyXpFevhXVh8jOPJrmtRpOTsYqUIkVdVooZYJS/z8WhfQUX8RJILmeuJNinGAMSu1veBr4asSHFt5/hng=="], + "tar": ["tar@7.5.15", "", { "dependencies": { "@isaacs/fs-minipass": "^4.0.0", "chownr": "^3.0.0", "minipass": "^7.1.2", "minizlib": "^3.1.0", "yallist": "^5.0.0" } }, "sha512-dzGK0boVlC4W5QFuQN1EFSl3bIDYsk7Tj40U6eIBnK2k/8ml7TZ5agbI5j5+qnoVcAA+rNtBml8SEiLxZpNqRQ=="], "tar-fs": ["tar-fs@2.1.4", "", { "dependencies": { "chownr": "^1.1.1", "mkdirp-classic": "^0.5.2", "pump": "^3.0.0", "tar-stream": "^2.1.4" } }, "sha512-mDAjwmZdh7LTT6pNleZ05Yt65HC3E+NiQzl672vQG38jIrehtJk/J3mNwIg+vShQPcLF/LV7CMnDW6vjj6sfYQ=="], @@ -4742,7 +4764,7 @@ "tarn": ["tarn@3.0.2", "", {}, "sha512-51LAVKUSZSVfI05vjPESNc5vwqqZpbXCsU+/+wxlOrUjk2SnFTt97v9ZgQrD4YmxYW1Px6w2KjaDitCfkvgxMQ=="], - "tedious": ["tedious@18.6.2", "", { "dependencies": { "@azure/core-auth": "^1.7.2", "@azure/identity": "^4.2.1", "@azure/keyvault-keys": "^4.4.0", "@js-joda/core": "^5.6.1", "@types/node": ">=18", "bl": "^6.0.11", "iconv-lite": "^0.6.3", "js-md4": "^0.3.2", "native-duplexpair": "^1.0.0", "sprintf-js": "^1.1.3" } }, "sha512-g7jC56o3MzLkE3lHkaFe2ZdOVFBahq5bsB60/M4NYUbocw/MCrS89IOEQUFr+ba6pb8ZHczZ/VqCyYeYq0xBAg=="], + "tedious": ["tedious@19.2.1", "", { "dependencies": { "@azure/core-auth": "^1.7.2", "@azure/identity": "^4.2.1", "@azure/keyvault-keys": "^4.4.0", "@js-joda/core": "^5.6.5", "@types/node": ">=18", "bl": "^6.1.4", "iconv-lite": "^0.7.0", "js-md4": "^0.3.2", "native-duplexpair": "^1.0.0", "sprintf-js": "^1.1.3" } }, "sha512-pk1Q16Yl62iocuQB+RWbg6rFUFkIyzqOFQ6NfysCltRvQqKwfurgj8v/f2X+CKvDhSL4IJ0cCOfCHDg9PWEEYA=="], "teex": ["teex@1.0.1", "", { "dependencies": { "streamx": "^2.12.5" } }, "sha512-eYE6iEI62Ni1H8oIa7KlDU6uQBtqr4Eajni3wX7rpfXD8ysFx8z0+dri+KWEPWpBsxXfxu58x/0jvTVT1ekOSg=="], @@ -4764,7 +4786,7 @@ "textextensions": ["textextensions@6.11.0", "", { "dependencies": { "editions": "^6.21.0" } }, "sha512-tXJwSr9355kFJI3lbCkPpUH5cP8/M0GGy2xLO34aZCjMXBaK3SoPnZwr/oWmo1FdCnELcs4npdCIOFtq9W3ruQ=="], - "thread-stream": ["thread-stream@4.0.0", "", { "dependencies": { "real-require": "^0.2.0" } }, "sha512-4iMVL6HAINXWf1ZKZjIPcz5wYaOdPhtO8ATvZ+Xqp3BTdaqtAwQkNmKORqcIo5YkQqGXq5cwfswDwMqqQNrpJA=="], + "thread-stream": ["thread-stream@4.1.0", "", { "dependencies": { "real-require": "^1.0.0" } }, "sha512-Bw6h2iBDt16v6iHLChBIoVYU8CBo9GPsW8TG7h1hRVhqKhIkH6N8qkxNSmiOZTKsCLPbtWG4ViWLkU6KeKXpig=="], "three": ["three@0.177.0", "", {}, "sha512-EiXv5/qWAaGI+Vz2A+JfavwYCMdGjxVsrn3oBwllUoqYeaBO75J63ZfyaQKoiLrqNHoTlUc6PFgMXnS0kI45zg=="], @@ -4862,7 +4884,7 @@ "type-fest": ["type-fest@4.41.0", "", {}, "sha512-TeTSQ6H5YHvpqVwBRcnLDCBnDOHWYu7IvGbHT6N8AOymcr9PJGjc1GTtiWZTYg0NCgYwvnYWEkVChQAr9bjfwA=="], - "type-is": ["type-is@2.0.1", "", { "dependencies": { "content-type": "^1.0.5", "media-typer": "^1.1.0", "mime-types": "^3.0.0" } }, "sha512-OZs6gsjF4vMp32qrCbiVSkrFmXtG/AZhY3t0iAMrMBiAZyV9oALtXO8hsrHbMXF9x6L3grlFuwW2oAz7cav+Gw=="], + "type-is": ["type-is@2.1.0", "", { "dependencies": { "content-type": "^2.0.0", "media-typer": "^1.1.0", "mime-types": "^3.0.0" } }, "sha512-faYHw0anBbc/kWF3zFTEnxSFOAGUX9GFbOBthvDdLsIlEoWOFOtS0zgCiQYwIskL9iGXZL3kAXD8OoZ4GmMATA=="], "typed-array-buffer": ["typed-array-buffer@1.0.3", "", { "dependencies": { "call-bound": "^1.0.3", "es-errors": "^1.3.0", "is-typed-array": "^1.1.14" } }, "sha512-nAYYwfY3qnzX30IkA6AQZjVbtK6duGontcQm1WSG1MD94YLqK0515GNApXkoxKOWMusVssAHWLh9SeaoefYFGw=="], @@ -4870,14 +4892,12 @@ "typescript": ["typescript@5.8.2", "", { "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" } }, "sha512-aJn6wq13/afZp/jT9QZmwEjDqqvSGp1VT5GVg+f/t6/oVyrgXM6BY1h9BRh/O5p3PlUPAe+WuiEZOmb/49RqoQ=="], - "typescript-eslint": ["typescript-eslint@8.59.2", "", { "dependencies": { "@typescript-eslint/eslint-plugin": "8.59.2", "@typescript-eslint/parser": "8.59.2", "@typescript-eslint/typescript-estree": "8.59.2", "@typescript-eslint/utils": "8.59.2" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", "typescript": ">=4.8.4 <6.1.0" } }, "sha512-pJw051uomb3ZeCzGTpRb8RbEqB5Y4WWet8gl/GcTlU35BSx0PVdZ86/bqkQCyKKuraVQEK7r6kBHQXF+fBhkoQ=="], + "typescript-eslint": ["typescript-eslint@8.59.3", "", { "dependencies": { "@typescript-eslint/eslint-plugin": "8.59.3", "@typescript-eslint/parser": "8.59.3", "@typescript-eslint/typescript-estree": "8.59.3", "@typescript-eslint/utils": "8.59.3" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", "typescript": ">=4.8.4 <6.1.0" } }, "sha512-KgusgyDgG4LI8Ih/sWaCtZ06tckLAS5CvT5A4D1Q7bYVoAAyzwiZvE4BmwDHkhRVkvhRBepKeASoFzQetha7Fg=="], "typical": ["typical@4.0.0", "", {}, "sha512-VAH4IvQ7BDFYglMd7BPRDfLgxZZX4O4TFcRDA6EN5X7erNJJq+McIEp8np9aVtxrCJ6qx4GTYVfOWNjcqwZgRw=="], "uc.micro": ["uc.micro@2.1.0", "", {}, "sha512-ARDJmphmdvUk6Glw7y9DQ2bFkKBHwQHLi2lsaH6PPmz/Ka9sFOBsBluozhDltWmnv9u/cF6Rt87znRTPV+yp/A=="], - "ufo": ["ufo@1.6.4", "", {}, "sha512-JFNbkD1Svwe0KvGi8GOeLcP4kAWQ609twvCdcHxq1oSL8svv39ZuSvajcD8B+5D0eL4+s1Is2D/O6KN3qcTeRA=="], - "uint8array-extras": ["uint8array-extras@1.5.0", "", {}, "sha512-rvKSBiC5zqCCiDZ9kAOszZcDvdAHwwIKJG33Ykj43OKcWsnmcBRL09YTU4nOeHZ8Y2a7l1MgTd08SBe9A8Qj6A=="], "ulid": ["ulid@3.0.1", "", { "bin": { "ulid": "dist/cli.js" } }, "sha512-dPJyqPzx8preQhqq24bBG1YNkvigm87K8kVEHCD+ruZg24t6IFEFv00xMWfxcC4djmFtiTLdFuADn4+DOz6R7Q=="], @@ -4886,7 +4906,7 @@ "undici": ["undici@5.29.0", "", { "dependencies": { "@fastify/busboy": "^2.0.0" } }, "sha512-raqeBD6NQK4SkWhQzeYKd1KmIG6dllBOTt55Rmkt4HtI9mwdWtJljnrXjAFUBLTSN67HWrOIZ3EPF4kjUw80Bg=="], - "undici-types": ["undici-types@6.20.0", "", {}, "sha512-Ny6QZ2Nju20vw1SRHe3d9jVu6gJ+4e3+MMpqu7pqE5HT6WsTSlce++GQmK5UXS8mzV8DSYHrQH+Xrf2jVcuKNg=="], + "undici-types": ["undici-types@7.21.0", "", {}, "sha512-w9IMgQrz4O0YN1LtB7K5P63vhlIOvC7opSmouCJ+ZywlPAlO9gIkJ+otk6LvGpAs2wg4econaCz3TvQ9xPoyuQ=="], "unicorn-magic": ["unicorn-magic@0.3.0", "", {}, "sha512-+QBBXBCvifc56fsbuxZQ6Sic3wqqc3WWaqxs58gvJrcOuN83HGTCwz3oS5phzU9LthRNE9VrJCFCLUgHeeFnfA=="], @@ -4936,7 +4956,7 @@ "v8-to-istanbul": ["v8-to-istanbul@9.3.0", "", { "dependencies": { "@jridgewell/trace-mapping": "^0.3.12", "@types/istanbul-lib-coverage": "^2.0.1", "convert-source-map": "^2.0.0" } }, "sha512-kiGUalWN+rgBJ/1OHZsBtU4rXZOfj/7rKQxULKlIzwzQSvMJUUNgPwJEEh7gU6xEVxC0ahoOBvN2YI8GH6FNgA=="], - "valibot": ["valibot@1.3.1", "", { "peerDependencies": { "typescript": ">=5" }, "optionalPeers": ["typescript"] }, "sha512-sfdRir/QFM0JaF22hqTroPc5xy4DimuGQVKFrzF1YfGwaS1nJot3Y8VqMdLO2Lg27fMzat2yD3pY5PbAYO39Gg=="], + "valibot": ["valibot@1.4.0", "", { "peerDependencies": { "typescript": ">=5" }, "optionalPeers": ["typescript"] }, "sha512-iC/x7fVcSyOwlm/VSt7RlHnzNGLGvR9GnxdifUeWoCJo0q4ZZvrVkIHC6faTlkxG47I2Y4UrFquPuVHCrOnrLg=="], "validate-npm-package-license": ["validate-npm-package-license@3.0.4", "", { "dependencies": { "spdx-correct": "^3.0.0", "spdx-expression-parse": "^3.0.0" } }, "sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew=="], @@ -4970,16 +4990,8 @@ "vscode-jsonrpc": ["vscode-jsonrpc@8.2.1", "", {}, "sha512-kdjOSJ2lLIn7r1rtrMbbNCHjyMPfRnowdKjBQ+mGq6NAW5QY2bEZC/khaC5OR8svbbjvLEaIXkOq45e2X9BIbQ=="], - "vscode-languageserver": ["vscode-languageserver@9.0.1", "", { "dependencies": { "vscode-languageserver-protocol": "3.17.5" }, "bin": { "installServerIntoExtension": "bin/installServerIntoExtension" } }, "sha512-woByF3PDpkHFUreUa7Hos7+pUWdeWMXRd26+ZX2A8cFx6v/JPTtd4/uN0/jB6XQHYaOlHbio03NTHCqrgG5n7g=="], - - "vscode-languageserver-protocol": ["vscode-languageserver-protocol@3.17.5", "", { "dependencies": { "vscode-jsonrpc": "8.2.0", "vscode-languageserver-types": "3.17.5" } }, "sha512-mb1bvRJN8SVznADSGWM9u/b07H7Ecg0I3OgXDuLdn307rl/J3A9YD6/eYOssqhecL27hK1IPZAsaqh00i/Jljg=="], - - "vscode-languageserver-textdocument": ["vscode-languageserver-textdocument@1.0.12", "", {}, "sha512-cxWNPesCnQCcMPeenjKKsOCKQZ/L6Tv19DTRIGuLWe32lyzWhihGVJ/rcckZXJxfdKCFvRLS3fpBIsV/ZGX4zA=="], - "vscode-languageserver-types": ["vscode-languageserver-types@3.17.5", "", {}, "sha512-Ld1VelNuX9pdF39h2Hgaeb5hEZM2Z3jUrrMgWQAu82jMtZp7p3vJT3BzToKtZI7NgQssZje5o0zryOrhQvzQAg=="], - "vscode-uri": ["vscode-uri@3.1.0", "", {}, "sha512-/BpdSx+yCQGnCvecbyXdxHDkuk55/G3xwnC0GqY4gmQ3j+A+g8kzzgB4Nk/SINjqn6+waqw3EgbVF2QKExkRxQ=="], - "walk-up-path": ["walk-up-path@4.0.0", "", {}, "sha512-3hu+tD8YzSLGuFYtPRb48vdhKMi0KQV5sn+uWr8+7dMEq/2G/dtLrdDinkLjqq5TIbIBjYJ4Ax/n3YiaW7QM8A=="], "web-streams-polyfill": ["web-streams-polyfill@4.0.0-beta.3", "", {}, "sha512-QW95TCTaHmsYfHDybGMwO5IJIM93I/6vTRk+daHTWFPhwh+C8Cg7j7XyKrwrj8Ib6vYXe0ocYNrmzY4xAAN6ug=="], @@ -5024,12 +5036,14 @@ "write-file-atomic": ["write-file-atomic@7.0.1", "", { "dependencies": { "signal-exit": "^4.0.1" } }, "sha512-OTIk8iR8/aCRWBqvxrzxR0hgxWpnYBblY1S5hDWBQfk/VFmJwzmJgQFN3WsoUKHISv2eAwe+PpbUzyL1CKTLXg=="], - "ws": ["ws@8.20.0", "", { "peerDependencies": { "bufferutil": "^4.0.1", "utf-8-validate": ">=5.0.2" }, "optionalPeers": ["bufferutil", "utf-8-validate"] }, "sha512-sAt8BhgNbzCtgGbt2OxmpuryO63ZoDk/sqaB/znQm94T4fCEsy/yV+7CdC1kJhOU9lboAEU7R3kquuycDoibVA=="], + "ws": ["ws@8.20.1", "", { "peerDependencies": { "bufferutil": "^4.0.1", "utf-8-validate": ">=5.0.2" }, "optionalPeers": ["bufferutil", "utf-8-validate"] }, "sha512-It4dO0K5v//JtTXuPkfEOaI3uUN87iYPnqo/ZzqCoG3g8uhA66QUMs/SrM0YK7/NAu+r4LMh/9dq2A7k+rHs+w=="], "wsl-utils": ["wsl-utils@0.1.0", "", { "dependencies": { "is-wsl": "^3.1.0" } }, "sha512-h3Fbisa2nKGPxCpm89Hk33lBLsnaGBvctQopaBSOW/uIs6FTe1ATyAnKFJrzVs9vpGdsTe73WF3V4lIsk4Gacw=="], "xdg-basedir": ["xdg-basedir@5.1.0", "", {}, "sha512-GCPAHLvrIH13+c0SuacwvRYj2SxJXQ4kaVTT5xgL3kPrz56XxkF21IGhjSE1+W0aw7gpBWRGXLCPnPby6lSpmQ=="], + "xml-naming": ["xml-naming@0.1.0", "", {}, "sha512-k8KO9hrMyNk6tUWqUfkTEZbezRRpONVOzUTnc97VnCvyj6Tf9lyUR9EDAIeiVLv56jsMcoXEwjW8Kv5yPY52lw=="], + "xml-parse-from-string": ["xml-parse-from-string@1.0.1", "", {}, "sha512-ErcKwJTF54uRzzNMXq2X5sMIy88zJvfN2DmdoQvy7PAFJ+tPRU6ydWuOKNMyfmOjdyBQTFREi60s0Y0SyI0G0g=="], "xml2js": ["xml2js@0.5.0", "", { "dependencies": { "sax": ">=0.6.0", "xmlbuilder": "~11.0.0" } }, "sha512-drPFnkQJik/O+uPKpqSgr22mpuFHqKdbS835iAQrUC73L2F5WkboIRd63ai/2Yg6I1jzifPFKH2NTK+cfglkIA=="], @@ -5066,6 +5080,8 @@ "zod-to-json-schema": ["zod-to-json-schema@3.24.5", "", { "peerDependencies": { "zod": "^3.24.1" } }, "sha512-/AuWwMP+YqiPbsJx5D6TfgRTc4kTLjsh5SOcd4bLsfUg2RcEXrFMJl1DGgdHy2aCfsIA/cr/1JM0xcB2GZji8g=="], + "zustand": ["zustand@4.5.7", "", { "dependencies": { "use-sync-external-store": "^1.2.2" }, "peerDependencies": { "@types/react": ">=16.8", "immer": ">=9.0.6", "react": ">=16.8" }, "optionalPeers": ["@types/react", "immer", "react"] }, "sha512-CHOUy7mu3lbD6o6LJLfllpjkzhHXSBlX8B9+qPddUsIfeF5S/UZ5q0kmCsnRqT1UHFQZchNFDDzMbQsuesHWlw=="], + "zwitch": ["zwitch@2.0.4", "", {}, "sha512-bXE4cR/kVZhKZX/RjPEflHaKVhUVl85noU3v6b8apfQEc1x4A+zBxjZ4lN8LqGd6WZ3dl98pY4o717VFmoPp+A=="], "@actions/artifact/@actions/core": ["@actions/core@2.0.3", "", { "dependencies": { "@actions/exec": "^2.0.0", "@actions/http-client": "^3.0.2" } }, "sha512-Od9Thc3T1mQJYddvVPM4QGiLUewdh+3txmDYHHxoNdkqysR1MbCT+rFOtNUxYAz+7+6RIsqipVahY2GJqGPyxA=="], @@ -5090,7 +5106,7 @@ "@ai-sdk/deepgram/@ai-sdk/provider": ["@ai-sdk/provider@3.0.10", "", { "dependencies": { "json-schema": "^0.4.0" } }, "sha512-Q3BZ27qfpYqnCYGvE3vt+Qi6LGOF9R5Nmzn+9JoM1lCRsD9mYaIhfJLkSunN48nfGXJ6n+XNV0J/XVpqGQl7Dw=="], - "@ai-sdk/deepgram/@ai-sdk/provider-utils": ["@ai-sdk/provider-utils@4.0.26", "", { "dependencies": { "@ai-sdk/provider": "3.0.10", "@standard-schema/spec": "^1.1.0", "eventsource-parser": "^3.0.8" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-CsKNLKsOpvPujRlIYvoz+Ybw+kGn7J4/fIZa/58+R7iWLLfwn6ifE2G6Yq8K9XvH/I/3bzaDAJ3NhRwEMsLBKQ=="], + "@ai-sdk/deepgram/@ai-sdk/provider-utils": ["@ai-sdk/provider-utils@4.0.27", "", { "dependencies": { "@ai-sdk/provider": "3.0.10", "@standard-schema/spec": "^1.1.0", "eventsource-parser": "^3.0.8" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-ubkAJ+xODouwtmN1tYlvTPphH1hPOBfZaEQe8U7skGvFAnIRs9PPpsq57bC2+Ky/MB4yzhd6YOsxTAx9sGpazw=="], "@ai-sdk/deepinfra/@ai-sdk/openai-compatible": ["@ai-sdk/openai-compatible@2.0.37", "", { "dependencies": { "@ai-sdk/provider": "3.0.8", "@ai-sdk/provider-utils": "4.0.21" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-+POSFVcgiu47BK64dhsI6OpcDC0/VAE2ZSaXdXGNNhpC/ava++uSRJYks0k2bpfY0wwCTgpAWZsXn/dG2Yppiw=="], @@ -5098,17 +5114,17 @@ "@ai-sdk/deepseek/@ai-sdk/provider": ["@ai-sdk/provider@3.0.10", "", { "dependencies": { "json-schema": "^0.4.0" } }, "sha512-Q3BZ27qfpYqnCYGvE3vt+Qi6LGOF9R5Nmzn+9JoM1lCRsD9mYaIhfJLkSunN48nfGXJ6n+XNV0J/XVpqGQl7Dw=="], - "@ai-sdk/deepseek/@ai-sdk/provider-utils": ["@ai-sdk/provider-utils@4.0.26", "", { "dependencies": { "@ai-sdk/provider": "3.0.10", "@standard-schema/spec": "^1.1.0", "eventsource-parser": "^3.0.8" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-CsKNLKsOpvPujRlIYvoz+Ybw+kGn7J4/fIZa/58+R7iWLLfwn6ifE2G6Yq8K9XvH/I/3bzaDAJ3NhRwEMsLBKQ=="], + "@ai-sdk/deepseek/@ai-sdk/provider-utils": ["@ai-sdk/provider-utils@4.0.27", "", { "dependencies": { "@ai-sdk/provider": "3.0.10", "@standard-schema/spec": "^1.1.0", "eventsource-parser": "^3.0.8" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-ubkAJ+xODouwtmN1tYlvTPphH1hPOBfZaEQe8U7skGvFAnIRs9PPpsq57bC2+Ky/MB4yzhd6YOsxTAx9sGpazw=="], "@ai-sdk/elevenlabs/@ai-sdk/provider": ["@ai-sdk/provider@3.0.10", "", { "dependencies": { "json-schema": "^0.4.0" } }, "sha512-Q3BZ27qfpYqnCYGvE3vt+Qi6LGOF9R5Nmzn+9JoM1lCRsD9mYaIhfJLkSunN48nfGXJ6n+XNV0J/XVpqGQl7Dw=="], - "@ai-sdk/elevenlabs/@ai-sdk/provider-utils": ["@ai-sdk/provider-utils@4.0.26", "", { "dependencies": { "@ai-sdk/provider": "3.0.10", "@standard-schema/spec": "^1.1.0", "eventsource-parser": "^3.0.8" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-CsKNLKsOpvPujRlIYvoz+Ybw+kGn7J4/fIZa/58+R7iWLLfwn6ifE2G6Yq8K9XvH/I/3bzaDAJ3NhRwEMsLBKQ=="], + "@ai-sdk/elevenlabs/@ai-sdk/provider-utils": ["@ai-sdk/provider-utils@4.0.27", "", { "dependencies": { "@ai-sdk/provider": "3.0.10", "@standard-schema/spec": "^1.1.0", "eventsource-parser": "^3.0.8" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-ubkAJ+xODouwtmN1tYlvTPphH1hPOBfZaEQe8U7skGvFAnIRs9PPpsq57bC2+Ky/MB4yzhd6YOsxTAx9sGpazw=="], - "@ai-sdk/fireworks/@ai-sdk/openai-compatible": ["@ai-sdk/openai-compatible@2.0.46", "", { "dependencies": { "@ai-sdk/provider": "3.0.10", "@ai-sdk/provider-utils": "4.0.26" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-23ExGdy3p0Grfz3BAjCbIOc74TjQc5nHu72e0+kx3hshvScp32a4nnQlzzG4VT1bDZxa9yPNNUNyb5nN6vJHcQ=="], + "@ai-sdk/fireworks/@ai-sdk/openai-compatible": ["@ai-sdk/openai-compatible@2.0.47", "", { "dependencies": { "@ai-sdk/provider": "3.0.10", "@ai-sdk/provider-utils": "4.0.27" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-Enm5UlL0zUCrW3792opk5h7hRWxZOZzDe6eQYVFqX9LUOGGCe1h8MZWAGim765nwzgnjlpeYOsuzZmLtRsTPlg=="], "@ai-sdk/fireworks/@ai-sdk/provider": ["@ai-sdk/provider@3.0.10", "", { "dependencies": { "json-schema": "^0.4.0" } }, "sha512-Q3BZ27qfpYqnCYGvE3vt+Qi6LGOF9R5Nmzn+9JoM1lCRsD9mYaIhfJLkSunN48nfGXJ6n+XNV0J/XVpqGQl7Dw=="], - "@ai-sdk/fireworks/@ai-sdk/provider-utils": ["@ai-sdk/provider-utils@4.0.26", "", { "dependencies": { "@ai-sdk/provider": "3.0.10", "@standard-schema/spec": "^1.1.0", "eventsource-parser": "^3.0.8" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-CsKNLKsOpvPujRlIYvoz+Ybw+kGn7J4/fIZa/58+R7iWLLfwn6ifE2G6Yq8K9XvH/I/3bzaDAJ3NhRwEMsLBKQ=="], + "@ai-sdk/fireworks/@ai-sdk/provider-utils": ["@ai-sdk/provider-utils@4.0.27", "", { "dependencies": { "@ai-sdk/provider": "3.0.10", "@standard-schema/spec": "^1.1.0", "eventsource-parser": "^3.0.8" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-ubkAJ+xODouwtmN1tYlvTPphH1hPOBfZaEQe8U7skGvFAnIRs9PPpsq57bC2+Ky/MB4yzhd6YOsxTAx9sGpazw=="], "@ai-sdk/google-vertex/@ai-sdk/google": ["@ai-sdk/google@3.0.64", "", { "dependencies": { "@ai-sdk/provider": "3.0.8", "@ai-sdk/provider-utils": "4.0.23" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-CbR82EgGPNrj/6q0HtclwuCqe0/pDShyv3nWDP/A9DroujzWXnLMlUJVrgPOsg4b40zQCwwVs2XSKCxvt/4QaA=="], @@ -5132,15 +5148,17 @@ "@antfu/install-pkg/tinyexec": ["tinyexec@1.1.2", "", {}, "sha512-dAqSqE/RabpBKI8+h26GfLq6Vb3JVXs30XYQjdMjaj/c2tS8IYYMbIzP599KtRj7c57/wYApb3QjgRgXmrCukA=="], + "@anthropic-ai/sdk/@types/node": ["@types/node@18.19.130", "", { "dependencies": { "undici-types": "~5.26.4" } }, "sha512-GRaXQx6jGfL8sKfaIDD6OupbIHBr9jv7Jnaml9tB7l4v068PAOXqfcujMMo5PhbIs6ggR1XODELqahT2R8v0fg=="], + "@aws-crypto/sha1-browser/@smithy/util-utf8": ["@smithy/util-utf8@2.3.0", "", { "dependencies": { "@smithy/util-buffer-from": "^2.2.0", "tslib": "^2.6.2" } }, "sha512-R8Rdn8Hy72KKcebgLiv8jQcQkXoLMOGGv5uI1/k0l+snqkOzQ1R0ChUBCxWMlBsFMekWjq0wRudIweFs7sKT5A=="], "@aws-crypto/sha256-browser/@smithy/util-utf8": ["@smithy/util-utf8@2.3.0", "", { "dependencies": { "@smithy/util-buffer-from": "^2.2.0", "tslib": "^2.6.2" } }, "sha512-R8Rdn8Hy72KKcebgLiv8jQcQkXoLMOGGv5uI1/k0l+snqkOzQ1R0ChUBCxWMlBsFMekWjq0wRudIweFs7sKT5A=="], "@aws-crypto/util/@smithy/util-utf8": ["@smithy/util-utf8@2.3.0", "", { "dependencies": { "@smithy/util-buffer-from": "^2.2.0", "tslib": "^2.6.2" } }, "sha512-R8Rdn8Hy72KKcebgLiv8jQcQkXoLMOGGv5uI1/k0l+snqkOzQ1R0ChUBCxWMlBsFMekWjq0wRudIweFs7sKT5A=="], - "@aws-sdk/credential-provider-node/@aws-sdk/credential-provider-ini": ["@aws-sdk/credential-provider-ini@3.972.38", "", { "dependencies": { "@aws-sdk/core": "^3.974.8", "@aws-sdk/credential-provider-env": "^3.972.34", "@aws-sdk/credential-provider-http": "^3.972.36", "@aws-sdk/credential-provider-login": "^3.972.38", "@aws-sdk/credential-provider-process": "^3.972.34", "@aws-sdk/credential-provider-sso": "^3.972.38", "@aws-sdk/credential-provider-web-identity": "^3.972.38", "@aws-sdk/nested-clients": "^3.997.6", "@aws-sdk/types": "^3.973.8", "@smithy/credential-provider-imds": "^4.2.14", "@smithy/property-provider": "^4.2.14", "@smithy/shared-ini-file-loader": "^4.4.9", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-oDzUBu2MGJFgoar05sPMCwSrhw44ASyccrHzj66vO69OZqi7I6hZZxXfuPLC8OCzW7C+sU+bI73XHij41yekgQ=="], + "@aws-sdk/credential-provider-node/@aws-sdk/credential-provider-ini": ["@aws-sdk/credential-provider-ini@3.972.39", "", { "dependencies": { "@aws-sdk/core": "^3.974.9", "@aws-sdk/credential-provider-env": "^3.972.35", "@aws-sdk/credential-provider-http": "^3.972.37", "@aws-sdk/credential-provider-login": "^3.972.39", "@aws-sdk/credential-provider-process": "^3.972.35", "@aws-sdk/credential-provider-sso": "^3.972.39", "@aws-sdk/credential-provider-web-identity": "^3.972.39", "@aws-sdk/nested-clients": "^3.997.7", "@aws-sdk/types": "^3.973.8", "@smithy/core": "^3.24.1", "@smithy/credential-provider-imds": "^4.3.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-QhRSrdkk+Gq0AFIylpiI0N6lcJqFYV9Jtr4Luz5FpYOYbjJSfyTG6iLhnK/UPIgN1Jnon8WAmSC//16XYGvwkA=="], - "@aws-sdk/credential-provider-sso/@aws-sdk/token-providers": ["@aws-sdk/token-providers@3.1041.0", "", { "dependencies": { "@aws-sdk/core": "^3.974.8", "@aws-sdk/nested-clients": "^3.997.6", "@aws-sdk/types": "^3.973.8", "@smithy/property-provider": "^4.2.14", "@smithy/shared-ini-file-loader": "^4.4.9", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-Th7kPI6YPtvJUcdznooXJMy+9rQWjmEF81LxaJssngBzuysK4a/x+l8kjm1zb7nYsUPbndnBdUnwng/3PLvtGw=="], + "@aws-sdk/credential-provider-sso/@aws-sdk/token-providers": ["@aws-sdk/token-providers@3.1046.0", "", { "dependencies": { "@aws-sdk/core": "^3.974.9", "@aws-sdk/nested-clients": "^3.997.7", "@aws-sdk/types": "^3.973.8", "@smithy/core": "^3.24.1", "@smithy/types": "^4.14.1", "tslib": "^2.6.2" } }, "sha512-9je8nZt+ntB8IjhpGNayU/AkBgvq/f4aFO2bH1LSNC5JX6K8zY4LUnr/ymqunePrwq+B5OVBpL7ILjYzMFSZAw=="], "@azure/core-http/@azure/abort-controller": ["@azure/abort-controller@1.1.0", "", { "dependencies": { "tslib": "^2.2.0" } }, "sha512-TrRLIoSQVzfAJX9H1JeFjzAoDGcoK1IYX1UImfceTZpsyYfWr09Ss1aHW1y5TrrR3iq6RZLBwJ3E24uwPhwahw=="], @@ -5170,6 +5188,8 @@ "@effect/platform-node/undici": ["undici@8.2.0", "", {}, "sha512-Z+4Hx9GE26Lh9Upwfnc8C7SsrpBPGaM/Gm6kMFtiG7c+5IvQKlXi/t+9x9DrrCh29cww5TSP9YdVaBcnLDs5fQ=="], + "@effect/platform-node-shared/effect": ["effect@4.0.0-beta.65", "", { "dependencies": { "@standard-schema/spec": "^1.1.0", "fast-check": "^4.6.0", "find-my-way-ts": "^0.1.6", "ini": "^6.0.0", "kubernetes-types": "^1.30.0", "msgpackr": "^1.11.9", "multipasta": "^0.2.7", "toml": "^4.1.1", "uuid": "^13.0.0", "yaml": "^2.8.3" } }, "sha512-QYKvQPAj3CmtsvWkHQww15wX4KG2gNsszDWEcOO5sZCMknp66u6Si/Opmt3wwWCwsyvRmDAdIg+JIz5qzbbFIw=="], + "@electron/asar/commander": ["commander@5.1.0", "", {}, "sha512-P0CysNDQ7rtVw4QIQtm+MRxV66vKFSvlsQvGYXZWR3qFU0jlMKHZZZgw8e+8DSah4UDKMqnknRDQz+xuQXQ/Zg=="], "@electron/asar/glob": ["glob@7.2.3", "", { "dependencies": { "fs.realpath": "^1.0.0", "inflight": "^1.0.4", "inherits": "2", "minimatch": "^3.1.1", "once": "^1.3.0", "path-is-absolute": "^1.0.0" } }, "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q=="], @@ -5190,11 +5210,11 @@ "@electron/osx-sign/isbinaryfile": ["isbinaryfile@4.0.10", "", {}, "sha512-iHrqe5shvBUcFbmZq9zOQHBoeOhZJu6RQGrDpBgenUm/Am+F3JM2MgQj+rK3Z601fzrL5gLZWtAPH2OBaSVcyw=="], - "@electron/universal/fs-extra": ["fs-extra@11.3.4", "", { "dependencies": { "graceful-fs": "^4.2.0", "jsonfile": "^6.0.1", "universalify": "^2.0.0" } }, "sha512-CTXd6rk/M3/ULNQj8FBqBWHYBVYybQ3VPBw0xGKFe3tuH7ytT6ACnvzpIQ3UZtB8yvUKC2cXn1a+x+5EVQLovA=="], + "@electron/universal/fs-extra": ["fs-extra@11.3.5", "", { "dependencies": { "graceful-fs": "^4.2.0", "jsonfile": "^6.0.1", "universalify": "^2.0.0" } }, "sha512-eKpRKAovdpZtR1WopLHxlBWvAgPny3c4gX1G5Jhwmmw4XJj0ifSD5qB5TOo8hmA0wlRKDAOAhEE1yVPgs6Fgcg=="], "@electron/universal/minimatch": ["minimatch@9.0.9", "", { "dependencies": { "brace-expansion": "^2.0.2" } }, "sha512-OBwBN9AL4dqmETlpS2zasx+vTeWclWzkblfZk7KTA5j3jeOONz/tRCnZomUyvNg83wL5Zv9Ss6HMJXAgL8R2Yg=="], - "@electron/windows-sign/fs-extra": ["fs-extra@11.3.4", "", { "dependencies": { "graceful-fs": "^4.2.0", "jsonfile": "^6.0.1", "universalify": "^2.0.0" } }, "sha512-CTXd6rk/M3/ULNQj8FBqBWHYBVYybQ3VPBw0xGKFe3tuH7ytT6ACnvzpIQ3UZtB8yvUKC2cXn1a+x+5EVQLovA=="], + "@electron/windows-sign/fs-extra": ["fs-extra@11.3.5", "", { "dependencies": { "graceful-fs": "^4.2.0", "jsonfile": "^6.0.1", "universalify": "^2.0.0" } }, "sha512-eKpRKAovdpZtR1WopLHxlBWvAgPny3c4gX1G5Jhwmmw4XJj0ifSD5qB5TOo8hmA0wlRKDAOAhEE1yVPgs6Fgcg=="], "@eslint-community/eslint-utils/eslint-visitor-keys": ["eslint-visitor-keys@3.4.3", "", {}, "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag=="], @@ -5212,6 +5232,8 @@ "@gitlab/gitlab-ai-provider/zod": ["zod@3.25.76", "", {}, "sha512-gzUt/qt81nXsFGKIFcC3YnfEAx5NkunCfnDlvuBSSFS02bcXu4Lmea0AFIUwbLWxWPx3d9p8S5QoaujKcNQxcQ=="], + "@happy-dom/global-registrator/@types/node": ["@types/node@20.19.41", "", { "dependencies": { "undici-types": "~6.21.0" } }, "sha512-ECymXOukMnOoVkC2bb1Vc/w/836DXncOg5m8Xj1RH7xSHZJWNYY6Zh7EH477vcnD5egKNNfy2RpNOmuChhFPgQ=="], + "@hey-api/openapi-ts/open": ["open@11.0.0", "", { "dependencies": { "default-browser": "^5.4.0", "define-lazy-prop": "^3.0.0", "is-in-ssh": "^1.0.0", "is-inside-container": "^1.0.0", "powershell-utils": "^0.1.0", "wsl-utils": "^0.3.0" } }, "sha512-smsWv2LzFjP03xmvFoJ331ss6h+jixfA4UUV/Bsiyuu4YJPfN+FIQGOIiv4w9/+MoHkfkJ22UIaQWRVFRfH6Vw=="], "@hey-api/openapi-ts/semver": ["semver@7.7.3", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q=="], @@ -5270,10 +5292,20 @@ "@kilocode/kilo-gateway/@opentui/solid": ["@opentui/solid@0.1.75", "", { "dependencies": { "@babel/core": "7.28.0", "@babel/preset-typescript": "7.27.1", "@opentui/core": "0.1.75", "babel-plugin-module-resolver": "5.0.2", "babel-preset-solid": "1.9.9", "s-js": "^0.4.9" }, "peerDependencies": { "solid-js": "1.9.9" } }, "sha512-WjKsZIfrm29znfRlcD9w3uUn/+uvoy2MmeoDwTvg1YOa0OjCTCmjZ43L9imp0m9S4HmVU8ma6o2bR4COzcyDdg=="], + "@kilocode/kilo-gateway/@types/node": ["@types/node@22.13.9", "", { "dependencies": { "undici-types": "~6.20.0" } }, "sha512-acBjXdRJ3A6Pb3tqnw9HZmyR3Fiol3aGxRCK1x3d+6CDAMjl7I649wpSd+yNURCjbOUGu9tqtLKnTGxmK6CyGw=="], + + "@kilocode/kilo-indexing/@types/node": ["@types/node@22.13.9", "", { "dependencies": { "undici-types": "~6.20.0" } }, "sha512-acBjXdRJ3A6Pb3tqnw9HZmyR3Fiol3aGxRCK1x3d+6CDAMjl7I649wpSd+yNURCjbOUGu9tqtLKnTGxmK6CyGw=="], + "@kilocode/kilo-indexing/glob": ["glob@13.0.6", "", { "dependencies": { "minimatch": "^10.2.2", "minipass": "^7.1.3", "path-scurry": "^2.0.2" } }, "sha512-Wjlyrolmm8uDpm/ogGyXZXb1Z+Ca2B8NbJwqBVg0axK9GbBeoS7yGV6vjXnYdGm6X53iehEuxxbyiKp8QmN4Vw=="], + "@kilocode/plugin/@types/node": ["@types/node@22.13.9", "", { "dependencies": { "undici-types": "~6.20.0" } }, "sha512-acBjXdRJ3A6Pb3tqnw9HZmyR3Fiol3aGxRCK1x3d+6CDAMjl7I649wpSd+yNURCjbOUGu9tqtLKnTGxmK6CyGw=="], + + "@kilocode/sdk/@types/node": ["@types/node@22.13.9", "", { "dependencies": { "undici-types": "~6.20.0" } }, "sha512-acBjXdRJ3A6Pb3tqnw9HZmyR3Fiol3aGxRCK1x3d+6CDAMjl7I649wpSd+yNURCjbOUGu9tqtLKnTGxmK6CyGw=="], + "@malept/flatpak-bundler/fs-extra": ["fs-extra@9.1.0", "", { "dependencies": { "at-least-node": "^1.0.0", "graceful-fs": "^4.2.0", "jsonfile": "^6.0.1", "universalify": "^2.0.0" } }, "sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ=="], + "@manypkg/find-root/@types/node": ["@types/node@12.20.55", "", {}, "sha512-J8xLz7q2OFulZ2cyGTLE1TbbZcjpno7FaN6zdJNrgAdrJ+DZzh/uFR6YrTb4C+nXakvud8Q4+rbhoIWlYQbUFQ=="], + "@manypkg/find-root/find-up": ["find-up@4.1.0", "", { "dependencies": { "locate-path": "^5.0.0", "path-exists": "^4.0.0" } }, "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw=="], "@manypkg/find-root/fs-extra": ["fs-extra@8.1.0", "", { "dependencies": { "graceful-fs": "^4.2.0", "jsonfile": "^4.0.0", "universalify": "^0.1.0" } }, "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g=="], @@ -5286,6 +5318,8 @@ "@modelcontextprotocol/sdk/zod-to-json-schema": ["zod-to-json-schema@3.25.2", "", { "peerDependencies": { "zod": "^3.25.28 || ^4" } }, "sha512-O/PgfnpT1xKSDeQYSCfRI5Gy3hPf91mKVDuYLUHZJMiDFptvP41MSnWofm8dnCm0256ZNfZIM7DSzuSMAFnjHA=="], + "@morphllm/morphsdk/diff": ["diff@7.0.0", "", {}, "sha512-PJWHUb1RFevKCwaFA9RlG5tCd+FO5iRh9A8HEtkmBH2Li03iJriB6m6JIN4rGz3K3JLawI7/veA1xzRKP6ISBw=="], + "@morphllm/morphsdk/openai": ["openai@4.104.0", "", { "dependencies": { "@types/node": "^18.11.18", "@types/node-fetch": "^2.6.4", "abort-controller": "^3.0.0", "agentkeepalive": "^4.2.1", "form-data-encoder": "1.7.2", "formdata-node": "^4.3.2", "node-fetch": "^2.6.7" }, "peerDependencies": { "ws": "^8.18.0", "zod": "^3.23.8" }, "optionalPeers": ["ws", "zod"], "bin": { "openai": "bin/cli" } }, "sha512-p99EFNsA/yX6UhVO93f5kJsDRLAg+CTA2RBqdHK4RtK8u5IJw32Hyb2dTGKbnnFmnuoBv5r7Z2CURI9sGZpSuA=="], "@npmcli/map-workspaces/glob": ["glob@13.0.6", "", { "dependencies": { "minimatch": "^10.2.2", "minipass": "^7.1.3", "path-scurry": "^2.0.2" } }, "sha512-Wjlyrolmm8uDpm/ogGyXZXb1Z+Ca2B8NbJwqBVg0axK9GbBeoS7yGV6vjXnYdGm6X53iehEuxxbyiKp8QmN4Vw=="], @@ -5302,7 +5336,7 @@ "@octokit/endpoint/universal-user-agent": ["universal-user-agent@6.0.1", "", {}, "sha512-yCzhz6FN2wU1NiiQRogkTQszlQSlpWaw8SvVegAc+bDxbzHgh1vX8uIe8OYyMH6DwH+sdTJsgMl36+mSMdRJIQ=="], - "@octokit/graphql/@octokit/request": ["@octokit/request@10.0.8", "", { "dependencies": { "@octokit/endpoint": "^11.0.3", "@octokit/request-error": "^7.0.2", "@octokit/types": "^16.0.0", "fast-content-type-parse": "^3.0.0", "json-with-bigint": "^3.5.3", "universal-user-agent": "^7.0.2" } }, "sha512-SJZNwY9pur9Agf7l87ywFi14W+Hd9Jg6Ifivsd33+/bGUQIjNujdFiXII2/qSlN2ybqUHfp5xpekMEjIBTjlSw=="], + "@octokit/graphql/@octokit/request": ["@octokit/request@10.0.9", "", { "dependencies": { "@octokit/endpoint": "^11.0.3", "@octokit/request-error": "^7.0.2", "@octokit/types": "^16.0.0", "content-type": "^2.0.0", "fast-content-type-parse": "^3.0.0", "json-with-bigint": "^3.5.3", "universal-user-agent": "^7.0.2" } }, "sha512-o8Bi3f608eyM+7BmBiUWxFsdjLb3/ym1cQek5LZOv9KkZcxRrHCPhhRzm6xjO6HVZ85ItD6+sTsjxo821SVa/A=="], "@octokit/plugin-paginate-rest/@octokit/types": ["@octokit/types@12.6.0", "", { "dependencies": { "@octokit/openapi-types": "^20.0.0" } }, "sha512-1rhSOfRa6H9w4YwK0yrf5faDaDTb+yLyBUKOCV4xtCDB5VmIPqd/v9yr9o6SAzOAlRxMiRiCic6JVM1/kunVkw=="], @@ -5332,6 +5366,8 @@ "@opencode-ai/app/@playwright/test": ["@playwright/test@1.59.1", "", { "dependencies": { "playwright": "1.59.1" }, "bin": { "playwright": "cli.js" } }, "sha512-PG6q63nQg5c9rIi4/Z5lR5IVF7yU5MqmKaPOe0HSc0O2cX1fPi96sUQu5j7eo4gKCkB2AnNGoWt7y4/Xx3Kcqg=="], + "@opencode-ai/app/@types/node": ["@types/node@22.13.9", "", { "dependencies": { "undici-types": "~6.20.0" } }, "sha512-acBjXdRJ3A6Pb3tqnw9HZmyR3Fiol3aGxRCK1x3d+6CDAMjl7I649wpSd+yNURCjbOUGu9tqtLKnTGxmK6CyGw=="], + "@opencode-ai/app/tailwindcss": ["tailwindcss@4.1.11", "", {}, "sha512-2E9TBm6MDD/xKYe+dvJZAmg3yxIEDNRc0jwlNyDg/4Fil2QcSLjFKGVff0lAf1jjeaArlG/M75Ey/EYr/OJtBA=="], "@opencode-ai/desktop/@actions/artifact": ["@actions/artifact@4.0.0", "", { "dependencies": { "@actions/core": "^1.10.0", "@actions/github": "^6.0.1", "@actions/http-client": "^2.1.0", "@azure/core-http": "^3.0.5", "@azure/storage-blob": "^12.15.0", "@octokit/core": "^5.2.1", "@octokit/plugin-request-log": "^1.0.4", "@octokit/plugin-retry": "^3.0.9", "@octokit/request": "^8.4.1", "@octokit/request-error": "^5.1.1", "@protobuf-ts/plugin": "^2.2.3-alpha.1", "archiver": "^7.0.1", "jwt-decode": "^3.1.2", "unzip-stream": "^0.3.1" } }, "sha512-HCc2jMJRAfviGFAh0FsOR/jNfWhirxl7W6z8zDtttt0GltwxBLdEIjLiweOPFl9WbyJRW1VWnPUSAixJqcWUMQ=="], @@ -5340,19 +5376,27 @@ "@opencode-ai/desktop-electron/@actions/artifact": ["@actions/artifact@4.0.0", "", { "dependencies": { "@actions/core": "^1.10.0", "@actions/github": "^6.0.1", "@actions/http-client": "^2.1.0", "@azure/core-http": "^3.0.5", "@azure/storage-blob": "^12.15.0", "@octokit/core": "^5.2.1", "@octokit/plugin-request-log": "^1.0.4", "@octokit/plugin-retry": "^3.0.9", "@octokit/request": "^8.4.1", "@octokit/request-error": "^5.1.1", "@protobuf-ts/plugin": "^2.2.3-alpha.1", "archiver": "^7.0.1", "jwt-decode": "^3.1.2", "unzip-stream": "^0.3.1" } }, "sha512-HCc2jMJRAfviGFAh0FsOR/jNfWhirxl7W6z8zDtttt0GltwxBLdEIjLiweOPFl9WbyJRW1VWnPUSAixJqcWUMQ=="], + "@opencode-ai/desktop-electron/@types/node": ["@types/node@22.13.9", "", { "dependencies": { "undici-types": "~6.20.0" } }, "sha512-acBjXdRJ3A6Pb3tqnw9HZmyR3Fiol3aGxRCK1x3d+6CDAMjl7I649wpSd+yNURCjbOUGu9tqtLKnTGxmK6CyGw=="], + "@opencode-ai/desktop-electron/marked": ["marked@15.0.12", "", { "bin": { "marked": "bin/marked.js" } }, "sha512-8dD6FusOQSrpv9Z1rdNMdlSgQOIP880DHqnohobOmYLElGEqAL/JvxvuxZO16r4HtjTlfPRDC1hbvxC9dPN2nA=="], "@opencode-ai/desktop-electron/typescript": ["typescript@5.6.3", "", { "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" } }, "sha512-hjcS1mhfuyi4WW8IWtjP7brDrG2cuDZukyrYrSauoXGNgx0S7zceP07adYkJycEr56BOUTNPzbInooiN3fn1qw=="], - "@opencode-ai/storybook/@storybook/addon-a11y": ["@storybook/addon-a11y@10.3.6", "", { "dependencies": { "@storybook/global": "^5.0.0", "axe-core": "^4.2.0" }, "peerDependencies": { "storybook": "^10.3.6" } }, "sha512-cbwXIT5CeHZ9AFbTKQ6YB7Ct6TAl/kKOgALbvzzVtFfRvm51JYygGaiJaB7PbPWn9wgJP2olJcFt+erlEc6cRw=="], + "@opencode-ai/plugin/effect": ["effect@4.0.0-beta.65", "", { "dependencies": { "@standard-schema/spec": "^1.1.0", "fast-check": "^4.6.0", "find-my-way-ts": "^0.1.6", "ini": "^6.0.0", "kubernetes-types": "^1.30.0", "msgpackr": "^1.11.9", "multipasta": "^0.2.7", "toml": "^4.1.1", "uuid": "^13.0.0", "yaml": "^2.8.3" } }, "sha512-QYKvQPAj3CmtsvWkHQww15wX4KG2gNsszDWEcOO5sZCMknp66u6Si/Opmt3wwWCwsyvRmDAdIg+JIz5qzbbFIw=="], + + "@opencode-ai/shared/semver": ["semver@7.7.4", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-vFKC2IEtQnVhpT78h1Yp8wzwrf8CM+MzKMHGJZfBtzhZNycRFnXsHk6E5TxIkkMsgNS7mdX3AGB7x2QM2di4lA=="], + + "@opencode-ai/storybook/@storybook/addon-a11y": ["@storybook/addon-a11y@10.4.0", "", { "dependencies": { "@storybook/global": "^5.0.0", "axe-core": "^4.2.0" }, "peerDependencies": { "storybook": "^10.4.0" } }, "sha512-N1QRmh+PMe5O81KDf8oPDv/csdLAmDCRCYLByukqdUXpTNlcULHDFUJNXl00/rFpbt7PbOZqzRzs72JJt6nWPA=="], + + "@opencode-ai/storybook/@storybook/addon-docs": ["@storybook/addon-docs@10.4.0", "", { "dependencies": { "@mdx-js/react": "^3.0.0", "@storybook/csf-plugin": "10.4.0", "@storybook/icons": "^2.0.2", "@storybook/react-dom-shim": "10.4.0", "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "react-dom": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "ts-dedent": "^2.0.0" }, "peerDependencies": { "@types/react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "storybook": "^10.4.0" }, "optionalPeers": ["@types/react"] }, "sha512-HJNvYGx/c3jjVwibnmbDgCZMYPI6xGUDjJSRi5CG0G9tpeoeijPo318f5N84RyYWK8LheHUrDN3Jv2UfVv8zwQ=="], - "@opencode-ai/storybook/@storybook/addon-docs": ["@storybook/addon-docs@10.3.6", "", { "dependencies": { "@mdx-js/react": "^3.0.0", "@storybook/csf-plugin": "10.3.6", "@storybook/icons": "^2.0.1", "@storybook/react-dom-shim": "10.3.6", "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "react-dom": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "ts-dedent": "^2.0.0" }, "peerDependencies": { "storybook": "^10.3.6" } }, "sha512-TvIdADVPtauxW0LzXIpIv7X6GxwetorhyNh+6+7MHC27XSBCWVxxRUwL63YeLlHTuXsIk0quG3b1xgwVRzWOJA=="], + "@opencode-ai/storybook/@types/node": ["@types/node@22.13.9", "", { "dependencies": { "undici-types": "~6.20.0" } }, "sha512-acBjXdRJ3A6Pb3tqnw9HZmyR3Fiol3aGxRCK1x3d+6CDAMjl7I649wpSd+yNURCjbOUGu9tqtLKnTGxmK6CyGw=="], "@opencode-ai/storybook/@types/react": ["@types/react@18.0.25", "", { "dependencies": { "@types/prop-types": "*", "@types/scheduler": "*", "csstype": "^3.0.2" } }, "sha512-xD6c0KDT4m7n9uD4ZHi02lzskaiqcBxf4zi+tXZY98a04wvc0hi/TcCPC2FOESZi51Nd7tlUeOJY8RofL799/g=="], "@opencode-ai/storybook/react": ["react@18.2.0", "", { "dependencies": { "loose-envify": "^1.1.0" } }, "sha512-/3IjMdb2L9QbBdWiW5e3P2/npwMBaU9mHCSCUzNln0ZCYbcfTsGbTJrU/kGemdH2IWmB2ioZ+zkxtmq6g09fGQ=="], - "@opencode-ai/storybook/storybook": ["storybook@10.3.6", "", { "dependencies": { "@storybook/global": "^5.0.0", "@storybook/icons": "^2.0.1", "@testing-library/jest-dom": "^6.9.1", "@testing-library/user-event": "^14.6.1", "@vitest/expect": "3.2.4", "@vitest/spy": "3.2.4", "@webcontainer/env": "^1.1.1", "esbuild": "^0.18.0 || ^0.19.0 || ^0.20.0 || ^0.21.0 || ^0.22.0 || ^0.23.0 || ^0.24.0 || ^0.25.0 || ^0.26.0 || ^0.27.0", "open": "^10.2.0", "recast": "^0.23.5", "semver": "^7.7.3", "use-sync-external-store": "^1.5.0", "ws": "^8.18.0" }, "peerDependencies": { "prettier": "^2 || ^3", "vite-plus": "^0.1.15" }, "optionalPeers": ["prettier", "vite-plus"], "bin": "./dist/bin/dispatcher.js" }, "sha512-vbSz7g/1rGMC1uAULqMZjALkIuLu2QABqfhRYhyr/11kzyesi+vAmwyJLukZP1FfecxGOgMwOh6GS0YsGpHAvQ=="], + "@opencode-ai/storybook/storybook": ["storybook@10.4.0", "", { "dependencies": { "@storybook/global": "^5.0.0", "@storybook/icons": "^2.0.2", "@testing-library/jest-dom": "^6.9.1", "@testing-library/user-event": "^14.6.1", "@vitest/expect": "3.2.4", "@vitest/spy": "3.2.4", "@webcontainer/env": "^1.1.1", "esbuild": "^0.18.0 || ^0.19.0 || ^0.20.0 || ^0.21.0 || ^0.22.0 || ^0.23.0 || ^0.24.0 || ^0.25.0 || ^0.26.0 || ^0.27.0", "open": "^10.2.0", "oxc-parser": "^0.127.0", "oxc-resolver": "^11.19.1", "recast": "^0.23.5", "semver": "^7.7.3", "use-sync-external-store": "^1.5.0", "ws": "^8.18.0" }, "peerDependencies": { "@types/react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "prettier": "^2 || ^3", "vite-plus": "^0.1.15" }, "optionalPeers": ["@types/react", "prettier", "vite-plus"], "bin": "./dist/bin/dispatcher.js" }, "sha512-zrtctbVa6xEXCXuE3vsiR0At31zLOtzj8QudN/GaBJLZl0Z2DfF1rDPtTxdAbAp11M2J/7JVHaTIQKXauQPbmg=="], "@opencode-ai/ui/@solid-primitives/resize-observer": ["@solid-primitives/resize-observer@2.1.3", "", { "dependencies": { "@solid-primitives/event-listener": "^2.4.3", "@solid-primitives/rootless": "^1.5.2", "@solid-primitives/static-store": "^0.1.2", "@solid-primitives/utils": "^6.3.2" }, "peerDependencies": { "solid-js": "^1.6.12" } }, "sha512-zBLje5E06TgOg93S7rGPldmhDnouNGhvfZVKOp+oG2XU8snA+GoCSSCz1M+jpNAg5Ek2EakU5UVQqL152WmdXQ=="], @@ -5374,10 +5418,16 @@ "@opentelemetry/sdk-logs/@opentelemetry/resources": ["@opentelemetry/resources@2.2.0", "", { "dependencies": { "@opentelemetry/core": "2.2.0", "@opentelemetry/semantic-conventions": "^1.29.0" }, "peerDependencies": { "@opentelemetry/api": ">=1.3.0 <1.10.0" } }, "sha512-1pNQf/JazQTMA0BiO5NINUzH0cbLbbl7mntLa4aJNmCCXSj0q03T5ZXXL0zw4G55TjdL9Tz32cznGClf+8zr5A=="], + "@opentui/core/diff": ["diff@9.0.0", "", {}, "sha512-svtcdpS8CgJyqAjEQIXdb3OjhFVVYjzGAPO8WGCmRbrml64SPw/jJD4GoE98aR7r25A0XcgrK3F02yw9R/vhQw=="], + "@opentui/solid/@babel/core": ["@babel/core@7.28.0", "", { "dependencies": { "@ampproject/remapping": "^2.2.0", "@babel/code-frame": "^7.27.1", "@babel/generator": "^7.28.0", "@babel/helper-compilation-targets": "^7.27.2", "@babel/helper-module-transforms": "^7.27.3", "@babel/helpers": "^7.27.6", "@babel/parser": "^7.28.0", "@babel/template": "^7.27.2", "@babel/traverse": "^7.28.0", "@babel/types": "^7.28.0", "convert-source-map": "^2.0.0", "debug": "^4.1.0", "gensync": "^1.0.0-beta.2", "json5": "^2.2.3", "semver": "^6.3.1" } }, "sha512-UlLAnTPrFdNGoFtbSXwcGFQBtQZJCNjaN6hQNP3UPvuNXT1i82N26KL3dZeIpNalWywr9IuQuncaAfUaS1g6sQ=="], "@oslojs/jwt/@oslojs/encoding": ["@oslojs/encoding@0.4.1", "", {}, "sha512-hkjo6MuIK/kQR5CrGNdAPZhS01ZCXuWDRJ187zh6qqF2+yMHZpD9fAYpX8q2bOO6Ryhl3XpCT6kUX76N8hhm4Q=="], + "@oxc-parser/binding-wasm32-wasi/@emnapi/runtime": ["@emnapi/runtime@1.9.2", "", { "dependencies": { "tslib": "^2.4.0" } }, "sha512-3U4+MIWHImeyu1wnmVygh5WlgfYDtyf0k8AbLhMFxOipihf6nrWC4syIm/SwEeec0mNSafiiNnMJwbza/Is6Lw=="], + + "@pierre/diffs/diff": ["diff@8.0.3", "", {}, "sha512-qejHi7bcSD4hQAZE0tNAawRK1ZtafHDmMTMkrrIGgSLl7hTnQHmKCeB45xAcbfTqK2zowkM3j3bHt/4b/ARbYQ=="], + "@protobuf-ts/plugin/typescript": ["typescript@3.9.10", "", { "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" } }, "sha512-w6fIxVE/H1PkLKcCPsFqKE7Kv7QUwhU8qQY2MueZXWx5cPZdwFupLgKK3vntcK98BtNHZtAF4LA/yl2a7k8R6Q=="], "@qdrant/js-client-rest/undici": ["undici@6.25.0", "", {}, "sha512-ZgpWDC5gmNiuY9CnLVXEH8rl50xhRCuLNA97fAUnKi8RRuV4E6KG31pDTsLVUKnohJE0I3XDrTeEydAXRw47xg=="], @@ -5394,15 +5444,15 @@ "@solid-primitives/scroll/@solid-primitives/rootless": ["@solid-primitives/rootless@1.5.3", "", { "dependencies": { "@solid-primitives/utils": "^6.4.0" }, "peerDependencies": { "solid-js": "^1.6.12" } }, "sha512-N8cIDAHbWcLahNRLr0knAAQvXyEdEMoAZvIMZKmhNb1mlx9e2UOv9BRD5YNwQUJwbNoYVhhLwFOEOcVXFx0HqA=="], - "@storybook/addon-links/storybook": ["storybook@10.3.6", "", { "dependencies": { "@storybook/global": "^5.0.0", "@storybook/icons": "^2.0.1", "@testing-library/jest-dom": "^6.9.1", "@testing-library/user-event": "^14.6.1", "@vitest/expect": "3.2.4", "@vitest/spy": "3.2.4", "@webcontainer/env": "^1.1.1", "esbuild": "^0.18.0 || ^0.19.0 || ^0.20.0 || ^0.21.0 || ^0.22.0 || ^0.23.0 || ^0.24.0 || ^0.25.0 || ^0.26.0 || ^0.27.0", "open": "^10.2.0", "recast": "^0.23.5", "semver": "^7.7.3", "use-sync-external-store": "^1.5.0", "ws": "^8.18.0" }, "peerDependencies": { "prettier": "^2 || ^3", "vite-plus": "^0.1.15" }, "optionalPeers": ["prettier", "vite-plus"], "bin": "./dist/bin/dispatcher.js" }, "sha512-vbSz7g/1rGMC1uAULqMZjALkIuLu2QABqfhRYhyr/11kzyesi+vAmwyJLukZP1FfecxGOgMwOh6GS0YsGpHAvQ=="], + "@storybook/addon-links/storybook": ["storybook@10.4.0", "", { "dependencies": { "@storybook/global": "^5.0.0", "@storybook/icons": "^2.0.2", "@testing-library/jest-dom": "^6.9.1", "@testing-library/user-event": "^14.6.1", "@vitest/expect": "3.2.4", "@vitest/spy": "3.2.4", "@webcontainer/env": "^1.1.1", "esbuild": "^0.18.0 || ^0.19.0 || ^0.20.0 || ^0.21.0 || ^0.22.0 || ^0.23.0 || ^0.24.0 || ^0.25.0 || ^0.26.0 || ^0.27.0", "open": "^10.2.0", "oxc-parser": "^0.127.0", "oxc-resolver": "^11.19.1", "recast": "^0.23.5", "semver": "^7.7.3", "use-sync-external-store": "^1.5.0", "ws": "^8.18.0" }, "peerDependencies": { "@types/react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "prettier": "^2 || ^3", "vite-plus": "^0.1.15" }, "optionalPeers": ["@types/react", "prettier", "vite-plus"], "bin": "./dist/bin/dispatcher.js" }, "sha512-zrtctbVa6xEXCXuE3vsiR0At31zLOtzj8QudN/GaBJLZl0Z2DfF1rDPtTxdAbAp11M2J/7JVHaTIQKXauQPbmg=="], - "@storybook/addon-onboarding/storybook": ["storybook@10.3.6", "", { "dependencies": { "@storybook/global": "^5.0.0", "@storybook/icons": "^2.0.1", "@testing-library/jest-dom": "^6.9.1", "@testing-library/user-event": "^14.6.1", "@vitest/expect": "3.2.4", "@vitest/spy": "3.2.4", "@webcontainer/env": "^1.1.1", "esbuild": "^0.18.0 || ^0.19.0 || ^0.20.0 || ^0.21.0 || ^0.22.0 || ^0.23.0 || ^0.24.0 || ^0.25.0 || ^0.26.0 || ^0.27.0", "open": "^10.2.0", "recast": "^0.23.5", "semver": "^7.7.3", "use-sync-external-store": "^1.5.0", "ws": "^8.18.0" }, "peerDependencies": { "prettier": "^2 || ^3", "vite-plus": "^0.1.15" }, "optionalPeers": ["prettier", "vite-plus"], "bin": "./dist/bin/dispatcher.js" }, "sha512-vbSz7g/1rGMC1uAULqMZjALkIuLu2QABqfhRYhyr/11kzyesi+vAmwyJLukZP1FfecxGOgMwOh6GS0YsGpHAvQ=="], + "@storybook/addon-onboarding/storybook": ["storybook@10.4.0", "", { "dependencies": { "@storybook/global": "^5.0.0", "@storybook/icons": "^2.0.2", "@testing-library/jest-dom": "^6.9.1", "@testing-library/user-event": "^14.6.1", "@vitest/expect": "3.2.4", "@vitest/spy": "3.2.4", "@webcontainer/env": "^1.1.1", "esbuild": "^0.18.0 || ^0.19.0 || ^0.20.0 || ^0.21.0 || ^0.22.0 || ^0.23.0 || ^0.24.0 || ^0.25.0 || ^0.26.0 || ^0.27.0", "open": "^10.2.0", "oxc-parser": "^0.127.0", "oxc-resolver": "^11.19.1", "recast": "^0.23.5", "semver": "^7.7.3", "use-sync-external-store": "^1.5.0", "ws": "^8.18.0" }, "peerDependencies": { "@types/react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "prettier": "^2 || ^3", "vite-plus": "^0.1.15" }, "optionalPeers": ["@types/react", "prettier", "vite-plus"], "bin": "./dist/bin/dispatcher.js" }, "sha512-zrtctbVa6xEXCXuE3vsiR0At31zLOtzj8QudN/GaBJLZl0Z2DfF1rDPtTxdAbAp11M2J/7JVHaTIQKXauQPbmg=="], - "@storybook/addon-vitest/storybook": ["storybook@10.3.6", "", { "dependencies": { "@storybook/global": "^5.0.0", "@storybook/icons": "^2.0.1", "@testing-library/jest-dom": "^6.9.1", "@testing-library/user-event": "^14.6.1", "@vitest/expect": "3.2.4", "@vitest/spy": "3.2.4", "@webcontainer/env": "^1.1.1", "esbuild": "^0.18.0 || ^0.19.0 || ^0.20.0 || ^0.21.0 || ^0.22.0 || ^0.23.0 || ^0.24.0 || ^0.25.0 || ^0.26.0 || ^0.27.0", "open": "^10.2.0", "recast": "^0.23.5", "semver": "^7.7.3", "use-sync-external-store": "^1.5.0", "ws": "^8.18.0" }, "peerDependencies": { "prettier": "^2 || ^3", "vite-plus": "^0.1.15" }, "optionalPeers": ["prettier", "vite-plus"], "bin": "./dist/bin/dispatcher.js" }, "sha512-vbSz7g/1rGMC1uAULqMZjALkIuLu2QABqfhRYhyr/11kzyesi+vAmwyJLukZP1FfecxGOgMwOh6GS0YsGpHAvQ=="], + "@storybook/addon-vitest/storybook": ["storybook@10.4.0", "", { "dependencies": { "@storybook/global": "^5.0.0", "@storybook/icons": "^2.0.2", "@testing-library/jest-dom": "^6.9.1", "@testing-library/user-event": "^14.6.1", "@vitest/expect": "3.2.4", "@vitest/spy": "3.2.4", "@webcontainer/env": "^1.1.1", "esbuild": "^0.18.0 || ^0.19.0 || ^0.20.0 || ^0.21.0 || ^0.22.0 || ^0.23.0 || ^0.24.0 || ^0.25.0 || ^0.26.0 || ^0.27.0", "open": "^10.2.0", "oxc-parser": "^0.127.0", "oxc-resolver": "^11.19.1", "recast": "^0.23.5", "semver": "^7.7.3", "use-sync-external-store": "^1.5.0", "ws": "^8.18.0" }, "peerDependencies": { "@types/react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "prettier": "^2 || ^3", "vite-plus": "^0.1.15" }, "optionalPeers": ["@types/react", "prettier", "vite-plus"], "bin": "./dist/bin/dispatcher.js" }, "sha512-zrtctbVa6xEXCXuE3vsiR0At31zLOtzj8QudN/GaBJLZl0Z2DfF1rDPtTxdAbAp11M2J/7JVHaTIQKXauQPbmg=="], - "@storybook/builder-vite/@storybook/csf-plugin": ["@storybook/csf-plugin@10.3.6", "", { "dependencies": { "unplugin": "^2.3.5" }, "peerDependencies": { "esbuild": "*", "rollup": "*", "storybook": "^10.3.6", "vite": "*", "webpack": "*" }, "optionalPeers": ["esbuild", "rollup", "vite", "webpack"] }, "sha512-9kBf7VRdRqTSIYo+rPtVn5yjYYyK8kP2QhEYx3oiXvfwy4RexmbJnhk/tXa/lNiTqukA1TqaWQ2+5MqF4fu6YQ=="], + "@storybook/builder-vite/@storybook/csf-plugin": ["@storybook/csf-plugin@10.4.0", "", { "dependencies": { "unplugin": "^2.3.5" }, "peerDependencies": { "esbuild": "*", "rollup": "*", "storybook": "^10.4.0", "vite": "*", "webpack": "*" }, "optionalPeers": ["esbuild", "rollup", "vite", "webpack"] }, "sha512-iSmrhMyEi2ohCWKu49ZUUf8l+k0OIStbWI1BTWt2FvKySlnqY/aHenus7839SgNL3aUNG5P0y9zlyN6/HlwlEQ=="], - "@storybook/builder-vite/storybook": ["storybook@10.3.6", "", { "dependencies": { "@storybook/global": "^5.0.0", "@storybook/icons": "^2.0.1", "@testing-library/jest-dom": "^6.9.1", "@testing-library/user-event": "^14.6.1", "@vitest/expect": "3.2.4", "@vitest/spy": "3.2.4", "@webcontainer/env": "^1.1.1", "esbuild": "^0.18.0 || ^0.19.0 || ^0.20.0 || ^0.21.0 || ^0.22.0 || ^0.23.0 || ^0.24.0 || ^0.25.0 || ^0.26.0 || ^0.27.0", "open": "^10.2.0", "recast": "^0.23.5", "semver": "^7.7.3", "use-sync-external-store": "^1.5.0", "ws": "^8.18.0" }, "peerDependencies": { "prettier": "^2 || ^3", "vite-plus": "^0.1.15" }, "optionalPeers": ["prettier", "vite-plus"], "bin": "./dist/bin/dispatcher.js" }, "sha512-vbSz7g/1rGMC1uAULqMZjALkIuLu2QABqfhRYhyr/11kzyesi+vAmwyJLukZP1FfecxGOgMwOh6GS0YsGpHAvQ=="], + "@storybook/builder-vite/storybook": ["storybook@10.4.0", "", { "dependencies": { "@storybook/global": "^5.0.0", "@storybook/icons": "^2.0.2", "@testing-library/jest-dom": "^6.9.1", "@testing-library/user-event": "^14.6.1", "@vitest/expect": "3.2.4", "@vitest/spy": "3.2.4", "@webcontainer/env": "^1.1.1", "esbuild": "^0.18.0 || ^0.19.0 || ^0.20.0 || ^0.21.0 || ^0.22.0 || ^0.23.0 || ^0.24.0 || ^0.25.0 || ^0.26.0 || ^0.27.0", "open": "^10.2.0", "oxc-parser": "^0.127.0", "oxc-resolver": "^11.19.1", "recast": "^0.23.5", "semver": "^7.7.3", "use-sync-external-store": "^1.5.0", "ws": "^8.18.0" }, "peerDependencies": { "@types/react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "prettier": "^2 || ^3", "vite-plus": "^0.1.15" }, "optionalPeers": ["@types/react", "prettier", "vite-plus"], "bin": "./dist/bin/dispatcher.js" }, "sha512-zrtctbVa6xEXCXuE3vsiR0At31zLOtzj8QudN/GaBJLZl0Z2DfF1rDPtTxdAbAp11M2J/7JVHaTIQKXauQPbmg=="], "@tailwindcss/oxide-wasm32-wasi/@emnapi/core": ["@emnapi/core@1.10.0", "", { "dependencies": { "@emnapi/wasi-threads": "1.2.1", "tslib": "^2.4.0" }, "bundled": true }, "sha512-yq6OkJ4p82CAfPl0u9mQebQHKPJkY7WrIuk205cTYnYe+k2Z8YBh11FrbRG/H6ihirqcacOgl2BIO8oyMQLeXw=="], @@ -5438,8 +5488,6 @@ "@typescript-eslint/visitor-keys/eslint-visitor-keys": ["eslint-visitor-keys@5.0.1", "", {}, "sha512-tD40eHxA35h0PEIZNeIjkHoDR4YjjJp34biM0mDvplBe//mB+IHCqHDGV7pxF+7MklTvighcCPPZC7ynWyjdTA=="], - "@vscode/ripgrep/yauzl": ["yauzl@2.10.0", "", { "dependencies": { "buffer-crc32": "~0.2.3", "fd-slicer": "~1.1.0" } }, "sha512-p4a9I6X6nu6IhoGmBqAcbJy1mlC4j27vEPZX9F4L4/vZT3Lyq1VkFHw/V/PUcB9Buo+DG3iHkT0x3Qya58zc3g=="], - "@vscode/test-cli/chokidar": ["chokidar@3.6.0", "", { "dependencies": { "anymatch": "~3.1.2", "braces": "~3.0.2", "glob-parent": "~5.1.2", "is-binary-path": "~2.1.0", "is-glob": "~4.0.1", "normalize-path": "~3.0.0", "readdirp": "~3.6.0" }, "optionalDependencies": { "fsevents": "~2.3.2" } }, "sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw=="], "@vscode/test-cli/glob": ["glob@10.5.0", "", { "dependencies": { "foreground-child": "^3.1.0", "jackspeak": "^3.1.2", "minimatch": "^9.0.4", "minipass": "^7.1.2", "package-json-from-dist": "^1.0.0", "path-scurry": "^1.11.1" }, "bin": { "glob": "dist/esm/bin.mjs" } }, "sha512-DfXN8DfhJ7NH3Oe7cFmu3NCu1wKbkReJ8TorzSAFbSKrlNaQSKfIzqYqVY8zlbs2NLBbWpRiU52GX2PbaBVNkg=="], @@ -5464,8 +5512,12 @@ "ajv-keywords/ajv": ["ajv@6.15.0", "", { "dependencies": { "fast-deep-equal": "^3.1.1", "fast-json-stable-stringify": "^2.0.0", "json-schema-traverse": "^0.4.1", "uri-js": "^4.2.2" } }, "sha512-fgFx7Hfoq60ytK2c7DhnF8jIvzYgOMxfugjLOSMHjLIPgenqa7S7oaagATUq99mV6IYvN2tRmC0wnTYX6iPbMw=="], + "anymatch/picomatch": ["picomatch@2.3.2", "", {}, "sha512-V7+vQEJ06Z+c5tSye8S+nHUfI51xoXIXjHQ99cQtKUkQqqO1kO/KCJUfZXuB47h/YBlDhah2H3hdUGXn8ie0oA=="], + "apache-arrow/@swc/helpers": ["@swc/helpers@0.5.21", "", { "dependencies": { "tslib": "^2.8.0" } }, "sha512-jI/VAmtdjB/RnI8GTnokyX7Ug8c+g+ffD6QRLa6XQewtnGyukKkKSk3wLTM3b5cjt1jNh9x0jfVlagdN2gDKQg=="], + "apache-arrow/@types/node": ["@types/node@20.19.41", "", { "dependencies": { "undici-types": "~6.21.0" } }, "sha512-ECymXOukMnOoVkC2bb1Vc/w/836DXncOg5m8Xj1RH7xSHZJWNYY6Zh7EH477vcnD5egKNNfy2RpNOmuChhFPgQ=="], + "app-builder-lib/@electron/get": ["@electron/get@3.1.0", "", { "dependencies": { "debug": "^4.1.1", "env-paths": "^2.2.0", "fs-extra": "^8.1.0", "got": "^11.8.5", "progress": "^2.0.3", "semver": "^6.2.0", "sumchecker": "^3.0.1" }, "optionalDependencies": { "global-agent": "^3.0.0" } }, "sha512-F+nKc0xW+kVbBRhFzaMgPy3KwmuNTYX1fx6+FxxoSnNgwYX6LD7AKBTWkU0MQ6IBoe7dz069CNkR673sPAgkCQ=="], "app-builder-lib/ci-info": ["ci-info@4.3.1", "", {}, "sha512-Wdy2Igu8OcBpI2pZePZ5oWjPC38tmDVx5WKUXKwlLYkA0ozo85sLsLvkBbBn/sZaSCMFOGZJ14fvW9t5/d7kdA=="], @@ -5474,6 +5526,8 @@ "app-builder-lib/hosted-git-info": ["hosted-git-info@4.1.0", "", { "dependencies": { "lru-cache": "^6.0.0" } }, "sha512-kyCuEOWjJqZuDbRHzL8V93NzQhwIB71oFWSyzVo+KPZI+pnQPPxucdkrOZvkLRnrf5URsQM+IJ09Dw29cRALIA=="], + "app-builder-lib/semver": ["semver@7.7.4", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-vFKC2IEtQnVhpT78h1Yp8wzwrf8CM+MzKMHGJZfBtzhZNycRFnXsHk6E5TxIkkMsgNS7mdX3AGB7x2QM2di4lA=="], + "app-builder-lib/which": ["which@5.0.0", "", { "dependencies": { "isexe": "^3.1.1" }, "bin": { "node-which": "bin/which.js" } }, "sha512-JEdGzHwwkrbWoGOlIHqQ5gtprKGOenpDHpxE9zVR1bWbOtYRyPPHMe9FaP6x61CmNaTThSkb0DAJte5jD+DmzQ=="], "archiver-utils/glob": ["glob@10.5.0", "", { "dependencies": { "foreground-child": "^3.1.0", "jackspeak": "^3.1.2", "minimatch": "^9.0.4", "minipass": "^7.1.2", "package-json-from-dist": "^1.0.0", "path-scurry": "^1.11.1" }, "bin": { "glob": "dist/esm/bin.mjs" } }, "sha512-DfXN8DfhJ7NH3Oe7cFmu3NCu1wKbkReJ8TorzSAFbSKrlNaQSKfIzqYqVY8zlbs2NLBbWpRiU52GX2PbaBVNkg=="], @@ -5492,6 +5546,8 @@ "ava/p-map": ["p-map@5.5.0", "", { "dependencies": { "aggregate-error": "^4.0.0" } }, "sha512-VFqfGDHlx87K66yZrNdI4YGtD70IRyd+zSvgks6mzHPRNkoKy+9EKP4SFC77/vTTQYmRmti7dvqC+m5jBrBAcg=="], + "ava/picomatch": ["picomatch@2.3.2", "", {}, "sha512-V7+vQEJ06Z+c5tSye8S+nHUfI51xoXIXjHQ99cQtKUkQqqO1kO/KCJUfZXuB47h/YBlDhah2H3hdUGXn8ie0oA=="], + "ava/write-file-atomic": ["write-file-atomic@5.0.1", "", { "dependencies": { "imurmurhash": "^0.1.4", "signal-exit": "^4.0.1" } }, "sha512-+QU2zd6OTD8XWIJCbffaiQeH9U73qIqafo1x6V1snCWYGJf6cVE0cDR4D8xRzcEnfI21IFrUPzPGtcPf8AC+Rw=="], "ava/yargs": ["yargs@17.7.2", "", { "dependencies": { "cliui": "^8.0.1", "escalade": "^3.1.1", "get-caller-file": "^2.0.5", "require-directory": "^2.1.1", "string-width": "^4.2.3", "y18n": "^5.0.5", "yargs-parser": "^21.1.1" } }, "sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w=="], @@ -5502,7 +5558,7 @@ "aws-sdk/xml2js": ["xml2js@0.6.2", "", { "dependencies": { "sax": ">=0.6.0", "xmlbuilder": "~11.0.0" } }, "sha512-T4rieHaC1EXcES0Kxxj4JWgaUQHDk+qwHcYOCFHfiwKz7tOVPLq7Hjq9dM1WCMhylqMEfP7hMcOIChvotiZegA=="], - "axios/proxy-from-env": ["proxy-from-env@2.1.0", "", {}, "sha512-cJ+oHTW1VAEa8cJslgmUZrc+sjRKgAKl3Zyse6+PV38hZe/V6Z14TbCuXcan9F9ghlz4QrFr2c92TNF82UkYHA=="], + "axios/https-proxy-agent": ["https-proxy-agent@5.0.1", "", { "dependencies": { "agent-base": "6", "debug": "4" } }, "sha512-dFcAjpTQFgoLMzC2VwU+C/CbS7uRL0lWmxDITmqm7C+7F0Odmj6s9l6alZc6AELXhrnggM2CeWSXHGOdX2YtwA=="], "babel-plugin-jsx-dom-expressions/@babel/helper-module-imports": ["@babel/helper-module-imports@7.18.6", "", { "dependencies": { "@babel/types": "^7.18.6" } }, "sha512-0NFvs3VkuSYbFi1x2Vd6tKrywq+z/cLeYC/RJNFrIX/30Bf5aiGYbtvGXolEktzJH8o5E5KJ3tT+nkxuuZFVlA=="], @@ -5510,8 +5566,6 @@ "bl/buffer": ["buffer@6.0.3", "", { "dependencies": { "base64-js": "^1.3.1", "ieee754": "^1.2.1" } }, "sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA=="], - "buffer/ieee754": ["ieee754@1.2.1", "", {}, "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA=="], - "builder-util/chalk": ["chalk@4.1.2", "", { "dependencies": { "ansi-styles": "^4.1.0", "supports-color": "^7.1.0" } }, "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA=="], "builder-util/fs-extra": ["fs-extra@10.1.0", "", { "dependencies": { "graceful-fs": "^4.2.0", "jsonfile": "^6.0.1", "universalify": "^2.0.0" } }, "sha512-oRXApq54ETRj4eMiFzGnHWGy+zo5raudjuxN0b8H7s/RU2oW0Wvsx9O0ACRN/kRq9E8Vu/ReskGB5o3ji+FzHQ=="], @@ -5522,8 +5576,6 @@ "c12/dotenv": ["dotenv@17.4.2", "", {}, "sha512-nI4U3TottKAcAD9LLud4Cb7b2QztQMUEfHbvhTH09bqXTxnSie8WnjPALV/WMCrJZ6UV/qHJ6L03OqO3LcdYZw=="], - "c12/pkg-types": ["pkg-types@2.3.1", "", { "dependencies": { "confbox": "^0.2.4", "exsolve": "^1.0.8", "pathe": "^2.0.3" } }, "sha512-y+ichcgc2LrADuhLNAx8DFjVfgz91pRxfZdI3UDhxHvcVEZsenLO+7XaU5vOp0u/7V/wZ+plyuQxtrDlZJ+yeg=="], - "c8/yargs": ["yargs@17.7.2", "", { "dependencies": { "cliui": "^8.0.1", "escalade": "^3.1.1", "get-caller-file": "^2.0.5", "require-directory": "^2.1.1", "string-width": "^4.2.3", "y18n": "^5.0.5", "yargs-parser": "^21.1.1" } }, "sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w=="], "c8/yargs-parser": ["yargs-parser@21.1.1", "", {}, "sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw=="], @@ -5584,6 +5636,8 @@ "effect/uuid": ["uuid@13.0.2", "", { "bin": { "uuid": "dist-node/bin/uuid" } }, "sha512-vzi9uRZ926x4XV73S/4qQaTwPXM2JBj6/6lI/byHH1jOpCzb0zDbfytgA9LcN/hzb2l7WQSQnxITOVx5un/wGw=="], + "electron/@types/node": ["@types/node@24.12.4", "", { "dependencies": { "undici-types": "~7.16.0" } }, "sha512-GUUEShf+PBCGW2KaXwcIt3Yk+e3pkKwWKb9GSyM9WQVE+ep2jzmHdGsHzu4wgcZy5fN9FBdVzjpBQsYlpfpgLA=="], + "electron-builder/chalk": ["chalk@4.1.2", "", { "dependencies": { "ansi-styles": "^4.1.0", "supports-color": "^7.1.0" } }, "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA=="], "electron-builder/fs-extra": ["fs-extra@10.1.0", "", { "dependencies": { "graceful-fs": "^4.2.0", "jsonfile": "^6.0.1", "universalify": "^2.0.0" } }, "sha512-oRXApq54ETRj4eMiFzGnHWGy+zo5raudjuxN0b8H7s/RU2oW0Wvsx9O0ACRN/kRq9E8Vu/ReskGB5o3ji+FzHQ=="], @@ -5598,6 +5652,8 @@ "electron-updater/fs-extra": ["fs-extra@10.1.0", "", { "dependencies": { "graceful-fs": "^4.2.0", "jsonfile": "^6.0.1", "universalify": "^2.0.0" } }, "sha512-oRXApq54ETRj4eMiFzGnHWGy+zo5raudjuxN0b8H7s/RU2oW0Wvsx9O0ACRN/kRq9E8Vu/ReskGB5o3ji+FzHQ=="], + "electron-updater/semver": ["semver@7.7.4", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-vFKC2IEtQnVhpT78h1Yp8wzwrf8CM+MzKMHGJZfBtzhZNycRFnXsHk6E5TxIkkMsgNS7mdX3AGB7x2QM2di4lA=="], + "electron-vite/esbuild": ["esbuild@0.25.12", "", { "optionalDependencies": { "@esbuild/aix-ppc64": "0.25.12", "@esbuild/android-arm": "0.25.12", "@esbuild/android-arm64": "0.25.12", "@esbuild/android-x64": "0.25.12", "@esbuild/darwin-arm64": "0.25.12", "@esbuild/darwin-x64": "0.25.12", "@esbuild/freebsd-arm64": "0.25.12", "@esbuild/freebsd-x64": "0.25.12", "@esbuild/linux-arm": "0.25.12", "@esbuild/linux-arm64": "0.25.12", "@esbuild/linux-ia32": "0.25.12", "@esbuild/linux-loong64": "0.25.12", "@esbuild/linux-mips64el": "0.25.12", "@esbuild/linux-ppc64": "0.25.12", "@esbuild/linux-riscv64": "0.25.12", "@esbuild/linux-s390x": "0.25.12", "@esbuild/linux-x64": "0.25.12", "@esbuild/netbsd-arm64": "0.25.12", "@esbuild/netbsd-x64": "0.25.12", "@esbuild/openbsd-arm64": "0.25.12", "@esbuild/openbsd-x64": "0.25.12", "@esbuild/openharmony-arm64": "0.25.12", "@esbuild/sunos-x64": "0.25.12", "@esbuild/win32-arm64": "0.25.12", "@esbuild/win32-ia32": "0.25.12", "@esbuild/win32-x64": "0.25.12" }, "bin": { "esbuild": "bin/esbuild" } }, "sha512-bbPBYYrtZbkt6Os6FiTLCTFxvq4tt3JKall1vRwshA3fdVztsLAatFaZobhkBC8/BrPetoa0oksYoKXoG4ryJg=="], "encoding-sniffer/iconv-lite": ["iconv-lite@0.6.3", "", { "dependencies": { "safer-buffer": ">= 2.1.2 < 3.0.0" } }, "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw=="], @@ -5606,6 +5662,8 @@ "enquirer/strip-ansi": ["strip-ansi@6.0.1", "", { "dependencies": { "ansi-regex": "^5.0.1" } }, "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A=="], + "esbuild-plugin-solid/@babel/preset-typescript": ["@babel/preset-typescript@7.28.5", "", { "dependencies": { "@babel/helper-plugin-utils": "^7.27.1", "@babel/helper-validator-option": "^7.27.1", "@babel/plugin-syntax-jsx": "^7.27.1", "@babel/plugin-transform-modules-commonjs": "^7.27.1", "@babel/plugin-transform-typescript": "^7.28.5" }, "peerDependencies": { "@babel/core": "^7.0.0-0" } }, "sha512-+bQy5WOI2V6LJZpPVxY+yp66XdZ2yifu0Mc1aP5CQKgjn4QM5IN2i5fAZ4xKop47pr8rpVhiAeu+nDQa12C8+g=="], + "eslint/ajv": ["ajv@6.15.0", "", { "dependencies": { "fast-deep-equal": "^3.1.1", "fast-json-stable-stringify": "^2.0.0", "json-schema-traverse": "^0.4.1", "uri-js": "^4.2.2" } }, "sha512-fgFx7Hfoq60ytK2c7DhnF8jIvzYgOMxfugjLOSMHjLIPgenqa7S7oaagATUq99mV6IYvN2tRmC0wnTYX6iPbMw=="], "eslint/chalk": ["chalk@4.1.2", "", { "dependencies": { "ansi-styles": "^4.1.0", "supports-color": "^7.1.0" } }, "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA=="], @@ -5634,7 +5692,7 @@ "framer-motion/motion-utils": ["motion-utils@12.36.0", "", {}, "sha512-eHWisygbiwVvf6PZ1vhaHCLamvkSbPIeAYxWUuL3a2PD/TROgE7FvfHWTIH4vMl798QLfMw15nRqIaRDXTlYRg=="], - "friendly-words/express": ["express@4.22.1", "", { "dependencies": { "accepts": "~1.3.8", "array-flatten": "1.1.1", "body-parser": "~1.20.3", "content-disposition": "~0.5.4", "content-type": "~1.0.4", "cookie": "~0.7.1", "cookie-signature": "~1.0.6", "debug": "2.6.9", "depd": "2.0.0", "encodeurl": "~2.0.0", "escape-html": "~1.0.3", "etag": "~1.8.1", "finalhandler": "~1.3.1", "fresh": "~0.5.2", "http-errors": "~2.0.0", "merge-descriptors": "1.0.3", "methods": "~1.1.2", "on-finished": "~2.4.1", "parseurl": "~1.3.3", "path-to-regexp": "~0.1.12", "proxy-addr": "~2.0.7", "qs": "~6.14.0", "range-parser": "~1.2.1", "safe-buffer": "5.2.1", "send": "~0.19.0", "serve-static": "~1.16.2", "setprototypeof": "1.2.0", "statuses": "~2.0.1", "type-is": "~1.6.18", "utils-merge": "1.0.1", "vary": "~1.1.2" } }, "sha512-F2X8g9P1X7uCPZMA3MVf9wcTqlyNp7IhH5qPCI0izhaOIYXaW9L535tGA3qmjRzpH+bZczqq7hVKxTR4NWnu+g=="], + "friendly-words/express": ["express@4.22.2", "", { "dependencies": { "accepts": "~1.3.8", "array-flatten": "1.1.1", "body-parser": "~1.20.5", "content-disposition": "~0.5.4", "content-type": "~1.0.4", "cookie": "~0.7.1", "cookie-signature": "~1.0.6", "debug": "2.6.9", "depd": "2.0.0", "encodeurl": "~2.0.0", "escape-html": "~1.0.3", "etag": "~1.8.1", "finalhandler": "~1.3.1", "fresh": "~0.5.2", "http-errors": "~2.0.0", "merge-descriptors": "1.0.3", "methods": "~1.1.2", "on-finished": "~2.4.1", "parseurl": "~1.3.3", "path-to-regexp": "~0.1.12", "proxy-addr": "~2.0.7", "qs": "~6.15.1", "range-parser": "~1.2.1", "safe-buffer": "5.2.1", "send": "~0.19.0", "serve-static": "~1.16.2", "setprototypeof": "1.2.0", "statuses": "~2.0.1", "type-is": "~1.6.18", "utils-merge": "1.0.1", "vary": "~1.1.2" } }, "sha512-IuL+Elrou2ZvCFHs18/CIzy2Nzvo25nZ1/D2eIZlz7c+QUayAcYoiM2BthCjs+EBHVpjYjcuLDAiCWgeIX3X1Q=="], "gaxios/node-fetch": ["node-fetch@3.3.2", "", { "dependencies": { "data-uri-to-buffer": "^4.0.0", "fetch-blob": "^3.1.4", "formdata-polyfill": "^4.0.10" } }, "sha512-dRB78srN/l6gqWulah9SrxeYnxeddIG30+GOqK/9OlLVyLg3HPnr6SqOWTWOXKRwC2eGYCkZ59NNuSgvSrpgOA=="], @@ -5654,6 +5712,8 @@ "iconv-corefoundation/node-addon-api": ["node-addon-api@1.7.2", "", {}, "sha512-ibPK3iA+vaY1eEjESkQkM0BbCqFOaZMiXRTtdB0u7b4djtY6JnsjvPdUHVMg6xQt3B8fpTTWHI9A+ADjM9frzg=="], + "image-q/@types/node": ["@types/node@16.9.1", "", {}, "sha512-QpLcX9ZSsq3YYUUnD3nFDY8H7wctAhQj/TFKL8Ya8v5fMm3CFXxo8zStsLAl780ltoYoo1WvKUVGBQK+1ifr7g=="], + "import-fresh/resolve-from": ["resolve-from@4.0.0", "", {}, "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g=="], "isomorphic-git/ignore": ["ignore@5.3.2", "", {}, "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g=="], @@ -5666,6 +5726,8 @@ "keytar/node-addon-api": ["node-addon-api@4.3.0", "", {}, "sha512-73sE9+3UaLYYFmDsFZnqCInzPyh3MqIwZO9cw58yIqAZhONrrabrYyYe3TuIqtIiOuTXVhsGau8hcrhhwSsDIQ=="], + "kilo-code/@types/node": ["@types/node@22.13.9", "", { "dependencies": { "undici-types": "~6.20.0" } }, "sha512-acBjXdRJ3A6Pb3tqnw9HZmyR3Fiol3aGxRCK1x3d+6CDAMjl7I649wpSd+yNURCjbOUGu9tqtLKnTGxmK6CyGw=="], + "kilo-code/openai": ["openai@4.104.0", "", { "dependencies": { "@types/node": "^18.11.18", "@types/node-fetch": "^2.6.4", "abort-controller": "^3.0.0", "agentkeepalive": "^4.2.1", "form-data-encoder": "1.7.2", "formdata-node": "^4.3.2", "node-fetch": "^2.6.7" }, "peerDependencies": { "ws": "^8.18.0", "zod": "^3.23.8" }, "optionalPeers": ["ws", "zod"], "bin": { "openai": "bin/cli" } }, "sha512-p99EFNsA/yX6UhVO93f5kJsDRLAg+CTA2RBqdHK4RtK8u5IJw32Hyb2dTGKbnnFmnuoBv5r7Z2CURI9sGZpSuA=="], "kilo-code/typescript": ["typescript@5.9.3", "", { "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" } }, "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw=="], @@ -5696,10 +5758,14 @@ "mermaid/uuid": ["uuid@11.1.1", "", { "bin": { "uuid": "dist/esm/bin/uuid" } }, "sha512-vIYxrBCC/N/K+Js3qSN88go7kIfNPssr/hHCesKCQNAjmgvYS2oqr69kIufEG+O4+PfezOH4EbIeHCfFov8ZgQ=="], + "micromatch/picomatch": ["picomatch@2.3.2", "", {}, "sha512-V7+vQEJ06Z+c5tSye8S+nHUfI51xoXIXjHQ99cQtKUkQqqO1kO/KCJUfZXuB47h/YBlDhah2H3hdUGXn8ie0oA=="], + "minipass-flush/minipass": ["minipass@3.3.6", "", { "dependencies": { "yallist": "^4.0.0" } }, "sha512-DxiNidxSEK+tHG6zOIklvNOwm3hvCrbUrdtzY74U6HKTJxvIDfOUL5W5P2Ghd3DTkhhKPYGqeNUIh5qcM4YBfw=="], "minipass-pipeline/minipass": ["minipass@3.3.6", "", { "dependencies": { "yallist": "^4.0.0" } }, "sha512-DxiNidxSEK+tHG6zOIklvNOwm3hvCrbUrdtzY74U6HKTJxvIDfOUL5W5P2Ghd3DTkhhKPYGqeNUIh5qcM4YBfw=="], + "mocha/diff": ["diff@7.0.0", "", {}, "sha512-PJWHUb1RFevKCwaFA9RlG5tCd+FO5iRh9A8HEtkmBH2Li03iJriB6m6JIN4rGz3K3JLawI7/veA1xzRKP6ISBw=="], + "mocha/glob": ["glob@10.5.0", "", { "dependencies": { "foreground-child": "^3.1.0", "jackspeak": "^3.1.2", "minimatch": "^9.0.4", "minipass": "^7.1.2", "package-json-from-dist": "^1.0.0", "path-scurry": "^1.11.1" }, "bin": { "glob": "dist/esm/bin.mjs" } }, "sha512-DfXN8DfhJ7NH3Oe7cFmu3NCu1wKbkReJ8TorzSAFbSKrlNaQSKfIzqYqVY8zlbs2NLBbWpRiU52GX2PbaBVNkg=="], "mocha/minimatch": ["minimatch@9.0.9", "", { "dependencies": { "brace-expansion": "^2.0.2" } }, "sha512-OBwBN9AL4dqmETlpS2zasx+vTeWclWzkblfZk7KTA5j3jeOONz/tRCnZomUyvNg83wL5Zv9Ss6HMJXAgL8R2Yg=="], @@ -5714,13 +5780,15 @@ "mssql/commander": ["commander@11.1.0", "", {}, "sha512-yPVavfyCcRhmorC7rWlkHn15b4wDVgVmBA7kV4QVBsF7kv/9TKJAbAXVTxvTnwP8HHKjRCJDClKbciiYS7p0DQ=="], + "mssql/tedious": ["tedious@18.6.2", "", { "dependencies": { "@azure/core-auth": "^1.7.2", "@azure/identity": "^4.2.1", "@azure/keyvault-keys": "^4.4.0", "@js-joda/core": "^5.6.1", "@types/node": ">=18", "bl": "^6.0.11", "iconv-lite": "^0.6.3", "js-md4": "^0.3.2", "native-duplexpair": "^1.0.0", "sprintf-js": "^1.1.3" } }, "sha512-g7jC56o3MzLkE3lHkaFe2ZdOVFBahq5bsB60/M4NYUbocw/MCrS89IOEQUFr+ba6pb8ZHczZ/VqCyYeYq0xBAg=="], + "next/postcss": ["postcss@8.4.31", "", { "dependencies": { "nanoid": "^3.3.6", "picocolors": "^1.0.0", "source-map-js": "^1.0.2" } }, "sha512-PS08Iboia9mts/2ygV3eLpY5ghnUcfLV/EXTOW1E2qYxJKGGBUtNjN76FYHnMs36RmARn41bC0AZmn+rR0OVpQ=="], "node-gyp/undici": ["undici@6.25.0", "", {}, "sha512-ZgpWDC5gmNiuY9CnLVXEH8rl50xhRCuLNA97fAUnKi8RRuV4E6KG31pDTsLVUKnohJE0I3XDrTeEydAXRw47xg=="], "node-gyp-build-optional-packages/detect-libc": ["detect-libc@2.1.2", "", {}, "sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ=="], - "node-sarif-builder/fs-extra": ["fs-extra@11.3.4", "", { "dependencies": { "graceful-fs": "^4.2.0", "jsonfile": "^6.0.1", "universalify": "^2.0.0" } }, "sha512-CTXd6rk/M3/ULNQj8FBqBWHYBVYybQ3VPBw0xGKFe3tuH7ytT6ACnvzpIQ3UZtB8yvUKC2cXn1a+x+5EVQLovA=="], + "node-sarif-builder/fs-extra": ["fs-extra@11.3.5", "", { "dependencies": { "graceful-fs": "^4.2.0", "jsonfile": "^6.0.1", "universalify": "^2.0.0" } }, "sha512-eKpRKAovdpZtR1WopLHxlBWvAgPny3c4gX1G5Jhwmmw4XJj0ifSD5qB5TOo8hmA0wlRKDAOAhEE1yVPgs6Fgcg=="], "normalize-package-data/hosted-git-info": ["hosted-git-info@7.0.2", "", { "dependencies": { "lru-cache": "^10.0.1" } }, "sha512-puUZAUKT5m8Zzvs72XWy3HtvVbTWljRE66cP60bxJzAqf2DgICo7lYTY2IHUmLnNpjYvw5bvmoHvPc0QO2a62w=="], @@ -5744,6 +5812,8 @@ "openid-client/lru-cache": ["lru-cache@6.0.0", "", { "dependencies": { "yallist": "^4.0.0" } }, "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA=="], + "opentui-spinner/@opentui/core": ["@opentui/core@0.1.75", "", { "dependencies": { "bun-ffi-structs": "0.1.2", "diff": "8.0.2", "jimp": "1.6.0", "marked": "17.0.1", "yoga-layout": "3.2.1" }, "optionalDependencies": { "@dimforge/rapier2d-simd-compat": "^0.17.3", "@opentui/core-darwin-arm64": "0.1.75", "@opentui/core-darwin-x64": "0.1.75", "@opentui/core-linux-arm64": "0.1.75", "@opentui/core-linux-x64": "0.1.75", "@opentui/core-win32-arm64": "0.1.75", "@opentui/core-win32-x64": "0.1.75", "bun-webgpu": "0.1.4", "planck": "^1.4.2", "three": "0.177.0" }, "peerDependencies": { "web-tree-sitter": "0.25.10" } }, "sha512-8ARRZxSG+BXkJmEVtM2DQ4se7DAF1ZCKD07d+AklgTr2mxCzmdxxPbOwRzboSQ6FM7qGuTVPVbV4O2W9DpUmoA=="], + "ora/cli-spinners": ["cli-spinners@2.9.2", "", {}, "sha512-ywqV+5MmyL4E7ybXgKys4DugZbX0FC6LnwrhjuykIjnK9k8OQacQ7axGKnjDXWNhns0xot3bZI5h55H8yo9cJg=="], "ora/log-symbols": ["log-symbols@6.0.0", "", { "dependencies": { "chalk": "^5.3.0", "is-unicode-supported": "^1.3.0" } }, "sha512-i24m8rpwhmPIS4zscNzK6MSEhk0DUWa/8iYQWxhffV8jkI4Phvs3F+quL5xvS0gdQR0FyTCMMH33Y78dDTzzIw=="], @@ -5762,8 +5832,6 @@ "pkg-conf/find-up": ["find-up@6.3.0", "", { "dependencies": { "locate-path": "^7.1.0", "path-exists": "^5.0.0" } }, "sha512-v2ZsoEuVHYy8ZIlYqwPe/39Cy+cFDzp4dXPaxNvkEuouymu+2Jbz0PxpKarJHYJTmv2HWT3O382qY8l4jMWthw=="], - "pkg-types/confbox": ["confbox@0.1.8", "", {}, "sha512-RMtmw0iFkeR4YV+fUOSucriAQNb9g8zFR52MWCtl+cCZOFRNL6zeB395vPzFhEjjn4fMxXudmELnl/KF/WrK6w=="], - "pkg-up/find-up": ["find-up@3.0.0", "", { "dependencies": { "locate-path": "^3.0.0" } }, "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg=="], "playwright/fsevents": ["fsevents@2.3.2", "", { "os": "darwin" }, "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA=="], @@ -5774,7 +5842,7 @@ "prebuild-install/detect-libc": ["detect-libc@2.1.2", "", {}, "sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ=="], - "prebuild-install/node-abi": ["node-abi@3.90.0", "", { "dependencies": { "semver": "^7.3.5" } }, "sha512-pZNQT7UnYlMwMBy5N1lV5X/YLTbZM5ncytN3xL7CHEzhDN8uVe0u55yaPUJICIJjaCW8NrM5BFdqr7HLweStNA=="], + "prebuild-install/node-abi": ["node-abi@3.92.0", "", { "dependencies": { "semver": "^7.3.5" } }, "sha512-KdHvFWZjEKDf0cakgFjebl371GPsISX2oZHcuyKqM7DtogIsHrqKeLTo8wBHxaXRAQlY2PsPlZmfo+9ZCxEREQ=="], "pretty-format/ansi-regex": ["ansi-regex@5.0.1", "", {}, "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ=="], @@ -5804,6 +5872,10 @@ "rimraf/glob": ["glob@7.2.3", "", { "dependencies": { "fs.realpath": "^1.0.0", "inflight": "^1.0.4", "inherits": "2", "minimatch": "^3.1.1", "once": "^1.3.0", "path-is-absolute": "^1.0.0" } }, "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q=="], + "rollup/@types/estree": ["@types/estree@1.0.8", "", {}, "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w=="], + + "router/path-to-regexp": ["path-to-regexp@8.4.2", "", {}, "sha512-qRcuIdP69NPm4qbACK+aDogI5CBDMi1jKe0ry5rSQJz8JVLsC7jV8XpiJjGRLLol3N+R5ihGYcrPLTno6pAdBA=="], + "secretlint/globby": ["globby@14.1.0", "", { "dependencies": { "@sindresorhus/merge-streams": "^2.1.0", "fast-glob": "^3.3.3", "ignore": "^7.0.3", "path-type": "^6.0.0", "slash": "^5.1.0", "unicorn-magic": "^0.3.0" } }, "sha512-0Ia46fDOaT7k4og1PDW4YbodWWr3scS2vAr2lTbsplOt2WkKp0vQbkI9wKis/T5LV/dqPjO3bpS/z6GTJB82LA=="], "serialize-error/type-fest": ["type-fest@0.13.1", "", {}, "sha512-34R7HTnG0XIJcBSn5XhDd7nNFPRcXYRZrBB2O2jdKqYODldSzBAqzsWoZYYvduky73toYS/ESqxPvkDf/F0XMg=="], @@ -5818,8 +5890,6 @@ "slice-ansi/is-fullwidth-code-point": ["is-fullwidth-code-point@4.0.0", "", {}, "sha512-O4L094N2/dZ7xqVdrXhh9r1KODPJpFms8B5sGdJLPy664AgvXsreZUyCQQNItZRDlYug4xStLjNp/sz3HvBowQ=="], - "socks/ip-address": ["ip-address@10.2.0", "", {}, "sha512-/+S6j4E9AHvW9SWMSEY9Xfy66O5PWvVEJ08O0y5JGyEKQpojb0K0GKpz/v5HJ/G0vi3D2sjGK78119oXZeE0qA=="], - "sort-keys/is-plain-obj": ["is-plain-obj@1.1.0", "", {}, "sha512-yvkRyxmFKEOQ4pNXCmJG5AEQNlXJS5LaONXo5/cLdTZdWvsZ1ioJEonLGAosKlMWE8lwUy/bJzMjcw8az73+Fg=="], "spdx-correct/spdx-expression-parse": ["spdx-expression-parse@3.0.1", "", { "dependencies": { "spdx-exceptions": "^2.1.0", "spdx-license-ids": "^3.0.0" } }, "sha512-cbqHunsQWnJNE6KhVSMsMeH5H/L9EpymbzqTQ3uLwNCLZ1Q481oWaofqH7nO6V07xlXwY6PhQdQ2IedWx/ZK4Q=="], @@ -5854,12 +5924,12 @@ "tar-fs/tar-stream": ["tar-stream@2.2.0", "", { "dependencies": { "bl": "^4.0.3", "end-of-stream": "^1.4.1", "fs-constants": "^1.0.0", "inherits": "^2.0.3", "readable-stream": "^3.1.1" } }, "sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ=="], - "tedious/iconv-lite": ["iconv-lite@0.6.3", "", { "dependencies": { "safer-buffer": ">= 2.1.2 < 3.0.0" } }, "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw=="], - "temp-file/fs-extra": ["fs-extra@10.1.0", "", { "dependencies": { "graceful-fs": "^4.2.0", "jsonfile": "^6.0.1", "universalify": "^2.0.0" } }, "sha512-oRXApq54ETRj4eMiFzGnHWGy+zo5raudjuxN0b8H7s/RU2oW0Wvsx9O0ACRN/kRq9E8Vu/ReskGB5o3ji+FzHQ=="], "test-exclude/glob": ["glob@10.5.0", "", { "dependencies": { "foreground-child": "^3.1.0", "jackspeak": "^3.1.2", "minimatch": "^9.0.4", "minipass": "^7.1.2", "package-json-from-dist": "^1.0.0", "path-scurry": "^1.11.1" }, "bin": { "glob": "dist/esm/bin.mjs" } }, "sha512-DfXN8DfhJ7NH3Oe7cFmu3NCu1wKbkReJ8TorzSAFbSKrlNaQSKfIzqYqVY8zlbs2NLBbWpRiU52GX2PbaBVNkg=="], + "thread-stream/real-require": ["real-require@1.0.0", "", {}, "sha512-P4nbQYQfePJxRSmY+v/KINxVucm4NF3p3s7pJveMTtom52FR4YGltUQLB8idDXwDDWW+eYrWDFbuzUnjoWHF7g=="], + "tiny-async-pool/semver": ["semver@5.7.2", "", { "bin": { "semver": "bin/semver" } }, "sha512-cBznnQ9KjJqU67B52RMC65CMarK2600WFnbkcaiwWq3xy/5haFJlshgnpjovMVJ+Hff49d8GEn0b87C5pDQ10g=="], "to-buffer/isarray": ["isarray@2.0.5", "", {}, "sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw=="], @@ -5868,6 +5938,8 @@ "tree-sitter-bash/node-addon-api": ["node-addon-api@8.7.0", "", {}, "sha512-9MdFxmkKaOYVTV+XVRG8ArDwwQ77XIgIPyKASB1k3JPq3M8fGQQQE3YpMOrKm6g//Ktx8ivZr8xo1Qmtqub+GA=="], + "type-is/content-type": ["content-type@2.0.0", "", {}, "sha512-j/O/d7GcZCyNl7/hwZAb606rzqkyvaDctLmckbxLzHvFBzTJHuGEdodATcP3yIRoDrLHkIATJuvzbFlp/ki2cQ=="], + "unused-filename/escape-string-regexp": ["escape-string-regexp@5.0.0", "", {}, "sha512-/veY75JbMK4j1yjvuUxuVsiS/hr/4iHs9FTT6cgTexxdE0Ly/glccBAkloH/DofkjRbZU3bnoj38mOmhkZ0lHw=="], "unused-filename/path-exists": ["path-exists@5.0.0", "", {}, "sha512-RjhtfwJOxzcFmNOi6ltcbcu4Iu+FL3zEj83dk4kAS+fVpTxXLO1b38RvJgT/0QwvV/L3aY9TAnyv0EOqW4GoMQ=="], @@ -5882,8 +5954,6 @@ "vitest/why-is-node-running": ["why-is-node-running@2.3.0", "", { "dependencies": { "siginfo": "^2.0.0", "stackback": "0.0.2" }, "bin": { "why-is-node-running": "cli.js" } }, "sha512-hUrmaWBdVDcxvYqnyh09zunKzROWjbZTiNy8dBEjkS7ehEDQibXJ7XvlmtbwuTclUiIyN+CyXQD4Vmko8fNm8w=="], - "vscode-languageserver-protocol/vscode-jsonrpc": ["vscode-jsonrpc@8.2.0", "", {}, "sha512-C+r0eKJUIfiDIfwJhria30+TYWPtuHJXHtI7J0YlOmKAo7ogxP20T0zxB7HZQIFhIyvoBPwWskjxrvAtfjyZfA=="], - "whatwg-encoding/iconv-lite": ["iconv-lite@0.6.3", "", { "dependencies": { "safer-buffer": ">= 2.1.2 < 3.0.0" } }, "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw=="], "wrap-ansi/ansi-styles": ["ansi-styles@6.2.3", "", {}, "sha512-4Dj6M28JB+oAH8kFkTLUo+a2jwOFkuqb3yucU0CANcRRUbxS0cP0nZYCGjcc3BNXwRIsUVmDGgzawme7zvJHvg=="], @@ -5930,16 +6000,16 @@ "@ai-sdk/vercel/@ai-sdk/provider-utils/@standard-schema/spec": ["@standard-schema/spec@1.1.0", "", {}, "sha512-l2aFy5jALhniG5HgqrD6jXLi/rUWrKvqN/qJx6yoJsgKhblVd+iqqU4RCXavm/jPityDo5TCvKMnpjKnOriy0w=="], - "@aws-crypto/sha1-browser/@smithy/util-utf8/@smithy/util-buffer-from": ["@smithy/util-buffer-from@2.2.0", "", { "dependencies": { "@smithy/is-array-buffer": "^2.2.0", "tslib": "^2.6.2" } }, "sha512-IJdWBbTcMQ6DA0gdNhh/BwrLkDR+ADW5Kr1aZmd4k3DIF6ezMV4R2NIAmT08wQJ3yUK82thHWmC/TnK/wpMMIA=="], - - "@aws-crypto/sha256-browser/@smithy/util-utf8/@smithy/util-buffer-from": ["@smithy/util-buffer-from@2.2.0", "", { "dependencies": { "@smithy/is-array-buffer": "^2.2.0", "tslib": "^2.6.2" } }, "sha512-IJdWBbTcMQ6DA0gdNhh/BwrLkDR+ADW5Kr1aZmd4k3DIF6ezMV4R2NIAmT08wQJ3yUK82thHWmC/TnK/wpMMIA=="], - - "@aws-crypto/util/@smithy/util-utf8/@smithy/util-buffer-from": ["@smithy/util-buffer-from@2.2.0", "", { "dependencies": { "@smithy/is-array-buffer": "^2.2.0", "tslib": "^2.6.2" } }, "sha512-IJdWBbTcMQ6DA0gdNhh/BwrLkDR+ADW5Kr1aZmd4k3DIF6ezMV4R2NIAmT08wQJ3yUK82thHWmC/TnK/wpMMIA=="], + "@anthropic-ai/sdk/@types/node/undici-types": ["undici-types@5.26.5", "", {}, "sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA=="], "@babel/helper-compilation-targets/lru-cache/yallist": ["yallist@3.1.1", "", {}, "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g=="], "@develar/schema-utils/ajv/json-schema-traverse": ["json-schema-traverse@0.4.1", "", {}, "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg=="], + "@effect/platform-node-shared/effect/@standard-schema/spec": ["@standard-schema/spec@1.1.0", "", {}, "sha512-l2aFy5jALhniG5HgqrD6jXLi/rUWrKvqN/qJx6yoJsgKhblVd+iqqU4RCXavm/jPityDo5TCvKMnpjKnOriy0w=="], + + "@effect/platform-node-shared/effect/uuid": ["uuid@13.0.2", "", { "bin": { "uuid": "dist-node/bin/uuid" } }, "sha512-vzi9uRZ926x4XV73S/4qQaTwPXM2JBj6/6lI/byHH1jOpCzb0zDbfytgA9LcN/hzb2l7WQSQnxITOVx5un/wGw=="], + "@electron/asar/minimatch/brace-expansion": ["brace-expansion@1.1.14", "", { "dependencies": { "balanced-match": "^1.0.0", "concat-map": "0.0.1" } }, "sha512-MWPGfDxnyzKU7rNOW9SP/c50vi3xrmrua/+6hfPbCS2ABNWfx24vPidzvC7krjU/RTo235sV776ymlsMtGKj8g=="], "@electron/fuses/chalk/supports-color": ["supports-color@7.2.0", "", { "dependencies": { "has-flag": "^4.0.0" } }, "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw=="], @@ -5972,6 +6042,8 @@ "@eslint/eslintrc/minimatch/brace-expansion": ["brace-expansion@1.1.14", "", { "dependencies": { "balanced-match": "^1.0.0", "concat-map": "0.0.1" } }, "sha512-MWPGfDxnyzKU7rNOW9SP/c50vi3xrmrua/+6hfPbCS2ABNWfx24vPidzvC7krjU/RTo235sV776ymlsMtGKj8g=="], + "@happy-dom/global-registrator/@types/node/undici-types": ["undici-types@6.21.0", "", {}, "sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ=="], + "@hey-api/openapi-ts/open/wsl-utils": ["wsl-utils@0.3.1", "", { "dependencies": { "is-wsl": "^3.1.0", "powershell-utils": "^0.1.0" } }, "sha512-g/eziiSUNBSsdDJtCLB8bdYEUMj4jR7AGeUo96p/3dTafgjHhpF4RiCFPiRILwjQoDXx5MqkBr4fwWtR3Ky4Wg=="], "@kilocode/kilo-gateway/@ai-sdk/openai/@ai-sdk/provider-utils": ["@ai-sdk/provider-utils@4.0.21", "", { "dependencies": { "@ai-sdk/provider": "3.0.8", "@standard-schema/spec": "^1.1.0", "eventsource-parser": "^3.0.6" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-MtFUYI1/8mgDvRmaBDjbLJPFFrMG777AvSgyIFQtZHIMzm88R/12vYBBpnk7pfiWLFE1DSZzY4WDYzGbKAcmiw=="], @@ -5990,18 +6062,30 @@ "@kilocode/kilo-gateway/@opentui/core/@opentui/core-win32-x64": ["@opentui/core-win32-x64@0.1.75", "", { "os": "win32", "cpu": "x64" }, "sha512-ESpVZVGewe3JkB2TwrG3VRbkxT909iPdtvgNT7xTCIYH2VB4jqZomJfvERPTE0tvqAZJm19mHECzJFI8asSJgQ=="], - "@kilocode/kilo-gateway/@opentui/core/bun-webgpu": ["bun-webgpu@0.1.4", "", { "dependencies": { "@webgpu/types": "^0.1.60" }, "optionalDependencies": { "bun-webgpu-darwin-arm64": "^0.1.4", "bun-webgpu-darwin-x64": "^0.1.4", "bun-webgpu-linux-x64": "^0.1.4", "bun-webgpu-win32-x64": "^0.1.4" } }, "sha512-Kw+HoXl1PMWJTh9wvh63SSRofTA8vYBFCw0XEP1V1fFdQEDhI8Sgf73sdndE/oDpN/7CMx0Yv/q8FCvO39ROMQ=="], + "@kilocode/kilo-gateway/@opentui/core/bun-ffi-structs": ["bun-ffi-structs@0.1.2", "", { "peerDependencies": { "typescript": "^5" } }, "sha512-Lh1oQAYHDcnesJauieA4UNkWGXY9hYck7OA5IaRwE3Bp6K2F2pJSNYqq+hIy7P3uOvo3km3oxS8304g5gDMl/w=="], + + "@kilocode/kilo-gateway/@opentui/core/diff": ["diff@8.0.2", "", {}, "sha512-sSuxWU5j5SR9QQji/o2qMvqRNYRDOcBTgsJ/DeCf4iSN4gW+gNMXM7wFIP+fdXZxoNiAnHUTGjCr+TSWXdRDKg=="], "@kilocode/kilo-gateway/@opentui/solid/@babel/core": ["@babel/core@7.28.0", "", { "dependencies": { "@ampproject/remapping": "^2.2.0", "@babel/code-frame": "^7.27.1", "@babel/generator": "^7.28.0", "@babel/helper-compilation-targets": "^7.27.2", "@babel/helper-module-transforms": "^7.27.3", "@babel/helpers": "^7.27.6", "@babel/parser": "^7.28.0", "@babel/template": "^7.27.2", "@babel/traverse": "^7.28.0", "@babel/types": "^7.28.0", "convert-source-map": "^2.0.0", "debug": "^4.1.0", "gensync": "^1.0.0-beta.2", "json5": "^2.2.3", "semver": "^6.3.1" } }, "sha512-UlLAnTPrFdNGoFtbSXwcGFQBtQZJCNjaN6hQNP3UPvuNXT1i82N26KL3dZeIpNalWywr9IuQuncaAfUaS1g6sQ=="], "@kilocode/kilo-gateway/@opentui/solid/babel-preset-solid": ["babel-preset-solid@1.9.9", "", { "dependencies": { "babel-plugin-jsx-dom-expressions": "^0.40.1" }, "peerDependencies": { "@babel/core": "^7.0.0", "solid-js": "^1.9.8" }, "optionalPeers": ["solid-js"] }, "sha512-pCnxWrciluXCeli/dj5PIEHgbNzim3evtTn12snjqqg8QZWJNMjH1AWIp4iG/tbVjqQ72aBEymMSagvmgxubXw=="], + "@kilocode/kilo-gateway/@types/node/undici-types": ["undici-types@6.20.0", "", {}, "sha512-Ny6QZ2Nju20vw1SRHe3d9jVu6gJ+4e3+MMpqu7pqE5HT6WsTSlce++GQmK5UXS8mzV8DSYHrQH+Xrf2jVcuKNg=="], + + "@kilocode/kilo-indexing/@types/node/undici-types": ["undici-types@6.20.0", "", {}, "sha512-Ny6QZ2Nju20vw1SRHe3d9jVu6gJ+4e3+MMpqu7pqE5HT6WsTSlce++GQmK5UXS8mzV8DSYHrQH+Xrf2jVcuKNg=="], + + "@kilocode/plugin/@types/node/undici-types": ["undici-types@6.20.0", "", {}, "sha512-Ny6QZ2Nju20vw1SRHe3d9jVu6gJ+4e3+MMpqu7pqE5HT6WsTSlce++GQmK5UXS8mzV8DSYHrQH+Xrf2jVcuKNg=="], + + "@kilocode/sdk/@types/node/undici-types": ["undici-types@6.20.0", "", {}, "sha512-Ny6QZ2Nju20vw1SRHe3d9jVu6gJ+4e3+MMpqu7pqE5HT6WsTSlce++GQmK5UXS8mzV8DSYHrQH+Xrf2jVcuKNg=="], + "@malept/flatpak-bundler/fs-extra/jsonfile": ["jsonfile@6.2.1", "", { "dependencies": { "universalify": "^2.0.0" }, "optionalDependencies": { "graceful-fs": "^4.1.6" } }, "sha512-zwOTdL3rFQ/lRdBnntKVOX6k5cKJwEc1HdilT71BWEu7J41gXIB2MRp+vxduPSwZJPWBxEzv4yH1wYLJGUHX4Q=="], "@malept/flatpak-bundler/fs-extra/universalify": ["universalify@2.0.1", "", {}, "sha512-gptHNQghINnc/vTGIk0SOFGFNXw7JVrlRUtConJRlvaw6DuX0wO5Jeko9sWrMBhh+PsYAZ7oXAiOnf/UKogyiw=="], "@manypkg/find-root/find-up/locate-path": ["locate-path@5.0.0", "", { "dependencies": { "p-locate": "^4.1.0" } }, "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g=="], + "@morphllm/morphsdk/openai/@types/node": ["@types/node@18.19.130", "", { "dependencies": { "undici-types": "~5.26.4" } }, "sha512-GRaXQx6jGfL8sKfaIDD6OupbIHBr9jv7Jnaml9tB7l4v068PAOXqfcujMMo5PhbIs6ggR1XODELqahT2R8v0fg=="], + "@morphllm/morphsdk/openai/zod": ["zod@3.25.76", "", {}, "sha512-gzUt/qt81nXsFGKIFcC3YnfEAx5NkunCfnDlvuBSSFS02bcXu4Lmea0AFIUwbLWxWPx3d9p8S5QoaujKcNQxcQ=="], "@octokit/core/@octokit/types/@octokit/openapi-types": ["@octokit/openapi-types@24.2.0", "", {}, "sha512-9sIH3nSUttelJSXUrmGzl7QUBFul0/mB8HRYl3fOlgHbIWG+WnYDXU3v/2zMtAvuzZ/ed00Ei6on975FhBfzrg=="], @@ -6014,6 +6098,8 @@ "@octokit/graphql/@octokit/request/@octokit/types": ["@octokit/types@16.0.0", "", { "dependencies": { "@octokit/openapi-types": "^27.0.0" } }, "sha512-sKq+9r1Mm4efXW1FCk7hFSeJo4QKreL/tTbR0rz/qx/r1Oa2VV83LTA/H/MuCOX7uCIJmQVRKBcbmWoySjAnSg=="], + "@octokit/graphql/@octokit/request/content-type": ["content-type@2.0.0", "", {}, "sha512-j/O/d7GcZCyNl7/hwZAb606rzqkyvaDctLmckbxLzHvFBzTJHuGEdodATcP3yIRoDrLHkIATJuvzbFlp/ki2cQ=="], + "@octokit/plugin-paginate-rest/@octokit/types/@octokit/openapi-types": ["@octokit/openapi-types@20.0.0", "", {}, "sha512-EtqRBEjp1dL/15V7WiX5LJMIxxkdiGJnabzYx5Apx4FkQIFgAfKumXeYAqqJCj1s+BMX4cPFIFC4OLCR6stlnA=="], "@octokit/plugin-rest-endpoint-methods/@octokit/types/@octokit/openapi-types": ["@octokit/openapi-types@20.0.0", "", {}, "sha512-EtqRBEjp1dL/15V7WiX5LJMIxxkdiGJnabzYx5Apx4FkQIFgAfKumXeYAqqJCj1s+BMX4cPFIFC4OLCR6stlnA=="], @@ -6028,7 +6114,7 @@ "@octokit/rest/@octokit/core/@octokit/graphql": ["@octokit/graphql@9.0.3", "", { "dependencies": { "@octokit/request": "^10.0.6", "@octokit/types": "^16.0.0", "universal-user-agent": "^7.0.0" } }, "sha512-grAEuupr/C1rALFnXTv6ZQhFuL1D8G5y8CN04RgrO4FIPMrtm+mcZzFG7dcBm+nq+1ppNixu+Jd78aeJOYxlGA=="], - "@octokit/rest/@octokit/core/@octokit/request": ["@octokit/request@10.0.8", "", { "dependencies": { "@octokit/endpoint": "^11.0.3", "@octokit/request-error": "^7.0.2", "@octokit/types": "^16.0.0", "fast-content-type-parse": "^3.0.0", "json-with-bigint": "^3.5.3", "universal-user-agent": "^7.0.2" } }, "sha512-SJZNwY9pur9Agf7l87ywFi14W+Hd9Jg6Ifivsd33+/bGUQIjNujdFiXII2/qSlN2ybqUHfp5xpekMEjIBTjlSw=="], + "@octokit/rest/@octokit/core/@octokit/request": ["@octokit/request@10.0.9", "", { "dependencies": { "@octokit/endpoint": "^11.0.3", "@octokit/request-error": "^7.0.2", "@octokit/types": "^16.0.0", "content-type": "^2.0.0", "fast-content-type-parse": "^3.0.0", "json-with-bigint": "^3.5.3", "universal-user-agent": "^7.0.2" } }, "sha512-o8Bi3f608eyM+7BmBiUWxFsdjLb3/ym1cQek5LZOv9KkZcxRrHCPhhRzm6xjO6HVZ85ItD6+sTsjxo821SVa/A=="], "@octokit/rest/@octokit/core/@octokit/request-error": ["@octokit/request-error@7.1.0", "", { "dependencies": { "@octokit/types": "^16.0.0" } }, "sha512-KMQIfq5sOPpkQYajXHwnhjCC0slzCNScLHs9JafXc4RAJI+9f+jNDlBNaIMTvazOPLgb4BnlhGJOTbnN0wIjPw=="], @@ -6038,15 +6124,25 @@ "@opencode-ai/app/@playwright/test/playwright": ["playwright@1.59.1", "", { "dependencies": { "playwright-core": "1.59.1" }, "optionalDependencies": { "fsevents": "2.3.2" }, "bin": { "playwright": "cli.js" } }, "sha512-C8oWjPR3F81yljW9o5OxcWzfh6avkVwDD2VYdwIGqTkl+OGFISgypqzfu7dOe4QNLL2aqcWBmI3PMtLIK233lw=="], + "@opencode-ai/app/@types/node/undici-types": ["undici-types@6.20.0", "", {}, "sha512-Ny6QZ2Nju20vw1SRHe3d9jVu6gJ+4e3+MMpqu7pqE5HT6WsTSlce++GQmK5UXS8mzV8DSYHrQH+Xrf2jVcuKNg=="], + "@opencode-ai/desktop-electron/@actions/artifact/@actions/http-client": ["@actions/http-client@2.2.3", "", { "dependencies": { "tunnel": "^0.0.6", "undici": "^5.25.4" } }, "sha512-mx8hyJi/hjFvbPokCg4uRd4ZX78t+YyRPtnKWwIl+RzNaVuFpQHfmlGVfsKEJN8LwTCvL+DfVgAM04XaHkm6bA=="], + "@opencode-ai/desktop-electron/@types/node/undici-types": ["undici-types@6.20.0", "", {}, "sha512-Ny6QZ2Nju20vw1SRHe3d9jVu6gJ+4e3+MMpqu7pqE5HT6WsTSlce++GQmK5UXS8mzV8DSYHrQH+Xrf2jVcuKNg=="], + "@opencode-ai/desktop/@actions/artifact/@actions/http-client": ["@actions/http-client@2.2.3", "", { "dependencies": { "tunnel": "^0.0.6", "undici": "^5.25.4" } }, "sha512-mx8hyJi/hjFvbPokCg4uRd4ZX78t+YyRPtnKWwIl+RzNaVuFpQHfmlGVfsKEJN8LwTCvL+DfVgAM04XaHkm6bA=="], - "@opencode-ai/storybook/@storybook/addon-docs/@storybook/csf-plugin": ["@storybook/csf-plugin@10.3.6", "", { "dependencies": { "unplugin": "^2.3.5" }, "peerDependencies": { "esbuild": "*", "rollup": "*", "storybook": "^10.3.6", "vite": "*", "webpack": "*" }, "optionalPeers": ["esbuild", "rollup", "vite", "webpack"] }, "sha512-9kBf7VRdRqTSIYo+rPtVn5yjYYyK8kP2QhEYx3oiXvfwy4RexmbJnhk/tXa/lNiTqukA1TqaWQ2+5MqF4fu6YQ=="], + "@opencode-ai/plugin/effect/@standard-schema/spec": ["@standard-schema/spec@1.1.0", "", {}, "sha512-l2aFy5jALhniG5HgqrD6jXLi/rUWrKvqN/qJx6yoJsgKhblVd+iqqU4RCXavm/jPityDo5TCvKMnpjKnOriy0w=="], + + "@opencode-ai/plugin/effect/uuid": ["uuid@13.0.2", "", { "bin": { "uuid": "dist-node/bin/uuid" } }, "sha512-vzi9uRZ926x4XV73S/4qQaTwPXM2JBj6/6lI/byHH1jOpCzb0zDbfytgA9LcN/hzb2l7WQSQnxITOVx5un/wGw=="], - "@opencode-ai/storybook/@storybook/addon-docs/@storybook/react-dom-shim": ["@storybook/react-dom-shim@10.3.6", "", { "peerDependencies": { "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "react-dom": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "storybook": "^10.3.6" } }, "sha512-/Tu1gPu+Fw+zOnAGmxRmOD30FX3a04LxcTAKflEtdpmtIMVR5bA3qpjy+f5YhoyDCecbXyKmL1OeIU2FIIZHqQ=="], + "@opencode-ai/storybook/@storybook/addon-docs/@storybook/csf-plugin": ["@storybook/csf-plugin@10.4.0", "", { "dependencies": { "unplugin": "^2.3.5" }, "peerDependencies": { "esbuild": "*", "rollup": "*", "storybook": "^10.4.0", "vite": "*", "webpack": "*" }, "optionalPeers": ["esbuild", "rollup", "vite", "webpack"] }, "sha512-iSmrhMyEi2ohCWKu49ZUUf8l+k0OIStbWI1BTWt2FvKySlnqY/aHenus7839SgNL3aUNG5P0y9zlyN6/HlwlEQ=="], - "@opencode-ai/storybook/@storybook/addon-docs/react": ["react@19.2.5", "", {}, "sha512-llUJLzz1zTUBrskt2pwZgLq59AemifIftw4aB7JxOqf1HY2FDaGDxgwpAPVzHU1kdWabH7FauP4i1oEeer2WCA=="], + "@opencode-ai/storybook/@storybook/addon-docs/@storybook/react-dom-shim": ["@storybook/react-dom-shim@10.4.0", "", { "peerDependencies": { "@types/react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "@types/react-dom": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "react-dom": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "storybook": "^10.4.0" }, "optionalPeers": ["@types/react", "@types/react-dom"] }, "sha512-dcYWzdPaJEHVlyOyyz0/0v3QJXmcnK2sjw4YiFwU9IVJhoJrBlE9lMtmbO3QqIbq4qA0hElYtGkKO7tMLSKDGw=="], + + "@opencode-ai/storybook/@storybook/addon-docs/react": ["react@19.2.6", "", {}, "sha512-sfWGGfavi0xr8Pg0sVsyHMAOziVYKgPLNrS7ig+ivMNb3wbCBw3KxtflsGBAwD3gYQlE/AEZsTLgToRrSCjb0Q=="], + + "@opencode-ai/storybook/@types/node/undici-types": ["undici-types@6.20.0", "", {}, "sha512-Ny6QZ2Nju20vw1SRHe3d9jVu6gJ+4e3+MMpqu7pqE5HT6WsTSlce++GQmK5UXS8mzV8DSYHrQH+Xrf2jVcuKNg=="], "@opencode-ai/storybook/storybook/open": ["open@10.2.0", "", { "dependencies": { "default-browser": "^5.2.1", "define-lazy-prop": "^3.0.0", "is-inside-container": "^1.0.0", "wsl-utils": "^0.1.0" } }, "sha512-YgBpdJHPyQ2UE5x+hlSXcnejzAvD0b22U2OuAP+8OnlJT+PjWPxtgmGqKKc+RgTM63U9gN0YzrYc71R2WT/hTA=="], @@ -6108,8 +6204,6 @@ "@textlint/linter-formatter/strip-ansi/ansi-regex": ["ansi-regex@5.0.1", "", {}, "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ=="], - "@vscode/ripgrep/yauzl/buffer-crc32": ["buffer-crc32@0.2.13", "", {}, "sha512-VO9Ht/+p3SN7SKWqcrgEzjGbRSJYTx+Q1pTQC0wrWqHx0vpJraQ6GtHx8tvcg1rlK1byhU5gccxgOgj7B0TDkQ=="], - "@vscode/test-cli/chokidar/glob-parent": ["glob-parent@5.1.2", "", { "dependencies": { "is-glob": "^4.0.1" } }, "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow=="], "@vscode/test-cli/chokidar/readdirp": ["readdirp@3.6.0", "", { "dependencies": { "picomatch": "^2.2.1" } }, "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA=="], @@ -6136,6 +6230,8 @@ "ajv-keywords/ajv/json-schema-traverse": ["json-schema-traverse@0.4.1", "", {}, "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg=="], + "apache-arrow/@types/node/undici-types": ["undici-types@6.21.0", "", {}, "sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ=="], + "app-builder-lib/@electron/get/fs-extra": ["fs-extra@8.1.0", "", { "dependencies": { "graceful-fs": "^4.2.0", "jsonfile": "^4.0.0", "universalify": "^0.1.0" } }, "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g=="], "app-builder-lib/@electron/get/semver": ["semver@6.3.1", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA=="], @@ -6170,7 +6266,7 @@ "ava/yargs/yargs-parser": ["yargs-parser@21.1.1", "", {}, "sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw=="], - "aws-sdk/xml2js/sax": ["sax@1.6.0", "", {}, "sha512-6R3J5M4AcbtLUdZmRv2SygeVaM7IhrLXu9BmnOGmmACak8fiUtOsYNWUS4uK7upbmHIBbLBeFeI//477BKLBzA=="], + "axios/https-proxy-agent/agent-base": ["agent-base@6.0.2", "", { "dependencies": { "debug": "4" } }, "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ=="], "babel-plugin-module-resolver/glob/minimatch": ["minimatch@8.0.7", "", { "dependencies": { "brace-expansion": "^2.0.1" } }, "sha512-V+1uQNdzybxa14e/p00HZnQNNcTjnRJjDxg2V8wtkjFctq4M7hXFws4oekyTP0Jebeq7QYtpFyOeBAjc88zvYg=="], @@ -6338,6 +6434,8 @@ "electron-vite/esbuild/@esbuild/win32-x64": ["@esbuild/win32-x64@0.25.12", "", { "os": "win32", "cpu": "x64" }, "sha512-alJC0uCZpTFrSL0CCDjcgleBXPnCrEAhTBILpeAp7M/OFgoqtAetfBzX0xM00MUsVVPpVjlPuMbREqnZCXaTnA=="], + "electron/@types/node/undici-types": ["undici-types@7.16.0", "", {}, "sha512-Zz+aZWSj8LE6zoxD+xrjh4VfkIG8Ya6LvYkZqtUQGJPZjYl53ypCaUwWqo7eI0x66KBGeRo+mlBEkMSeSZ38Nw=="], + "enquirer/strip-ansi/ansi-regex": ["ansi-regex@5.0.1", "", {}, "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ=="], "eslint/ajv/json-schema-traverse": ["json-schema-traverse@0.4.1", "", {}, "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg=="], @@ -6368,8 +6466,6 @@ "friendly-words/express/merge-descriptors": ["merge-descriptors@1.0.3", "", {}, "sha512-gaNvAS7TZ897/rVaZ0nMtAyxNyi/pdbjbAwUpFQpN70GqnVfOiXpeUUMKRBmzXaSQ8DdTX4/0ms62r2K+hE6mQ=="], - "friendly-words/express/qs": ["qs@6.14.2", "", { "dependencies": { "side-channel": "^1.1.0" } }, "sha512-V/yCWTTF7VJ9hIh18Ugr2zhJMP01MY7c5kh4J870L7imm6/DIzBsNLTXzMwUA3yZ5b/KBqLx8Kp3uRvd7xSe3Q=="], - "friendly-words/express/send": ["send@0.19.2", "", { "dependencies": { "debug": "2.6.9", "depd": "2.0.0", "destroy": "1.2.0", "encodeurl": "~2.0.0", "escape-html": "~1.0.3", "etag": "~1.8.1", "fresh": "~0.5.2", "http-errors": "~2.0.1", "mime": "1.6.0", "ms": "2.1.3", "on-finished": "~2.4.1", "range-parser": "~1.2.1", "statuses": "~2.0.2" } }, "sha512-VMbMxbDeehAxpOtWJXlcUS5E8iXh6QmN+BkRX1GARS3wRaXEEgzCcB10gTQazO42tpNIya8xIyNx8fll1OFPrg=="], "friendly-words/express/serve-static": ["serve-static@1.16.3", "", { "dependencies": { "encodeurl": "~2.0.0", "escape-html": "~1.0.3", "parseurl": "~1.3.3", "send": "~0.19.1" } }, "sha512-x0RTqQel6g5SY7Lg6ZreMmsOzncHFU7nhnRWkKgWuMTu5NN0DR5oruckMqRvacAN9d5w6ARnRBXl9xhDCgfMeA=="], @@ -6386,6 +6482,10 @@ "jszip/readable-stream/string_decoder": ["string_decoder@1.1.1", "", { "dependencies": { "safe-buffer": "~5.1.0" } }, "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg=="], + "kilo-code/@types/node/undici-types": ["undici-types@6.20.0", "", {}, "sha512-Ny6QZ2Nju20vw1SRHe3d9jVu6gJ+4e3+MMpqu7pqE5HT6WsTSlce++GQmK5UXS8mzV8DSYHrQH+Xrf2jVcuKNg=="], + + "kilo-code/openai/@types/node": ["@types/node@18.19.130", "", { "dependencies": { "undici-types": "~5.26.4" } }, "sha512-GRaXQx6jGfL8sKfaIDD6OupbIHBr9jv7Jnaml9tB7l4v068PAOXqfcujMMo5PhbIs6ggR1XODELqahT2R8v0fg=="], + "lazystream/readable-stream/safe-buffer": ["safe-buffer@5.1.2", "", {}, "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g=="], "lazystream/readable-stream/string_decoder": ["string_decoder@1.1.1", "", { "dependencies": { "safe-buffer": "~5.1.0" } }, "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg=="], @@ -6402,6 +6502,8 @@ "mocha/yargs/string-width": ["string-width@4.2.3", "", { "dependencies": { "emoji-regex": "^8.0.0", "is-fullwidth-code-point": "^3.0.0", "strip-ansi": "^6.0.1" } }, "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g=="], + "mssql/tedious/iconv-lite": ["iconv-lite@0.6.3", "", { "dependencies": { "safer-buffer": ">= 2.1.2 < 3.0.0" } }, "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw=="], + "node-sarif-builder/fs-extra/jsonfile": ["jsonfile@6.2.1", "", { "dependencies": { "universalify": "^2.0.0" }, "optionalDependencies": { "graceful-fs": "^4.1.6" } }, "sha512-zwOTdL3rFQ/lRdBnntKVOX6k5cKJwEc1HdilT71BWEu7J41gXIB2MRp+vxduPSwZJPWBxEzv4yH1wYLJGUHX4Q=="], "node-sarif-builder/fs-extra/universalify": ["universalify@2.0.1", "", {}, "sha512-gptHNQghINnc/vTGIk0SOFGFNXw7JVrlRUtConJRlvaw6DuX0wO5Jeko9sWrMBhh+PsYAZ7oXAiOnf/UKogyiw=="], @@ -6416,6 +6518,22 @@ "opencontrol/@modelcontextprotocol/sdk/zod-to-json-schema": ["zod-to-json-schema@3.25.2", "", { "peerDependencies": { "zod": "^3.25.28 || ^4" } }, "sha512-O/PgfnpT1xKSDeQYSCfRI5Gy3hPf91mKVDuYLUHZJMiDFptvP41MSnWofm8dnCm0256ZNfZIM7DSzuSMAFnjHA=="], + "opentui-spinner/@opentui/core/@opentui/core-darwin-arm64": ["@opentui/core-darwin-arm64@0.1.75", "", { "os": "darwin", "cpu": "arm64" }, "sha512-gGaGZjkFpqcXJk6321JzhRl66pM2VxBlI470L8W4DQUW4S6iDT1R9L7awSzGB4Cn9toUl7DTV8BemaXZYXV4SA=="], + + "opentui-spinner/@opentui/core/@opentui/core-darwin-x64": ["@opentui/core-darwin-x64@0.1.75", "", { "os": "darwin", "cpu": "x64" }, "sha512-tPlvqQI0whZ76amHydpJs5kN+QeWAIcFbI8RAtlAo9baj2EbxTDC+JGwgb9Fnt0/YQx831humbtaNDhV2Jt1bw=="], + + "opentui-spinner/@opentui/core/@opentui/core-linux-arm64": ["@opentui/core-linux-arm64@0.1.75", "", { "os": "linux", "cpu": "arm64" }, "sha512-nVxIQ4Hqf84uBergDpWiVzU6pzpjy6tqBHRQpySxZ2flkJ/U6/aMEizVrQ1jcgIdxZtvqWDETZhzxhG0yDx+cw=="], + + "opentui-spinner/@opentui/core/@opentui/core-linux-x64": ["@opentui/core-linux-x64@0.1.75", "", { "os": "linux", "cpu": "x64" }, "sha512-1CnApef4kxA+ORyLfbuCLgZfEjp4wr3HjFnt7FAfOb73kIZH82cb7JYixeqRyy9eOcKfKqxLmBYy3o8IDkc4Rg=="], + + "opentui-spinner/@opentui/core/@opentui/core-win32-arm64": ["@opentui/core-win32-arm64@0.1.75", "", { "os": "win32", "cpu": "arm64" }, "sha512-j0UB95nmkYGNzmOrs6GqaddO1S90R0YC6IhbKnbKBdjchFPNVLz9JpexAs6MBDXPZwdKAywMxtwG2h3aTJtxng=="], + + "opentui-spinner/@opentui/core/@opentui/core-win32-x64": ["@opentui/core-win32-x64@0.1.75", "", { "os": "win32", "cpu": "x64" }, "sha512-ESpVZVGewe3JkB2TwrG3VRbkxT909iPdtvgNT7xTCIYH2VB4jqZomJfvERPTE0tvqAZJm19mHECzJFI8asSJgQ=="], + + "opentui-spinner/@opentui/core/bun-ffi-structs": ["bun-ffi-structs@0.1.2", "", { "peerDependencies": { "typescript": "^5" } }, "sha512-Lh1oQAYHDcnesJauieA4UNkWGXY9hYck7OA5IaRwE3Bp6K2F2pJSNYqq+hIy7P3uOvo3km3oxS8304g5gDMl/w=="], + + "opentui-spinner/@opentui/core/diff": ["diff@8.0.2", "", {}, "sha512-sSuxWU5j5SR9QQji/o2qMvqRNYRDOcBTgsJ/DeCf4iSN4gW+gNMXM7wFIP+fdXZxoNiAnHUTGjCr+TSWXdRDKg=="], + "ora/log-symbols/is-unicode-supported": ["is-unicode-supported@1.3.0", "", {}, "sha512-43r2mRvz+8JRIKnWJ+3j8JtjRKZ6GmjzfaE/qiBJnikNnYv/6bagRJ1kUhNk8R5EX/GkobD+r+sfxCPJsiKBLQ=="], "p-locate/p-limit/yocto-queue": ["yocto-queue@0.1.0", "", {}, "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q=="], @@ -6474,12 +6592,6 @@ "@actions/artifact/@actions/core/@actions/exec/@actions/io": ["@actions/io@2.0.0", "", {}, "sha512-Jv33IN09XLO+0HS79aaODsvIRyduiF7NY/F6LYeK5oeUmrsz7aFdRphQjFoESF4jS7lMauDOttKALcpapVDIAg=="], - "@aws-crypto/sha1-browser/@smithy/util-utf8/@smithy/util-buffer-from/@smithy/is-array-buffer": ["@smithy/is-array-buffer@2.2.0", "", { "dependencies": { "tslib": "^2.6.2" } }, "sha512-GGP3O9QFD24uGeAXYUjwSTXARoqpZykHadOmA8G5vfJPK0/DC67qa//0qvqrJzL1xc8WQWX7/yc7fwudjPHPhA=="], - - "@aws-crypto/sha256-browser/@smithy/util-utf8/@smithy/util-buffer-from/@smithy/is-array-buffer": ["@smithy/is-array-buffer@2.2.0", "", { "dependencies": { "tslib": "^2.6.2" } }, "sha512-GGP3O9QFD24uGeAXYUjwSTXARoqpZykHadOmA8G5vfJPK0/DC67qa//0qvqrJzL1xc8WQWX7/yc7fwudjPHPhA=="], - - "@aws-crypto/util/@smithy/util-utf8/@smithy/util-buffer-from/@smithy/is-array-buffer": ["@smithy/is-array-buffer@2.2.0", "", { "dependencies": { "tslib": "^2.6.2" } }, "sha512-GGP3O9QFD24uGeAXYUjwSTXARoqpZykHadOmA8G5vfJPK0/DC67qa//0qvqrJzL1xc8WQWX7/yc7fwudjPHPhA=="], - "@electron/asar/minimatch/brace-expansion/balanced-match": ["balanced-match@1.0.2", "", {}, "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw=="], "@electron/universal/minimatch/brace-expansion/balanced-match": ["balanced-match@1.0.2", "", {}, "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw=="], @@ -6496,10 +6608,14 @@ "@manypkg/find-root/find-up/locate-path/p-locate": ["p-locate@4.1.0", "", { "dependencies": { "p-limit": "^2.2.0" } }, "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A=="], + "@morphllm/morphsdk/openai/@types/node/undici-types": ["undici-types@5.26.5", "", {}, "sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA=="], + "@octokit/graphql/@octokit/request/@octokit/types/@octokit/openapi-types": ["@octokit/openapi-types@27.0.0", "", {}, "sha512-whrdktVs1h6gtR+09+QsNk2+FO+49j6ga1c55YZudfEG+oKJVvJLQi3zkOm5JjiUXAagWK2tI2kTGKJ2Ys7MGA=="], "@octokit/rest/@octokit/core/@octokit/request/@octokit/endpoint": ["@octokit/endpoint@11.0.3", "", { "dependencies": { "@octokit/types": "^16.0.0", "universal-user-agent": "^7.0.2" } }, "sha512-FWFlNxghg4HrXkD3ifYbS/IdL/mDHjh9QcsNyhQjN8dplUoZbejsdpmuqdA76nxj2xoWPs7p8uX2SNr9rYu0Ag=="], + "@octokit/rest/@octokit/core/@octokit/request/content-type": ["content-type@2.0.0", "", {}, "sha512-j/O/d7GcZCyNl7/hwZAb606rzqkyvaDctLmckbxLzHvFBzTJHuGEdodATcP3yIRoDrLHkIATJuvzbFlp/ki2cQ=="], + "@octokit/rest/@octokit/core/@octokit/types/@octokit/openapi-types": ["@octokit/openapi-types@27.0.0", "", {}, "sha512-whrdktVs1h6gtR+09+QsNk2+FO+49j6ga1c55YZudfEG+oKJVvJLQi3zkOm5JjiUXAagWK2tI2kTGKJ2Ys7MGA=="], "@opencode-ai/app/@playwright/test/playwright/fsevents": ["fsevents@2.3.2", "", { "os": "darwin" }, "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA=="], @@ -6540,6 +6656,8 @@ "@tailwindcss/vite/@tailwindcss/oxide/@tailwindcss/oxide-wasm32-wasi/tslib": ["tslib@2.8.1", "", { "bundled": true }, "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w=="], + "@vscode/test-cli/chokidar/readdirp/picomatch": ["picomatch@2.3.2", "", {}, "sha512-V7+vQEJ06Z+c5tSye8S+nHUfI51xoXIXjHQ99cQtKUkQqqO1kO/KCJUfZXuB47h/YBlDhah2H3hdUGXn8ie0oA=="], + "@vscode/test-cli/glob/jackspeak/@isaacs/cliui": ["@isaacs/cliui@8.0.2", "", { "dependencies": { "string-width": "^5.1.2", "string-width-cjs": "npm:string-width@^4.2.0", "strip-ansi": "^7.0.1", "strip-ansi-cjs": "npm:strip-ansi@^6.0.1", "wrap-ansi": "^8.1.0", "wrap-ansi-cjs": "npm:wrap-ansi@^7.0.0" } }, "sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA=="], "@vscode/test-cli/glob/path-scurry/lru-cache": ["lru-cache@10.4.3", "", {}, "sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ=="], @@ -6606,8 +6724,6 @@ "friendly-words/express/body-parser/iconv-lite": ["iconv-lite@0.4.24", "", { "dependencies": { "safer-buffer": ">= 2.1.2 < 3" } }, "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA=="], - "friendly-words/express/body-parser/qs": ["qs@6.15.1", "", { "dependencies": { "side-channel": "^1.1.0" } }, "sha512-6YHEFRL9mfgcAvql/XhwTvf5jKcOiiupt2FiJxHkiX1z4j7WL8J/jRHYLluORvc1XxB5rV20KoeK00gVJamspg=="], - "friendly-words/express/body-parser/raw-body": ["raw-body@2.5.3", "", { "dependencies": { "bytes": "~3.1.2", "http-errors": "~2.0.1", "iconv-lite": "~0.4.24", "unpipe": "~1.0.0" } }, "sha512-s4VSOf6yN0rvbRZGxs8Om5CWj6seneMwK3oDb4lWDH0UPhWcxwOWw5+qk24bxq87szX1ydrwylIOp2uG1ojUpA=="], "friendly-words/express/debug/ms": ["ms@2.0.0", "", {}, "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A=="], @@ -6624,6 +6740,8 @@ "iconv-corefoundation/cli-truncate/string-width/strip-ansi": ["strip-ansi@6.0.1", "", { "dependencies": { "ansi-regex": "^5.0.1" } }, "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A=="], + "kilo-code/openai/@types/node/undici-types": ["undici-types@5.26.5", "", {}, "sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA=="], + "mocha/glob/jackspeak/@isaacs/cliui": ["@isaacs/cliui@8.0.2", "", { "dependencies": { "string-width": "^5.1.2", "string-width-cjs": "npm:string-width@^4.2.0", "strip-ansi": "^7.0.1", "strip-ansi-cjs": "npm:strip-ansi@^6.0.1", "wrap-ansi": "^8.1.0", "wrap-ansi-cjs": "npm:wrap-ansi@^7.0.0" } }, "sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA=="], "mocha/glob/path-scurry/lru-cache": ["lru-cache@10.4.3", "", {}, "sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ=="], diff --git a/bunfig.toml b/bunfig.toml index 36a21d9332..0fde17500d 100644 --- a/bunfig.toml +++ b/bunfig.toml @@ -1,5 +1,6 @@ [install] exact = true +minimumReleaseAge = 410520 # seconds (~4.75 days / ~114 hours) [test] root = "./do-not-run-tests-from-root" diff --git a/flake.nix b/flake.nix index 6380c23d27..541454f974 100644 --- a/flake.nix +++ b/flake.nix @@ -267,13 +267,9 @@ opencode = final.callPackage ./nix/opencode.nix { inherit node_modules; }; - desktop = final.callPackage ./nix/desktop.nix { - inherit opencode; - }; in { inherit opencode; - opencode-desktop = desktop; }; }; @@ -286,13 +282,10 @@ kilo = pkgs.callPackage ./nix/kilo.nix { inherit node_modules; }; - desktop = pkgs.callPackage ./nix/desktop.nix { - inherit kilo; - }; in { default = kilo; - inherit kilo desktop; + inherit kilo; # Updater derivation with fakeHash - build fails and reveals correct hash node_modules_updater = node_modules.override { hash = pkgs.lib.fakeHash; diff --git a/install b/install index c418668825..c6d0e37e0b 100755 --- a/install +++ b/install @@ -385,6 +385,57 @@ add_to_path() { fi } +# Persistently add $INSTALL_DIR to the Windows user PATH via the registry. +# Works from Git Bash (MINGW64), MSYS2, and Cygwin. +# Converts the MSYS/Cygwin path to native Windows form so that cmd and +# PowerShell also see the entry. +add_to_windows_path() { + local win_install_dir + if command -v cygpath >/dev/null 2>&1; then + win_install_dir=$(cygpath -w "$INSTALL_DIR") + else + # Fallback: naive conversion /c/Users/... -> C:\Users\... + win_install_dir=$(echo "$INSTALL_DIR" | sed -e 's|^/\([a-zA-Z]\)/|\1:\\|' -e 's|/|\\|g') + fi + + # Escape single-quotes for PowerShell single-quoted strings ('' is the escape sequence) + local win_install_dir_ps="${win_install_dir//\'/\'\'}" + + local ps_cmd + ps_cmd=$(cat </dev/null 2>&1; then + ps_output=$(powershell.exe -NoProfile -NonInteractive -Command "$ps_cmd" 2>/dev/null) || return 1 + elif command -v pwsh >/dev/null 2>&1; then + ps_output=$(pwsh -NoProfile -NonInteractive -Command "$ps_cmd" 2>/dev/null) || return 1 + else + return 1 + fi + + if [[ "$ps_output" == *"added"* ]]; then + print_message info "${MUTED}Successfully added ${NC}kilo ${MUTED}to Windows user PATH${NC}" + fi + + # Also add to the current bash session so `kilo` works immediately + # without the user needing to open a new terminal. + if [[ ":$PATH:" != *":$INSTALL_DIR:"* ]]; then + export PATH="$INSTALL_DIR:$PATH" + fi + + return 0 +} + XDG_CONFIG_HOME=${XDG_CONFIG_HOME:-$HOME/.config} current_shell=$(basename "$SHELL") @@ -411,6 +462,13 @@ case $current_shell in esac if [[ "$no_modify_path" != "true" ]]; then + # Detect if we are running on Windows (MINGW, MSYS, Cygwin) regardless + # of whether the binary was installed from download or --binary. + _running_on_windows=false + case "$(uname -s)" in + MINGW*|MSYS*|CYGWIN*) _running_on_windows=true ;; + esac + config_file="" for file in $config_files; do if [[ -f $file ]]; then @@ -419,7 +477,12 @@ if [[ "$no_modify_path" != "true" ]]; then fi done - if [[ -z $config_file ]]; then + if [[ "$_running_on_windows" == "true" ]] && add_to_windows_path; then + # Handled via the Windows user-level PATH in the registry. + # The change is visible in new cmd / PowerShell / Git Bash sessions. + # We also exported it into the current session above. + : + elif [[ -z $config_file ]]; then print_message warning "No config file found for $current_shell. You may need to manually add to PATH:" print_message info " export PATH=$INSTALL_DIR:\$PATH" elif [[ ":$PATH:" != *":$INSTALL_DIR:"* ]]; then diff --git a/nix/desktop.nix b/nix/desktop.nix deleted file mode 100644 index 9bc5734d5b..0000000000 --- a/nix/desktop.nix +++ /dev/null @@ -1,100 +0,0 @@ -{ - lib, - stdenv, - rustPlatform, - pkg-config, - cargo-tauri, - bun, - nodejs, - cargo, - rustc, - jq, - wrapGAppsHook4, - makeWrapper, - dbus, - glib, - gtk4, - libsoup_3, - librsvg, - libappindicator, - glib-networking, - openssl, - webkitgtk_4_1, - gst_all_1, - kilo, -}: -rustPlatform.buildRustPackage (finalAttrs: { - pname = "kilo-desktop"; - inherit (kilo) - version - src - node_modules - patches - ; - - cargoRoot = "packages/desktop/src-tauri"; - cargoLock.lockFile = ../packages/desktop/src-tauri/Cargo.lock; - buildAndTestSubdir = finalAttrs.cargoRoot; - - nativeBuildInputs = [ - pkg-config - cargo-tauri.hook - bun - nodejs # for patchShebangs node_modules - cargo - rustc - jq - makeWrapper - ] ++ lib.optionals stdenv.hostPlatform.isLinux [ wrapGAppsHook4 ]; - - buildInputs = lib.optionals stdenv.isLinux [ - dbus - glib - gtk4 - libsoup_3 - librsvg - libappindicator - glib-networking - openssl - webkitgtk_4_1 - gst_all_1.gstreamer - gst_all_1.gst-plugins-base - gst_all_1.gst-plugins-good - gst_all_1.gst-plugins-bad - ]; - - strictDeps = true; - - preBuild = '' - cp -a ${finalAttrs.node_modules}/{node_modules,packages} . - chmod -R u+w node_modules packages - patchShebangs node_modules - patchShebangs packages/desktop/node_modules - - mkdir -p packages/desktop/src-tauri/sidecars - cp ${kilo}/bin/kilo packages/desktop/src-tauri/sidecars/kilo-cli-${stdenv.hostPlatform.rust.rustcTarget} - ''; - - # see publish-tauri job in .github/workflows/publish.yml - tauriBuildFlags = [ - "--config" - "tauri.prod.conf.json" - "--no-sign" # no code signing or auto updates - ]; - - # FIXME: workaround for concerns about case insensitive filesystems - # should be removed once binary is renamed or decided otherwise - # darwin output is a .app bundle so no conflict - postFixup = lib.optionalString stdenv.hostPlatform.isLinux '' - mv $out/bin/Kilo $out/bin/kilo-desktop - sed -i 's|^Exec=Kilo$|Exec=kilo-desktop|' $out/share/applications/Kilo.desktop - ''; - - meta = { - description = "Kilo Desktop App"; - homepage = "https://kilo.ai"; - license = lib.licenses.mit; - mainProgram = "kilo-desktop"; - inherit (kilo.meta) platforms; - }; -}) diff --git a/nix/hashes.json b/nix/hashes.json index a0a3123799..3aa8ec21fe 100644 --- a/nix/hashes.json +++ b/nix/hashes.json @@ -1,8 +1,8 @@ { "nodeModules": { - "x86_64-linux": "sha256-uDEFWmG84/Ml/hXqV0U80f6re45sAUExyHzgRLWqcZg=", - "aarch64-linux": "sha256-4r/OG9jq17HXLxehUneUB1Y7cjblgprmi8FRvwYgdjY=", - "aarch64-darwin": "sha256-X5/ejqcGD8Ea8sk0uRpgE/TEKdq8gdCd9lHHqalKyTM=", - "x86_64-darwin": "sha256-4YG1MNzMRJNNOx2phFQyiNc09G9ksgitJ2PNbfhM1e4=" + "x86_64-linux": "sha256-n8qDAnmhHLO3PICGFAL6UzPi6k+Dwtl9Wg87PKqa/vY=", + "aarch64-linux": "sha256-XBCHg6BYL1J3M5JCUZda9e8ZKi8EEpI2hez6w3ICOYs=", + "aarch64-darwin": "sha256-3K35v/UghrM8CSpWvbvJVLeoF94MALWVA7F5B++Po00=", + "x86_64-darwin": "sha256-aTjNOPvGygJ34A1Q5+qziyyuSFa0w6WMUjUcsXSLtBU=" } } diff --git a/nix/node_modules.nix b/nix/node_modules.nix index e92a8a4884..26b7364408 100644 --- a/nix/node_modules.nix +++ b/nix/node_modules.nix @@ -30,7 +30,7 @@ stdenvNoCC.mkDerivation { ../bun.lock ../package.json ../patches - ../install # required by desktop build (cli.rs include_str!) + ../install ] ); }; @@ -52,9 +52,6 @@ stdenvNoCC.mkDerivation { --os="${bunOs}" \ --filter '!./' \ --filter './packages/opencode' \ - --filter './packages/desktop' \ - --filter './packages/app' \ - --filter './packages/shared' \ --frozen-lockfile \ --ignore-scripts \ --no-progress diff --git a/package.json b/package.json index ba7bae8527..cf45589710 100644 --- a/package.json +++ b/package.json @@ -6,16 +6,12 @@ "type": "module", "packageManager": "bun@1.3.13", "scripts": { - "dev": "~/.bun/bin/bun run --cwd packages/opencode --conditions=browser src/index.ts", - "dev-setup": "~/.bun/bin/bun run --cwd packages/opencode --conditions=browser src/index.ts dev-setup", - "test:local": "~/.bun/bin/bun run --cwd packages/opencode script/build.ts --single --skip-install && mkdir -p /tmp/czcode-test && sh -c 'cd /tmp/czcode-test && [ -f .env ] && set -a && . ./.env && set +a; exec $(find $OLDPWD/packages/opencode/dist -name czcode -type f | head -1)'", - "dev:desktop": "bun --cwd packages/desktop-electron dev", - "dev:web": "bun --cwd packages/app dev", - "dev:console": "ulimit -n 10240 2>/dev/null; bun run --cwd packages/console/app dev", + "dev": "bun run --cwd packages/opencode --conditions=browser src/index.ts", + "dev-setup": "bun run --cwd packages/opencode --conditions=browser src/index.ts dev-setup", "dev:storybook": "bun --cwd packages/storybook storybook", "lint": "oxlint", "typecheck": "bun turbo typecheck", - "postinstall": "bun run --cwd packages/opencode fix-node-pty", + "postinstall": "bun run --cwd packages/opencode fix-node-pty && bun run script/setup-git.ts", "prepare": "husky", "random": "echo 'Random script'", "hello": "echo 'Hello World!'", @@ -35,8 +31,8 @@ "@types/cross-spawn": "6.0.6", "@octokit/rest": "22.0.0", "@hono/zod-validator": "0.4.2", - "@opentui/core": "0.1.105", - "@opentui/solid": "0.1.105", + "@opentui/core": "0.2.2", + "@opentui/solid": "0.2.2", "ulid": "3.0.1", "@kobalte/core": "0.13.11", "@types/luxon": "3.7.1", @@ -51,7 +47,7 @@ "@solid-primitives/storage": "4.3.3", "@tailwindcss/vite": "4.1.11", "diff": "8.0.4", - "dompurify": "3.3.3", + "dompurify": "3.4.2", "drizzle-kit": "1.0.0-beta.19-d95b7a4", "drizzle-orm": "1.0.0-beta.19-d95b7a4", "effect": "4.0.0-beta.57", @@ -86,7 +82,6 @@ "devDependencies": { "@actions/artifact": "5.0.1", "@tsconfig/bun": "catalog:", - "@types/bun": "catalog:", "@types/mime-types": "3.0.1", "@typescript/native-preview": "catalog:", "glob": "13.0.5", @@ -97,6 +92,7 @@ "semver": "^7.6.0", "sst": "3.18.10", "turbo": "2.8.13", + "@types/bun": "catalog:", "@changesets/changelog-github": "^0.5.1", "@changesets/cli": "^2.27.10" }, @@ -118,16 +114,6 @@ "semi": false, "printWidth": 120 }, - "trustedDependencies": [ - "esbuild", - "node-pty", - "protobufjs", - "tree-sitter", - "tree-sitter-bash", - "tree-sitter-powershell", - "web-tree-sitter", - "electron" - ], "overrides": { "poe-oauth": "0.0.6", "@types/bun": "catalog:", @@ -142,7 +128,7 @@ "smol-toml": ">=1.6.1", "fastify": ">=5.8.3", "diff": "8.0.4", - "dompurify": "3.3.3", + "dompurify": "3.4.2", "happy-dom": ">=20.8.9" }, "patchedDependencies": { @@ -151,6 +137,6 @@ "solid-js@1.9.10": "patches/solid-js@1.9.10.patch", "stream-chat@9.38.0": "patches/stream-chat@9.38.0.patch" }, - "version": "7.2.34", + "version": "7.2.49", "peerDependencies": {} } diff --git a/packages/app/.gitignore b/packages/app/.gitignore deleted file mode 100644 index d699efb38d..0000000000 --- a/packages/app/.gitignore +++ /dev/null @@ -1,3 +0,0 @@ -src/assets/theme.css -e2e/test-results -e2e/playwright-report diff --git a/packages/app/AGENTS.md b/packages/app/AGENTS.md deleted file mode 100644 index 765e960c81..0000000000 --- a/packages/app/AGENTS.md +++ /dev/null @@ -1,30 +0,0 @@ -## Debugging - -- NEVER try to restart the app, or the server process, EVER. - -## Local Dev - -- `opencode dev web` proxies `https://app.opencode.ai`, so local UI/CSS changes will not show there. -- For local UI changes, run the backend and app dev servers separately. -- Backend (from `packages/opencode`): `bun run --conditions=browser ./src/index.ts serve --port 4096` -- App (from `packages/app`): `bun dev -- --port 4444` -- Open `http://localhost:4444` to verify UI changes (it targets the backend at `http://localhost:4096`). - -## SolidJS - -- Always prefer `createStore` over multiple `createSignal` calls - -## Tool Calling - -- ALWAYS USE PARALLEL TOOLS WHEN APPLICABLE. - -## Browser Automation - -Use `agent-browser` for web automation. Run `agent-browser --help` for all commands. - -Core workflow: - -1. `agent-browser open ` - Navigate to page -2. `agent-browser snapshot -i` - Get interactive elements with refs (@e1, @e2) -3. `agent-browser click @e1` / `fill @e2 "text"` - Interact using refs -4. Re-snapshot after page changes diff --git a/packages/app/README.md b/packages/app/README.md deleted file mode 100644 index 304e272cd0..0000000000 --- a/packages/app/README.md +++ /dev/null @@ -1,50 +0,0 @@ -## Usage - -Dependencies for these templates are managed with [pnpm](https://pnpm.io) using `pnpm up -Lri`. - -This is the reason you see a `pnpm-lock.yaml`. That said, any package manager will work. This file can safely be removed once you clone a template. - -```bash -$ npm install # or pnpm install or yarn install -``` - -### Learn more on the [Solid Website](https://solidjs.com) and come chat with us on our [Discord](https://discord.com/invite/solidjs) - -## Available Scripts - -In the project directory, you can run: - -### `npm run dev` or `npm start` - -Runs the app in the development mode.
-Open [http://localhost:3000](http://localhost:3000) to view it in the browser. - -The page will reload if you make edits.
- -### `npm run build` - -Builds the app for production to the `dist` folder.
-It correctly bundles Solid in production mode and optimizes the build for the best performance. - -The build is minified and the filenames include the hashes.
-Your app is ready to be deployed! - -## E2E Testing - -Playwright starts the Vite dev server automatically via `webServer`, and UI tests expect an opencode backend at `localhost:4096` by default. - -```bash -bunx playwright install chromium -bun run test:e2e:local -bun run test:e2e:local -- --grep "settings" -``` - -Environment options: - -- `PLAYWRIGHT_SERVER_HOST` / `PLAYWRIGHT_SERVER_PORT` (backend address, default: `localhost:4096`) -- `PLAYWRIGHT_PORT` (Vite dev server port, default: `3000`) -- `PLAYWRIGHT_BASE_URL` (override base URL, default: `http://localhost:`) - -## Deployment - -You can deploy the `dist` folder to any static host provider (netlify, surge, now, etc.) diff --git a/packages/app/bunfig.toml b/packages/app/bunfig.toml deleted file mode 100644 index f1caabbcce..0000000000 --- a/packages/app/bunfig.toml +++ /dev/null @@ -1,3 +0,0 @@ -[test] -root = "./src" -preload = ["./happydom.ts"] diff --git a/packages/app/create-effect-simplification-spec.md b/packages/app/create-effect-simplification-spec.md deleted file mode 100644 index cc101ab059..0000000000 --- a/packages/app/create-effect-simplification-spec.md +++ /dev/null @@ -1,515 +0,0 @@ -# CreateEffect Simplification Implementation Spec - -Reduce reactive misuse across `packages/app`. - ---- - -## Context - -This work targets `packages/app/src`, which currently has 101 `createEffect` calls across 37 files. - -The biggest clusters are `pages/session.tsx` (19), `pages/layout.tsx` (13), `pages/session/file-tabs.tsx` (6), and several context providers that mirror one store into another. - -Key issues from the audit: - -- Derived state is being written through effects instead of computed directly -- Session and file resets are handled by watch-and-clear effects instead of keyed state boundaries -- User-driven actions are hidden inside reactive effects -- Context layers mirror and hydrate child stores with multiple sync effects -- Several areas repeat the same imperative trigger pattern in multiple effects - -Keep the implementation focused on removing unnecessary effects, not on broad UI redesign. - -## Goals - -- Cut high-churn `createEffect` usage in the hottest files first -- Replace effect-driven derived state with reactive derivation -- Replace reset-on-key effects with keyed ownership boundaries -- Move event-driven work to direct actions and write paths -- Remove mirrored store hydration where a single source of truth can exist -- Leave necessary external sync effects in place, but make them narrower and clearer - -## Non-Goals - -- Do not rewrite unrelated component structure just to reduce the count -- Do not change product behavior, navigation flow, or persisted data shape unless required for a cleaner write boundary -- Do not remove effects that bridge to DOM, editors, polling, or external APIs unless there is a clearly safer equivalent -- Do not attempt a repo-wide cleanup outside `packages/app` - -## Effect Taxonomy And Replacement Rules - -Use these rules during implementation. - -### Prefer `createMemo` - -Use `createMemo` when the target value is pure derived state from other signals or stores. - -Do this when an effect only reads reactive inputs and writes another reactive value that could be computed instead. - -Apply this to: - -- `packages/app/src/pages/session.tsx:141` -- `packages/app/src/pages/layout.tsx:557` -- `packages/app/src/components/terminal.tsx:261` -- `packages/app/src/components/session/session-header.tsx:309` - -Rules: - -- If no external system is touched, do not use `createEffect` -- Derive once, then read the memo where needed -- If normalization is required, prefer normalizing at the write boundary before falling back to a memo - -### Prefer Keyed Remounts - -Use keyed remounts when local UI state should reset because an identity changed. - -Do this with `sessionKey`, `scope()`, or another stable identity instead of watching the key and manually clearing signals. - -Apply this to: - -- `packages/app/src/pages/session.tsx:325` -- `packages/app/src/pages/session.tsx:336` -- `packages/app/src/pages/session.tsx:477` -- `packages/app/src/pages/session.tsx:869` -- `packages/app/src/pages/session.tsx:963` -- `packages/app/src/pages/session/message-timeline.tsx:149` -- `packages/app/src/context/file.tsx:100` - -Rules: - -- If the desired behavior is "new identity, fresh local state," key the owner subtree -- Keep state local to the keyed boundary so teardown and recreation handle the reset naturally - -### Prefer Event Handlers And Actions - -Use direct handlers, store actions, and async command functions when work happens because a user clicked, selected, reloaded, or navigated. - -Do this when an effect is just watching for a flag change, command token, or event-bus signal to trigger imperative logic. - -Apply this to: - -- `packages/app/src/pages/layout.tsx:484` -- `packages/app/src/pages/layout.tsx:652` -- `packages/app/src/pages/layout.tsx:776` -- `packages/app/src/pages/layout.tsx:1489` -- `packages/app/src/pages/layout.tsx:1519` -- `packages/app/src/components/file-tree.tsx:328` -- `packages/app/src/pages/session/terminal-panel.tsx:55` -- `packages/app/src/context/global-sync.tsx:148` -- Duplicated trigger sets in: - - `packages/app/src/pages/session/review-tab.tsx:122` - - `packages/app/src/pages/session/review-tab.tsx:130` - - `packages/app/src/pages/session/review-tab.tsx:138` - - `packages/app/src/pages/session/file-tabs.tsx:367` - - `packages/app/src/pages/session/file-tabs.tsx:378` - - `packages/app/src/pages/session/file-tabs.tsx:389` - - `packages/app/src/pages/session/use-session-hash-scroll.ts:144` - - `packages/app/src/pages/session/use-session-hash-scroll.ts:149` - - `packages/app/src/pages/session/use-session-hash-scroll.ts:167` - -Rules: - -- If the trigger is user intent, call the action at the source of that intent -- If the same imperative work is triggered from multiple places, extract one function and call it directly - -### Prefer `onMount` And `onCleanup` - -Use `onMount` and `onCleanup` for lifecycle-only setup and teardown. - -This is the right fit for subscriptions, one-time wiring, timers, and imperative integration that should not rerun for ordinary reactive changes. - -Use this when: - -- Setup should happen once per owner lifecycle -- Cleanup should always pair with teardown -- The work is not conceptually derived state - -### Keep `createEffect` When It Is A Real Bridge - -Keep `createEffect` when it synchronizes reactive data to an external imperative sink. - -Examples that should remain, though they may be narrowed or split: - -- DOM/editor sync in `packages/app/src/components/prompt-input.tsx:690` -- Scroll sync in `packages/app/src/pages/session.tsx:685` -- Scroll/hash sync in `packages/app/src/pages/session/use-session-hash-scroll.ts:149` -- External sync in: - - `packages/app/src/context/language.tsx:207` - - `packages/app/src/context/settings.tsx:110` - - `packages/app/src/context/sdk.tsx:26` -- Polling in: - - `packages/app/src/components/status-popover.tsx:59` - - `packages/app/src/components/dialog-select-server.tsx:273` - -Rules: - -- Keep the effect single-purpose -- Make dependencies explicit and narrow -- Avoid writing back into the same reactive graph unless absolutely required - -## Implementation Plan - -### Phase 0: Classification Pass - -Before changing code, tag each targeted effect as one of: derive, reset, event, lifecycle, or external bridge. - -Acceptance criteria: - -- Every targeted effect in this spec is tagged with a replacement strategy before refactoring starts -- Shared helpers to be introduced are identified up front to avoid repeating patterns - -### Phase 1: Derived-State Cleanup - -Tackle highest-value, lowest-risk derived-state cleanup first. - -Priority items: - -- Normalize tabs at write boundaries and remove `packages/app/src/pages/session.tsx:141` -- Stop syncing `workspaceOrder` in `packages/app/src/pages/layout.tsx:557` -- Make prompt slash filtering reactive so `packages/app/src/components/prompt-input.tsx:652` can be removed -- Replace other obvious derived-state effects in terminal and session header - -Acceptance criteria: - -- No behavior change in tab ordering, prompt filtering, terminal display, or header state -- Targeted derived-state effects are deleted, not just moved - -### Phase 2: Keyed Reset Cleanup - -Replace reset-on-key effects with keyed ownership boundaries. - -Priority items: - -- Key session-scoped UI and state by `sessionKey` -- Key file-scoped state by `scope()` -- Remove manual clear-and-reseed effects in session and file context - -Acceptance criteria: - -- Switching session or file scope recreates the intended local state cleanly -- No stale state leaks across session or scope changes -- Target reset effects are deleted - -### Phase 3: Event-Driven Work Extraction - -Move event-driven work out of reactive effects. - -Priority items: - -- Replace `globalStore.reload` effect dispatching with direct calls -- Split mixed-responsibility effect in `packages/app/src/pages/layout.tsx:1489` -- Collapse duplicated imperative trigger triplets into single functions -- Move file-tree and terminal-panel imperative work to explicit handlers - -Acceptance criteria: - -- User-triggered behavior still fires exactly once per intended action -- No effect remains whose only job is to notice a command-like state and trigger an imperative function - -### Phase 4: Context Ownership Cleanup - -Remove mirrored child-store hydration patterns. - -Priority items: - -- Remove child-store hydration mirrors in `packages/app/src/context/global-sync/child-store.ts:184`, `:190`, `:193` -- Simplify mirror logic in `packages/app/src/context/global-sync.tsx:130`, `:138` -- Revisit `packages/app/src/context/layout.tsx:424` if it still mirrors instead of deriving - -Acceptance criteria: - -- There is one clear source of truth for each synced value -- Child stores no longer need effect-based hydration to stay consistent -- Initialization and updates both work without manual mirror effects - -### Phase 5: Cleanup And Keeper Review - -Clean up remaining targeted hotspots and narrow the effects that should stay. - -Acceptance criteria: - -- Remaining `createEffect` calls in touched files are all true bridges or clearly justified lifecycle sync -- Mixed-responsibility effects are split into smaller units where still needed - -## Detailed Work Items By Area - -### 1. Normalize Tab State - -Files: - -- `packages/app/src/pages/session.tsx:141` - -Work: - -- Move tab normalization into the functions that create, load, or update tab state -- Make readers consume already-normalized tab data -- Remove the effect that rewrites derived tab state after the fact - -Rationale: - -- Tabs should become valid when written, not be repaired later -- This removes a feedback loop and makes state easier to trust - -Acceptance criteria: - -- The effect at `packages/app/src/pages/session.tsx:141` is removed -- Newly created and restored tabs are normalized before they enter local state -- Tab rendering still matches current behavior for valid and edge-case inputs - -### 2. Key Session-Owned State - -Files: - -- `packages/app/src/pages/session.tsx:325` -- `packages/app/src/pages/session.tsx:336` -- `packages/app/src/pages/session.tsx:477` -- `packages/app/src/pages/session.tsx:869` -- `packages/app/src/pages/session.tsx:963` -- `packages/app/src/pages/session/message-timeline.tsx:149` - -Work: - -- Identify state that should reset when `sessionKey` changes -- Move that state under a keyed subtree or keyed owner boundary -- Remove effects that watch `sessionKey` just to clear local state, refs, or temporary UI flags - -Rationale: - -- Session identity already defines the lifetime of this UI state -- Keyed ownership makes reset behavior automatic and easier to reason about - -Acceptance criteria: - -- The targeted reset effects are removed -- Changing sessions resets only the intended session-local state -- Scroll and editor state that should persist are not accidentally reset - -### 3. Derive Workspace Order - -Files: - -- `packages/app/src/pages/layout.tsx:557` - -Work: - -- Stop writing `workspaceOrder` from live workspace data in an effect -- Represent user overrides separately from live workspace data -- Compute effective order from current data plus overrides with a memo or pure helper - -Rationale: - -- Persisted user intent and live source data should not mirror each other through an effect -- A computed effective order avoids drift and racey resync behavior - -Acceptance criteria: - -- The effect at `packages/app/src/pages/layout.tsx:557` is removed -- Workspace order updates correctly when workspaces appear, disappear, or are reordered by the user -- User overrides persist without requiring a sync-back effect - -### 4. Remove Child-Store Mirrors - -Files: - -- `packages/app/src/context/global-sync.tsx:130` -- `packages/app/src/context/global-sync.tsx:138` -- `packages/app/src/context/global-sync.tsx:148` -- `packages/app/src/context/global-sync/child-store.ts:184` -- `packages/app/src/context/global-sync/child-store.ts:190` -- `packages/app/src/context/global-sync/child-store.ts:193` -- `packages/app/src/context/layout.tsx:424` - -Work: - -- Trace the actual ownership of global and child store values -- Replace hydration and mirror effects with explicit initialization and direct updates -- Remove the `globalStore.reload` event-bus pattern and call the needed reload paths directly - -Rationale: - -- Mirrors make it hard to tell which state is authoritative -- Event-bus style state toggles hide control flow and create accidental reruns - -Acceptance criteria: - -- Child store hydration no longer depends on effect-based copying -- Reload work can be followed from the event source to the handler without a reactive relay -- State remains correct on first load, child creation, and subsequent updates - -### 5. Key File-Scoped State - -Files: - -- `packages/app/src/context/file.tsx:100` - -Work: - -- Move file-scoped local state under a boundary keyed by `scope()` -- Remove any effect that watches `scope()` only to reset file-local state - -Rationale: - -- File scope changes are identity changes -- Keyed ownership gives a cleaner reset than manual clear logic - -Acceptance criteria: - -- The effect at `packages/app/src/context/file.tsx:100` is removed -- Switching scopes resets only scope-local state -- No previous-scope data appears after a scope change - -### 6. Split Layout Side Effects - -Files: - -- `packages/app/src/pages/layout.tsx:1489` -- Related event-driven effects near `packages/app/src/pages/layout.tsx:484`, `:652`, `:776`, `:1519` - -Work: - -- Break the mixed-responsibility effect at `:1489` into direct actions and smaller bridge effects only where required -- Move user-triggered branches into the actual command or handler that causes them -- Remove any branch that only exists because one effect is handling unrelated concerns - -Rationale: - -- Mixed effects hide cause and make reruns hard to predict -- Smaller units reduce accidental coupling and make future cleanup safer - -Acceptance criteria: - -- The effect at `packages/app/src/pages/layout.tsx:1489` no longer mixes unrelated responsibilities -- Event-driven branches execute from direct handlers -- Remaining effects in this area each have one clear external sync purpose - -### 7. Remove Duplicate Triggers - -Files: - -- `packages/app/src/pages/session/review-tab.tsx:122` -- `packages/app/src/pages/session/review-tab.tsx:130` -- `packages/app/src/pages/session/review-tab.tsx:138` -- `packages/app/src/pages/session/file-tabs.tsx:367` -- `packages/app/src/pages/session/file-tabs.tsx:378` -- `packages/app/src/pages/session/file-tabs.tsx:389` -- `packages/app/src/pages/session/use-session-hash-scroll.ts:144` -- `packages/app/src/pages/session/use-session-hash-scroll.ts:149` -- `packages/app/src/pages/session/use-session-hash-scroll.ts:167` - -Work: - -- Extract one explicit imperative function per behavior -- Call that function from each source event instead of replicating the same effect pattern multiple times -- Preserve the scroll-sync effect that is truly syncing with the DOM, but remove duplicate trigger scaffolding around it - -Rationale: - -- Duplicate triggers make it easy to miss a case or fire twice -- One named action is easier to test and reason about - -Acceptance criteria: - -- Repeated imperative effect triplets are collapsed into shared functions -- Scroll behavior still works, including hash-based navigation -- No duplicate firing is introduced - -### 8. Make Prompt Filtering Reactive - -Files: - -- `packages/app/src/components/prompt-input.tsx:652` -- Keep `packages/app/src/components/prompt-input.tsx:690` as needed - -Work: - -- Convert slash filtering into a pure reactive derivation from the current input and candidate command list -- Keep only the editor or DOM bridge effect if it is still needed for imperative syncing - -Rationale: - -- Filtering is classic derived state -- It should not need an effect if it can be computed from current inputs - -Acceptance criteria: - -- The effect at `packages/app/src/components/prompt-input.tsx:652` is removed -- Filtered slash-command results update correctly as the input changes -- The editor sync effect at `:690` still behaves correctly - -### 9. Clean Up Smaller Derived-State Cases - -Files: - -- `packages/app/src/components/terminal.tsx:261` -- `packages/app/src/components/session/session-header.tsx:309` - -Work: - -- Replace effect-written local state with memos or inline derivation -- Remove intermediate setters when the value can be computed directly - -Rationale: - -- These are low-risk wins that reinforce the same pattern -- They also help keep follow-up cleanup consistent - -Acceptance criteria: - -- Targeted effects are removed -- UI output remains unchanged under the same inputs - -## Verification And Regression Checks - -Run focused checks after each phase, not only at the end. - -### Suggested Verification - -- Switch between sessions rapidly and confirm local session UI resets only where intended -- Open, close, and reorder tabs and confirm order and normalization remain stable -- Change workspaces, reload workspace data, and verify effective ordering is correct -- Change file scope and confirm stale file state does not bleed across scopes -- Trigger layout actions that previously depended on effects and confirm they still fire once -- Use slash commands in the prompt and verify filtering updates as you type -- Test review tab, file tab, and hash-scroll flows for duplicate or missing triggers -- Verify global sync initialization, reload, and child-store creation paths - -### Regression Checks - -- No accidental infinite reruns -- No double-firing network or command actions -- No lost cleanup for listeners, timers, or scroll handlers -- No preserved stale state after identity changes -- No removed effect that was actually bridging to DOM or an external API - -If available, add or update tests around pure helpers introduced during this cleanup. - -Favor tests for derived ordering, normalization, and action extraction, since those are easiest to lock down. - -## Definition Of Done - -This work is done when all of the following are true: - -- The highest-leverage targets in this spec are implemented -- Each removed effect has been replaced by a clearer pattern: memo, keyed boundary, direct action, or lifecycle hook -- The "should remain" effects still exist only where they serve a real external sync purpose -- Touched files have fewer mixed-responsibility effects and clearer ownership of state -- Manual verification covers session switching, file scope changes, workspace ordering, prompt filtering, and reload flows -- No behavior regressions are found in the targeted areas - -A reduced raw `createEffect` count is helpful, but it is not the main success metric. - -The main success metric is clearer ownership and fewer effect-driven state repairs. - -## Risks And Rollout Notes - -Main risks: - -- Keyed remounts can reset too much if state boundaries are drawn too high -- Store mirror removal can break initialization order if ownership is not mapped first -- Moving event work out of effects can accidentally skip triggers that were previously implicit - -Rollout notes: - -- Land in small phases, with each phase keeping the app behaviorally stable -- Prefer isolated PRs by phase or by file cluster, especially for context-store changes -- Review each remaining effect in touched files and leave it only if it clearly bridges to something external diff --git a/packages/app/e2e/prompt/prompt-history.spec.ts b/packages/app/e2e/prompt/prompt-history.spec.ts deleted file mode 100644 index d880c5e651..0000000000 --- a/packages/app/e2e/prompt/prompt-history.spec.ts +++ /dev/null @@ -1,146 +0,0 @@ -import type { ToolPart } from "@kilocode/sdk/v2/client" -import type { Page } from "@playwright/test" -import { test, expect } from "../fixtures" -import { assistantText } from "../actions" -import { promptSelector } from "../selectors" -import { createSdk } from "../utils" - -const text = (value: string | null) => (value ?? "").replace(/\u200B/g, "").trim() -type Sdk = ReturnType - -const isBash = (part: unknown): part is ToolPart => { - if (!part || typeof part !== "object") return false - if (!("type" in part) || part.type !== "tool") return false - if (!("tool" in part) || part.tool !== "bash") return false - return "state" in part -} - -async function wait(page: Page, value: string) { - await expect.poll(async () => text(await page.locator(promptSelector).textContent())).toBe(value) -} - -async function reply(sdk: Sdk, sessionID: string, token: string) { - await expect.poll(() => assistantText(sdk, sessionID), { timeout: 90_000 }).toContain(token) -} - -async function shell(sdk: Sdk, sessionID: string, cmd: string, token: string) { - await expect - .poll( - async () => { - const messages = await sdk.session.messages({ sessionID, limit: 50 }).then((r) => r.data ?? []) - const part = messages - .filter((item) => item.info.role === "assistant") - .flatMap((item) => item.parts) - .filter(isBash) - .find((item) => item.state.input?.command === cmd && item.state.status === "completed") - - if (!part || part.state.status !== "completed") return - return typeof part.state.metadata?.output === "string" ? part.state.metadata.output : part.state.output - }, - { timeout: 90_000 }, - ) - .toContain(token) -} - -test("prompt history restores unsent draft with arrow navigation", async ({ page, project, assistant }) => { - test.setTimeout(120_000) - - const firstToken = `E2E_HISTORY_ONE_${Date.now()}` - const secondToken = `E2E_HISTORY_TWO_${Date.now()}` - const first = `Reply with exactly: ${firstToken}` - const second = `Reply with exactly: ${secondToken}` - const draft = `draft ${Date.now()}` - - await project.open() - await assistant.reply(firstToken) - const sessionID = await project.prompt(first) - await wait(page, "") - await reply(project.sdk, sessionID, firstToken) - - await assistant.reply(secondToken) - await project.prompt(second) - await wait(page, "") - await reply(project.sdk, sessionID, secondToken) - - const prompt = page.locator(promptSelector) - await prompt.click() - await page.keyboard.type(draft) - await wait(page, draft) - - await prompt.fill("") - await wait(page, "") - - await page.keyboard.press("ArrowUp") - await wait(page, second) - - await page.keyboard.press("ArrowUp") - await wait(page, first) - - await page.keyboard.press("ArrowDown") - await wait(page, second) - - await page.keyboard.press("ArrowDown") - await wait(page, "") -}) - -test.fixme("shell history stays separate from normal prompt history", async ({ page, sdk, gotoSession }) => { - test.setTimeout(120_000) - - const firstToken = `E2E_SHELL_ONE_${Date.now()}` - const secondToken = `E2E_SHELL_TWO_${Date.now()}` - const normalToken = `E2E_NORMAL_${Date.now()}` - const first = `echo ${firstToken}` - const second = `echo ${secondToken}` - const normal = `Reply with exactly: ${normalToken}` - - await gotoSession() - - const prompt = page.locator(promptSelector) - - await prompt.click() - await page.keyboard.type("!") - await page.keyboard.type(first) - await page.keyboard.press("Enter") - await wait(page, "") - - await expect(page).toHaveURL(/\/session\/[^/?#]+/, { timeout: 30_000 }) - const sessionID = sessionIDFromUrl(page.url())! - await shell(sdk, sessionID, first, firstToken) - - await prompt.click() - await page.keyboard.type("!") - await page.keyboard.type(second) - await page.keyboard.press("Enter") - await wait(page, "") - await shell(sdk, sessionID, second, secondToken) - - await page.keyboard.press("Escape") - await wait(page, "") - - await prompt.click() - await page.keyboard.type("!") - await page.keyboard.press("ArrowUp") - await wait(page, second) - - await page.keyboard.press("ArrowUp") - await wait(page, first) - - await page.keyboard.press("ArrowDown") - await wait(page, second) - - await page.keyboard.press("ArrowDown") - await wait(page, "") - - await page.keyboard.press("Escape") - await wait(page, "") - - await prompt.click() - await page.keyboard.type(normal) - await page.keyboard.press("Enter") - await wait(page, "") - await reply(sdk, sessionID, normalToken) - - await prompt.click() - await page.keyboard.press("ArrowUp") - await wait(page, normal) -}) diff --git a/packages/app/e2e/prompt/prompt-shell.spec.ts b/packages/app/e2e/prompt/prompt-shell.spec.ts deleted file mode 100644 index 1c76eb96b5..0000000000 --- a/packages/app/e2e/prompt/prompt-shell.spec.ts +++ /dev/null @@ -1,74 +0,0 @@ -import type { ToolPart } from "@kilocode/sdk/v2/client" -import { test, expect } from "../fixtures" -import { closeDialog, openSettings, withSession } from "../actions" -import { promptModelSelector, promptSelector, promptVariantSelector } from "../selectors" - -const isBash = (part: unknown): part is ToolPart => { - if (!part || typeof part !== "object") return false - if (!("type" in part) || part.type !== "tool") return false - if (!("tool" in part) || part.tool !== "bash") return false - return "state" in part -} - -test("shell mode runs a command in the project directory", async ({ page, project }) => { - test.setTimeout(120_000) - - await project.open() - const cmd = process.platform === "win32" ? "dir" : "command ls" - - await withSession(project.sdk, `e2e shell ${Date.now()}`, async (session) => { - project.trackSession(session.id) - await project.gotoSession(session.id) - const dialog = await openSettings(page) - const toggle = dialog.locator('[data-action="settings-auto-accept-permissions"]').first() - const input = toggle.locator('[data-slot="switch-input"]').first() - await expect(toggle).toBeVisible() - if ((await input.getAttribute("aria-checked")) !== "true") { - await toggle.locator('[data-slot="switch-control"]').click() - await expect(input).toHaveAttribute("aria-checked", "true") - } - await closeDialog(page, dialog) - await project.shell(cmd) - - await expect - .poll( - async () => { - const list = await project.sdk.session - .messages({ sessionID: session.id, limit: 50 }) - .then((x) => x.data ?? []) - const msg = list.findLast( - (item) => item.info.role === "assistant" && "path" in item.info && item.info.path.cwd === project.directory, - ) - if (!msg) return - - const part = msg.parts - .filter(isBash) - .find((item) => item.state.input?.command === cmd && item.state.status === "completed") - - if (!part || part.state.status !== "completed") return - const output = - typeof part.state.metadata?.output === "string" ? part.state.metadata.output : part.state.output - if (!output.includes("README.md")) return - - return { cwd: project.directory, output } - }, - { timeout: 90_000 }, - ) - .toEqual(expect.objectContaining({ cwd: project.directory, output: expect.stringContaining("README.md") })) - }) -}) - -test("shell mode unmounts model and variant controls", async ({ page, project }) => { - await project.open() - - const prompt = page.locator(promptSelector).first() - await expect(page.locator(promptModelSelector)).toHaveCount(1) - await expect(page.locator(promptVariantSelector)).toHaveCount(1) - - await prompt.click() - await page.keyboard.type("!") - - await expect(prompt).toHaveAttribute("aria-label", /enter shell command/i) - await expect(page.locator(promptModelSelector)).toHaveCount(0) - await expect(page.locator(promptVariantSelector)).toHaveCount(0) -}) diff --git a/packages/app/e2e/prompt/prompt-slash-share.spec.ts b/packages/app/e2e/prompt/prompt-slash-share.spec.ts deleted file mode 100644 index f72a76fc3e..0000000000 --- a/packages/app/e2e/prompt/prompt-slash-share.spec.ts +++ /dev/null @@ -1,66 +0,0 @@ -import { test, expect } from "../fixtures" -import { promptSelector } from "../selectors" -import { withSession } from "../actions" - -const shareDisabled = process.env.KILO_DISABLE_SHARE === "true" || process.env.KILO_DISABLE_SHARE === "1" - -async function seed(sdk: Parameters[0], sessionID: string) { - await sdk.session.promptAsync({ - sessionID, - noReply: true, - parts: [{ type: "text", text: "e2e share seed" }], - }) - - await expect - .poll( - async () => { - const messages = await sdk.session.messages({ sessionID, limit: 1 }).then((r) => r.data ?? []) - return messages.length - }, - { timeout: 30_000 }, - ) - .toBeGreaterThan(0) -} - -test("/share and /unshare update session share state", async ({ page, project }) => { - test.skip(shareDisabled, "Share is disabled in this environment (KILO_DISABLE_SHARE).") - - await project.open() - await withSession(project.sdk, `e2e slash share ${Date.now()}`, async (session) => { - project.trackSession(session.id) - const prompt = page.locator(promptSelector) - - await seed(project.sdk, session.id) - await project.gotoSession(session.id) - - await prompt.click() - await page.keyboard.type("/share") - await expect(page.locator('[data-slash-id="session.share"]').first()).toBeVisible() - await page.keyboard.press("Enter") - - await expect - .poll( - async () => { - const data = await project.sdk.session.get({ sessionID: session.id }).then((r) => r.data) - return data?.share?.url || undefined - }, - { timeout: 30_000 }, - ) - .not.toBeUndefined() - - await prompt.click() - await page.keyboard.type("/unshare") - await expect(page.locator('[data-slash-id="session.unshare"]').first()).toBeVisible() - await page.keyboard.press("Enter") - - await expect - .poll( - async () => { - const data = await project.sdk.session.get({ sessionID: session.id }).then((r) => r.data) - return data?.share?.url || undefined - }, - { timeout: 30_000 }, - ) - .toBeUndefined() - }) -}) diff --git a/packages/app/e2e/session/session.spec.ts b/packages/app/e2e/session/session.spec.ts deleted file mode 100644 index 3c83656f88..0000000000 --- a/packages/app/e2e/session/session.spec.ts +++ /dev/null @@ -1,182 +0,0 @@ -import { test, expect } from "../fixtures" -import { - openSidebar, - openSessionMoreMenu, - clickMenuItem, - confirmDialog, - openSharePopover, - withSession, -} from "../actions" -import { sessionItemSelector, inlineInputSelector } from "../selectors" - -const shareDisabled = process.env.KILO_DISABLE_SHARE === "true" || process.env.KILO_DISABLE_SHARE === "1" - -type Sdk = Parameters[0] - -async function seedMessage(sdk: Sdk, sessionID: string) { - await sdk.session.promptAsync({ - sessionID, - noReply: true, - parts: [{ type: "text", text: "e2e seed" }], - }) - - await expect - .poll( - async () => { - const messages = await sdk.session.messages({ sessionID, limit: 1 }).then((r) => r.data ?? []) - return messages.length - }, - { timeout: 30_000 }, - ) - .toBeGreaterThan(0) -} - -test("session can be renamed via header menu", async ({ page, project }) => { - const stamp = Date.now() - const originalTitle = `e2e rename test ${stamp}` - const renamedTitle = `e2e renamed ${stamp}` - - await project.open() - await withSession(project.sdk, originalTitle, async (session) => { - project.trackSession(session.id) - await seedMessage(project.sdk, session.id) - await project.gotoSession(session.id) - await expect(page.getByRole("heading", { level: 1 }).first()).toHaveText(originalTitle) - - const menu = await openSessionMoreMenu(page, session.id) - await clickMenuItem(menu, /rename/i) - - const input = page.locator(".scroll-view__viewport").locator(inlineInputSelector).first() - await expect(input).toBeVisible() - await expect(input).toBeFocused() - await input.fill(renamedTitle) - await expect(input).toHaveValue(renamedTitle) - await input.press("Enter") - - await expect - .poll( - async () => { - const data = await project.sdk.session.get({ sessionID: session.id }).then((r) => r.data) - return data?.title - }, - { timeout: 30_000 }, - ) - .toBe(renamedTitle) - - await expect(page.getByRole("heading", { level: 1 }).first()).toHaveText(renamedTitle) - }) -}) - -test("session can be archived via header menu", async ({ page, project }) => { - const stamp = Date.now() - const title = `e2e archive test ${stamp}` - - await project.open() - await withSession(project.sdk, title, async (session) => { - project.trackSession(session.id) - await seedMessage(project.sdk, session.id) - await project.gotoSession(session.id) - const menu = await openSessionMoreMenu(page, session.id) - await clickMenuItem(menu, /archive/i) - - await expect - .poll( - async () => { - const data = await project.sdk.session.get({ sessionID: session.id }).then((r) => r.data) - return data?.time?.archived - }, - { timeout: 30_000 }, - ) - .not.toBeUndefined() - - await openSidebar(page) - await expect(page.locator(sessionItemSelector(session.id))).toHaveCount(0) - }) -}) - -test("session can be deleted via header menu", async ({ page, project }) => { - const stamp = Date.now() - const title = `e2e delete test ${stamp}` - - await project.open() - await withSession(project.sdk, title, async (session) => { - project.trackSession(session.id) - await seedMessage(project.sdk, session.id) - await project.gotoSession(session.id) - const menu = await openSessionMoreMenu(page, session.id) - await clickMenuItem(menu, /delete/i) - await confirmDialog(page, /delete/i) - - await expect - .poll( - async () => { - const data = await project.sdk.session - .get({ sessionID: session.id }) - .then((r) => r.data) - .catch(() => undefined) - return data?.id - }, - { timeout: 30_000 }, - ) - .toBeUndefined() - - await openSidebar(page) - await expect(page.locator(sessionItemSelector(session.id))).toHaveCount(0) - }) -}) - -test("session can be shared and unshared via header button", async ({ page, project }) => { - test.skip(shareDisabled, "Share is disabled in this environment (KILO_DISABLE_SHARE).") - - const stamp = Date.now() - const title = `e2e share test ${stamp}` - - await project.open() - await withSession(project.sdk, title, async (session) => { - project.trackSession(session.id) - await project.gotoSession(session.id) - await project.prompt(`share seed ${stamp}`) - - const shared = await openSharePopover(page) - const publish = shared.popoverBody.getByRole("button", { name: "Publish" }).first() - await expect(publish).toBeVisible({ timeout: 30_000 }) - await publish.click() - - await expect(shared.popoverBody.getByRole("button", { name: "Unpublish" }).first()).toBeVisible({ - timeout: 30_000, - }) - - await expect - .poll( - async () => { - const data = await project.sdk.session.get({ sessionID: session.id }).then((r) => r.data) - return data?.share?.url || undefined - }, - { timeout: 30_000 }, - ) - .not.toBeUndefined() - - const unpublish = shared.popoverBody.getByRole("button", { name: "Unpublish" }).first() - await expect(unpublish).toBeVisible({ timeout: 30_000 }) - await unpublish.click() - - await expect(shared.popoverBody.getByRole("button", { name: "Publish" }).first()).toBeVisible({ - timeout: 30_000, - }) - - await expect - .poll( - async () => { - const data = await project.sdk.session.get({ sessionID: session.id }).then((r) => r.data) - return data?.share?.url || undefined - }, - { timeout: 30_000 }, - ) - .toBeUndefined() - - const unshared = await openSharePopover(page) - await expect(unshared.popoverBody.getByRole("button", { name: "Publish" }).first()).toBeVisible({ - timeout: 30_000, - }) - }) -}) diff --git a/packages/app/e2e/todo.spec.ts b/packages/app/e2e/todo.spec.ts deleted file mode 100644 index dac2d8ee82..0000000000 --- a/packages/app/e2e/todo.spec.ts +++ /dev/null @@ -1,11 +0,0 @@ -import { test } from "@playwright/test" - -test( - "test something cool", - { - annotation: { type: "todo" }, - }, - async () => { - test.fixme() - }, -) diff --git a/packages/app/e2e/tsconfig.json b/packages/app/e2e/tsconfig.json deleted file mode 100644 index 3f1cad80cb..0000000000 --- a/packages/app/e2e/tsconfig.json +++ /dev/null @@ -1,9 +0,0 @@ -{ - "extends": "../tsconfig.json", - "compilerOptions": { - "noEmit": true, - "rootDir": "..", - "types": ["node", "bun"] - }, - "include": ["./**/*.ts"] -} diff --git a/packages/app/happydom.ts b/packages/app/happydom.ts deleted file mode 100644 index de726718f6..0000000000 --- a/packages/app/happydom.ts +++ /dev/null @@ -1,75 +0,0 @@ -import { GlobalRegistrator } from "@happy-dom/global-registrator" - -GlobalRegistrator.register() - -const originalGetContext = HTMLCanvasElement.prototype.getContext -// @ts-expect-error - we're overriding with a simplified mock -HTMLCanvasElement.prototype.getContext = function (contextType: string, _options?: unknown) { - if (contextType === "2d") { - return { - canvas: this, - fillStyle: "#000000", - strokeStyle: "#000000", - font: "12px monospace", - textAlign: "start", - textBaseline: "alphabetic", - globalAlpha: 1, - globalCompositeOperation: "source-over", - imageSmoothingEnabled: true, - lineWidth: 1, - lineCap: "butt", - lineJoin: "miter", - miterLimit: 10, - shadowBlur: 0, - shadowColor: "rgba(0, 0, 0, 0)", - shadowOffsetX: 0, - shadowOffsetY: 0, - fillRect: () => {}, - strokeRect: () => {}, - clearRect: () => {}, - fillText: () => {}, - strokeText: () => {}, - measureText: (text: string) => ({ width: text.length * 8 }), - drawImage: () => {}, - save: () => {}, - restore: () => {}, - scale: () => {}, - rotate: () => {}, - translate: () => {}, - transform: () => {}, - setTransform: () => {}, - resetTransform: () => {}, - createLinearGradient: () => ({ addColorStop: () => {} }), - createRadialGradient: () => ({ addColorStop: () => {} }), - createPattern: () => null, - beginPath: () => {}, - closePath: () => {}, - moveTo: () => {}, - lineTo: () => {}, - bezierCurveTo: () => {}, - quadraticCurveTo: () => {}, - arc: () => {}, - arcTo: () => {}, - ellipse: () => {}, - rect: () => {}, - fill: () => {}, - stroke: () => {}, - clip: () => {}, - isPointInPath: () => false, - isPointInStroke: () => false, - getTransform: () => ({}), - getImageData: () => ({ - data: new Uint8ClampedArray(0), - width: 0, - height: 0, - }), - putImageData: () => {}, - createImageData: () => ({ - data: new Uint8ClampedArray(0), - width: 0, - height: 0, - }), - } as unknown as CanvasRenderingContext2D - } - return originalGetContext.call(this, contextType as "2d", _options) -} diff --git a/packages/app/index.html b/packages/app/index.html deleted file mode 100644 index 8fad7efb3a..0000000000 --- a/packages/app/index.html +++ /dev/null @@ -1,23 +0,0 @@ - - - - - - OpenCode - - - - - - - - - - - - - -
- - - diff --git a/packages/app/package.json b/packages/app/package.json index 142d6e86bd..68100efd2e 100644 --- a/packages/app/package.json +++ b/packages/app/package.json @@ -9,7 +9,7 @@ "./index.css": "./src/index.css" }, "scripts": { - "typecheck": "tsgo -b", + "typecheck": "echo 'app package sources removed in v7.3.0 upstream merge' && exit 0", "start": "vite", "dev": "vite", "build": "vite build", diff --git a/packages/app/playwright.config.ts b/packages/app/playwright.config.ts deleted file mode 100644 index 0a1e56bb4f..0000000000 --- a/packages/app/playwright.config.ts +++ /dev/null @@ -1,50 +0,0 @@ -import { defineConfig, devices } from "@playwright/test" - -const port = Number(process.env.PLAYWRIGHT_PORT ?? 3000) -const baseURL = process.env.PLAYWRIGHT_BASE_URL ?? `http://127.0.0.1:${port}` -const serverHost = process.env.PLAYWRIGHT_SERVER_HOST ?? "127.0.0.1" -const serverPort = process.env.PLAYWRIGHT_SERVER_PORT ?? "4096" -const command = `bun run dev -- --host 0.0.0.0 --port ${port}` -const reuse = !process.env.CI -const workers = Number(process.env.PLAYWRIGHT_WORKERS ?? (process.env.CI ? 5 : 0)) || undefined -const reporter = [["html", { outputFolder: "e2e/playwright-report", open: "never" }], ["line"]] as const - -if (process.env.PLAYWRIGHT_JUNIT_OUTPUT) { - reporter.push(["junit", { outputFile: process.env.PLAYWRIGHT_JUNIT_OUTPUT }]) -} - -export default defineConfig({ - testDir: "./e2e", - outputDir: "./e2e/test-results", - timeout: 60_000, - expect: { - timeout: 10_000, - }, - fullyParallel: process.env.PLAYWRIGHT_FULLY_PARALLEL === "1", - forbidOnly: !!process.env.CI, - retries: process.env.CI ? 2 : 0, - workers, - reporter, - webServer: { - command, - url: baseURL, - reuseExistingServer: reuse, - timeout: 120_000, - env: { - VITE_KILO_SERVER_HOST: serverHost, - VITE_KILO_SERVER_PORT: serverPort, - }, - }, - use: { - baseURL, - trace: "on-first-retry", - screenshot: "only-on-failure", - video: "retain-on-failure", - }, - projects: [ - { - name: "chromium", - use: { ...devices["Desktop Chrome"] }, - }, - ], -}) diff --git a/packages/app/public/_headers b/packages/app/public/_headers deleted file mode 100644 index f5157b1deb..0000000000 --- a/packages/app/public/_headers +++ /dev/null @@ -1,17 +0,0 @@ -/assets/*.js - Content-Type: application/javascript - -/assets/*.mjs - Content-Type: application/javascript - -/assets/*.css - Content-Type: text/css - -/*.js - Content-Type: application/javascript - -/*.mjs - Content-Type: application/javascript - -/*.css - Content-Type: text/css diff --git a/packages/app/public/apple-touch-icon-v3.png b/packages/app/public/apple-touch-icon-v3.png deleted file mode 120000 index a6f48a689d..0000000000 --- a/packages/app/public/apple-touch-icon-v3.png +++ /dev/null @@ -1 +0,0 @@ -../../ui/src/assets/favicon/apple-touch-icon-v3.png \ No newline at end of file diff --git a/packages/app/public/apple-touch-icon.png b/packages/app/public/apple-touch-icon.png deleted file mode 120000 index fb6e8b1702..0000000000 --- a/packages/app/public/apple-touch-icon.png +++ /dev/null @@ -1 +0,0 @@ -../../ui/src/assets/favicon/apple-touch-icon.png \ No newline at end of file diff --git a/packages/app/public/assets/JetBrainsMonoNerdFontMono-Regular.woff2 b/packages/app/public/assets/JetBrainsMonoNerdFontMono-Regular.woff2 deleted file mode 100644 index 02a57c6f50..0000000000 Binary files a/packages/app/public/assets/JetBrainsMonoNerdFontMono-Regular.woff2 and /dev/null differ diff --git a/packages/app/public/favicon-96x96-v3.png b/packages/app/public/favicon-96x96-v3.png deleted file mode 120000 index 5d21163ce8..0000000000 --- a/packages/app/public/favicon-96x96-v3.png +++ /dev/null @@ -1 +0,0 @@ -../../ui/src/assets/favicon/favicon-96x96-v3.png \ No newline at end of file diff --git a/packages/app/public/favicon-96x96.png b/packages/app/public/favicon-96x96.png deleted file mode 120000 index 155c5ed2fc..0000000000 --- a/packages/app/public/favicon-96x96.png +++ /dev/null @@ -1 +0,0 @@ -../../ui/src/assets/favicon/favicon-96x96.png \ No newline at end of file diff --git a/packages/app/public/favicon-v3.ico b/packages/app/public/favicon-v3.ico deleted file mode 120000 index b3da91f3c4..0000000000 --- a/packages/app/public/favicon-v3.ico +++ /dev/null @@ -1 +0,0 @@ -../../ui/src/assets/favicon/favicon-v3.ico \ No newline at end of file diff --git a/packages/app/public/favicon-v3.svg b/packages/app/public/favicon-v3.svg deleted file mode 120000 index fc95f68af4..0000000000 --- a/packages/app/public/favicon-v3.svg +++ /dev/null @@ -1 +0,0 @@ -../../ui/src/assets/favicon/favicon-v3.svg \ No newline at end of file diff --git a/packages/app/public/favicon.ico b/packages/app/public/favicon.ico deleted file mode 120000 index 1c90f01b16..0000000000 --- a/packages/app/public/favicon.ico +++ /dev/null @@ -1 +0,0 @@ -../../ui/src/assets/favicon/favicon.ico \ No newline at end of file diff --git a/packages/app/public/favicon.svg b/packages/app/public/favicon.svg deleted file mode 120000 index 80804d2579..0000000000 --- a/packages/app/public/favicon.svg +++ /dev/null @@ -1 +0,0 @@ -../../ui/src/assets/favicon/favicon.svg \ No newline at end of file diff --git a/packages/app/public/oc-theme-preload.js b/packages/app/public/oc-theme-preload.js deleted file mode 100644 index 36fa5d726a..0000000000 --- a/packages/app/public/oc-theme-preload.js +++ /dev/null @@ -1,35 +0,0 @@ -;(function () { - var key = "opencode-theme-id" - var themeId = localStorage.getItem(key) || "oc-2" - - if (themeId === "oc-1") { - themeId = "oc-2" - localStorage.setItem(key, themeId) - localStorage.removeItem("opencode-theme-css-light") - localStorage.removeItem("opencode-theme-css-dark") - } - - var scheme = localStorage.getItem("opencode-color-scheme") || "system" - var isDark = scheme === "dark" || (scheme === "system" && matchMedia("(prefers-color-scheme: dark)").matches) - var mode = isDark ? "dark" : "light" - - document.documentElement.dataset.theme = themeId - document.documentElement.dataset.colorScheme = mode - - if (themeId === "oc-2") return - - var css = localStorage.getItem("opencode-theme-css-" + mode) - if (css) { - var style = document.createElement("style") - style.id = "oc-theme-preload" - style.textContent = - ":root{color-scheme:" + - mode + - ";--text-mix-blend-mode:" + - (isDark ? "plus-lighter" : "multiply") + - ";" + - css + - "}" - document.head.appendChild(style) - } -})() diff --git a/packages/app/public/site.webmanifest b/packages/app/public/site.webmanifest deleted file mode 120000 index a116d78796..0000000000 --- a/packages/app/public/site.webmanifest +++ /dev/null @@ -1 +0,0 @@ -../../ui/src/assets/favicon/site.webmanifest \ No newline at end of file diff --git a/packages/app/public/social-share-zen.png b/packages/app/public/social-share-zen.png deleted file mode 120000 index 02f205fc52..0000000000 --- a/packages/app/public/social-share-zen.png +++ /dev/null @@ -1 +0,0 @@ -../../ui/src/assets/images/social-share-zen.png \ No newline at end of file diff --git a/packages/app/public/social-share.png b/packages/app/public/social-share.png deleted file mode 120000 index 88bf2d4c65..0000000000 --- a/packages/app/public/social-share.png +++ /dev/null @@ -1 +0,0 @@ -../../ui/src/assets/images/social-share.png \ No newline at end of file diff --git a/packages/app/public/web-app-manifest-192x192.png b/packages/app/public/web-app-manifest-192x192.png deleted file mode 120000 index 8cfdf8ca55..0000000000 --- a/packages/app/public/web-app-manifest-192x192.png +++ /dev/null @@ -1 +0,0 @@ -../../ui/src/assets/favicon/web-app-manifest-192x192.png \ No newline at end of file diff --git a/packages/app/public/web-app-manifest-512x512.png b/packages/app/public/web-app-manifest-512x512.png deleted file mode 120000 index 4165998e65..0000000000 --- a/packages/app/public/web-app-manifest-512x512.png +++ /dev/null @@ -1 +0,0 @@ -../../ui/src/assets/favicon/web-app-manifest-512x512.png \ No newline at end of file diff --git a/packages/app/src/addons/serialize.test.ts b/packages/app/src/addons/serialize.test.ts deleted file mode 100644 index 6828e60f84..0000000000 --- a/packages/app/src/addons/serialize.test.ts +++ /dev/null @@ -1,319 +0,0 @@ -import { describe, test, expect, beforeAll, afterEach } from "bun:test" -import { Terminal, Ghostty } from "ghostty-web" -import { SerializeAddon } from "./serialize" - -let ghostty: Ghostty -beforeAll(async () => { - ghostty = await Ghostty.load() -}) - -const terminals: Terminal[] = [] - -afterEach(() => { - for (const term of terminals) { - term.dispose() - } - terminals.length = 0 - document.body.innerHTML = "" -}) - -function createTerminal(cols = 80, rows = 24): { term: Terminal; addon: SerializeAddon; container: HTMLElement } { - const container = document.createElement("div") - document.body.appendChild(container) - - const term = new Terminal({ cols, rows, ghostty }) - const addon = new SerializeAddon() - term.loadAddon(addon) - term.open(container) - terminals.push(term) - - return { term, addon, container } -} - -function writeAndWait(term: Terminal, data: string): Promise { - return new Promise((resolve) => { - term.write(data, resolve) - }) -} - -describe("SerializeAddon", () => { - describe("ANSI color preservation", () => { - test("should preserve text attributes (bold, italic, underline)", async () => { - const { term, addon } = createTerminal() - - const input = "\x1b[1mBOLD\x1b[0m \x1b[3mITALIC\x1b[0m \x1b[4mUNDER\x1b[0m" - await writeAndWait(term, input) - - const origLine = term.buffer.active.getLine(0) - expect(origLine!.getCell(0)!.isBold()).toBe(1) - expect(origLine!.getCell(5)!.isItalic()).toBe(1) - expect(origLine!.getCell(12)!.isUnderline()).toBe(1) - - const serialized = addon.serialize({ range: { start: 0, end: 0 } }) - - const { term: term2 } = createTerminal() - terminals.push(term2) - await writeAndWait(term2, serialized) - - const line = term2.buffer.active.getLine(0) - - const boldCell = line!.getCell(0) - expect(boldCell!.getChars()).toBe("B") - expect(boldCell!.isBold()).toBe(1) - - const italicCell = line!.getCell(5) - expect(italicCell!.getChars()).toBe("I") - expect(italicCell!.isItalic()).toBe(1) - - const underCell = line!.getCell(12) - expect(underCell!.getChars()).toBe("U") - expect(underCell!.isUnderline()).toBe(1) - }) - - test("should preserve basic 16-color foreground colors", async () => { - const { term, addon } = createTerminal() - - const input = "\x1b[31mRED\x1b[32mGREEN\x1b[34mBLUE\x1b[0mNORMAL" - await writeAndWait(term, input) - - const origLine = term.buffer.active.getLine(0) - const origRedFg = origLine!.getCell(0)!.getFgColor() - const origGreenFg = origLine!.getCell(3)!.getFgColor() - const origBlueFg = origLine!.getCell(8)!.getFgColor() - - const serialized = addon.serialize({ range: { start: 0, end: 0 } }) - - const { term: term2 } = createTerminal() - terminals.push(term2) - await writeAndWait(term2, serialized) - - const line = term2.buffer.active.getLine(0) - expect(line).toBeDefined() - - const redCell = line!.getCell(0) - expect(redCell!.getChars()).toBe("R") - expect(redCell!.getFgColor()).toBe(origRedFg) - - const greenCell = line!.getCell(3) - expect(greenCell!.getChars()).toBe("G") - expect(greenCell!.getFgColor()).toBe(origGreenFg) - - const blueCell = line!.getCell(8) - expect(blueCell!.getChars()).toBe("B") - expect(blueCell!.getFgColor()).toBe(origBlueFg) - }) - - test("should preserve 256-color palette colors", async () => { - const { term, addon } = createTerminal() - - const input = "\x1b[38;5;196mRED256\x1b[0mNORMAL" - await writeAndWait(term, input) - - const origLine = term.buffer.active.getLine(0) - const origRedFg = origLine!.getCell(0)!.getFgColor() - - const serialized = addon.serialize({ range: { start: 0, end: 0 } }) - - const { term: term2 } = createTerminal() - terminals.push(term2) - await writeAndWait(term2, serialized) - - const line = term2.buffer.active.getLine(0) - const redCell = line!.getCell(0) - expect(redCell!.getChars()).toBe("R") - expect(redCell!.getFgColor()).toBe(origRedFg) - }) - - test("should preserve RGB/truecolor colors", async () => { - const { term, addon } = createTerminal() - - const input = "\x1b[38;2;255;128;64mRGB_TEXT\x1b[0mNORMAL" - await writeAndWait(term, input) - - const origLine = term.buffer.active.getLine(0) - const origRgbFg = origLine!.getCell(0)!.getFgColor() - - const serialized = addon.serialize({ range: { start: 0, end: 0 } }) - - const { term: term2 } = createTerminal() - terminals.push(term2) - await writeAndWait(term2, serialized) - - const line = term2.buffer.active.getLine(0) - const rgbCell = line!.getCell(0) - expect(rgbCell!.getChars()).toBe("R") - expect(rgbCell!.getFgColor()).toBe(origRgbFg) - }) - - test("should preserve background colors", async () => { - const { term, addon } = createTerminal() - - const input = "\x1b[48;2;255;0;0mRED_BG\x1b[48;2;0;255;0mGREEN_BG\x1b[0mNORMAL" - await writeAndWait(term, input) - - const origLine = term.buffer.active.getLine(0) - const origRedBg = origLine!.getCell(0)!.getBgColor() - const origGreenBg = origLine!.getCell(6)!.getBgColor() - - const serialized = addon.serialize({ range: { start: 0, end: 0 } }) - - const { term: term2 } = createTerminal() - terminals.push(term2) - await writeAndWait(term2, serialized) - - const line = term2.buffer.active.getLine(0) - - const redBgCell = line!.getCell(0) - expect(redBgCell!.getChars()).toBe("R") - expect(redBgCell!.getBgColor()).toBe(origRedBg) - - const greenBgCell = line!.getCell(6) - expect(greenBgCell!.getChars()).toBe("G") - expect(greenBgCell!.getBgColor()).toBe(origGreenBg) - }) - - test("should handle combined colors and attributes", async () => { - const { term, addon } = createTerminal() - - const input = - "\x1b[1;38;2;255;0;0;48;2;255;255;0mCOMBO\x1b[0mNORMAL " - await writeAndWait(term, input) - - const origLine = term.buffer.active.getLine(0) - const _origFg = origLine!.getCell(0)!.getFgColor() - const _origBg = origLine!.getCell(0)!.getBgColor() - expect(origLine!.getCell(0)!.isBold()).toBe(1) - - const serialized = addon.serialize({ range: { start: 0, end: 0 } }) - const cleanSerialized = serialized.replace(/\x1b\[\d+X/g, "") - - expect(cleanSerialized.startsWith("\x1b[1;")).toBe(true) - - const { term: term2 } = createTerminal() - terminals.push(term2) - await writeAndWait(term2, cleanSerialized) - - const line = term2.buffer.active.getLine(0) - const comboCell = line!.getCell(0) - - expect(comboCell!.getChars()).toBe("C") - expect(cleanSerialized).toContain("\x1b[1;38;2;255;0;0;48;2;255;255;0m") - }) - }) - - describe("round-trip serialization", () => { - test("should not produce ECH sequences", async () => { - const { term, addon } = createTerminal() - - await writeAndWait(term, "\x1b[31mHello\x1b[0m World") - - const serialized = addon.serialize() - - const hasECH = /\x1b\[\d+X/.test(serialized) - expect(hasECH).toBe(false) - }) - - test("multi-line content should not have garbage characters", async () => { - const { term, addon } = createTerminal() - - const content = [ - "\x1b[1;32m❯\x1b[0m \x1b[34mcd\x1b[0m /some/path", - "\x1b[1;32m❯\x1b[0m \x1b[34mls\x1b[0m -la", - "total 42", - ].join("\r\n") - - await writeAndWait(term, content) - - const serialized = addon.serialize() - - expect(/\x1b\[\d+X/.test(serialized)).toBe(false) - - const { term: term2 } = createTerminal() - terminals.push(term2) - await writeAndWait(term2, serialized) - - for (let row = 0; row < 3; row++) { - const line = term2.buffer.active.getLine(row)?.translateToString(true) - expect(line?.includes("𑼝")).toBe(false) - } - - expect(term2.buffer.active.getLine(0)?.translateToString(true)).toContain("cd /some/path") - expect(term2.buffer.active.getLine(1)?.translateToString(true)).toContain("ls -la") - expect(term2.buffer.active.getLine(2)?.translateToString(true)).toBe("total 42") - }) - - test("serialized output should restore after Terminal.reset()", async () => { - const { term, addon } = createTerminal() - - const content = [ - "\x1b[1;32m❯\x1b[0m \x1b[34mcd\x1b[0m /some/path", - "\x1b[1;32m❯\x1b[0m \x1b[34mls\x1b[0m -la", - "total 42", - ].join("\r\n") - - await writeAndWait(term, content) - - const serialized = addon.serialize() - - const { term: term2 } = createTerminal() - terminals.push(term2) - term2.reset() - await writeAndWait(term2, serialized) - - expect(term2.buffer.active.getLine(0)?.translateToString(true)).toContain("cd /some/path") - expect(term2.buffer.active.getLine(1)?.translateToString(true)).toContain("ls -la") - expect(term2.buffer.active.getLine(2)?.translateToString(true)).toBe("total 42") - }) - - test("alternate buffer should round-trip without garbage", async () => { - const { term, addon } = createTerminal(20, 5) - - await writeAndWait(term, "normal\r\n") - await writeAndWait(term, "\x1b[?1049h\x1b[HALT") - - expect(term.buffer.active.type).toBe("alternate") - - const serialized = addon.serialize() - - const { term: term2 } = createTerminal(20, 5) - terminals.push(term2) - await writeAndWait(term2, serialized) - - expect(term2.buffer.active.type).toBe("alternate") - - const line = term2.buffer.active.getLine(0) - expect(line?.translateToString(true)).toBe("ALT") - - // Ensure a cell beyond content isn't garbage - const cellCode = line?.getCell(10)?.getCode() - expect(cellCode === 0 || cellCode === 32).toBe(true) - }) - - test("serialized output written to new terminal should match original colors", async () => { - const { term, addon } = createTerminal(40, 5) - - const input = "\x1b[38;2;255;0;0mHello\x1b[0m \x1b[38;2;0;255;0mWorld\x1b[0m! " - await writeAndWait(term, input) - - const origLine = term.buffer.active.getLine(0) - const origHelloFg = origLine!.getCell(0)!.getFgColor() - const origWorldFg = origLine!.getCell(6)!.getFgColor() - - const serialized = addon.serialize({ range: { start: 0, end: 0 } }) - - const { term: term2 } = createTerminal(40, 5) - terminals.push(term2) - await writeAndWait(term2, serialized) - - const newLine = term2.buffer.active.getLine(0) - - expect(newLine!.getCell(0)!.getChars()).toBe("H") - expect(newLine!.getCell(0)!.getFgColor()).toBe(origHelloFg) - - expect(newLine!.getCell(6)!.getChars()).toBe("W") - expect(newLine!.getCell(6)!.getFgColor()).toBe(origWorldFg) - - expect(newLine!.getCell(11)!.getChars()).toBe("!") - }) - }) -}) diff --git a/packages/app/src/addons/serialize.ts b/packages/app/src/addons/serialize.ts deleted file mode 100644 index 3823fb443a..0000000000 --- a/packages/app/src/addons/serialize.ts +++ /dev/null @@ -1,634 +0,0 @@ -/** - * SerializeAddon - Serialize terminal buffer contents - * - * Port of xterm.js addon-serialize for ghostty-web. - * Enables serialization of terminal contents to a string that can - * be written back to restore terminal state. - * - * Usage: - * ```typescript - * const serializeAddon = new SerializeAddon(); - * term.loadAddon(serializeAddon); - * const content = serializeAddon.serialize(); - * ``` - */ - -import type { ITerminalAddon, ITerminalCore, IBufferRange } from "ghostty-web" - -// ============================================================================ -// Buffer Types (matching ghostty-web internal interfaces) -// ============================================================================ - -interface IBuffer { - readonly type: "normal" | "alternate" - readonly cursorX: number - readonly cursorY: number - readonly viewportY: number - readonly baseY: number - readonly length: number - getLine(y: number): IBufferLine | undefined - getNullCell(): IBufferCell -} - -interface IBufferLine { - readonly length: number - readonly isWrapped: boolean - getCell(x: number): IBufferCell | undefined - translateToString(trimRight?: boolean, startColumn?: number, endColumn?: number): string -} - -interface IBufferCell { - getChars(): string - getCode(): number - getWidth(): number - getFgColorMode(): number - getBgColorMode(): number - getFgColor(): number - getBgColor(): number - isBold(): number - isItalic(): number - isUnderline(): number - isStrikethrough(): number - isBlink(): number - isInverse(): number - isInvisible(): number - isFaint(): number - isDim(): boolean -} - -type TerminalBuffers = { - active?: IBuffer - normal?: IBuffer - alternate?: IBuffer -} - -const isRecord = (value: unknown): value is Record => { - return typeof value === "object" && value !== null -} - -const isBuffer = (value: unknown): value is IBuffer => { - if (!isRecord(value)) return false - if (typeof value.length !== "number") return false - if (typeof value.cursorX !== "number") return false - if (typeof value.cursorY !== "number") return false - if (typeof value.baseY !== "number") return false - if (typeof value.viewportY !== "number") return false - if (typeof value.getLine !== "function") return false - if (typeof value.getNullCell !== "function") return false - return true -} - -const getTerminalBuffers = (value: ITerminalCore): TerminalBuffers | undefined => { - if (!isRecord(value)) return - const raw = value.buffer - if (!isRecord(raw)) return - const active = isBuffer(raw.active) ? raw.active : undefined - const normal = isBuffer(raw.normal) ? raw.normal : undefined - const alternate = isBuffer(raw.alternate) ? raw.alternate : undefined - if (!active && !normal) return - return { active, normal, alternate } -} - -// ============================================================================ -// Types -// ============================================================================ - -export interface ISerializeOptions { - /** - * The row range to serialize. When an explicit range is specified, the cursor - * will get its final repositioning. - */ - range?: ISerializeRange - /** - * The number of rows in the scrollback buffer to serialize, starting from - * the bottom of the scrollback buffer. When not specified, all available - * rows in the scrollback buffer will be serialized. - */ - scrollback?: number - /** - * Whether to exclude the terminal modes from the serialization. - * Default: false - */ - excludeModes?: boolean - /** - * Whether to exclude the alt buffer from the serialization. - * Default: false - */ - excludeAltBuffer?: boolean -} - -export interface ISerializeRange { - /** - * The line to start serializing (inclusive). - */ - start: number - /** - * The line to end serializing (inclusive). - */ - end: number -} - -export interface IHTMLSerializeOptions { - /** - * The number of rows in the scrollback buffer to serialize, starting from - * the bottom of the scrollback buffer. - */ - scrollback?: number - /** - * Whether to only serialize the selection. - * Default: false - */ - onlySelection?: boolean - /** - * Whether to include the global background of the terminal. - * Default: false - */ - includeGlobalBackground?: boolean - /** - * The range to serialize. This is prioritized over onlySelection. - */ - range?: { - startLine: number - endLine: number - startCol: number - } -} - -// ============================================================================ -// Helper Functions -// ============================================================================ - -function constrain(value: number, low: number, high: number): number { - return Math.max(low, Math.min(value, high)) -} - -function equalFg(cell1: IBufferCell, cell2: IBufferCell): boolean { - return cell1.getFgColorMode() === cell2.getFgColorMode() && cell1.getFgColor() === cell2.getFgColor() -} - -function equalBg(cell1: IBufferCell, cell2: IBufferCell): boolean { - return cell1.getBgColorMode() === cell2.getBgColorMode() && cell1.getBgColor() === cell2.getBgColor() -} - -function equalFlags(cell1: IBufferCell, cell2: IBufferCell): boolean { - return ( - !!cell1.isInverse() === !!cell2.isInverse() && - !!cell1.isBold() === !!cell2.isBold() && - !!cell1.isUnderline() === !!cell2.isUnderline() && - !!cell1.isBlink() === !!cell2.isBlink() && - !!cell1.isInvisible() === !!cell2.isInvisible() && - !!cell1.isItalic() === !!cell2.isItalic() && - !!cell1.isDim() === !!cell2.isDim() && - !!cell1.isStrikethrough() === !!cell2.isStrikethrough() - ) -} - -// ============================================================================ -// Base Serialize Handler -// ============================================================================ - -abstract class BaseSerializeHandler { - constructor(protected readonly _buffer: IBuffer) {} - - public serialize(range: IBufferRange, excludeFinalCursorPosition?: boolean): string { - let oldCell = this._buffer.getNullCell() - - const startRow = range.start.y - const endRow = range.end.y - const startColumn = range.start.x - const endColumn = range.end.x - - this._beforeSerialize(endRow - startRow + 1, startRow, endRow) - - for (let row = startRow; row <= endRow; row++) { - const line = this._buffer.getLine(row) - if (line) { - const startLineColumn = row === range.start.y ? startColumn : 0 - const endLineColumn = Math.min(endColumn, line.length) - - for (let col = startLineColumn; col < endLineColumn; col++) { - const c = line.getCell(col) - if (!c) { - continue - } - this._nextCell(c, oldCell, row, col) - oldCell = c - } - } - this._rowEnd(row, row === endRow) - } - - this._afterSerialize() - - return this._serializeString(excludeFinalCursorPosition) - } - - protected _nextCell(_cell: IBufferCell, _oldCell: IBufferCell, _row: number, _col: number): void {} - protected _rowEnd(_row: number, _isLastRow: boolean): void {} - protected _beforeSerialize(_rows: number, _startRow: number, _endRow: number): void {} - protected _afterSerialize(): void {} - protected _serializeString(_excludeFinalCursorPosition?: boolean): string { - return "" - } -} - -// ============================================================================ -// String Serialize Handler -// ============================================================================ - -class StringSerializeHandler extends BaseSerializeHandler { - private _rowIndex: number = 0 - private _allRows: string[] = [] - private _allRowSeparators: string[] = [] - private _currentRow: string = "" - private _nullCellCount: number = 0 - private _cursorStyle: IBufferCell - private _firstRow: number = 0 - private _lastCursorRow: number = 0 - private _lastCursorCol: number = 0 - private _lastContentCursorRow: number = 0 - private _lastContentCursorCol: number = 0 - - constructor( - buffer: IBuffer, - private readonly _terminal: ITerminalCore, - ) { - super(buffer) - this._cursorStyle = this._buffer.getNullCell() - } - - protected _beforeSerialize(rows: number, start: number, _end: number): void { - this._allRows = Array.from({ length: rows }) - this._allRowSeparators = Array.from({ length: rows }) - this._rowIndex = 0 - - this._currentRow = "" - this._nullCellCount = 0 - this._cursorStyle = this._buffer.getNullCell() - - this._lastContentCursorRow = start - this._lastCursorRow = start - this._firstRow = start - } - - protected _rowEnd(row: number, isLastRow: boolean): void { - let rowSeparator = "" - - const nextLine = isLastRow ? undefined : this._buffer.getLine(row + 1) - const wrapped = !!nextLine?.isWrapped - - if (this._nullCellCount > 0 && wrapped) { - this._currentRow += " ".repeat(this._nullCellCount) - } - - this._nullCellCount = 0 - - if (!isLastRow && !wrapped) { - rowSeparator = "\r\n" - this._lastCursorRow = row + 1 - this._lastCursorCol = 0 - } - - this._allRows[this._rowIndex] = this._currentRow - this._allRowSeparators[this._rowIndex++] = rowSeparator - this._currentRow = "" - this._nullCellCount = 0 - } - - private _diffStyle(cell: IBufferCell, oldCell: IBufferCell): number[] { - const sgrSeq: number[] = [] - const fgChanged = !equalFg(cell, oldCell) - const bgChanged = !equalBg(cell, oldCell) - const flagsChanged = !equalFlags(cell, oldCell) - - if (fgChanged || bgChanged || flagsChanged) { - if (this._isAttributeDefault(cell)) { - if (!this._isAttributeDefault(oldCell)) { - sgrSeq.push(0) - } - } else { - if (flagsChanged) { - if (!!cell.isInverse() !== !!oldCell.isInverse()) { - sgrSeq.push(cell.isInverse() ? 7 : 27) - } - if (!!cell.isBold() !== !!oldCell.isBold()) { - sgrSeq.push(cell.isBold() ? 1 : 22) - } - if (!!cell.isUnderline() !== !!oldCell.isUnderline()) { - sgrSeq.push(cell.isUnderline() ? 4 : 24) - } - if (!!cell.isBlink() !== !!oldCell.isBlink()) { - sgrSeq.push(cell.isBlink() ? 5 : 25) - } - if (!!cell.isInvisible() !== !!oldCell.isInvisible()) { - sgrSeq.push(cell.isInvisible() ? 8 : 28) - } - if (!!cell.isItalic() !== !!oldCell.isItalic()) { - sgrSeq.push(cell.isItalic() ? 3 : 23) - } - if (!!cell.isDim() !== !!oldCell.isDim()) { - sgrSeq.push(cell.isDim() ? 2 : 22) - } - if (!!cell.isStrikethrough() !== !!oldCell.isStrikethrough()) { - sgrSeq.push(cell.isStrikethrough() ? 9 : 29) - } - } - if (fgChanged) { - const color = cell.getFgColor() - const mode = cell.getFgColorMode() - if (mode === 2 || mode === 3 || mode === -1) { - sgrSeq.push(38, 2, (color >>> 16) & 0xff, (color >>> 8) & 0xff, color & 0xff) - } else if (mode === 1) { - // Palette - if (color >= 16) { - sgrSeq.push(38, 5, color) - } else { - sgrSeq.push(color & 8 ? 90 + (color & 7) : 30 + (color & 7)) - } - } else { - sgrSeq.push(39) - } - } - if (bgChanged) { - const color = cell.getBgColor() - const mode = cell.getBgColorMode() - if (mode === 2 || mode === 3 || mode === -1) { - sgrSeq.push(48, 2, (color >>> 16) & 0xff, (color >>> 8) & 0xff, color & 0xff) - } else if (mode === 1) { - // Palette - if (color >= 16) { - sgrSeq.push(48, 5, color) - } else { - sgrSeq.push(color & 8 ? 100 + (color & 7) : 40 + (color & 7)) - } - } else { - sgrSeq.push(49) - } - } - } - } - - return sgrSeq - } - - private _isAttributeDefault(cell: IBufferCell): boolean { - const mode = cell.getFgColorMode() - const bgMode = cell.getBgColorMode() - - if (mode === 0 && bgMode === 0) { - return ( - !cell.isBold() && - !cell.isItalic() && - !cell.isUnderline() && - !cell.isBlink() && - !cell.isInverse() && - !cell.isInvisible() && - !cell.isDim() && - !cell.isStrikethrough() - ) - } - - const fgColor = cell.getFgColor() - const bgColor = cell.getBgColor() - const nullCell = this._buffer.getNullCell() - const nullFg = nullCell.getFgColor() - const nullBg = nullCell.getBgColor() - - return ( - fgColor === nullFg && - bgColor === nullBg && - !cell.isBold() && - !cell.isItalic() && - !cell.isUnderline() && - !cell.isBlink() && - !cell.isInverse() && - !cell.isInvisible() && - !cell.isDim() && - !cell.isStrikethrough() - ) - } - - protected _nextCell(cell: IBufferCell, _oldCell: IBufferCell, row: number, col: number): void { - const isPlaceHolderCell = cell.getWidth() === 0 - - if (isPlaceHolderCell) { - return - } - - const codepoint = cell.getCode() - const isInvalidCodepoint = codepoint > 0x10ffff || (codepoint >= 0xd800 && codepoint <= 0xdfff) - const isGarbage = isInvalidCodepoint || (codepoint >= 0xf000 && cell.getWidth() === 1) - const isEmptyCell = codepoint === 0 || cell.getChars() === "" || isGarbage - - const sgrSeq = this._diffStyle(cell, this._cursorStyle) - - const styleChanged = sgrSeq.length > 0 - - if (styleChanged) { - if (this._nullCellCount > 0) { - this._currentRow += " ".repeat(this._nullCellCount) - this._nullCellCount = 0 - } - - this._lastContentCursorRow = this._lastCursorRow = row - this._lastContentCursorCol = this._lastCursorCol = col - - this._currentRow += `\u001b[${sgrSeq.join(";")}m` - - const line = this._buffer.getLine(row) - const cellFromLine = line?.getCell(col) - if (cellFromLine) { - this._cursorStyle = cellFromLine - } - } - - if (isEmptyCell) { - this._nullCellCount += cell.getWidth() - } else { - if (this._nullCellCount > 0) { - this._currentRow += " ".repeat(this._nullCellCount) - this._nullCellCount = 0 - } - - this._currentRow += cell.getChars() - - this._lastContentCursorRow = this._lastCursorRow = row - this._lastContentCursorCol = this._lastCursorCol = col + cell.getWidth() - } - } - - protected _serializeString(excludeFinalCursorPosition?: boolean): string { - let rowEnd = this._allRows.length - - if (this._buffer.length - this._firstRow <= this._terminal.rows) { - rowEnd = this._lastContentCursorRow + 1 - this._firstRow - this._lastCursorCol = this._lastContentCursorCol - this._lastCursorRow = this._lastContentCursorRow - } - - let content = "" - - for (let i = 0; i < rowEnd; i++) { - content += this._allRows[i] - if (i + 1 < rowEnd) { - content += this._allRowSeparators[i] - } - } - - if (excludeFinalCursorPosition) return content - - const absoluteCursorRow = (this._buffer.baseY ?? 0) + this._buffer.cursorY - const cursorRow = constrain(absoluteCursorRow - this._firstRow + 1, 1, Number.MAX_SAFE_INTEGER) - const cursorCol = this._buffer.cursorX + 1 - content += `\u001b[${cursorRow};${cursorCol}H` - - const line = this._buffer.getLine(absoluteCursorRow) - const cell = line?.getCell(this._buffer.cursorX) - const style = (() => { - if (!cell) return this._buffer.getNullCell() - if (cell.getWidth() !== 0) return cell - if (this._buffer.cursorX > 0) return line?.getCell(this._buffer.cursorX - 1) ?? cell - return cell - })() - - const sgrSeq = this._diffStyle(style, this._cursorStyle) - if (sgrSeq.length) content += `\u001b[${sgrSeq.join(";")}m` - - return content - } -} - -// ============================================================================ -// SerializeAddon Class -// ============================================================================ - -export class SerializeAddon implements ITerminalAddon { - private _terminal?: ITerminalCore - - /** - * Activate the addon (called by Terminal.loadAddon) - */ - public activate(terminal: ITerminalCore): void { - this._terminal = terminal - } - - /** - * Dispose the addon and clean up resources - */ - public dispose(): void { - this._terminal = undefined - } - - /** - * Serializes terminal rows into a string that can be written back to the - * terminal to restore the state. The cursor will also be positioned to the - * correct cell. - * - * @param options Custom options to allow control over what gets serialized. - */ - public serialize(options?: ISerializeOptions): string { - if (!this._terminal) { - throw new Error("Cannot use addon until it has been loaded") - } - - const buffer = getTerminalBuffers(this._terminal) - - if (!buffer) { - return "" - } - - const normalBuffer = buffer.normal ?? buffer.active - const altBuffer = buffer.alternate - - if (!normalBuffer) { - return "" - } - - let content = options?.range - ? this._serializeBufferByRange(normalBuffer, options.range, true) - : this._serializeBufferByScrollback(normalBuffer, options?.scrollback) - - if (!options?.excludeAltBuffer && buffer.active?.type === "alternate" && altBuffer) { - const alternateContent = this._serializeBufferByScrollback(altBuffer, undefined) - content += `\u001b[?1049h\u001b[H${alternateContent}` - } - - return content - } - - /** - * Serializes terminal content as plain text (no escape sequences) - * @param options Custom options to allow control over what gets serialized. - */ - public serializeAsText(options?: { scrollback?: number; trimWhitespace?: boolean }): string { - if (!this._terminal) { - throw new Error("Cannot use addon until it has been loaded") - } - - const buffer = getTerminalBuffers(this._terminal) - - if (!buffer) { - return "" - } - - const activeBuffer = buffer.active ?? buffer.normal - if (!activeBuffer) { - return "" - } - - const maxRows = activeBuffer.length - const scrollback = options?.scrollback - const correctRows = scrollback === undefined ? maxRows : constrain(scrollback + this._terminal.rows, 0, maxRows) - - const startRow = maxRows - correctRows - const endRow = maxRows - 1 - const lines: string[] = [] - - for (let row = startRow; row <= endRow; row++) { - const line = activeBuffer.getLine(row) - if (line) { - const text = line.translateToString(options?.trimWhitespace ?? true) - lines.push(text) - } - } - - // Trim trailing empty lines if requested - if (options?.trimWhitespace) { - while (lines.length > 0 && lines[lines.length - 1] === "") { - lines.pop() - } - } - - return lines.join("\n") - } - - private _serializeBufferByScrollback(buffer: IBuffer, scrollback?: number): string { - const maxRows = buffer.length - const rows = this._terminal?.rows ?? 24 - const correctRows = scrollback === undefined ? maxRows : constrain(scrollback + rows, 0, maxRows) - return this._serializeBufferByRange( - buffer, - { - start: maxRows - correctRows, - end: maxRows - 1, - }, - false, - ) - } - - private _serializeBufferByRange( - buffer: IBuffer, - range: ISerializeRange, - excludeFinalCursorPosition: boolean, - ): string { - const handler = new StringSerializeHandler(buffer, this._terminal!) - const cols = this._terminal?.cols ?? 80 - return handler.serialize( - { - start: { x: 0, y: range.start }, - end: { x: cols, y: range.end }, - }, - excludeFinalCursorPosition, - ) - } -} diff --git a/packages/app/src/components/debug-bar.tsx b/packages/app/src/components/debug-bar.tsx deleted file mode 100644 index 11f9f59e4e..0000000000 --- a/packages/app/src/components/debug-bar.tsx +++ /dev/null @@ -1,443 +0,0 @@ -import { useIsRouting, useLocation } from "@solidjs/router" -import { batch, createEffect, onCleanup, onMount } from "solid-js" -import { createStore } from "solid-js/store" -import { makeEventListener } from "@solid-primitives/event-listener" -import { Tooltip } from "@opencode-ai/ui/tooltip" -import { useLanguage } from "@/context/language" - -type Mem = Performance & { - memory?: { - usedJSHeapSize: number - jsHeapSizeLimit: number - } -} - -type Evt = PerformanceEntry & { - interactionId?: number - processingStart?: number -} - -type Shift = PerformanceEntry & { - hadRecentInput: boolean - value: number -} - -type Obs = PerformanceObserverInit & { - durationThreshold?: number -} - -const span = 5000 - -const ms = (n?: number, d = 0) => { - if (n === undefined || Number.isNaN(n)) return - return `${n.toFixed(d)}ms` -} - -const time = (n?: number) => { - if (n === undefined || Number.isNaN(n)) return - return `${Math.round(n)}` -} - -const mb = (n?: number) => { - if (n === undefined || Number.isNaN(n)) return - const v = n / 1024 / 1024 - return `${v >= 1024 ? v.toFixed(0) : v.toFixed(1)}MB` -} - -const bad = (n: number | undefined, limit: number, low = false) => { - if (n === undefined || Number.isNaN(n)) return false - return low ? n < limit : n > limit -} - -const session = (path: string) => path.includes("/session") - -function Cell(props: { bad?: boolean; dim?: boolean; label: string; tip: string; value: string; wide?: boolean }) { - return ( - -
-
{props.label}
-
- {props.value} -
-
-
- ) -} - -export function DebugBar() { - const language = useLanguage() - const location = useLocation() - const routing = useIsRouting() - const [state, setState] = createStore({ - cls: undefined as number | undefined, - delay: undefined as number | undefined, - fps: undefined as number | undefined, - gap: undefined as number | undefined, - heap: { - limit: undefined as number | undefined, - used: undefined as number | undefined, - }, - inp: undefined as number | undefined, - jank: undefined as number | undefined, - long: { - block: undefined as number | undefined, - count: undefined as number | undefined, - max: undefined as number | undefined, - }, - nav: { - dur: undefined as number | undefined, - pending: false, - }, - }) - - const na = () => language.t("debugBar.na") - const heap = () => (state.heap.limit ? (state.heap.used ?? 0) / state.heap.limit : undefined) - const heapv = () => { - const value = heap() - if (value === undefined) return na() - return `${Math.round(value * 100)}%` - } - const longv = () => (state.long.count === undefined ? na() : `${time(state.long.block) ?? na()}/${state.long.count}`) - const navv = () => (state.nav.pending ? "..." : (time(state.nav.dur) ?? na())) - - let prev = "" - let start = 0 - let init = false - let one = 0 - let two = 0 - - createEffect(() => { - const busy = routing() - const next = `${location.pathname}${location.search}` - - if (!init) { - init = true - prev = next - return - } - - if (busy) { - if (one !== 0) cancelAnimationFrame(one) - if (two !== 0) cancelAnimationFrame(two) - one = 0 - two = 0 - if (start !== 0) return - start = performance.now() - if (session(prev)) setState("nav", { dur: undefined, pending: true }) - return - } - - if (start === 0) { - prev = next - return - } - - const at = start - const from = prev - start = 0 - prev = next - - if (!(session(from) || session(next))) return - - if (one !== 0) cancelAnimationFrame(one) - if (two !== 0) cancelAnimationFrame(two) - one = requestAnimationFrame(() => { - one = 0 - two = requestAnimationFrame(() => { - two = 0 - setState("nav", { dur: performance.now() - at, pending: false }) - }) - }) - }) - - onMount(() => { - const obs: PerformanceObserver[] = [] - const fps: Array<{ at: number; dur: number }> = [] - const long: Array<{ at: number; dur: number }> = [] - const seen = new Map() - let hasLong = false - let poll: number | undefined - let raf = 0 - let last = 0 - let snap = 0 - - const trim = (list: Array<{ at: number; dur: number }>, span: number, at: number) => { - while (list[0] && at - list[0].at > span) list.shift() - } - - const syncFrame = (at: number) => { - trim(fps, span, at) - const total = fps.reduce((sum, entry) => sum + entry.dur, 0) - const gap = fps.reduce((max, entry) => Math.max(max, entry.dur), 0) - const jank = fps.filter((entry) => entry.dur > 32).length - batch(() => { - setState("fps", total > 0 ? (fps.length * 1000) / total : undefined) - setState("gap", gap > 0 ? gap : undefined) - setState("jank", jank) - }) - } - - const syncLong = (at = performance.now()) => { - if (!hasLong) return - trim(long, span, at) - const block = long.reduce((sum, entry) => sum + Math.max(0, entry.dur - 50), 0) - const max = long.reduce((hi, entry) => Math.max(hi, entry.dur), 0) - setState("long", { block, count: long.length, max }) - } - - const syncInp = (at = performance.now()) => { - for (const [key, entry] of seen) { - if (at - entry.at > span) seen.delete(key) - } - let delay = 0 - let inp = 0 - for (const entry of seen.values()) { - delay = Math.max(delay, entry.delay) - inp = Math.max(inp, entry.dur) - } - batch(() => { - setState("delay", delay > 0 ? delay : undefined) - setState("inp", inp > 0 ? inp : undefined) - }) - } - - const syncHeap = () => { - const mem = (performance as Mem).memory - if (!mem) return - setState("heap", { limit: mem.jsHeapSizeLimit, used: mem.usedJSHeapSize }) - } - - const reset = () => { - fps.length = 0 - long.length = 0 - seen.clear() - last = 0 - snap = 0 - batch(() => { - setState("fps", undefined) - setState("gap", undefined) - setState("jank", undefined) - setState("delay", undefined) - setState("inp", undefined) - if (hasLong) setState("long", { block: 0, count: 0, max: 0 }) - }) - } - - const watch = (type: string, init: Obs, fn: (entries: PerformanceEntry[]) => void) => { - if (typeof PerformanceObserver === "undefined") return false - if (!(PerformanceObserver.supportedEntryTypes ?? []).includes(type)) return false - const ob = new PerformanceObserver((list) => fn(list.getEntries())) - try { - ob.observe(init) - obs.push(ob) - return true - } catch { - ob.disconnect() - return false - } - } - - if ( - watch("layout-shift", { buffered: true, type: "layout-shift" }, (entries) => { - const add = entries.reduce((sum, entry) => { - const item = entry as Shift - if (item.hadRecentInput) return sum - return sum + item.value - }, 0) - if (add === 0) return - setState("cls", (value) => (value ?? 0) + add) - }) - ) { - setState("cls", 0) - } - - if ( - watch("longtask", { buffered: true, type: "longtask" }, (entries) => { - const at = performance.now() - long.push(...entries.map((entry) => ({ at: entry.startTime, dur: entry.duration }))) - syncLong(at) - }) - ) { - hasLong = true - setState("long", { block: 0, count: 0, max: 0 }) - } - - watch("event", { buffered: true, durationThreshold: 16, type: "event" }, (entries) => { - for (const raw of entries) { - const entry = raw as Evt - if (entry.duration < 16) continue - const key = - entry.interactionId && entry.interactionId > 0 - ? entry.interactionId - : `${entry.name}:${Math.round(entry.startTime)}` - const prev = seen.get(key) - const delay = Math.max(0, (entry.processingStart ?? entry.startTime) - entry.startTime) - seen.set(key, { - at: entry.startTime, - delay: Math.max(prev?.delay ?? 0, delay), - dur: Math.max(prev?.dur ?? 0, entry.duration), - }) - if (seen.size <= 200) continue - const first = seen.keys().next().value - if (first !== undefined) seen.delete(first) - } - syncInp() - }) - - const loop = (at: number) => { - if (document.visibilityState !== "visible") { - raf = 0 - return - } - - if (last === 0) { - last = at - raf = requestAnimationFrame(loop) - return - } - - fps.push({ at, dur: at - last }) - last = at - - if (at - snap >= 250) { - snap = at - syncFrame(at) - } - - raf = requestAnimationFrame(loop) - } - - const stop = () => { - if (raf !== 0) cancelAnimationFrame(raf) - raf = 0 - if (poll === undefined) return - clearInterval(poll) - poll = undefined - } - - const start = () => { - if (document.visibilityState !== "visible") return - if (poll === undefined) { - poll = window.setInterval(() => { - syncLong() - syncInp() - syncHeap() - }, 1000) - } - if (raf !== 0) return - raf = requestAnimationFrame(loop) - } - - const vis = () => { - if (document.visibilityState !== "visible") { - stop() - return - } - reset() - start() - } - - syncHeap() - start() - makeEventListener(document, "visibilitychange", vis) - - onCleanup(() => { - if (one !== 0) cancelAnimationFrame(one) - if (two !== 0) cancelAnimationFrame(two) - stop() - for (const ob of obs) ob.disconnect() - }) - }) - - return ( - - ) -} diff --git a/packages/app/src/components/dialog-connect-provider.tsx b/packages/app/src/components/dialog-connect-provider.tsx deleted file mode 100644 index 4822fa7266..0000000000 --- a/packages/app/src/components/dialog-connect-provider.tsx +++ /dev/null @@ -1,656 +0,0 @@ -import type { ProviderAuthAuthorization, ProviderAuthMethod } from "@kilocode/sdk/v2/client" -import { Button } from "@opencode-ai/ui/button" -import { useDialog } from "@opencode-ai/ui/context/dialog" -import { Dialog } from "@opencode-ai/ui/dialog" -import { Icon } from "@opencode-ai/ui/icon" -import { IconButton } from "@opencode-ai/ui/icon-button" -import { List, type ListRef } from "@opencode-ai/ui/list" -import { ProviderIcon } from "@opencode-ai/ui/provider-icon" -import { Spinner } from "@opencode-ai/ui/spinner" -import { TextField } from "@opencode-ai/ui/text-field" -import { showToast } from "@opencode-ai/ui/toast" -import { createEffect, createMemo, createResource, Match, onCleanup, onMount, Switch } from "solid-js" -import { createStore, produce } from "solid-js/store" -import { Link } from "@/components/link" -import { useGlobalSDK } from "@/context/global-sdk" -import { useGlobalSync } from "@/context/global-sync" -import { useLanguage } from "@/context/language" -import { useProviders } from "@/hooks/use-providers" - -export function DialogConnectProvider(props: { provider: string }) { - const dialog = useDialog() - const globalSync = useGlobalSync() - const globalSDK = useGlobalSDK() - const language = useLanguage() - const providers = useProviders() - - const all = () => { - void import("./dialog-select-provider").then((x) => { - dialog.show(() => ) - }) - } - - const alive = { value: true } - const timer = { current: undefined as ReturnType | undefined } - - onCleanup(() => { - alive.value = false - if (timer.current === undefined) return - clearTimeout(timer.current) - timer.current = undefined - }) - - const provider = createMemo( - () => - providers.all().find((x) => x.id === props.provider) ?? - globalSync.data.provider.all.find((x) => x.id === props.provider)!, - ) - const fallback = createMemo(() => [ - { - type: "api" as const, - label: language.t("provider.connect.method.apiKey"), - }, - ]) - const [auth] = createResource( - () => props.provider, - async () => { - const cached = globalSync.data.provider_auth[props.provider] - if (cached) return cached - const res = await globalSDK.client.provider.auth() - if (!alive.value) return fallback() - globalSync.set("provider_auth", res.data ?? {}) - return res.data?.[props.provider] ?? fallback() - }, - ) - const loading = createMemo(() => auth.loading && !globalSync.data.provider_auth[props.provider]) - const methods = createMemo(() => auth.latest ?? globalSync.data.provider_auth[props.provider] ?? fallback()) - const [store, setStore] = createStore({ - methodIndex: undefined as undefined | number, - authorization: undefined as undefined | ProviderAuthAuthorization, - state: "pending" as undefined | "pending" | "complete" | "error" | "prompt", - error: undefined as string | undefined, - }) - - type Action = - | { type: "method.select"; index: number } - | { type: "method.reset" } - | { type: "auth.prompt" } - | { type: "auth.pending" } - | { type: "auth.complete"; authorization: ProviderAuthAuthorization } - | { type: "auth.error"; error: string } - - function dispatch(action: Action) { - setStore( - produce((draft) => { - if (action.type === "method.select") { - draft.methodIndex = action.index - draft.authorization = undefined - draft.state = undefined - draft.error = undefined - return - } - if (action.type === "method.reset") { - draft.methodIndex = undefined - draft.authorization = undefined - draft.state = undefined - draft.error = undefined - return - } - if (action.type === "auth.prompt") { - draft.state = "prompt" - draft.error = undefined - return - } - if (action.type === "auth.pending") { - draft.state = "pending" - draft.error = undefined - return - } - if (action.type === "auth.complete") { - draft.state = "complete" - draft.authorization = action.authorization - draft.error = undefined - return - } - draft.state = "error" - draft.error = action.error - }), - ) - } - - const method = createMemo(() => (store.methodIndex !== undefined ? methods().at(store.methodIndex!) : undefined)) - - const methodLabel = (value?: { type?: string; label?: string }) => { - if (!value) return "" - if (value.type === "api") return language.t("provider.connect.method.apiKey") - return value.label ?? "" - } - - function formatError(value: unknown, fallback: string): string { - if (value && typeof value === "object" && "data" in value) { - const data = (value as { data?: { message?: unknown } }).data - if (typeof data?.message === "string" && data.message) return data.message - } - if (value && typeof value === "object" && "error" in value) { - const nested = formatError((value as { error?: unknown }).error, "") - if (nested) return nested - } - if (value && typeof value === "object" && "message" in value) { - const message = (value as { message?: unknown }).message - if (typeof message === "string" && message) return message - } - if (value instanceof Error && value.message) return value.message - if (typeof value === "string" && value) return value - return fallback - } - - async function selectMethod(index: number, inputs?: Record) { - if (timer.current !== undefined) { - clearTimeout(timer.current) - timer.current = undefined - } - - const method = methods()[index] - dispatch({ type: "method.select", index }) - - if (method.type === "oauth") { - if (method.prompts?.length && !inputs) { - dispatch({ type: "auth.prompt" }) - return - } - dispatch({ type: "auth.pending" }) - const start = Date.now() - await globalSDK.client.provider.oauth - .authorize( - { - providerID: props.provider, - method: index, - inputs, - }, - { throwOnError: true }, - ) - .then((x) => { - if (!alive.value) return - const elapsed = Date.now() - start - const delay = 1000 - elapsed - - if (delay > 0) { - if (timer.current !== undefined) clearTimeout(timer.current) - timer.current = setTimeout(() => { - timer.current = undefined - if (!alive.value) return - dispatch({ type: "auth.complete", authorization: x.data! }) - }, delay) - return - } - dispatch({ type: "auth.complete", authorization: x.data! }) - }) - .catch((e) => { - if (!alive.value) return - dispatch({ type: "auth.error", error: formatError(e, language.t("common.requestFailed")) }) - }) - } - } - - function OAuthPromptsView() { - const [formStore, setFormStore] = createStore({ - value: {} as Record, - index: 0, - }) - - const prompts = createMemo>(() => { - const value = method() - if (value?.type !== "oauth") return [] - return value.prompts ?? [] - }) - const matches = (prompt: NonNullable[number]>, value: Record) => { - if (!prompt.when) return true - const actual = value[prompt.when.key] - if (actual === undefined) return false - return prompt.when.op === "eq" ? actual === prompt.when.value : actual !== prompt.when.value - } - const current = createMemo(() => { - const all = prompts() - const index = all.findIndex((prompt, index) => index >= formStore.index && matches(prompt, formStore.value)) - if (index === -1) return - return { - index, - prompt: all[index], - } - }) - const valid = createMemo(() => { - const item = current() - if (!item || item.prompt.type !== "text") return false - const value = formStore.value[item.prompt.key] ?? "" - return value.trim().length > 0 - }) - - async function next(index: number, value: Record) { - if (store.methodIndex === undefined) return - const next = prompts().findIndex((prompt, i) => i > index && matches(prompt, value)) - if (next !== -1) { - setFormStore("index", next) - return - } - await selectMethod(store.methodIndex, value) - } - - async function handleSubmit(e: SubmitEvent) { - e.preventDefault() - const item = current() - if (!item || item.prompt.type !== "text") return - if (!valid()) return - await next(item.index, formStore.value) - } - - const item = () => current() - const text = createMemo(() => { - const prompt = item()?.prompt - if (!prompt || prompt.type !== "text") return - return prompt - }) - const select = createMemo(() => { - const prompt = item()?.prompt - if (!prompt || prompt.type !== "select") return - return prompt - }) - - return ( -
- - - { - const prompt = text() - if (!prompt) return - setFormStore("value", prompt.key, value) - }} - /> - - - -
-
{select()?.message}
-
- x.value} - current={select()?.options.find((x) => x.value === formStore.value[select()!.key])} - onSelect={(value) => { - if (!value) return - const prompt = select() - if (!prompt) return - const nextValue = { - ...formStore.value, - [prompt.key]: value.value, - } - setFormStore("value", prompt.key, value.value) - void next(item()!.index, nextValue) - }} - > - {(option) => ( -
-
- - {option.label} - {option.hint} -
- )} - -
-
- - - - ) - } - - let listRef: ListRef | undefined - function handleKey(e: KeyboardEvent) { - if (e.key === "Enter" && e.target instanceof HTMLInputElement) { - return - } - if (e.key === "Escape") return - listRef?.onKeyDown(e) - } - - let auto = false - createEffect(() => { - if (auto) return - if (loading()) return - if (methods().length === 1) { - auto = true - void selectMethod(0) - } - }) - - async function complete() { - await globalSDK.client.global.dispose() - dialog.close() - showToast({ - variant: "success", - icon: "circle-check", - title: language.t("provider.connect.toast.connected.title", { provider: provider().name }), - description: language.t("provider.connect.toast.connected.description", { provider: provider().name }), - }) - } - - function goBack() { - if (methods().length === 1) { - all() - return - } - if (store.authorization) { - dispatch({ type: "method.reset" }) - return - } - if (store.methodIndex !== undefined) { - dispatch({ type: "method.reset" }) - return - } - all() - } - - function MethodSelection() { - return ( - <> -
- {language.t("provider.connect.selectMethod", { provider: provider().name })} -
-
- { - listRef = ref - }} - items={methods} - key={(m) => m?.label} - onSelect={async (selected, index) => { - if (!selected) return - void selectMethod(index) - }} - > - {(i) => ( -
-
- - {methodLabel(i)} -
- )} - -
- - ) - } - - function ApiAuthView() { - const [formStore, setFormStore] = createStore({ - value: "", - error: undefined as string | undefined, - }) - - async function handleSubmit(e: SubmitEvent) { - e.preventDefault() - - const form = e.currentTarget as HTMLFormElement - const formData = new FormData(form) - const apiKey = formData.get("apiKey") as string - - if (!apiKey?.trim()) { - setFormStore("error", language.t("provider.connect.apiKey.required")) - return - } - - setFormStore("error", undefined) - await globalSDK.client.auth.set({ - providerID: props.provider, - auth: { - type: "api", - key: apiKey, - }, - }) - await complete() - } - - return ( -
- - {/* kilocode_change start */} - -
-
{language.t("provider.connect.kiloGateway.line1")}
-
{language.t("provider.connect.kiloGateway.line2")}
-
- {language.t("provider.connect.kiloGateway.visit.prefix")} - - {language.t("provider.connect.kiloGateway.visit.link")} - - {language.t("provider.connect.kiloGateway.visit.suffix")} -
-
-
- {/* kilocode_change end */} - -
- {language.t("provider.connect.apiKey.description", { provider: provider().name })} -
-
-
-
- setFormStore("value", v)} - validationState={formStore.error ? "invalid" : undefined} - error={formStore.error} - /> - - -
- ) - } - - function OAuthCodeView() { - const [formStore, setFormStore] = createStore({ - value: "", - error: undefined as string | undefined, - }) - - async function handleSubmit(e: SubmitEvent) { - e.preventDefault() - - const form = e.currentTarget as HTMLFormElement - const formData = new FormData(form) - const code = formData.get("code") as string - - if (!code?.trim()) { - setFormStore("error", language.t("provider.connect.oauth.code.required")) - return - } - - setFormStore("error", undefined) - const result = await globalSDK.client.provider.oauth - .callback({ - providerID: props.provider, - method: store.methodIndex, - code, - }) - .then((value) => (value.error ? { ok: false as const, error: value.error } : { ok: true as const })) - .catch((error) => ({ ok: false as const, error })) - if (result.ok) { - await complete() - return - } - setFormStore("error", formatError(result.error, language.t("provider.connect.oauth.code.invalid"))) - } - - return ( -
-
- {language.t("provider.connect.oauth.code.visit.prefix")} - {language.t("provider.connect.oauth.code.visit.link")} - {language.t("provider.connect.oauth.code.visit.suffix", { provider: provider().name })} -
-
- setFormStore("value", v)} - validationState={formStore.error ? "invalid" : undefined} - error={formStore.error} - /> - - -
- ) - } - - function OAuthAutoView() { - const code = createMemo(() => { - const instructions = store.authorization?.instructions - if (instructions?.includes(":")) { - return instructions.split(":")[1]?.trim() - } - return instructions - }) - - onMount(() => { - void (async () => { - const result = await globalSDK.client.provider.oauth - .callback({ - providerID: props.provider, - method: store.methodIndex, - }) - .then((value) => (value.error ? { ok: false as const, error: value.error } : { ok: true as const })) - .catch((error) => ({ ok: false as const, error })) - - if (!alive.value) return - - if (!result.ok) { - const message = formatError(result.error, language.t("common.requestFailed")) - dispatch({ type: "auth.error", error: message }) - return - } - - await complete() - })() - }) - - return ( -
-
- {language.t("provider.connect.oauth.auto.visit.prefix")} - {language.t("provider.connect.oauth.auto.visit.link")} - {language.t("provider.connect.oauth.auto.visit.suffix", { provider: provider().name })} -
- -
- - {language.t("provider.connect.status.waiting")} -
-
- ) - } - - return ( - - } - > -
-
- -
- - - {language.t("provider.connect.title.anthropicProMax")} - - {language.t("provider.connect.title", { provider: provider().name })} - -
-
-
-
- - -
-
- - {language.t("provider.connect.status.inProgress")} -
-
-
- - - - -
-
- - {language.t("provider.connect.status.inProgress")} -
-
-
- - - - -
-
- - {language.t("provider.connect.status.failed", { error: store.error ?? "" })} -
-
-
- - - - - - - - - - - - - -
-
-
-
-
- ) -} diff --git a/packages/app/src/components/dialog-custom-provider-form.ts b/packages/app/src/components/dialog-custom-provider-form.ts deleted file mode 100644 index e26dcb0971..0000000000 --- a/packages/app/src/components/dialog-custom-provider-form.ts +++ /dev/null @@ -1,158 +0,0 @@ -const PROVIDER_ID = /^[a-z0-9][a-z0-9-_]*$/ -const OPENAI_COMPATIBLE = "@ai-sdk/openai-compatible" - -type Translator = (key: string, vars?: Record) => string - -export type ModelErr = { - id?: string - name?: string -} - -export type HeaderErr = { - key?: string - value?: string -} - -export type ModelRow = { - row: string - id: string - name: string - err: ModelErr -} - -export type HeaderRow = { - row: string - key: string - value: string - err: HeaderErr -} - -export type FormState = { - providerID: string - name: string - baseURL: string - apiKey: string - models: ModelRow[] - headers: HeaderRow[] - err: { - providerID?: string - name?: string - baseURL?: string - } -} - -type ValidateArgs = { - form: FormState - t: Translator - disabledProviders: string[] - existingProviderIDs: Set -} - -export function validateCustomProvider(input: ValidateArgs) { - const providerID = input.form.providerID.trim() - const name = input.form.name.trim() - const baseURL = input.form.baseURL.trim() - const apiKey = input.form.apiKey.trim() - - const env = apiKey.match(/^\{env:([^}]+)\}$/)?.[1]?.trim() - const key = apiKey && !env ? apiKey : undefined - - const idError = !providerID - ? input.t("provider.custom.error.providerID.required") - : !PROVIDER_ID.test(providerID) - ? input.t("provider.custom.error.providerID.format") - : undefined - - const nameError = !name ? input.t("provider.custom.error.name.required") : undefined - const urlError = !baseURL - ? input.t("provider.custom.error.baseURL.required") - : !/^https?:\/\//.test(baseURL) - ? input.t("provider.custom.error.baseURL.format") - : undefined - - const disabled = input.disabledProviders.includes(providerID) - const existsError = idError - ? undefined - : input.existingProviderIDs.has(providerID) && !disabled - ? input.t("provider.custom.error.providerID.exists") - : undefined - - const seenModels = new Set() - const models = input.form.models.map((m) => { - const id = m.id.trim() - const idError = !id - ? input.t("provider.custom.error.required") - : seenModels.has(id) - ? input.t("provider.custom.error.duplicate") - : (() => { - seenModels.add(id) - return undefined - })() - const nameError = !m.name.trim() ? input.t("provider.custom.error.required") : undefined - return { id: idError, name: nameError } - }) - const modelsValid = models.every((m) => !m.id && !m.name) - const modelConfig = Object.fromEntries(input.form.models.map((m) => [m.id.trim(), { name: m.name.trim() }])) - - const seenHeaders = new Set() - const headers = input.form.headers.map((h) => { - const key = h.key.trim() - const value = h.value.trim() - - if (!key && !value) return {} - const keyError = !key - ? input.t("provider.custom.error.required") - : seenHeaders.has(key.toLowerCase()) - ? input.t("provider.custom.error.duplicate") - : (() => { - seenHeaders.add(key.toLowerCase()) - return undefined - })() - const valueError = !value ? input.t("provider.custom.error.required") : undefined - return { key: keyError, value: valueError } - }) - const headersValid = headers.every((h) => !h.key && !h.value) - const headerConfig = Object.fromEntries( - input.form.headers - .map((h) => ({ key: h.key.trim(), value: h.value.trim() })) - .filter((h) => !!h.key && !!h.value) - .map((h) => [h.key, h.value]), - ) - - const err = { - providerID: idError ?? existsError, - name: nameError, - baseURL: urlError, - } - - const ok = !idError && !existsError && !nameError && !urlError && modelsValid && headersValid - if (!ok) return { err, models, headers } - - return { - err, - models, - headers, - result: { - providerID, - name, - key, - config: { - npm: OPENAI_COMPATIBLE, - name, - ...(env ? { env: [env] } : {}), - options: { - baseURL, - ...(Object.keys(headerConfig).length ? { headers: headerConfig } : {}), - }, - models: modelConfig, - }, - }, - } -} - -let row = 0 - -const nextRow = () => `row-${row++}` - -export const modelRow = (): ModelRow => ({ row: nextRow(), id: "", name: "", err: {} }) -export const headerRow = (): HeaderRow => ({ row: nextRow(), key: "", value: "", err: {} }) diff --git a/packages/app/src/components/dialog-custom-provider.test.ts b/packages/app/src/components/dialog-custom-provider.test.ts deleted file mode 100644 index 07dd26ecd6..0000000000 --- a/packages/app/src/components/dialog-custom-provider.test.ts +++ /dev/null @@ -1,80 +0,0 @@ -import { describe, expect, test } from "bun:test" -import { validateCustomProvider } from "./dialog-custom-provider-form" - -const t = (key: string) => key - -describe("validateCustomProvider", () => { - test("builds trimmed config payload", () => { - const result = validateCustomProvider({ - form: { - providerID: "custom-provider", - name: " Custom Provider ", - baseURL: "https://api.example.com ", - apiKey: " {env: CUSTOM_PROVIDER_KEY} ", - models: [{ row: "m0", id: " model-a ", name: " Model A ", err: {} }], - headers: [ - { row: "h0", key: " X-Test ", value: " enabled ", err: {} }, - { row: "h1", key: "", value: "", err: {} }, - ], - err: {}, - }, - t, - disabledProviders: [], - existingProviderIDs: new Set(), - }) - - expect(result.result).toEqual({ - providerID: "custom-provider", - name: "Custom Provider", - key: undefined, - config: { - npm: "@ai-sdk/openai-compatible", - name: "Custom Provider", - env: ["CUSTOM_PROVIDER_KEY"], - options: { - baseURL: "https://api.example.com", - headers: { - "X-Test": "enabled", - }, - }, - models: { - "model-a": { name: "Model A" }, - }, - }, - }) - }) - - test("flags duplicate rows and allows reconnecting disabled providers", () => { - const result = validateCustomProvider({ - form: { - providerID: "custom-provider", - name: "Provider", - baseURL: "https://api.example.com", - apiKey: "secret", - models: [ - { row: "m0", id: "model-a", name: "Model A", err: {} }, - { row: "m1", id: "model-a", name: "Model A 2", err: {} }, - ], - headers: [ - { row: "h0", key: "Authorization", value: "one", err: {} }, - { row: "h1", key: "authorization", value: "two", err: {} }, - ], - err: {}, - }, - t, - disabledProviders: ["custom-provider"], - existingProviderIDs: new Set(["custom-provider"]), - }) - - expect(result.result).toBeUndefined() - expect(result.err.providerID).toBeUndefined() - expect(result.models[1]).toEqual({ - id: "provider.custom.error.duplicate", - name: undefined, - }) - expect(result.headers[1]).toEqual({ - key: "provider.custom.error.duplicate", - value: undefined, - }) - }) -}) diff --git a/packages/app/src/components/dialog-custom-provider.tsx b/packages/app/src/components/dialog-custom-provider.tsx deleted file mode 100644 index 6f38651e6d..0000000000 --- a/packages/app/src/components/dialog-custom-provider.tsx +++ /dev/null @@ -1,329 +0,0 @@ -import { Button } from "@opencode-ai/ui/button" -import { useDialog } from "@opencode-ai/ui/context/dialog" -import { Dialog } from "@opencode-ai/ui/dialog" -import { IconButton } from "@opencode-ai/ui/icon-button" -import { ProviderIcon } from "@opencode-ai/ui/provider-icon" -import { useMutation } from "@tanstack/solid-query" -import { TextField } from "@opencode-ai/ui/text-field" -import { showToast } from "@opencode-ai/ui/toast" -import { batch, For } from "solid-js" -import { createStore, produce } from "solid-js/store" -import { Link } from "@/components/link" -import { useGlobalSDK } from "@/context/global-sdk" -import { useGlobalSync } from "@/context/global-sync" -import { useLanguage } from "@/context/language" -import { type FormState, headerRow, modelRow, validateCustomProvider } from "./dialog-custom-provider-form" -import { DialogSelectProvider } from "./dialog-select-provider" - -type Props = { - back?: "providers" | "close" -} - -export function DialogCustomProvider(props: Props) { - const dialog = useDialog() - const globalSync = useGlobalSync() - const globalSDK = useGlobalSDK() - const language = useLanguage() - - const [form, setForm] = createStore({ - providerID: "", - name: "", - baseURL: "", - apiKey: "", - models: [modelRow()], - headers: [headerRow()], - err: {}, - }) - - const goBack = () => { - if (props.back === "close") { - dialog.close() - return - } - dialog.show(() => ) - } - - const addModel = () => { - setForm( - "models", - produce((rows) => { - rows.push(modelRow()) - }), - ) - } - - const removeModel = (index: number) => { - if (form.models.length <= 1) return - setForm( - "models", - produce((rows) => { - rows.splice(index, 1) - }), - ) - } - - const addHeader = () => { - setForm( - "headers", - produce((rows) => { - rows.push(headerRow()) - }), - ) - } - - const removeHeader = (index: number) => { - if (form.headers.length <= 1) return - setForm( - "headers", - produce((rows) => { - rows.splice(index, 1) - }), - ) - } - - const setField = (key: "providerID" | "name" | "baseURL" | "apiKey", value: string) => { - setForm(key, value) - if (key === "apiKey") return - setForm("err", key, undefined) - } - - const setModel = (index: number, key: "id" | "name", value: string) => { - batch(() => { - setForm("models", index, key, value) - setForm("models", index, "err", key, undefined) - }) - } - - const setHeader = (index: number, key: "key" | "value", value: string) => { - batch(() => { - setForm("headers", index, key, value) - setForm("headers", index, "err", key, undefined) - }) - } - - const validate = () => { - const output = validateCustomProvider({ - form, - t: language.t, - disabledProviders: globalSync.data.config.disabled_providers ?? [], - existingProviderIDs: new Set(globalSync.data.provider.all.map((p) => p.id)), - }) - batch(() => { - setForm("err", output.err) - output.models.forEach((err, index) => setForm("models", index, "err", err)) - output.headers.forEach((err, index) => setForm("headers", index, "err", err)) - }) - return output.result - } - - const saveMutation = useMutation(() => ({ - mutationFn: async (result: NonNullable>) => { - const disabledProviders = globalSync.data.config.disabled_providers ?? [] - const nextDisabled = disabledProviders.filter((id) => id !== result.providerID) - - if (result.key) { - await globalSDK.client.auth.set({ - providerID: result.providerID, - auth: { - type: "api", - key: result.key, - }, - }) - } - - await globalSync.updateConfig({ - provider: { [result.providerID]: result.config }, - disabled_providers: nextDisabled, - }) - return result - }, - onSuccess: (result) => { - dialog.close() - showToast({ - variant: "success", - icon: "circle-check", - title: language.t("provider.connect.toast.connected.title", { provider: result.name }), - description: language.t("provider.connect.toast.connected.description", { provider: result.name }), - }) - }, - onError: (err) => { - const message = err instanceof Error ? err.message : String(err) - showToast({ title: language.t("common.requestFailed"), description: message }) - }, - })) - - const save = (e: SubmitEvent) => { - e.preventDefault() - if (saveMutation.isPending) return - - const result = validate() - if (!result) return - saveMutation.mutate(result) - } - - return ( - - } - transition - > -
-
- -
{language.t("provider.custom.title")}
-
- -
-

- {language.t("provider.custom.description.prefix")} - - {language.t("provider.custom.description.link")} - - {language.t("provider.custom.description.suffix")} -

- -
- setField("providerID", v)} - validationState={form.err.providerID ? "invalid" : undefined} - error={form.err.providerID} - /> - setField("name", v)} - validationState={form.err.name ? "invalid" : undefined} - error={form.err.name} - /> - setField("baseURL", v)} - validationState={form.err.baseURL ? "invalid" : undefined} - error={form.err.baseURL} - /> - setField("apiKey", v)} - /> -
- -
- - - {(m, i) => ( -
-
- setModel(i(), "id", v)} - validationState={m.err.id ? "invalid" : undefined} - error={m.err.id} - /> -
-
- setModel(i(), "name", v)} - validationState={m.err.name ? "invalid" : undefined} - error={m.err.name} - /> -
- removeModel(i())} - disabled={form.models.length <= 1} - aria-label={language.t("provider.custom.models.remove")} - /> -
- )} -
- -
- -
- - - {(h, i) => ( -
-
- setHeader(i(), "key", v)} - validationState={h.err.key ? "invalid" : undefined} - error={h.err.key} - /> -
-
- setHeader(i(), "value", v)} - validationState={h.err.value ? "invalid" : undefined} - error={h.err.value} - /> -
- removeHeader(i())} - disabled={form.headers.length <= 1} - aria-label={language.t("provider.custom.headers.remove")} - /> -
- )} -
- -
- - -
-
-
- ) -} diff --git a/packages/app/src/components/dialog-manage-models.tsx b/packages/app/src/components/dialog-manage-models.tsx deleted file mode 100644 index ace79e38a7..0000000000 --- a/packages/app/src/components/dialog-manage-models.tsx +++ /dev/null @@ -1,101 +0,0 @@ -import { Dialog } from "@opencode-ai/ui/dialog" -import { List } from "@opencode-ai/ui/list" -import { Switch } from "@opencode-ai/ui/switch" -import { Tooltip } from "@opencode-ai/ui/tooltip" -import { Button } from "@opencode-ai/ui/button" -import type { Component } from "solid-js" -import { useLocal } from "@/context/local" -import { popularProviders } from "@/hooks/use-providers" -import { useLanguage } from "@/context/language" -import { useDialog } from "@opencode-ai/ui/context/dialog" -import { DialogSelectProvider } from "./dialog-select-provider" - -export const DialogManageModels: Component = () => { - const local = useLocal() - const language = useLanguage() - const dialog = useDialog() - - const handleConnectProvider = () => { - dialog.show(() => ) - } - const providerRank = (id: string) => popularProviders.indexOf(id) - const providerList = (providerID: string) => local.model.list().filter((x) => x.provider.id === providerID) - const providerVisible = (providerID: string) => - providerList(providerID).every((x) => local.model.visible({ modelID: x.id, providerID: x.provider.id })) - const setProviderVisibility = (providerID: string, checked: boolean) => { - providerList(providerID).forEach((x) => { - local.model.setVisibility({ modelID: x.id, providerID: x.provider.id }, checked) - }) - } - - return ( - - {language.t("command.provider.connect")} - - } - > - `${x?.provider?.id}:${x?.id}`} - items={local.model.list()} - filterKeys={["provider.name", "name", "id"]} - sortBy={(a, b) => a.name.localeCompare(b.name)} - groupBy={(x) => x.provider.id} - groupHeader={(group) => { - const provider = group.items[0].provider - return ( - <> - {provider.name} - - setProviderVisibility(provider.id, checked)} - hideLabel - > - {provider.name} - - - - ) - }} - sortGroupsBy={(a, b) => { - const aRank = providerRank(a.items[0].provider.id) - const bRank = providerRank(b.items[0].provider.id) - const aPopular = aRank >= 0 - const bPopular = bRank >= 0 - if (aPopular && !bPopular) return -1 - if (!aPopular && bPopular) return 1 - return aRank - bRank - }} - onSelect={(x) => { - if (!x) return - const key = { modelID: x.id, providerID: x.provider.id } - local.model.setVisibility(key, !local.model.visible(key)) - }} - > - {(i) => ( -
- {i.name} -
e.stopPropagation()}> - { - local.model.setVisibility({ modelID: i.id, providerID: i.provider.id }, checked) - }} - /> -
-
- )} -
-
- ) -} diff --git a/packages/app/src/components/dialog-release-notes.tsx b/packages/app/src/components/dialog-release-notes.tsx deleted file mode 100644 index d0a35b71be..0000000000 --- a/packages/app/src/components/dialog-release-notes.tsx +++ /dev/null @@ -1,144 +0,0 @@ -import { createSignal } from "solid-js" -import { Dialog } from "@opencode-ai/ui/dialog" -import { Button } from "@opencode-ai/ui/button" -import { useDialog } from "@opencode-ai/ui/context/dialog" -import { useLanguage } from "@/context/language" -import { useSettings } from "@/context/settings" - -export type Highlight = { - title: string - description: string - media?: { - type: "image" | "video" - src: string - alt?: string - } -} - -export function DialogReleaseNotes(props: { highlights: Highlight[] }) { - const dialog = useDialog() - const language = useLanguage() - const settings = useSettings() - const [index, setIndex] = createSignal(0) - - const total = () => props.highlights.length - const last = () => Math.max(0, total() - 1) - const feature = () => props.highlights[index()] ?? props.highlights[last()] - const isFirst = () => index() === 0 - const isLast = () => index() >= last() - const paged = () => total() > 1 - - function handleNext() { - if (isLast()) return - setIndex(index() + 1) - } - - function handleClose() { - dialog.close() - } - - function handleDisable() { - settings.general.setReleaseNotes(false) - handleClose() - } - - function handleKeyDown(e: KeyboardEvent) { - if (e.key === "Escape") { - e.preventDefault() - handleClose() - return - } - - if (!paged()) return - if (e.key === "ArrowLeft" && !isFirst()) { - e.preventDefault() - setIndex(index() - 1) - } - if (e.key === "ArrowRight" && !isLast()) { - e.preventDefault() - setIndex(index() + 1) - } - } - - return ( - -
- {/* Left side - Text content */} -
- {/* Top section - feature content (fixed position from top) */} -
-
-

{feature()?.title ?? ""}

-
-

{feature()?.description ?? ""}

-
- - {/* Spacer to push buttons to bottom */} -
- - {/* Bottom section - buttons and indicators (fixed position) */} -
-
- {isLast() ? ( - - ) : ( - - )} - - -
- - {paged() && ( -
- {props.highlights.map((_, i) => ( - - ))} -
- )} -
-
- - {/* Right side - Media content (edge to edge) */} - {feature()?.media && ( -
- {feature()!.media!.type === "image" ? ( - {feature()!.media!.alt - ) : ( -
- )} -
-
- ) -} diff --git a/packages/app/src/components/dialog-select-model-unpaid.tsx b/packages/app/src/components/dialog-select-model-unpaid.tsx deleted file mode 100644 index e25e8f0c17..0000000000 --- a/packages/app/src/components/dialog-select-model-unpaid.tsx +++ /dev/null @@ -1,145 +0,0 @@ -import { Button } from "@opencode-ai/ui/button" -import { useDialog } from "@opencode-ai/ui/context/dialog" -import { Dialog } from "@opencode-ai/ui/dialog" -import { List, type ListRef } from "@opencode-ai/ui/list" -import { ProviderIcon } from "@opencode-ai/ui/provider-icon" -import { Tag } from "@opencode-ai/ui/tag" -import { Tooltip } from "@opencode-ai/ui/tooltip" -import { type Component, Show } from "solid-js" -import { useLocal } from "@/context/local" -import { popularProviders, useProviders } from "@/hooks/use-providers" -import { ModelTooltip } from "./model-tooltip" -import { useLanguage } from "@/context/language" - -type ModelState = ReturnType["model"] - -export const DialogSelectModelUnpaid: Component<{ model?: ModelState }> = (props) => { - const model = props.model ?? useLocal().model - const dialog = useDialog() - const providers = useProviders() - const language = useLanguage() - - const connect = (provider: string) => { - void import("./dialog-connect-provider").then((x) => { - dialog.show(() => ) - }) - } - - const all = () => { - void import("./dialog-select-provider").then((x) => { - dialog.show(() => ) - }) - } - - let listRef: ListRef | undefined - const handleKeyDown = (e: KeyboardEvent) => { - if (e.key === "Escape") return - listRef?.onKeyDown(e) - } - - return ( - -
-
{language.t("dialog.model.unpaid.freeModels.title")}
- (listRef = ref)} - items={model.list} - current={model.current()} - key={(x) => `${x.provider.id}:${x.id}`} - itemWrapper={(item, node) => ( - - } - > - {node} - - )} - onSelect={(x) => { - model.set(x ? { modelID: x.id, providerID: x.provider.id } : undefined, { - recent: true, - }) - dialog.close() - }} - > - {(i) => ( -
- {i.name} - {language.t("model.tag.free")} - - {language.t("model.tag.latest")} - -
- )} -
-
-
-
-
-
{language.t("dialog.model.unpaid.addMore.title")}
-
- x?.id} - items={providers.popular} - activeIcon="plus-small" - sortBy={(a, b) => { - if (popularProviders.includes(a.id) && popularProviders.includes(b.id)) - return popularProviders.indexOf(a.id) - popularProviders.indexOf(b.id) - return a.name.localeCompare(b.name) - }} - onSelect={(x) => { - if (!x) return - connect(x.id) - }} - > - {(i) => ( -
- - {i.name} - -
{language.t("dialog.provider.opencode.tagline")}
-
- - {language.t("dialog.provider.tag.recommended")} - - - <> -
- {language.t("dialog.provider.opencodeGo.tagline")} -
- {language.t("dialog.provider.tag.recommended")} - -
- -
{language.t("dialog.provider.anthropic.note")}
-
-
- )} -
- -
-
-
-
-
- ) -} diff --git a/packages/app/src/components/dialog-select-model.tsx b/packages/app/src/components/dialog-select-model.tsx deleted file mode 100644 index e9c99e08fb..0000000000 --- a/packages/app/src/components/dialog-select-model.tsx +++ /dev/null @@ -1,231 +0,0 @@ -import { Popover as Kobalte } from "@kobalte/core/popover" -import { Component, ComponentProps, createMemo, JSX, Show, ValidComponent } from "solid-js" -import { createStore } from "solid-js/store" -import { useLocal } from "@/context/local" -import { useDialog } from "@opencode-ai/ui/context/dialog" -import { popularProviders } from "@/hooks/use-providers" -import { Button } from "@opencode-ai/ui/button" -import { IconButton } from "@opencode-ai/ui/icon-button" -import { Tag } from "@opencode-ai/ui/tag" -import { Dialog } from "@opencode-ai/ui/dialog" -import { List } from "@opencode-ai/ui/list" -import { Tooltip } from "@opencode-ai/ui/tooltip" -import { ModelTooltip } from "./model-tooltip" -import { useLanguage } from "@/context/language" - -const isFree = (provider: string, cost: { input: number } | undefined) => - provider === "opencode" && (!cost || cost.input === 0) - -type ModelState = ReturnType["model"] - -const ModelList: Component<{ - provider?: string - class?: string - onSelect: () => void - action?: JSX.Element - model?: ModelState -}> = (props) => { - const model = props.model ?? useLocal().model - const language = useLanguage() - - const models = createMemo(() => - model - .list() - .filter((m) => model.visible({ modelID: m.id, providerID: m.provider.id })) - .filter((m) => (props.provider ? m.provider.id === props.provider : true)), - ) - - return ( - `${x.provider.id}:${x.id}`} - items={models} - current={model.current()} - filterKeys={["provider.name", "name", "id"]} - sortBy={(a, b) => a.name.localeCompare(b.name)} - groupBy={(x) => x.provider.name} - sortGroupsBy={(a, b) => { - const aProvider = a.items[0].provider.id - const bProvider = b.items[0].provider.id - if (popularProviders.includes(aProvider) && !popularProviders.includes(bProvider)) return -1 - if (!popularProviders.includes(aProvider) && popularProviders.includes(bProvider)) return 1 - return popularProviders.indexOf(aProvider) - popularProviders.indexOf(bProvider) - }} - itemWrapper={(item, node) => ( - } - > - {node} - - )} - onSelect={(x) => { - model.set(x ? { modelID: x.id, providerID: x.provider.id } : undefined, { - recent: true, - }) - props.onSelect() - }} - > - {(i) => ( -
- {i.name} - - {language.t("model.tag.free")} - - - {language.t("model.tag.latest")} - -
- )} -
- ) -} - -type ModelSelectorTriggerProps = Omit, "as" | "ref"> -type Dismiss = "escape" | "outside" | "select" | "manage" | "provider" - -export function ModelSelectorPopover(props: { - provider?: string - model?: ModelState - children?: JSX.Element - triggerAs?: ValidComponent - triggerProps?: ModelSelectorTriggerProps - onClose?: (cause: "escape" | "select") => void -}) { - const [store, setStore] = createStore<{ - open: boolean - dismiss: Dismiss | null - }>({ - open: false, - dismiss: null, - }) - const dialog = useDialog() - - const close = (dismiss: Dismiss) => { - setStore("dismiss", dismiss) - setStore("open", false) - } - - const handleManage = () => { - close("manage") - void import("./dialog-manage-models").then((x) => { - dialog.show(() => ) - }) - } - - const handleConnectProvider = () => { - close("provider") - void import("./dialog-select-provider").then((x) => { - dialog.show(() => ) - }) - } - const language = useLanguage() - - return ( - { - if (next) setStore("dismiss", null) - setStore("open", next) - }} - modal={false} - placement="top-start" - gutter={4} - > - - {props.children} - - - { - close("escape") - event.preventDefault() - event.stopPropagation() - }} - onPointerDownOutside={() => close("outside")} - onFocusOutside={() => close("outside")} - onCloseAutoFocus={(event) => { - const dismiss = store.dismiss - if (dismiss === "outside") event.preventDefault() - if (dismiss === "escape" || dismiss === "select") { - event.preventDefault() - props.onClose?.(dismiss) - } - setStore("dismiss", null) - }} - > - {language.t("dialog.model.select.title")} - close("select")} - class="p-1" - action={ -
- - - - - - -
- } - /> -
-
-
- ) -} - -export const DialogSelectModel: Component<{ provider?: string; model?: ModelState }> = (props) => { - const dialog = useDialog() - const language = useLanguage() - - const provider = () => { - void import("./dialog-select-provider").then((x) => { - dialog.show(() => ) - }) - } - - const manage = () => { - void import("./dialog-manage-models").then((x) => { - dialog.show(() => ) - }) - } - - return ( - - {language.t("command.provider.connect")} - - } - > - dialog.close()} /> - - - ) -} diff --git a/packages/app/src/components/dialog-select-provider.tsx b/packages/app/src/components/dialog-select-provider.tsx deleted file mode 100644 index 9dcf81e1f0..0000000000 --- a/packages/app/src/components/dialog-select-provider.tsx +++ /dev/null @@ -1,91 +0,0 @@ -import { Component, Show } from "solid-js" -import { useDialog } from "@opencode-ai/ui/context/dialog" -import { popularProviders, useProviders } from "@/hooks/use-providers" -import { Dialog } from "@opencode-ai/ui/dialog" -import { List } from "@opencode-ai/ui/list" -import { Tag } from "@opencode-ai/ui/tag" -import { ProviderIcon } from "@opencode-ai/ui/provider-icon" -import { DialogConnectProvider } from "./dialog-connect-provider" -import { useLanguage } from "@/context/language" -import { DialogCustomProvider } from "./dialog-custom-provider" - -const CUSTOM_ID = "_custom" - -export const DialogSelectProvider: Component = () => { - const dialog = useDialog() - const providers = useProviders() - const language = useLanguage() - - // kilocode_change start - Use "Recommended" terminology to match kilocode - const popularGroup = () => language.t("dialog.provider.group.recommended") - // kilocode_change end - const otherGroup = () => language.t("dialog.provider.group.other") - const customLabel = () => language.t("settings.providers.tag.custom") - const note = (id: string) => { - if (id === "anthropic") return language.t("dialog.provider.anthropic.note") - if (id === "openai") return language.t("dialog.provider.openai.note") - if (id.startsWith("github-copilot")) return language.t("dialog.provider.copilot.note") - if (id === "opencode-go") return language.t("dialog.provider.opencodeGo.tagline") - } - - return ( - - x?.id} - items={() => { - language.locale() - return [{ id: CUSTOM_ID, name: customLabel() }, ...providers.all()] - }} - filterKeys={["id", "name"]} - groupBy={(x) => (popularProviders.includes(x.id) ? popularGroup() : otherGroup())} - sortBy={(a, b) => { - if (a.id === CUSTOM_ID) return -1 - if (b.id === CUSTOM_ID) return 1 - if (popularProviders.includes(a.id) && popularProviders.includes(b.id)) - return popularProviders.indexOf(a.id) - popularProviders.indexOf(b.id) - return a.name.localeCompare(b.name) - }} - sortGroupsBy={(a, b) => { - const popular = popularGroup() - if (a.category === popular && b.category !== popular) return -1 - if (b.category === popular && a.category !== popular) return 1 - return 0 - }} - onSelect={(x) => { - if (!x) return - if (x.id === CUSTOM_ID) { - dialog.show(() => ) - return - } - dialog.show(() => ) - }} - > - {(i) => ( -
- - {i.name} - {/* kilocode_change start - Provider tags and notes */} - - {language.t("dialog.provider.tag.recommended")} -
{language.t("dialog.provider.kilo.note")}
-
- {/* kilocode_change end */} - - {language.t("settings.providers.tag.custom")} - - - {language.t("dialog.provider.tag.recommended")} - - {(value) =>
{value()}
}
- - {language.t("dialog.provider.tag.recommended")} - -
- )} -
-
- ) -} diff --git a/packages/app/src/components/dialog-select-server.tsx b/packages/app/src/components/dialog-select-server.tsx deleted file mode 100644 index 0cb5a2d604..0000000000 --- a/packages/app/src/components/dialog-select-server.tsx +++ /dev/null @@ -1,649 +0,0 @@ -import { Button } from "@opencode-ai/ui/button" -import { useDialog } from "@opencode-ai/ui/context/dialog" -import { Dialog } from "@opencode-ai/ui/dialog" -import { DropdownMenu } from "@opencode-ai/ui/dropdown-menu" -import { Icon } from "@opencode-ai/ui/icon" -import { IconButton } from "@opencode-ai/ui/icon-button" -import { List } from "@opencode-ai/ui/list" -import { TextField } from "@opencode-ai/ui/text-field" -import { useMutation } from "@tanstack/solid-query" -import { showToast } from "@opencode-ai/ui/toast" -import { useNavigate } from "@solidjs/router" -import { createEffect, createMemo, createResource, onCleanup, Show } from "solid-js" -import { createStore, reconcile } from "solid-js/store" -import { ServerHealthIndicator, ServerRow } from "@/components/server/server-row" -import { useLanguage } from "@/context/language" -import { usePlatform } from "@/context/platform" -import { normalizeServerUrl, ServerConnection, useServer } from "@/context/server" -import { type ServerHealth, useCheckServerHealth } from "@/utils/server-health" - -const DEFAULT_USERNAME = "opencode" - -interface ServerFormProps { - value: string - name: string - username: string - password: string - placeholder: string - busy: boolean - error: string - status: boolean | undefined - onChange: (value: string) => void - onNameChange: (value: string) => void - onUsernameChange: (value: string) => void - onPasswordChange: (value: string) => void - onSubmit: () => void - onBack: () => void -} - -function showRequestError(language: ReturnType, err: unknown) { - showToast({ - variant: "error", - title: language.t("common.requestFailed"), - description: err instanceof Error ? err.message : String(err), - }) -} - -function useDefaultServer() { - const language = useLanguage() - const platform = usePlatform() - const [defaultKey, defaultUrlActions] = createResource( - async () => { - try { - const key = await platform.getDefaultServer?.() - if (!key) return null - return key - } catch (err) { - showRequestError(language, err) - return null - } - }, - { initialValue: null }, - ) - - const canDefault = createMemo(() => !!platform.getDefaultServer && !!platform.setDefaultServer) - const setDefault = async (key: ServerConnection.Key | null) => { - try { - await platform.setDefaultServer?.(key) - defaultUrlActions.mutate(key) - } catch (err) { - showRequestError(language, err) - } - } - - return { defaultKey, canDefault, setDefault } -} - -function useServerPreview() { - const checkServerHealth = useCheckServerHealth() - - const looksComplete = (value: string) => { - const normalized = normalizeServerUrl(value) - if (!normalized) return false - const host = normalized.replace(/^https?:\/\//, "").split("/")[0] - if (!host) return false - if (host.includes("localhost") || host.startsWith("127.0.0.1")) return true - return host.includes(".") || host.includes(":") - } - - const previewStatus = async ( - value: string, - username: string, - password: string, - setStatus: (value: boolean | undefined) => void, - ) => { - setStatus(undefined) - if (!looksComplete(value)) return - const normalized = normalizeServerUrl(value) - if (!normalized) return - const http: ServerConnection.HttpBase = { url: normalized } - if (username) http.username = username - if (password) http.password = password - const result = await checkServerHealth(http) - setStatus(result.healthy) - } - - return { previewStatus } -} - -function ServerForm(props: ServerFormProps) { - const language = useLanguage() - const keyDown = (event: KeyboardEvent) => { - event.stopPropagation() - if (event.key === "Escape") { - event.preventDefault() - props.onBack() - return - } - if (event.key !== "Enter" || event.isComposing) return - event.preventDefault() - props.onSubmit() - } - - return ( -
-
-
- -
- -
- - -
-
-
- ) -} - -export function DialogSelectServer() { - const navigate = useNavigate() - const dialog = useDialog() - const server = useServer() - const platform = usePlatform() - const language = useLanguage() - const { defaultKey, canDefault, setDefault } = useDefaultServer() - const { previewStatus } = useServerPreview() - const checkServerHealth = useCheckServerHealth() - const [store, setStore] = createStore({ - status: {} as Record, - addServer: { - url: "", - name: "", - username: DEFAULT_USERNAME, - password: "", - error: "", - showForm: false, - status: undefined as boolean | undefined, - }, - editServer: { - id: undefined as string | undefined, - value: "", - name: "", - username: "", - password: "", - error: "", - status: undefined as boolean | undefined, - }, - }) - - const resetAdd = () => { - setStore("addServer", { - url: "", - name: "", - username: DEFAULT_USERNAME, - password: "", - error: "", - showForm: false, - status: undefined, - }) - } - const resetEdit = () => { - setStore("editServer", { - id: undefined, - value: "", - name: "", - username: "", - password: "", - error: "", - status: undefined, - }) - } - - const addMutation = useMutation(() => ({ - mutationFn: async (value: string) => { - const normalized = normalizeServerUrl(value) - if (!normalized) { - resetAdd() - return - } - - const conn: ServerConnection.Http = { - type: "http", - http: { url: normalized }, - } - if (store.addServer.name.trim()) conn.displayName = store.addServer.name.trim() - if (store.addServer.password) conn.http.password = store.addServer.password - if (store.addServer.password && store.addServer.username) conn.http.username = store.addServer.username - const result = await checkServerHealth(conn.http) - if (!result.healthy) { - setStore("addServer", { error: language.t("dialog.server.add.error") }) - return - } - - resetAdd() - await select(conn, true) - }, - })) - - const editMutation = useMutation(() => ({ - mutationFn: async (input: { original: ServerConnection.Any; value: string }) => { - if (input.original.type !== "http") return - const normalized = normalizeServerUrl(input.value) - if (!normalized) { - resetEdit() - return - } - - const name = store.editServer.name.trim() || undefined - const username = store.editServer.username || undefined - const password = store.editServer.password || undefined - const existingName = input.original.displayName - if ( - normalized === input.original.http.url && - name === existingName && - username === input.original.http.username && - password === input.original.http.password - ) { - resetEdit() - return - } - - const conn: ServerConnection.Http = { - type: "http", - displayName: name, - http: { url: normalized, username, password }, - } - const result = await checkServerHealth(conn.http) - if (!result.healthy) { - setStore("editServer", { error: language.t("dialog.server.add.error") }) - return - } - if (normalized === input.original.http.url) { - server.add(conn) - } else { - replaceServer(input.original, conn) - } - - resetEdit() - }, - })) - - const replaceServer = (original: ServerConnection.Http, next: ServerConnection.Http) => { - const active = server.key - const newConn = server.add(next) - if (!newConn) return - const nextActive = active === ServerConnection.key(original) ? ServerConnection.key(newConn) : active - if (nextActive) server.setActive(nextActive) - server.remove(ServerConnection.key(original)) - } - - const items = createMemo(() => { - const current = server.current - const list = server.list - if (!current) return list - if (!list.includes(current)) return [current, ...list] - return [current, ...list.filter((x) => x !== current)] - }) - - const current = createMemo(() => items().find((x) => ServerConnection.key(x) === server.key) ?? items()[0]) - - const sortedItems = createMemo(() => { - const list = items() - if (!list.length) return list - const active = current() - const order = new Map(list.map((url, index) => [url, index] as const)) - const rank = (value?: ServerHealth) => { - if (value?.healthy === true) return 0 - if (value?.healthy === false) return 2 - return 1 - } - return list.slice().sort((a, b) => { - if (a === active) return -1 - if (b === active) return 1 - const diff = rank(store.status[ServerConnection.key(a)]) - rank(store.status[ServerConnection.key(b)]) - if (diff !== 0) return diff - return (order.get(a) ?? 0) - (order.get(b) ?? 0) - }) - }) - - async function refreshHealth() { - const results: Record = {} - await Promise.all( - items().map(async (conn) => { - results[ServerConnection.key(conn)] = await checkServerHealth(conn.http) - }), - ) - setStore("status", reconcile(results)) - } - - createEffect(() => { - items() - void refreshHealth() - const interval = setInterval(refreshHealth, 10_000) - onCleanup(() => clearInterval(interval)) - }) - - async function select(conn: ServerConnection.Any, persist?: boolean) { - if (!persist && store.status[ServerConnection.key(conn)]?.healthy === false) return - dialog.close() - if (persist && conn.type === "http") { - server.add(conn) - navigate("/") - return - } - navigate("/") - queueMicrotask(() => server.setActive(ServerConnection.key(conn))) - } - - const handleAddChange = (value: string) => { - if (addMutation.isPending) return - setStore("addServer", { url: value, error: "" }) - void previewStatus(value, store.addServer.username, store.addServer.password, (next) => - setStore("addServer", { status: next }), - ) - } - - const handleAddNameChange = (value: string) => { - if (addMutation.isPending) return - setStore("addServer", { name: value, error: "" }) - } - - const handleAddUsernameChange = (value: string) => { - if (addMutation.isPending) return - setStore("addServer", { username: value, error: "" }) - void previewStatus(store.addServer.url, value, store.addServer.password, (next) => - setStore("addServer", { status: next }), - ) - } - - const handleAddPasswordChange = (value: string) => { - if (addMutation.isPending) return - setStore("addServer", { password: value, error: "" }) - void previewStatus(store.addServer.url, store.addServer.username, value, (next) => - setStore("addServer", { status: next }), - ) - } - - const handleEditChange = (value: string) => { - if (editMutation.isPending) return - setStore("editServer", { value, error: "" }) - void previewStatus(value, store.editServer.username, store.editServer.password, (next) => - setStore("editServer", { status: next }), - ) - } - - const handleEditNameChange = (value: string) => { - if (editMutation.isPending) return - setStore("editServer", { name: value, error: "" }) - } - - const handleEditUsernameChange = (value: string) => { - if (editMutation.isPending) return - setStore("editServer", { username: value, error: "" }) - void previewStatus(store.editServer.value, value, store.editServer.password, (next) => - setStore("editServer", { status: next }), - ) - } - - const handleEditPasswordChange = (value: string) => { - if (editMutation.isPending) return - setStore("editServer", { password: value, error: "" }) - void previewStatus(store.editServer.value, store.editServer.username, value, (next) => - setStore("editServer", { status: next }), - ) - } - - const mode = createMemo<"list" | "add" | "edit">(() => { - if (store.editServer.id) return "edit" - if (store.addServer.showForm) return "add" - return "list" - }) - - const editing = createMemo(() => { - if (!store.editServer.id) return - return items().find((x) => x.type === "http" && x.http.url === store.editServer.id) - }) - - const resetForm = () => { - resetAdd() - resetEdit() - } - - const startAdd = () => { - resetEdit() - setStore("addServer", { - showForm: true, - url: "", - name: "", - username: DEFAULT_USERNAME, - password: "", - error: "", - status: undefined, - }) - } - - const startEdit = (conn: ServerConnection.Http) => { - resetAdd() - setStore("editServer", { - id: conn.http.url, - value: conn.http.url, - name: conn.displayName ?? "", - username: conn.http.username ?? "", - password: conn.http.password ?? "", - error: "", - status: store.status[ServerConnection.key(conn)]?.healthy, - }) - } - - const submitForm = () => { - if (mode() === "add") { - if (addMutation.isPending) return - setStore("addServer", { error: "" }) - addMutation.mutate(store.addServer.url) - return - } - const original = editing() - if (!original) return - if (editMutation.isPending) return - setStore("editServer", { error: "" }) - editMutation.mutate({ original, value: store.editServer.value }) - } - - const isFormMode = createMemo(() => mode() !== "list") - const isAddMode = createMemo(() => mode() === "add") - const formBusy = createMemo(() => (isAddMode() ? addMutation.isPending : editMutation.isPending)) - - const formTitle = createMemo(() => { - if (!isFormMode()) return language.t("dialog.server.title") - return ( -
- - {isAddMode() ? language.t("dialog.server.add.title") : language.t("dialog.server.edit.title")} -
- ) - }) - - createEffect(() => { - if (!store.editServer.id) return - if (editing()) return - resetEdit() - }) - - async function handleRemove(url: ServerConnection.Key) { - server.remove(url) - if ((await platform.getDefaultServer?.()) === url) { - void platform.setDefaultServer?.(null) - } - } - - return ( - -
- - } - > - x.http.url} - onSelect={(x) => { - if (x) void select(x) - }} - divider={true} - class="flex-1 min-h-0 px-5 [&_[data-slot=list-search-wrapper]]:w-full [&_[data-slot=list-scroll]]:flex-1 [&_[data-slot=list-scroll]]:overflow-y-auto [&_[data-slot=list-items]]:bg-surface-base [&_[data-slot=list-items]]:rounded-md [&_[data-slot=list-item]]:min-h-14 [&_[data-slot=list-item]]:p-3 [&_[data-slot=list-item]]:!bg-transparent" - > - {(i) => { - const key = ServerConnection.key(i) - return ( -
-
- -
- - - {language.t("dialog.server.status.default")} - - - } - showCredentials - /> -
- - - - - - - e.stopPropagation()} - onPointerDown={(e: PointerEvent) => e.stopPropagation()} - /> - - - { - if (i.type !== "http") return - startEdit(i) - }} - > - {language.t("dialog.server.menu.edit")} - - - setDefault(key)}> - - {language.t("dialog.server.menu.default")} - - - - - setDefault(null)}> - - {language.t("dialog.server.menu.defaultRemove")} - - - - - handleRemove(ServerConnection.key(i))} - class="text-text-on-critical-base hover:bg-surface-critical-weak" - > - {language.t("dialog.server.menu.delete")} - - - - - -
-
- ) - }} -
-
- -
- - {language.t("dialog.server.add.button")} - - } - > - - -
-
-
- ) -} diff --git a/packages/app/src/components/dialog-settings.tsx b/packages/app/src/components/dialog-settings.tsx deleted file mode 100644 index 83cea131f5..0000000000 --- a/packages/app/src/components/dialog-settings.tsx +++ /dev/null @@ -1,73 +0,0 @@ -import { Component } from "solid-js" -import { Dialog } from "@opencode-ai/ui/dialog" -import { Tabs } from "@opencode-ai/ui/tabs" -import { Icon } from "@opencode-ai/ui/icon" -import { useLanguage } from "@/context/language" -import { usePlatform } from "@/context/platform" -import { SettingsGeneral } from "./settings-general" -import { SettingsKeybinds } from "./settings-keybinds" -import { SettingsProviders } from "./settings-providers" -import { SettingsModels } from "./settings-models" - -export const DialogSettings: Component = () => { - const language = useLanguage() - const platform = usePlatform() - - return ( - - - -
-
-
-
- {language.t("settings.section.desktop")} -
- - - {language.t("settings.tab.general")} - - - - {language.t("settings.tab.shortcuts")} - -
-
- -
- {language.t("settings.section.server")} -
- - - {language.t("settings.providers.title")} - - - - {language.t("settings.models.title")} - -
-
-
-
-
- {language.t("app.name.desktop")} - v{platform.version} -
-
-
- - - - - - - - - - - - -
-
- ) -} diff --git a/packages/app/src/components/file-tree.test.ts b/packages/app/src/components/file-tree.test.ts deleted file mode 100644 index 29e20b4807..0000000000 --- a/packages/app/src/components/file-tree.test.ts +++ /dev/null @@ -1,78 +0,0 @@ -import { beforeAll, describe, expect, mock, test } from "bun:test" - -let shouldListRoot: typeof import("./file-tree").shouldListRoot -let shouldListExpanded: typeof import("./file-tree").shouldListExpanded -let dirsToExpand: typeof import("./file-tree").dirsToExpand - -beforeAll(async () => { - mock.module("@solidjs/router", () => ({ - useNavigate: () => () => undefined, - useParams: () => ({}), - })) - mock.module("@/context/file", () => ({ - useFile: () => ({ - tree: { - state: () => undefined, - list: () => Promise.resolve(), - children: () => [], - expand: () => {}, - collapse: () => {}, - }, - }), - })) - mock.module("@opencode-ai/ui/collapsible", () => ({ - Collapsible: { - Trigger: (props: { children?: unknown }) => props.children, - Content: (props: { children?: unknown }) => props.children, - }, - })) - mock.module("@opencode-ai/ui/file-icon", () => ({ FileIcon: () => null })) - mock.module("@opencode-ai/ui/icon", () => ({ Icon: () => null })) - mock.module("@opencode-ai/ui/tooltip", () => ({ Tooltip: (props: { children?: unknown }) => props.children })) - const mod = await import("./file-tree") - shouldListRoot = mod.shouldListRoot - shouldListExpanded = mod.shouldListExpanded - dirsToExpand = mod.dirsToExpand -}) - -describe("file tree fetch discipline", () => { - test("root lists on mount unless already loaded or loading", () => { - expect(shouldListRoot({ level: 0 })).toBe(true) - expect(shouldListRoot({ level: 0, dir: { loaded: true } })).toBe(false) - expect(shouldListRoot({ level: 0, dir: { loading: true } })).toBe(false) - expect(shouldListRoot({ level: 1 })).toBe(false) - }) - - test("nested dirs list only when expanded and stale", () => { - expect(shouldListExpanded({ level: 1 })).toBe(false) - expect(shouldListExpanded({ level: 1, dir: { expanded: false } })).toBe(false) - expect(shouldListExpanded({ level: 1, dir: { expanded: true } })).toBe(true) - expect(shouldListExpanded({ level: 1, dir: { expanded: true, loaded: true } })).toBe(false) - expect(shouldListExpanded({ level: 1, dir: { expanded: true, loading: true } })).toBe(false) - expect(shouldListExpanded({ level: 0, dir: { expanded: true } })).toBe(false) - }) - - test("allowed auto-expand picks only collapsed dirs", () => { - const expanded = new Set() - const filter = { dirs: new Set(["src", "src/components"]) } - - const first = dirsToExpand({ - level: 0, - filter, - expanded: (dir) => expanded.has(dir), - }) - - expect(first).toEqual(["src", "src/components"]) - - for (const dir of first) expanded.add(dir) - - const second = dirsToExpand({ - level: 0, - filter, - expanded: (dir) => expanded.has(dir), - }) - - expect(second).toEqual([]) - expect(dirsToExpand({ level: 1, filter, expanded: () => false })).toEqual([]) - }) -}) diff --git a/packages/app/src/components/file-tree.tsx b/packages/app/src/components/file-tree.tsx deleted file mode 100644 index f4148dfe4e..0000000000 --- a/packages/app/src/components/file-tree.tsx +++ /dev/null @@ -1,506 +0,0 @@ -import { useFile } from "@/context/file" -import { encodeFilePath } from "@/context/file/path" -import { Collapsible } from "@opencode-ai/ui/collapsible" -import { FileIcon } from "@opencode-ai/ui/file-icon" -import { Icon } from "@opencode-ai/ui/icon" -import { - createEffect, - createMemo, - For, - Match, - on, - Show, - splitProps, - Switch, - untrack, - type ComponentProps, - type ParentProps, -} from "solid-js" -import { Dynamic } from "solid-js/web" -import type { FileNode } from "@kilocode/sdk/v2" - -const MAX_DEPTH = 128 - -function pathToFileUrl(filepath: string): string { - return `file://${encodeFilePath(filepath)}` -} - -type Kind = "add" | "del" | "mix" - -type Filter = { - files: Set - dirs: Set -} - -export function shouldListRoot(input: { level: number; dir?: { loaded?: boolean; loading?: boolean } }) { - if (input.level !== 0) return false - if (input.dir?.loaded) return false - if (input.dir?.loading) return false - return true -} - -export function shouldListExpanded(input: { - level: number - dir?: { expanded?: boolean; loaded?: boolean; loading?: boolean } -}) { - if (input.level === 0) return false - if (!input.dir?.expanded) return false - if (input.dir.loaded) return false - if (input.dir.loading) return false - return true -} - -export function dirsToExpand(input: { - level: number - filter?: { dirs: Set } - expanded: (dir: string) => boolean -}) { - if (input.level !== 0) return [] - if (!input.filter) return [] - return [...input.filter.dirs].filter((dir) => !input.expanded(dir)) -} - -const kindLabel = (kind: Kind) => { - if (kind === "add") return "A" - if (kind === "del") return "D" - return "M" -} - -const kindTextColor = (kind: Kind) => { - if (kind === "add") return "color: var(--icon-diff-add-base)" - if (kind === "del") return "color: var(--icon-diff-delete-base)" - return "color: var(--icon-diff-modified-base)" -} - -const kindDotColor = (kind: Kind) => { - if (kind === "add") return "background-color: var(--icon-diff-add-base)" - if (kind === "del") return "background-color: var(--icon-diff-delete-base)" - return "background-color: var(--icon-diff-modified-base)" -} - -const visibleKind = (node: FileNode, kinds?: ReadonlyMap, marks?: Set) => { - const kind = kinds?.get(node.path) - if (!kind) return - if (!marks?.has(node.path)) return - return kind -} - -const buildDragImage = (target: HTMLElement) => { - const icon = target.querySelector('[data-component="file-icon"]') ?? target.querySelector("svg") - const text = target.querySelector("span") - if (!icon || !text) return - - const image = document.createElement("div") - image.className = - "flex items-center gap-x-2 px-2 py-1 bg-surface-raised-base rounded-md border border-border-base text-12-regular text-text-strong" - image.style.position = "absolute" - image.style.top = "-1000px" - image.innerHTML = (icon as SVGElement).outerHTML + (text as HTMLSpanElement).outerHTML - return image -} - -const withFileDragImage = (event: DragEvent) => { - const image = buildDragImage(event.currentTarget as HTMLElement) - if (!image) return - document.body.appendChild(image) - event.dataTransfer?.setDragImage(image, 0, 12) - setTimeout(() => document.body.removeChild(image), 0) -} - -const FileTreeNode = ( - p: ParentProps & - ComponentProps<"div"> & - ComponentProps<"button"> & { - node: FileNode - level: number - active?: string - nodeClass?: string - draggable: boolean - kinds?: ReadonlyMap - marks?: Set - as?: "div" | "button" - }, -) => { - const [local, rest] = splitProps(p, [ - "node", - "level", - "active", - "nodeClass", - "draggable", - "kinds", - "marks", - "as", - "children", - "class", - "classList", - ]) - const kind = () => visibleKind(local.node, local.kinds, local.marks) - const active = () => !!kind() && !local.node.ignored - const color = () => { - const value = kind() - if (!value) return - return kindTextColor(value) - } - - return ( - { - if (!local.draggable) return - event.dataTransfer?.setData("text/plain", `file:${local.node.path}`) - event.dataTransfer?.setData("text/uri-list", pathToFileUrl(local.node.path)) - if (event.dataTransfer) event.dataTransfer.effectAllowed = "copy" - withFileDragImage(event) - }} - {...rest} - > - {local.children} - - {local.node.name} - - {(() => { - const value = kind() - if (!value) return null - if (local.node.type === "file") { - return ( - - {kindLabel(value)} - - ) - } - return
- })()} - - ) -} - -export default function FileTree(props: { - path: string - class?: string - nodeClass?: string - active?: string - level?: number - allowed?: readonly string[] - modified?: readonly string[] - kinds?: ReadonlyMap - draggable?: boolean - onFileClick?: (file: FileNode) => void - - _filter?: Filter - _marks?: Set - _deeps?: Map - _kinds?: ReadonlyMap - _chain?: readonly string[] -}) { - const file = useFile() - const level = props.level ?? 0 - const draggable = () => props.draggable ?? true - - const key = (p: string) => - file - .normalize(p) - .replace(/[\\/]+$/, "") - .replaceAll("\\", "/") - const chain = props._chain ? [...props._chain, key(props.path)] : [key(props.path)] - - const filter = createMemo(() => { - if (props._filter) return props._filter - - const allowed = props.allowed - if (!allowed) return - - const files = new Set(allowed) - const dirs = new Set() - - for (const item of allowed) { - const parts = item.split("/") - const parents = parts.slice(0, -1) - for (const [idx] of parents.entries()) { - const dir = parents.slice(0, idx + 1).join("/") - if (dir) dirs.add(dir) - } - } - - return { files, dirs } - }) - - const marks = createMemo(() => { - if (props._marks) return props._marks - - const out = new Set() - for (const item of props.modified ?? []) out.add(item) - for (const item of props.kinds?.keys() ?? []) out.add(item) - if (out.size === 0) return - return out - }) - - const kinds = createMemo(() => { - if (props._kinds) return props._kinds - return props.kinds - }) - - const deeps = createMemo(() => { - if (props._deeps) return props._deeps - - const out = new Map() - - const root = props.path - if (!(file.tree.state(root)?.expanded ?? false)) return out - - const seen = new Set() - const stack: { dir: string; lvl: number; i: number; kids: string[]; max: number }[] = [] - - const push = (dir: string, lvl: number) => { - const id = key(dir) - if (seen.has(id)) return - seen.add(id) - - const kids = file.tree - .children(dir) - .filter((node) => node.type === "directory" && (file.tree.state(node.path)?.expanded ?? false)) - .map((node) => node.path) - - stack.push({ dir, lvl, i: 0, kids, max: lvl }) - } - - push(root, level - 1) - - while (stack.length > 0) { - const top = stack[stack.length - 1]! - - if (top.i < top.kids.length) { - const next = top.kids[top.i]! - top.i++ - push(next, top.lvl + 1) - continue - } - - out.set(top.dir, top.max) - stack.pop() - - const parent = stack[stack.length - 1] - if (!parent) continue - parent.max = Math.max(parent.max, top.max) - } - - return out - }) - - createEffect(() => { - const current = filter() - const dirs = dirsToExpand({ - level, - filter: current, - expanded: (dir) => untrack(() => file.tree.state(dir)?.expanded) ?? false, - }) - for (const dir of dirs) file.tree.expand(dir) - }) - - createEffect( - on( - () => props.path, - (path) => { - const dir = untrack(() => file.tree.state(path)) - if (!shouldListRoot({ level, dir })) return - void file.tree.list(path) - }, - { defer: false }, - ), - ) - - const nodes = createMemo(() => { - const nodes = file.tree.children(props.path) - const current = filter() - if (!current) return nodes - - const parent = (path: string) => { - const idx = path.lastIndexOf("/") - if (idx === -1) return "" - return path.slice(0, idx) - } - - const leaf = (path: string) => { - const idx = path.lastIndexOf("/") - return idx === -1 ? path : path.slice(idx + 1) - } - - const out = nodes.filter((node) => { - if (node.type === "file") return current.files.has(node.path) - return current.dirs.has(node.path) - }) - - const seen = new Set(out.map((node) => node.path)) - - for (const dir of current.dirs) { - if (parent(dir) !== props.path) continue - if (seen.has(dir)) continue - out.push({ - name: leaf(dir), - path: dir, - absolute: dir, - type: "directory", - ignored: false, - }) - seen.add(dir) - } - - for (const item of current.files) { - if (parent(item) !== props.path) continue - if (seen.has(item)) continue - out.push({ - name: leaf(item), - path: item, - absolute: item, - type: "file", - ignored: false, - }) - seen.add(item) - } - - out.sort((a, b) => { - if (a.type !== b.type) { - return a.type === "directory" ? -1 : 1 - } - return a.name.localeCompare(b.name) - }) - - return out - }) - - return ( -
- - {(node) => { - const expanded = () => file.tree.state(node.path)?.expanded ?? false - const deep = () => deeps().get(node.path) ?? -1 - const kind = () => visibleKind(node, kinds(), marks()) - const active = () => !!kind() && !node.ignored - - return ( - - - (open ? file.tree.expand(node.path) : file.tree.collapse(node.path))} - > - - -
- -
-
-
- -
- ...
} - > - - -
-
-
- - props.onFileClick?.(node)} - > -
- - - - - - - - - - - - - - - - - - ) - }} - -
- ) -} diff --git a/packages/app/src/components/link.tsx b/packages/app/src/components/link.tsx deleted file mode 100644 index 85f7efc539..0000000000 --- a/packages/app/src/components/link.tsx +++ /dev/null @@ -1,26 +0,0 @@ -import { ComponentProps, splitProps } from "solid-js" -import { usePlatform } from "@/context/platform" - -export interface LinkProps extends Omit, "href"> { - href: string -} - -export function Link(props: LinkProps) { - const platform = usePlatform() - const [local, rest] = splitProps(props, ["href", "children", "class"]) - - return ( - { - if (!local.href) return - event.preventDefault() - platform.openLink(local.href) - }} - {...rest} - > - {local.children} - - ) -} diff --git a/packages/app/src/components/model-tooltip.tsx b/packages/app/src/components/model-tooltip.tsx deleted file mode 100644 index 53164dae85..0000000000 --- a/packages/app/src/components/model-tooltip.tsx +++ /dev/null @@ -1,91 +0,0 @@ -import { Show, type Component } from "solid-js" -import { useLanguage } from "@/context/language" - -type InputKey = "text" | "image" | "audio" | "video" | "pdf" -type InputMap = Record - -type ModelInfo = { - id: string - name: string - provider: { - name: string - } - capabilities?: { - reasoning: boolean - input: InputMap - } - modalities?: { - input: Array - } - reasoning?: boolean - limit: { - context: number - } -} - -export const ModelTooltip: Component<{ model: ModelInfo; latest?: boolean; free?: boolean }> = (props) => { - const language = useLanguage() - const sourceName = (model: ModelInfo) => { - const value = `${model.id} ${model.name}`.toLowerCase() - - if (/claude|anthropic/.test(value)) return language.t("model.provider.anthropic") - if (/gpt|o[1-4]|codex|openai/.test(value)) return language.t("model.provider.openai") - if (/gemini|palm|bard|google/.test(value)) return language.t("model.provider.google") - if (/grok|xai/.test(value)) return language.t("model.provider.xai") - if (/llama|meta/.test(value)) return language.t("model.provider.meta") - - return model.provider.name - } - const inputLabel = (value: string) => { - if (value === "text") return language.t("model.input.text") - if (value === "image") return language.t("model.input.image") - if (value === "audio") return language.t("model.input.audio") - if (value === "video") return language.t("model.input.video") - if (value === "pdf") return language.t("model.input.pdf") - return value - } - const title = () => { - const tags: Array = [] - if (props.latest) tags.push(language.t("model.tag.latest")) - if (props.free) tags.push(language.t("model.tag.free")) - const suffix = tags.length ? ` (${tags.join(", ")})` : "" - return `${sourceName(props.model)} ${props.model.name}${suffix}` - } - const inputs = () => { - if (props.model.capabilities) { - const input = props.model.capabilities.input - const order: Array = ["text", "image", "audio", "video", "pdf"] - const entries = order.filter((key) => input[key]).map((key) => inputLabel(key)) - return entries.length ? entries.join(", ") : undefined - } - const raw = props.model.modalities?.input - if (!raw) return - const entries = raw.map((value) => inputLabel(value)) - return entries.length ? entries.join(", ") : undefined - } - const reasoning = () => { - if (props.model.capabilities) - return props.model.capabilities.reasoning - ? language.t("model.tooltip.reasoning.allowed") - : language.t("model.tooltip.reasoning.none") - return props.model.reasoning - ? language.t("model.tooltip.reasoning.allowed") - : language.t("model.tooltip.reasoning.none") - } - const context = () => language.t("model.tooltip.context", { limit: props.model.limit.context.toLocaleString() }) - - return ( -
-
{title()}
- - {(value) => ( -
- {language.t("model.tooltip.allows", { inputs: value() })} -
- )} -
-
{reasoning()}
-
{context()}
-
- ) -} diff --git a/packages/app/src/components/prompt-input.tsx b/packages/app/src/components/prompt-input.tsx deleted file mode 100644 index 9d7e2394bf..0000000000 --- a/packages/app/src/components/prompt-input.tsx +++ /dev/null @@ -1,1619 +0,0 @@ -import { useFilteredList } from "@opencode-ai/ui/hooks" -import { useSpring } from "@opencode-ai/ui/motion-spring" -import { createEffect, on, Component, Show, onCleanup, createMemo, createSignal, createResource } from "solid-js" -import { createStore } from "solid-js/store" -import { useLocal } from "@/context/local" -import { selectionFromLines, type SelectedLineRange, useFile } from "@/context/file" -import { - ContentPart, - DEFAULT_PROMPT, - isPromptEqual, - Prompt, - usePrompt, - ImageAttachmentPart, - AgentPart, - FileAttachmentPart, -} from "@/context/prompt" -import { useLayout } from "@/context/layout" -import { useSDK } from "@/context/sdk" -import { useSync } from "@/context/sync" -import { useComments } from "@/context/comments" -import { Button } from "@opencode-ai/ui/button" -import { DockShellForm, DockTray } from "@opencode-ai/ui/dock-surface" -import { Icon } from "@opencode-ai/ui/icon" -import { ProviderIcon } from "@opencode-ai/ui/provider-icon" -import { Tooltip, TooltipKeybind } from "@opencode-ai/ui/tooltip" -import { IconButton } from "@opencode-ai/ui/icon-button" -import { Select } from "@opencode-ai/ui/select" -import { useDialog } from "@opencode-ai/ui/context/dialog" -import { ModelSelectorPopover } from "@/components/dialog-select-model" -import { useProviders } from "@/hooks/use-providers" -import { useCommand } from "@/context/command" -import { Persist, persisted } from "@/utils/persist" -import { usePermission } from "@/context/permission" -import { useLanguage } from "@/context/language" -import { usePlatform } from "@/context/platform" -import { useSessionLayout } from "@/pages/session/session-layout" -import { createSessionTabs } from "@/pages/session/helpers" -import { createTextFragment, getCursorPosition, setCursorPosition, setRangeEdge } from "./prompt-input/editor-dom" -import { createPromptAttachments } from "./prompt-input/attachments" -import { ACCEPTED_FILE_TYPES } from "./prompt-input/files" -import { - canNavigateHistoryAtCursor, - navigatePromptHistory, - prependHistoryEntry, - type PromptHistoryComment, - type PromptHistoryEntry, - type PromptHistoryStoredEntry, - promptLength, -} from "./prompt-input/history" -import { createPromptSubmit, type FollowupDraft } from "./prompt-input/submit" -import { PromptPopover, type AtOption, type SlashCommand } from "./prompt-input/slash-popover" -import { PromptContextItems } from "./prompt-input/context-items" -import { PromptImageAttachments } from "./prompt-input/image-attachments" -import { PromptDragOverlay } from "./prompt-input/drag-overlay" -import { promptPlaceholder } from "./prompt-input/placeholder" -import { ImagePreview } from "@opencode-ai/ui/image-preview" -import { useQueries } from "@tanstack/solid-query" -import { loadAgentsQuery, loadProvidersQuery } from "@/context/global-sync/bootstrap" - -interface PromptInputProps { - class?: string - ref?: (el: HTMLDivElement) => void - newSessionWorktree?: string - onNewSessionWorktreeReset?: () => void - edit?: { id: string; prompt: Prompt; context: FollowupDraft["context"] } - onEditLoaded?: () => void - shouldQueue?: () => boolean - onQueue?: (draft: FollowupDraft) => void - onAbort?: () => void - onSubmit?: () => void -} - -const EXAMPLES = [ - "prompt.example.1", - "prompt.example.2", - "prompt.example.3", - "prompt.example.4", - "prompt.example.5", - "prompt.example.6", - "prompt.example.7", - "prompt.example.8", - "prompt.example.9", - "prompt.example.10", - "prompt.example.11", - "prompt.example.12", - "prompt.example.13", - "prompt.example.14", - "prompt.example.15", - "prompt.example.16", - "prompt.example.17", - "prompt.example.18", - "prompt.example.19", - "prompt.example.20", - "prompt.example.21", - "prompt.example.22", - "prompt.example.23", - "prompt.example.24", - "prompt.example.25", -] as const - -const NON_EMPTY_TEXT = /[^\s\u200B]/ - -export const PromptInput: Component = (props) => { - const sdk = useSDK() - - const sync = useSync() - const local = useLocal() - const files = useFile() - const prompt = usePrompt() - const layout = useLayout() - const comments = useComments() - const dialog = useDialog() - const providers = useProviders() - const command = useCommand() - const permission = usePermission() - const language = useLanguage() - const platform = usePlatform() - const { params, tabs, view } = useSessionLayout() - let editorRef!: HTMLDivElement - let fileInputRef: HTMLInputElement | undefined - let scrollRef!: HTMLDivElement - let slashPopoverRef!: HTMLDivElement - - const mirror = { input: false } - const inset = 56 - const space = `${inset}px` - - const scrollCursorIntoView = () => { - const container = scrollRef - const selection = window.getSelection() - if (!container || !selection || selection.rangeCount === 0) return - - const range = selection.getRangeAt(0) - if (!editorRef.contains(range.startContainer)) return - - const cursor = getCursorPosition(editorRef) - const length = promptLength(prompt.current().filter((part) => part.type !== "image")) - if (cursor >= length) { - container.scrollTop = container.scrollHeight - return - } - - const rect = range.getClientRects().item(0) ?? range.getBoundingClientRect() - if (!rect.height) return - - const containerRect = container.getBoundingClientRect() - const top = rect.top - containerRect.top + container.scrollTop - const bottom = rect.bottom - containerRect.top + container.scrollTop - const padding = 12 - - if (top < container.scrollTop + padding) { - container.scrollTop = Math.max(0, top - padding) - return - } - - if (bottom > container.scrollTop + container.clientHeight - inset) { - container.scrollTop = bottom - container.clientHeight + inset - } - } - - const queueScroll = (count = 2) => { - requestAnimationFrame(() => { - scrollCursorIntoView() - if (count > 1) queueScroll(count - 1) - }) - } - - const activeFileTab = createSessionTabs({ - tabs, - pathFromTab: files.pathFromTab, - normalizeTab: (tab) => (tab.startsWith("file://") ? files.tab(tab) : tab), - }).activeFileTab - - const commentInReview = (path: string) => { - const sessionID = params.id - if (!sessionID) return false - - const diffs = sync.data.session_diff[sessionID] - if (!diffs) return false - return diffs.some((diff) => diff.file === path) - } - - const openComment = (item: { path: string; commentID?: string; commentOrigin?: "review" | "file" }) => { - if (!item.commentID) return - - const focus = { file: item.path, id: item.commentID } - comments.setActive(focus) - - const queueCommentFocus = (attempts = 6) => { - const schedule = (left: number) => { - requestAnimationFrame(() => { - comments.setFocus({ ...focus }) - if (left <= 0) return - requestAnimationFrame(() => { - const current = comments.focus() - if (!current) return - if (current.file !== focus.file || current.id !== focus.id) return - schedule(left - 1) - }) - }) - } - - schedule(attempts) - } - - const wantsReview = item.commentOrigin === "review" || (item.commentOrigin !== "file" && commentInReview(item.path)) - if (wantsReview) { - if (!view().reviewPanel.opened()) view().reviewPanel.open() - layout.fileTree.setTab("changes") - tabs().setActive("review") - queueCommentFocus() - return - } - - if (!view().reviewPanel.opened()) view().reviewPanel.open() - layout.fileTree.setTab("all") - const tab = files.tab(item.path) - void tabs().open(tab) - tabs().setActive(tab) - void Promise.resolve(files.load(item.path)).finally(() => queueCommentFocus()) - } - - const recent = createMemo(() => { - const all = tabs().all() - const active = activeFileTab() - const order = active ? [active, ...all.filter((x) => x !== active)] : all - const seen = new Set() - const paths: string[] = [] - - for (const tab of order) { - const path = files.pathFromTab(tab) - if (!path) continue - if (seen.has(path)) continue - seen.add(path) - paths.push(path) - } - - return paths - }) - const info = createMemo(() => (params.id ? sync.session.get(params.id) : undefined)) - const status = createMemo( - () => - sync.data.session_status[params.id ?? ""] ?? { - type: "idle", - }, - ) - const working = createMemo(() => status()?.type !== "idle") - const imageAttachments = createMemo(() => - prompt.current().filter((part): part is ImageAttachmentPart => part.type === "image"), - ) - - const [store, setStore] = createStore<{ - popover: "at" | "slash" | null - historyIndex: number - savedPrompt: PromptHistoryEntry | null - placeholder: number - draggingType: "image" | "@mention" | null - mode: "normal" | "shell" - applyingHistory: boolean - }>({ - popover: null, - historyIndex: -1, - savedPrompt: null as PromptHistoryEntry | null, - placeholder: Math.floor(Math.random() * EXAMPLES.length), - draggingType: null, - mode: "normal", - applyingHistory: false, - }) - - const buttonsSpring = useSpring(() => (store.mode === "normal" ? 1 : 0), { visualDuration: 0.2, bounce: 0 }) - const motion = (value: number) => ({ - opacity: value, - transform: `scale(${0.98 + value * 0.02})`, - filter: `blur(${(1 - value) * 2}px)`, - "pointer-events": value > 0.5 ? ("auto" as const) : ("none" as const), - }) - const buttons = createMemo(() => motion(buttonsSpring())) - const shell = createMemo(() => motion(1 - buttonsSpring())) - const control = createMemo(() => ({ height: "28px", ...buttons() })) - - const commentCount = createMemo(() => { - if (store.mode === "shell") return 0 - return prompt.context.items().filter((item) => !!item.comment?.trim()).length - }) - const blank = createMemo(() => { - const text = prompt - .current() - .map((part) => ("content" in part ? part.content : "")) - .join("") - return text.trim().length === 0 && imageAttachments().length === 0 && commentCount() === 0 - }) - const stopping = createMemo(() => working() && blank()) - const tip = () => { - if (stopping()) { - return ( -
- {language.t("prompt.action.stop")} - {language.t("common.key.esc")} -
- ) - } - - return ( -
- {language.t("prompt.action.send")} - -
- ) - } - - const contextItems = createMemo(() => { - const items = prompt.context.items() - if (store.mode !== "shell") return items - return items.filter((item) => !item.comment?.trim()) - }) - - const hasUserPrompt = createMemo(() => { - const sessionID = params.id - if (!sessionID) return false - const messages = sync.data.message[sessionID] - if (!messages) return false - return messages.some((m) => m.role === "user") - }) - - const [history, setHistory] = persisted( - Persist.global("prompt-history", ["prompt-history.v1"]), - createStore<{ - entries: PromptHistoryStoredEntry[] - }>({ - entries: [], - }), - ) - const [shellHistory, setShellHistory] = persisted( - Persist.global("prompt-history-shell", ["prompt-history-shell.v1"]), - createStore<{ - entries: PromptHistoryStoredEntry[] - }>({ - entries: [], - }), - ) - - const suggest = createMemo(() => !hasUserPrompt()) - - const placeholder = createMemo(() => - promptPlaceholder({ - mode: store.mode, - commentCount: commentCount(), - example: suggest() ? (store.mode === "shell" ? "git status" : language.t(EXAMPLES[store.placeholder])) : "", - suggest: suggest(), - t: (key, params) => language.t(key as Parameters[0], params as never), - }), - ) - - const historyComments = () => { - const byID = new Map(comments.all().map((item) => [`${item.file}\n${item.id}`, item] as const)) - return prompt.context.items().flatMap((item) => { - if (item.type !== "file") return [] - const comment = item.comment?.trim() - if (!comment) return [] - - const selection = item.commentID ? byID.get(`${item.path}\n${item.commentID}`)?.selection : undefined - const nextSelection = - selection ?? - (item.selection - ? ({ - start: item.selection.startLine, - end: item.selection.endLine, - } satisfies SelectedLineRange) - : undefined) - if (!nextSelection) return [] - - return [ - { - id: item.commentID ?? item.key, - path: item.path, - selection: { ...nextSelection }, - comment, - time: item.commentID ? (byID.get(`${item.path}\n${item.commentID}`)?.time ?? Date.now()) : Date.now(), - origin: item.commentOrigin, - preview: item.preview, - } satisfies PromptHistoryComment, - ] - }) - } - - const applyHistoryComments = (items: PromptHistoryComment[]) => { - comments.replace( - items.map((item) => ({ - id: item.id, - file: item.path, - selection: { ...item.selection }, - comment: item.comment, - time: item.time, - })), - ) - prompt.context.replaceComments( - items.map((item) => ({ - type: "file" as const, - path: item.path, - selection: selectionFromLines(item.selection), - comment: item.comment, - commentID: item.id, - commentOrigin: item.origin, - preview: item.preview, - })), - ) - } - - const applyHistoryPrompt = (entry: PromptHistoryEntry, position: "start" | "end") => { - const p = entry.prompt - const length = position === "start" ? 0 : promptLength(p) - setStore("applyingHistory", true) - applyHistoryComments(entry.comments) - prompt.set(p, length) - requestAnimationFrame(() => { - editorRef.focus() - setCursorPosition(editorRef, length) - setStore("applyingHistory", false) - queueScroll() - }) - } - - const getCaretState = () => { - const selection = window.getSelection() - const textLength = promptLength(prompt.current()) - if (!selection || selection.rangeCount === 0) { - return { collapsed: false, cursorPosition: 0, textLength } - } - const anchorNode = selection.anchorNode - if (!anchorNode || !editorRef.contains(anchorNode)) { - return { collapsed: false, cursorPosition: 0, textLength } - } - return { - collapsed: selection.isCollapsed, - cursorPosition: getCursorPosition(editorRef), - textLength, - } - } - - const escBlur = () => platform.platform === "desktop" && platform.os === "macos" - - const pick = () => fileInputRef?.click() - - const setMode = (mode: "normal" | "shell") => { - setStore("mode", mode) - setStore("popover", null) - requestAnimationFrame(() => editorRef?.focus()) - } - - const shellModeKey = "mod+shift+x" - const normalModeKey = "mod+shift+e" - - command.register("prompt-input", () => [ - { - id: "file.attach", - title: language.t("prompt.action.attachFile"), - category: language.t("command.category.file"), - keybind: "mod+u", - disabled: store.mode !== "normal", - onSelect: pick, - }, - { - id: "prompt.mode.shell", - title: language.t("command.prompt.mode.shell"), - category: language.t("command.category.session"), - keybind: shellModeKey, - disabled: store.mode === "shell", - onSelect: () => setMode("shell"), - }, - { - id: "prompt.mode.normal", - title: language.t("command.prompt.mode.normal"), - category: language.t("command.category.session"), - keybind: normalModeKey, - disabled: store.mode === "normal", - onSelect: () => setMode("normal"), - }, - ]) - - const closePopover = () => setStore("popover", null) - - const resetHistoryNavigation = (force = false) => { - if (!force && (store.historyIndex < 0 || store.applyingHistory)) return - setStore("historyIndex", -1) - setStore("savedPrompt", null) - } - - const clearEditor = () => { - editorRef.innerHTML = "" - } - - const setEditorText = (text: string) => { - clearEditor() - editorRef.textContent = text - } - - const focusEditorEnd = () => { - requestAnimationFrame(() => { - editorRef.focus() - const range = document.createRange() - const selection = window.getSelection() - range.selectNodeContents(editorRef) - range.collapse(false) - selection?.removeAllRanges() - selection?.addRange(range) - }) - } - - const currentCursor = () => { - const selection = window.getSelection() - if (!selection || selection.rangeCount === 0 || !editorRef.contains(selection.anchorNode)) return null - return getCursorPosition(editorRef) - } - - const restoreFocus = () => { - requestAnimationFrame(() => { - const cursor = prompt.cursor() ?? promptLength(prompt.current()) - editorRef.focus() - setCursorPosition(editorRef, cursor) - queueScroll() - }) - } - - const renderEditorWithCursor = (parts: Prompt) => { - const cursor = currentCursor() - renderEditor(parts) - if (cursor !== null) setCursorPosition(editorRef, cursor) - } - - createEffect(() => { - params.id - if (params.id) return - if (!suggest()) return - const interval = setInterval(() => { - setStore("placeholder", (prev) => (prev + 1) % EXAMPLES.length) - }, 6500) - onCleanup(() => clearInterval(interval)) - }) - - const [composing, setComposing] = createSignal(false) - const isImeComposing = (event: KeyboardEvent) => event.isComposing || composing() || event.keyCode === 229 - - const handleBlur = () => { - closePopover() - setComposing(false) - } - - const handleCompositionStart = () => { - setComposing(true) - } - - const handleCompositionEnd = () => { - setComposing(false) - requestAnimationFrame(() => { - if (composing()) return - reconcile(prompt.current().filter((part) => part.type !== "image")) - }) - } - - const agentList = createMemo(() => - sync.data.agent - .filter((agent) => !agent.hidden && agent.mode !== "primary") - .map((agent): AtOption => ({ type: "agent", name: agent.name, display: agent.name })), - ) - const agentNames = createMemo(() => local.agent.list().map((agent) => agent.name)) - - const handleAtSelect = (option: AtOption | undefined) => { - if (!option) return - if (option.type === "agent") { - addPart({ type: "agent", name: option.name, content: "@" + option.name, start: 0, end: 0 }) - } else { - addPart({ type: "file", path: option.path, content: "@" + option.path, start: 0, end: 0 }) - } - } - - const atKey = (x: AtOption | undefined) => { - if (!x) return "" - return x.type === "agent" ? `agent:${x.name}` : `file:${x.path}` - } - - const { - flat: atFlat, - active: atActive, - setActive: setAtActive, - onInput: atOnInput, - onKeyDown: atOnKeyDown, - } = useFilteredList({ - items: async (query) => { - const agents = agentList() - const open = recent() - const seen = new Set(open) - const pinned: AtOption[] = open.map((path) => ({ type: "file", path, display: path, recent: true })) - if (!query.trim()) return [...agents, ...pinned] - const paths = await files.searchFilesAndDirectories(query) - const fileOptions: AtOption[] = paths - .filter((path) => !seen.has(path)) - .map((path) => ({ type: "file", path, display: path })) - return [...agents, ...pinned, ...fileOptions] - }, - key: atKey, - filterKeys: ["display"], - groupBy: (item) => { - if (item.type === "agent") return "agent" - if (item.recent) return "recent" - return "file" - }, - sortGroupsBy: (a, b) => { - const rank = (category: string) => { - if (category === "agent") return 0 - if (category === "recent") return 1 - return 2 - } - return rank(a.category) - rank(b.category) - }, - onSelect: handleAtSelect, - }) - - const slashCommands = createMemo(() => { - const builtin = command.options - .filter((opt) => !opt.disabled && !opt.id.startsWith("suggested.") && opt.slash) - .map((opt) => ({ - id: opt.id, - trigger: opt.slash!, - title: opt.title, - description: opt.description, - keybind: opt.keybind, - type: "builtin" as const, - })) - - const custom = sync.data.command.map((cmd) => ({ - id: `custom.${cmd.name}`, - trigger: cmd.name, - title: cmd.name, - description: cmd.description, - type: "custom" as const, - source: cmd.source, - })) - - return [...custom, ...builtin] - }) - - const handleSlashSelect = (cmd: SlashCommand | undefined) => { - if (!cmd) return - closePopover() - const images = imageAttachments() - - if (cmd.type === "custom") { - const text = `/${cmd.trigger} ` - setEditorText(text) - prompt.set([{ type: "text", content: text, start: 0, end: text.length }, ...images], text.length) - focusEditorEnd() - return - } - - clearEditor() - prompt.set([...DEFAULT_PROMPT, ...images], 0) - command.trigger(cmd.id, "slash") - } - - const { - flat: slashFlat, - active: slashActive, - setActive: setSlashActive, - onInput: slashOnInput, - onKeyDown: slashOnKeyDown, - } = useFilteredList({ - items: slashCommands, - key: (x) => x?.id, - filterKeys: ["trigger", "title"], - onSelect: handleSlashSelect, - }) - - const createPill = (part: FileAttachmentPart | AgentPart) => { - const pill = document.createElement("span") - pill.textContent = part.content - pill.setAttribute("data-type", part.type) - if (part.type === "file") pill.setAttribute("data-path", part.path) - if (part.type === "agent") pill.setAttribute("data-name", part.name) - pill.setAttribute("contenteditable", "false") - pill.style.userSelect = "text" - pill.style.cursor = "default" - return pill - } - - const isNormalizedEditor = () => - Array.from(editorRef.childNodes).every((node) => { - if (node.nodeType === Node.TEXT_NODE) { - const text = node.textContent ?? "" - if (!text.includes("\u200B")) return true - if (text !== "\u200B") return false - - const prev = node.previousSibling - const next = node.nextSibling - const prevIsBr = prev?.nodeType === Node.ELEMENT_NODE && (prev as HTMLElement).tagName === "BR" - return !!prevIsBr && !next - } - if (node.nodeType !== Node.ELEMENT_NODE) return false - const el = node as HTMLElement - if (el.dataset.type === "file") return true - if (el.dataset.type === "agent") return true - return el.tagName === "BR" - }) - - const renderEditor = (parts: Prompt) => { - clearEditor() - for (const part of parts) { - if (part.type === "text") { - editorRef.appendChild(createTextFragment(part.content)) - continue - } - if (part.type === "file" || part.type === "agent") { - editorRef.appendChild(createPill(part)) - } - } - - const last = editorRef.lastChild - if (last?.nodeType === Node.ELEMENT_NODE && (last as HTMLElement).tagName === "BR") { - editorRef.appendChild(document.createTextNode("\u200B")) - } - } - - // Auto-scroll active command into view when navigating with keyboard - createEffect(() => { - const activeId = slashActive() - if (!activeId || !slashPopoverRef) return - - requestAnimationFrame(() => { - const element = slashPopoverRef.querySelector(`[data-slash-id="${activeId}"]`) - element?.scrollIntoView({ block: "nearest", behavior: "smooth" }) - }) - }) - const selectPopoverActive = () => { - if (store.popover === "at") { - const items = atFlat() - if (items.length === 0) return - const active = atActive() - const item = items.find((entry) => atKey(entry) === active) ?? items[0] - handleAtSelect(item) - return - } - - if (store.popover === "slash") { - const items = slashFlat() - if (items.length === 0) return - const active = slashActive() - const item = items.find((entry) => entry.id === active) ?? items[0] - handleSlashSelect(item) - } - } - - const reconcile = (input: Prompt) => { - if (mirror.input) { - mirror.input = false - if (isNormalizedEditor()) return - - renderEditorWithCursor(input) - return - } - - const dom = parseFromDOM() - if (isNormalizedEditor() && isPromptEqual(input, dom)) return - - renderEditorWithCursor(input) - } - - createEffect( - on( - () => prompt.current(), - (parts) => { - if (composing()) return - reconcile(parts.filter((part) => part.type !== "image")) - }, - ), - ) - - const parseFromDOM = (): Prompt => { - const parts: Prompt = [] - let position = 0 - let buffer = "" - - const flushText = () => { - let content = buffer - if (content.includes("\r")) content = content.replace(/\r\n?/g, "\n") - if (content.includes("\u200B")) content = content.replace(/\u200B/g, "") - buffer = "" - if (!content) return - parts.push({ type: "text", content, start: position, end: position + content.length }) - position += content.length - } - - const pushFile = (file: HTMLElement) => { - const content = file.textContent ?? "" - parts.push({ - type: "file", - path: file.dataset.path!, - content, - start: position, - end: position + content.length, - }) - position += content.length - } - - const pushAgent = (agent: HTMLElement) => { - const content = agent.textContent ?? "" - parts.push({ - type: "agent", - name: agent.dataset.name!, - content, - start: position, - end: position + content.length, - }) - position += content.length - } - - const visit = (node: Node) => { - if (node.nodeType === Node.TEXT_NODE) { - buffer += node.textContent ?? "" - return - } - if (node.nodeType !== Node.ELEMENT_NODE) return - - const el = node as HTMLElement - if (el.dataset.type === "file") { - flushText() - pushFile(el) - return - } - if (el.dataset.type === "agent") { - flushText() - pushAgent(el) - return - } - if (el.tagName === "BR") { - buffer += "\n" - return - } - - for (const child of Array.from(el.childNodes)) { - visit(child) - } - } - - const children = Array.from(editorRef.childNodes) - children.forEach((child, index) => { - const isBlock = child.nodeType === Node.ELEMENT_NODE && ["DIV", "P"].includes((child as HTMLElement).tagName) - visit(child) - if (isBlock && index < children.length - 1) { - buffer += "\n" - } - }) - - flushText() - - if (parts.length === 0) parts.push(...DEFAULT_PROMPT) - return parts - } - - const handleInput = () => { - const rawParts = parseFromDOM() - const images = imageAttachments() - const cursorPosition = getCursorPosition(editorRef) - const rawText = - rawParts.length === 1 && rawParts[0]?.type === "text" - ? rawParts[0].content - : rawParts.map((p) => ("content" in p ? p.content : "")).join("") - const hasNonText = rawParts.some((part) => part.type !== "text") - const shouldReset = !NON_EMPTY_TEXT.test(rawText) && !hasNonText && images.length === 0 - - if (shouldReset) { - closePopover() - resetHistoryNavigation() - if (prompt.dirty()) { - mirror.input = true - prompt.set(DEFAULT_PROMPT, 0) - } - queueScroll() - return - } - - const shellMode = store.mode === "shell" - - if (!shellMode) { - const atMatch = rawText.substring(0, cursorPosition).match(/@(\S*)$/) - const slashMatch = rawText.match(/^\/(\S*)$/) - - if (atMatch) { - atOnInput(atMatch[1]) - setStore("popover", "at") - } else if (slashMatch) { - slashOnInput(slashMatch[1]) - setStore("popover", "slash") - } else { - closePopover() - } - } else { - closePopover() - } - - resetHistoryNavigation() - - mirror.input = true - prompt.set([...rawParts, ...images], cursorPosition) - queueScroll() - } - - const addPart = (part: ContentPart) => { - if (part.type === "image") return false - - const selection = window.getSelection() - if (!selection) return false - - if (selection.rangeCount === 0 || !editorRef.contains(selection.anchorNode)) { - editorRef.focus() - const cursor = prompt.cursor() ?? promptLength(prompt.current()) - setCursorPosition(editorRef, cursor) - } - - if (selection.rangeCount === 0) return false - const range = selection.getRangeAt(0) - if (!editorRef.contains(range.startContainer)) return false - - if (part.type === "file" || part.type === "agent") { - const cursorPosition = getCursorPosition(editorRef) - const rawText = prompt - .current() - .map((p) => ("content" in p ? p.content : "")) - .join("") - const textBeforeCursor = rawText.substring(0, cursorPosition) - const atMatch = textBeforeCursor.match(/@(\S*)$/) - const pill = createPill(part) - const gap = document.createTextNode(" ") - - if (atMatch) { - const start = atMatch.index ?? cursorPosition - atMatch[0].length - setRangeEdge(editorRef, range, "start", start) - setRangeEdge(editorRef, range, "end", cursorPosition) - } - - range.deleteContents() - range.insertNode(gap) - range.insertNode(pill) - range.setStartAfter(gap) - range.collapse(true) - selection.removeAllRanges() - selection.addRange(range) - } - - if (part.type === "text") { - const fragment = createTextFragment(part.content) - const last = fragment.lastChild - range.deleteContents() - range.insertNode(fragment) - if (last) { - if (last.nodeType === Node.TEXT_NODE) { - const text = last.textContent ?? "" - if (text === "\u200B") { - range.setStart(last, 0) - } - if (text !== "\u200B") { - range.setStart(last, text.length) - } - } - if (last.nodeType !== Node.TEXT_NODE) { - const isBreak = last.nodeType === Node.ELEMENT_NODE && (last as HTMLElement).tagName === "BR" - const next = last.nextSibling - const emptyText = next?.nodeType === Node.TEXT_NODE && (next.textContent ?? "") === "" - if (isBreak && (!next || emptyText)) { - const placeholder = next && emptyText ? next : document.createTextNode("\u200B") - if (!next) last.parentNode?.insertBefore(placeholder, null) - placeholder.textContent = "\u200B" - range.setStart(placeholder, 0) - } else { - range.setStartAfter(last) - } - } - } - range.collapse(true) - selection.removeAllRanges() - selection.addRange(range) - } - - handleInput() - closePopover() - return true - } - - const addToHistory = (prompt: Prompt, mode: "normal" | "shell") => { - const currentHistory = mode === "shell" ? shellHistory : history - const setCurrentHistory = mode === "shell" ? setShellHistory : setHistory - const next = prependHistoryEntry(currentHistory.entries, prompt, mode === "shell" ? [] : historyComments()) - if (next === currentHistory.entries) return - setCurrentHistory("entries", next) - } - - createEffect( - on( - () => props.edit?.id, - (id) => { - const edit = props.edit - if (!id || !edit) return - - for (const item of prompt.context.items()) { - prompt.context.remove(item.key) - } - - for (const item of edit.context) { - prompt.context.add({ - type: item.type, - path: item.path, - selection: item.selection, - comment: item.comment, - commentID: item.commentID, - commentOrigin: item.commentOrigin, - preview: item.preview, - }) - } - - setStore("mode", "normal") - setStore("popover", null) - setStore("historyIndex", -1) - setStore("savedPrompt", null) - prompt.set(edit.prompt, promptLength(edit.prompt)) - requestAnimationFrame(() => { - editorRef.focus() - setCursorPosition(editorRef, promptLength(edit.prompt)) - queueScroll() - }) - props.onEditLoaded?.() - }, - { defer: true }, - ), - ) - - const navigateHistory = (direction: "up" | "down") => { - const result = navigatePromptHistory({ - direction, - entries: store.mode === "shell" ? shellHistory.entries : history.entries, - historyIndex: store.historyIndex, - currentPrompt: prompt.current(), - currentComments: historyComments(), - savedPrompt: store.savedPrompt, - }) - if (!result.handled) return false - setStore("historyIndex", result.historyIndex) - setStore("savedPrompt", result.savedPrompt) - applyHistoryPrompt(result.entry, result.cursor) - return true - } - - const { addAttachments, removeAttachment, handlePaste } = createPromptAttachments({ - editor: () => editorRef, - isDialogActive: () => !!dialog.active, - setDraggingType: (type) => setStore("draggingType", type), - focusEditor: () => { - editorRef.focus() - setCursorPosition(editorRef, promptLength(prompt.current())) - }, - addPart, - readClipboardImage: platform.readClipboardImage, - }) - - const variants = createMemo(() => ["default", ...local.model.variant.list()]) - const accepting = createMemo(() => { - const id = params.id - if (!id) return permission.isAutoAcceptingDirectory(sdk.directory) - return permission.isAutoAccepting(id, sdk.directory) - }) - - const { abort, handleSubmit } = createPromptSubmit({ - info, - imageAttachments, - commentCount, - autoAccept: () => accepting(), - mode: () => store.mode, - working, - editor: () => editorRef, - queueScroll, - promptLength, - addToHistory, - resetHistoryNavigation: () => { - resetHistoryNavigation(true) - }, - setMode: (mode) => setStore("mode", mode), - setPopover: (popover) => setStore("popover", popover), - newSessionWorktree: () => props.newSessionWorktree, - onNewSessionWorktreeReset: props.onNewSessionWorktreeReset, - shouldQueue: props.shouldQueue, - onQueue: props.onQueue, - onAbort: props.onAbort, - onSubmit: props.onSubmit, - }) - - const handleKeyDown = (event: KeyboardEvent) => { - if ((event.metaKey || event.ctrlKey) && !event.altKey && !event.shiftKey && event.key.toLowerCase() === "u") { - event.preventDefault() - if (store.mode !== "normal") return - pick() - return - } - - if (event.key === "Backspace") { - const selection = window.getSelection() - if (selection && selection.isCollapsed) { - const node = selection.anchorNode - const offset = selection.anchorOffset - if (node && node.nodeType === Node.TEXT_NODE) { - const text = node.textContent ?? "" - if (/^\u200B+$/.test(text) && offset > 0) { - const range = document.createRange() - range.setStart(node, 0) - range.collapse(true) - selection.removeAllRanges() - selection.addRange(range) - } - } - } - } - - if (event.key === "!" && store.mode === "normal") { - const cursorPosition = getCursorPosition(editorRef) - if (cursorPosition === 0) { - setStore("mode", "shell") - setStore("popover", null) - event.preventDefault() - return - } - } - - if (event.key === "Escape") { - if (store.popover) { - closePopover() - event.preventDefault() - event.stopPropagation() - return - } - - if (store.mode === "shell") { - setStore("mode", "normal") - event.preventDefault() - event.stopPropagation() - return - } - - if (working()) { - void abort() - event.preventDefault() - event.stopPropagation() - return - } - - if (escBlur()) { - editorRef.blur() - event.preventDefault() - event.stopPropagation() - return - } - } - - if (store.mode === "shell") { - const { collapsed, cursorPosition, textLength } = getCaretState() - if (event.key === "Backspace" && collapsed && cursorPosition === 0 && textLength === 0) { - setStore("mode", "normal") - event.preventDefault() - return - } - } - - // Handle Shift+Enter BEFORE IME check - Shift+Enter is never used for IME input - // and should always insert a newline regardless of composition state - if (event.key === "Enter" && event.shiftKey) { - addPart({ type: "text", content: "\n", start: 0, end: 0 }) - event.preventDefault() - return - } - - if (event.key === "Enter" && isImeComposing(event)) { - return - } - - const ctrl = event.ctrlKey && !event.metaKey && !event.altKey && !event.shiftKey - - if (store.popover) { - if (event.key === "Tab") { - selectPopoverActive() - event.preventDefault() - return - } - const nav = event.key === "ArrowUp" || event.key === "ArrowDown" || event.key === "Enter" - const ctrlNav = ctrl && (event.key === "n" || event.key === "p") - if (nav || ctrlNav) { - if (store.popover === "at") { - atOnKeyDown(event) - event.preventDefault() - return - } - if (store.popover === "slash") { - slashOnKeyDown(event) - } - event.preventDefault() - return - } - } - - if (ctrl && event.code === "KeyG") { - if (store.popover) { - closePopover() - event.preventDefault() - return - } - if (working()) { - void abort() - event.preventDefault() - } - return - } - - if (event.key === "ArrowUp" || event.key === "ArrowDown") { - if (event.altKey || event.ctrlKey || event.metaKey) return - const { collapsed } = getCaretState() - if (!collapsed) return - - const cursorPosition = getCursorPosition(editorRef) - const textContent = prompt - .current() - .map((part) => ("content" in part ? part.content : "")) - .join("") - const direction = event.key === "ArrowUp" ? "up" : "down" - if (!canNavigateHistoryAtCursor(direction, textContent, cursorPosition, store.historyIndex >= 0)) return - if (navigateHistory(direction)) { - event.preventDefault() - } - return - } - - // Note: Shift+Enter is handled earlier, before IME check - if (event.key === "Enter" && !event.shiftKey) { - event.preventDefault() - if (event.repeat) return - if ( - working() && - prompt - .current() - .map((part) => ("content" in part ? part.content : "")) - .join("") - .trim().length === 0 && - imageAttachments().length === 0 && - commentCount() === 0 - ) { - return - } - void handleSubmit(event) - } - } - - const [agentsQuery, globalProvidersQuery, providersQuery] = useQueries(() => ({ - queries: [loadAgentsQuery(sdk.directory), loadProvidersQuery(null), loadProvidersQuery(sdk.directory)], - })) - - const agentsLoading = () => agentsQuery.isLoading - const agentsShouldFadeIn = createMemo((prev) => prev ?? agentsLoading()) - const providersLoading = () => agentsLoading() || providersQuery.isLoading || globalProvidersQuery.isLoading - const providersShouldFadeIn = createMemo((prev) => prev ?? providersLoading()) - - const [promptReady] = createResource( - () => prompt.ready().promise, - (p) => p, - ) - - return ( -
- {(promptReady(), null)} - (slashPopoverRef = el)} - atFlat={atFlat()} - atActive={atActive() ?? undefined} - atKey={atKey} - setAtActive={setAtActive} - onAtSelect={handleAtSelect} - slashFlat={slashFlat()} - slashActive={slashActive() ?? undefined} - setSlashActive={setSlashActive} - onSlashSelect={handleSlashSelect} - commandKeybind={command.keybind} - t={(key) => language.t(key as Parameters[0])} - /> - - - { - const active = comments.active() - return !!item.commentID && item.commentID === active?.id && item.path === active?.file - }} - openComment={openComment} - remove={(item) => { - if (item.commentID) comments.remove(item.path, item.commentID) - prompt.context.remove(item.key) - }} - t={(key) => language.t(key as Parameters[0])} - /> - - dialog.show(() => ) - } - onRemove={removeAttachment} - removeLabel={language.t("prompt.attachment.remove")} - /> -
{ - const target = e.target - if (!(target instanceof HTMLElement)) return - if (target.closest('[data-action="prompt-attach"], [data-action="prompt-submit"]')) { - return - } - editorRef?.focus() - }} - > -
(scrollRef = el)} - style={{ "scroll-padding-bottom": space }} - > -
{ - editorRef = el - props.ref?.(el) - }} - role="textbox" - aria-multiline="true" - aria-label={placeholder()} - contenteditable="true" - autocapitalize={store.mode === "normal" ? "sentences" : "off"} - autocorrect={store.mode === "normal" ? "on" : "off"} - spellcheck={store.mode === "normal"} - inputMode="text" - // @ts-expect-error - autocomplete="off" - onInput={handleInput} - onPaste={handlePaste} - onCompositionStart={handleCompositionStart} - onCompositionEnd={handleCompositionEnd} - onBlur={handleBlur} - onKeyDown={handleKeyDown} - classList={{ - "select-text": true, - "w-full pl-3 pr-2 pt-2 text-14-regular text-text-strong focus:outline-none whitespace-pre-wrap": true, - "[&_[data-type=file]]:text-syntax-property": true, - "[&_[data-type=agent]]:text-syntax-type": true, - "font-mono!": store.mode === "shell", - }} - style={{ "padding-bottom": space }} - /> -
- {placeholder()} -
-
- - - - - -
-
-
- - {language.t("prompt.mode.shell")} -
- -
-
- -
- - (x === "default" ? language.t("common.default") : x)} - onSelect={(value) => { - local.model.variant.set(value === "default" ? undefined : value) - restoreFocus() - }} - class="capitalize max-w-[160px] text-text-base" - valueClass="truncate text-13-regular text-text-base" - triggerStyle={control()} - triggerProps={{ "data-action": "prompt-model-variant" }} - variant="ghost" - /> - -
-
- - -
-
-
- - -
- ) -} diff --git a/packages/app/src/components/prompt-input/attachments.test.ts b/packages/app/src/components/prompt-input/attachments.test.ts deleted file mode 100644 index 43f7d425bd..0000000000 --- a/packages/app/src/components/prompt-input/attachments.test.ts +++ /dev/null @@ -1,44 +0,0 @@ -import { describe, expect, test } from "bun:test" -import { attachmentMime } from "./files" -import { pasteMode } from "./paste" - -describe("attachmentMime", () => { - test("keeps PDFs when the browser reports the mime", async () => { - const file = new File(["%PDF-1.7"], "guide.pdf", { type: "application/pdf" }) - expect(await attachmentMime(file)).toBe("application/pdf") - }) - - test("normalizes structured text types to text/plain", async () => { - const file = new File(['{"ok":true}\n'], "data.json", { type: "application/json" }) - expect(await attachmentMime(file)).toBe("text/plain") - }) - - test("accepts text files even with a misleading browser mime", async () => { - const file = new File(["export const x = 1\n"], "main.ts", { type: "video/mp2t" }) - expect(await attachmentMime(file)).toBe("text/plain") - }) - - test("rejects binary files", async () => { - const file = new File([Uint8Array.of(0, 255, 1, 2)], "blob.bin", { type: "application/octet-stream" }) - expect(await attachmentMime(file)).toBeUndefined() - }) -}) - -describe("pasteMode", () => { - test("uses native paste for short single-line text", () => { - expect(pasteMode("hello world")).toBe("native") - }) - - test("uses manual paste for multiline text", () => { - expect( - pasteMode(`{ - "ok": true -}`), - ).toBe("manual") - expect(pasteMode("a\r\nb")).toBe("manual") - }) - - test("uses manual paste for large text", () => { - expect(pasteMode("x".repeat(8000))).toBe("manual") - }) -}) diff --git a/packages/app/src/components/prompt-input/attachments.ts b/packages/app/src/components/prompt-input/attachments.ts deleted file mode 100644 index f12a4210c0..0000000000 --- a/packages/app/src/components/prompt-input/attachments.ts +++ /dev/null @@ -1,196 +0,0 @@ -import { onMount } from "solid-js" -import { makeEventListener } from "@solid-primitives/event-listener" -import { showToast } from "@opencode-ai/ui/toast" -import { usePrompt, type ContentPart, type ImageAttachmentPart } from "@/context/prompt" -import { useLanguage } from "@/context/language" -import { uuid } from "@/utils/uuid" -import { getCursorPosition } from "./editor-dom" -import { attachmentMime } from "./files" -import { normalizePaste, pasteMode } from "./paste" - -function dataUrl(file: File, mime: string) { - return new Promise((resolve) => { - const reader = new FileReader() - reader.addEventListener("error", () => resolve("")) - reader.addEventListener("load", () => { - const value = typeof reader.result === "string" ? reader.result : "" - const idx = value.indexOf(",") - if (idx === -1) { - resolve(value) - return - } - resolve(`data:${mime};base64,${value.slice(idx + 1)}`) - }) - reader.readAsDataURL(file) - }) -} - -type PromptAttachmentsInput = { - editor: () => HTMLDivElement | undefined - isDialogActive: () => boolean - setDraggingType: (type: "image" | "@mention" | null) => void - focusEditor: () => void - addPart: (part: ContentPart) => boolean - readClipboardImage?: () => Promise -} - -export function createPromptAttachments(input: PromptAttachmentsInput) { - const prompt = usePrompt() - const language = useLanguage() - - const warn = () => { - showToast({ - title: language.t("prompt.toast.pasteUnsupported.title"), - description: language.t("prompt.toast.pasteUnsupported.description"), - }) - } - - const add = async (file: File, toast = true) => { - const mime = await attachmentMime(file) - if (!mime) { - if (toast) warn() - return false - } - - const editor = input.editor() - if (!editor) return false - - const url = await dataUrl(file, mime) - if (!url) return false - - const attachment: ImageAttachmentPart = { - type: "image", - id: uuid(), - filename: file.name, - mime, - dataUrl: url, - } - const cursor = prompt.cursor() ?? getCursorPosition(editor) - prompt.set([...prompt.current(), attachment], cursor) - return true - } - - const addAttachment = (file: File) => add(file) - - const addAttachments = async (files: File[], toast = true) => { - let found = false - - for (const file of files) { - const ok = await add(file, false) - if (ok) found = true - } - - if (!found && files.length > 0 && toast) warn() - return found - } - - const removeAttachment = (id: string) => { - const current = prompt.current() - const next = current.filter((part) => part.type !== "image" || part.id !== id) - prompt.set(next, prompt.cursor()) - } - - const handlePaste = async (event: ClipboardEvent) => { - const clipboardData = event.clipboardData - if (!clipboardData) return - - event.preventDefault() - event.stopPropagation() - - const files = Array.from(clipboardData.items).flatMap((item) => { - if (item.kind !== "file") return [] - const file = item.getAsFile() - return file ? [file] : [] - }) - - if (files.length > 0) { - await addAttachments(files) - return - } - - const plainText = clipboardData.getData("text/plain") ?? "" - - // Desktop: Browser clipboard has no images and no text, try platform's native clipboard for images - if (input.readClipboardImage && !plainText) { - const file = await input.readClipboardImage() - if (file) { - await addAttachment(file) - return - } - } - - if (!plainText) return - - const text = normalizePaste(plainText) - - const put = () => { - if (input.addPart({ type: "text", content: text, start: 0, end: 0 })) return true - input.focusEditor() - return input.addPart({ type: "text", content: text, start: 0, end: 0 }) - } - - if (pasteMode(text) === "manual") { - put() - return - } - - const inserted = typeof document.execCommand === "function" && document.execCommand("insertText", false, text) - if (inserted) return - - put() - } - - const handleGlobalDragOver = (event: DragEvent) => { - if (input.isDialogActive()) return - - event.preventDefault() - const hasFiles = event.dataTransfer?.types.includes("Files") - const hasText = event.dataTransfer?.types.includes("text/plain") - if (hasFiles) { - input.setDraggingType("image") - } else if (hasText) { - input.setDraggingType("@mention") - } - } - - const handleGlobalDragLeave = (event: DragEvent) => { - if (input.isDialogActive()) return - if (!event.relatedTarget) { - input.setDraggingType(null) - } - } - - const handleGlobalDrop = async (event: DragEvent) => { - if (input.isDialogActive()) return - - event.preventDefault() - input.setDraggingType(null) - - const plainText = event.dataTransfer?.getData("text/plain") - const filePrefix = "file:" - if (plainText?.startsWith(filePrefix)) { - const filePath = plainText.slice(filePrefix.length) - input.focusEditor() - input.addPart({ type: "file", path: filePath, content: "@" + filePath, start: 0, end: 0 }) - return - } - - const dropped = event.dataTransfer?.files - if (!dropped) return - - await addAttachments(Array.from(dropped)) - } - - onMount(() => { - makeEventListener(document, "dragover", handleGlobalDragOver) - makeEventListener(document, "dragleave", handleGlobalDragLeave) - makeEventListener(document, "drop", handleGlobalDrop) - }) - - return { - addAttachment, - addAttachments, - removeAttachment, - handlePaste, - } -} diff --git a/packages/app/src/components/prompt-input/build-request-parts.test.ts b/packages/app/src/components/prompt-input/build-request-parts.test.ts deleted file mode 100644 index 06c3773310..0000000000 --- a/packages/app/src/components/prompt-input/build-request-parts.test.ts +++ /dev/null @@ -1,336 +0,0 @@ -import { describe, expect, test } from "bun:test" -import type { Prompt } from "@/context/prompt" -import { buildRequestParts } from "./build-request-parts" - -describe("buildRequestParts", () => { - test("builds typed request and optimistic parts without cast path", () => { - const prompt: Prompt = [ - { type: "text", content: "hello", start: 0, end: 5 }, - { - type: "file", - path: "src/foo.ts", - content: "@src/foo.ts", - start: 5, - end: 16, - selection: { startLine: 4, startChar: 1, endLine: 6, endChar: 1 }, - }, - { type: "agent", name: "planner", content: "@planner", start: 16, end: 24 }, - ] - - const result = buildRequestParts({ - prompt, - context: [{ key: "ctx:1", type: "file", path: "src/bar.ts", comment: "check this" }], - images: [ - { type: "image", id: "img_1", filename: "a.png", mime: "image/png", dataUrl: "data:image/png;base64,AAA" }, - ], - text: "hello @src/foo.ts @planner", - messageID: "msg_1", - sessionID: "ses_1", - sessionDirectory: "/repo", - }) - - expect(result.requestParts[0]?.type).toBe("text") - expect(result.requestParts.some((part) => part.type === "agent")).toBe(true) - expect( - result.requestParts.some((part) => part.type === "file" && part.url.startsWith("file:///repo/src/foo.ts")), - ).toBe(true) - expect(result.requestParts.some((part) => part.type === "text" && part.synthetic)).toBe(true) - expect( - result.requestParts.some( - (part) => - part.type === "text" && - part.synthetic && - part.metadata?.opencodeComment && - (part.metadata.opencodeComment as { comment?: string }).comment === "check this", - ), - ).toBe(true) - - expect(result.optimisticParts).toHaveLength(result.requestParts.length) - expect(result.optimisticParts.every((part) => part.sessionID === "ses_1" && part.messageID === "msg_1")).toBe(true) - }) - - test("keeps multiple uploaded attachments in order", () => { - const result = buildRequestParts({ - prompt: [{ type: "text", content: "check these", start: 0, end: 11 }], - context: [], - images: [ - { type: "image", id: "img_1", filename: "a.png", mime: "image/png", dataUrl: "data:image/png;base64,AAA" }, - { - type: "image", - id: "img_2", - filename: "b.pdf", - mime: "application/pdf", - dataUrl: "data:application/pdf;base64,BBB", - }, - ], - text: "check these", - messageID: "msg_multi", - sessionID: "ses_multi", - sessionDirectory: "/repo", - }) - - const files = result.requestParts.filter((part) => part.type === "file" && part.url.startsWith("data:")) - - expect(files).toHaveLength(2) - expect(files.map((part) => (part.type === "file" ? part.filename : ""))).toEqual(["a.png", "b.pdf"]) - }) - - test("deduplicates context files when prompt already includes same path", () => { - const prompt: Prompt = [{ type: "file", path: "src/foo.ts", content: "@src/foo.ts", start: 0, end: 11 }] - - const result = buildRequestParts({ - prompt, - context: [ - { key: "ctx:dup", type: "file", path: "src/foo.ts" }, - { key: "ctx:comment", type: "file", path: "src/foo.ts", comment: "focus here" }, - ], - images: [], - text: "@src/foo.ts", - messageID: "msg_2", - sessionID: "ses_2", - sessionDirectory: "/repo", - }) - - const fooFiles = result.requestParts.filter( - (part) => part.type === "file" && part.url.startsWith("file:///repo/src/foo.ts"), - ) - const synthetic = result.requestParts.filter((part) => part.type === "text" && part.synthetic) - - expect(fooFiles).toHaveLength(2) - expect(synthetic).toHaveLength(1) - }) - - test("adds file parts for @mentions inside comment text", () => { - const result = buildRequestParts({ - prompt: [{ type: "text", content: "look", start: 0, end: 4 }], - context: [ - { - key: "ctx:comment-mention", - type: "file", - path: "src/review.ts", - comment: "Compare with @src/shared.ts and @src/review.ts.", - }, - ], - images: [], - text: "look", - messageID: "msg_comment_mentions", - sessionID: "ses_comment_mentions", - sessionDirectory: "/repo", - }) - - const files = result.requestParts.filter((part) => part.type === "file") - expect(files).toHaveLength(2) - expect(files.some((part) => part.type === "file" && part.url === "file:///repo/src/review.ts")).toBe(true) - expect(files.some((part) => part.type === "file" && part.url === "file:///repo/src/shared.ts")).toBe(true) - }) - - test("handles Windows paths correctly (simulated on macOS)", () => { - const prompt: Prompt = [{ type: "file", path: "src\\foo.ts", content: "@src\\foo.ts", start: 0, end: 11 }] - - const result = buildRequestParts({ - prompt, - context: [], - images: [], - text: "@src\\foo.ts", - messageID: "msg_win_1", - sessionID: "ses_win_1", - sessionDirectory: "D:\\projects\\myapp", // Windows path - }) - - // Should create valid file URLs - const filePart = result.requestParts.find((part) => part.type === "file") - expect(filePart).toBeDefined() - if (filePart?.type === "file") { - // URL should be parseable - expect(() => new URL(filePart.url)).not.toThrow() - // Should not have encoded backslashes in wrong place - expect(filePart.url).not.toContain("%5C") - // Should have normalized to forward slashes - expect(filePart.url).toContain("/src/foo.ts") - } - }) - - test("handles Windows absolute path with special characters", () => { - const prompt: Prompt = [{ type: "file", path: "file#name.txt", content: "@file#name.txt", start: 0, end: 14 }] - - const result = buildRequestParts({ - prompt, - context: [], - images: [], - text: "@file#name.txt", - messageID: "msg_win_2", - sessionID: "ses_win_2", - sessionDirectory: "C:\\Users\\test\\Documents", // Windows path - }) - - const filePart = result.requestParts.find((part) => part.type === "file") - expect(filePart).toBeDefined() - if (filePart?.type === "file") { - // URL should be parseable - expect(() => new URL(filePart.url)).not.toThrow() - // Special chars should be encoded - expect(filePart.url).toContain("file%23name.txt") - // Should have Windows drive letter properly encoded - expect(filePart.url).toMatch(/file:\/\/\/[A-Z]:/) - } - }) - - test("handles Linux absolute paths correctly", () => { - const prompt: Prompt = [{ type: "file", path: "src/app.ts", content: "@src/app.ts", start: 0, end: 10 }] - - const result = buildRequestParts({ - prompt, - context: [], - images: [], - text: "@src/app.ts", - messageID: "msg_linux_1", - sessionID: "ses_linux_1", - sessionDirectory: "/home/user/project", - }) - - const filePart = result.requestParts.find((part) => part.type === "file") - expect(filePart).toBeDefined() - if (filePart?.type === "file") { - // URL should be parseable - expect(() => new URL(filePart.url)).not.toThrow() - // Should be a normal Unix path - expect(filePart.url).toBe("file:///home/user/project/src/app.ts") - } - }) - - test("handles macOS paths correctly", () => { - const prompt: Prompt = [{ type: "file", path: "README.md", content: "@README.md", start: 0, end: 9 }] - - const result = buildRequestParts({ - prompt, - context: [], - images: [], - text: "@README.md", - messageID: "msg_mac_1", - sessionID: "ses_mac_1", - sessionDirectory: "/Users/kelvin/Projects/opencode", - }) - - const filePart = result.requestParts.find((part) => part.type === "file") - expect(filePart).toBeDefined() - if (filePart?.type === "file") { - // URL should be parseable - expect(() => new URL(filePart.url)).not.toThrow() - // Should be a normal Unix path - expect(filePart.url).toBe("file:///Users/kelvin/Projects/opencode/README.md") - } - }) - - test("handles context files with Windows paths", () => { - const prompt: Prompt = [] - - const result = buildRequestParts({ - prompt, - context: [ - { key: "ctx:1", type: "file", path: "src\\utils\\helper.ts" }, - { key: "ctx:2", type: "file", path: "test\\unit.test.ts", comment: "check tests" }, - ], - images: [], - text: "test", - messageID: "msg_win_ctx", - sessionID: "ses_win_ctx", - sessionDirectory: "D:\\workspace\\app", - }) - - const fileParts = result.requestParts.filter((part) => part.type === "file") - expect(fileParts).toHaveLength(2) - - // All file URLs should be valid - fileParts.forEach((part) => { - if (part.type === "file") { - expect(() => new URL(part.url)).not.toThrow() - expect(part.url).not.toContain("%5C") // No encoded backslashes - } - }) - }) - - test("handles absolute Windows paths (user manually specifies full path)", () => { - const prompt: Prompt = [ - { type: "file", path: "D:\\other\\project\\file.ts", content: "@D:\\other\\project\\file.ts", start: 0, end: 25 }, - ] - - const result = buildRequestParts({ - prompt, - context: [], - images: [], - text: "@D:\\other\\project\\file.ts", - messageID: "msg_abs", - sessionID: "ses_abs", - sessionDirectory: "C:\\current\\project", - }) - - const filePart = result.requestParts.find((part) => part.type === "file") - expect(filePart).toBeDefined() - if (filePart?.type === "file") { - // Should handle absolute path that differs from sessionDirectory - expect(() => new URL(filePart.url)).not.toThrow() - expect(filePart.url).toContain("/D:/other/project/file.ts") - } - }) - - test("handles selection with query parameters on Windows", () => { - const prompt: Prompt = [ - { - type: "file", - path: "src\\App.tsx", - content: "@src\\App.tsx", - start: 0, - end: 11, - selection: { startLine: 10, startChar: 0, endLine: 20, endChar: 5 }, - }, - ] - - const result = buildRequestParts({ - prompt, - context: [], - images: [], - text: "@src\\App.tsx", - messageID: "msg_sel", - sessionID: "ses_sel", - sessionDirectory: "C:\\project", - }) - - const filePart = result.requestParts.find((part) => part.type === "file") - expect(filePart).toBeDefined() - if (filePart?.type === "file") { - // Should have query parameters - expect(filePart.url).toContain("?start=10&end=20") - // Should be valid URL - expect(() => new URL(filePart.url)).not.toThrow() - // Query params should parse correctly - const url = new URL(filePart.url) - expect(url.searchParams.get("start")).toBe("10") - expect(url.searchParams.get("end")).toBe("20") - } - }) - - test("handles file paths with dots and special segments on Windows", () => { - const prompt: Prompt = [ - { type: "file", path: "..\\..\\shared\\util.ts", content: "@..\\..\\shared\\util.ts", start: 0, end: 21 }, - ] - - const result = buildRequestParts({ - prompt, - context: [], - images: [], - text: "@..\\..\\shared\\util.ts", - messageID: "msg_dots", - sessionID: "ses_dots", - sessionDirectory: "C:\\projects\\myapp\\src", - }) - - const filePart = result.requestParts.find((part) => part.type === "file") - expect(filePart).toBeDefined() - if (filePart?.type === "file") { - // Should be valid URL - expect(() => new URL(filePart.url)).not.toThrow() - // Should preserve .. segments (backend normalizes) - expect(filePart.url).toContain("/..") - } - }) -}) diff --git a/packages/app/src/components/prompt-input/drag-overlay.tsx b/packages/app/src/components/prompt-input/drag-overlay.tsx deleted file mode 100644 index 41962ce536..0000000000 --- a/packages/app/src/components/prompt-input/drag-overlay.tsx +++ /dev/null @@ -1,25 +0,0 @@ -import { Component, Show } from "solid-js" -import { Icon } from "@opencode-ai/ui/icon" - -type PromptDragOverlayProps = { - type: "image" | "@mention" | null - label: string -} - -const kindToIcon = { - image: "photo", - "@mention": "link", -} as const - -export const PromptDragOverlay: Component = (props) => { - return ( - -
-
- - {props.label} -
-
-
- ) -} diff --git a/packages/app/src/components/prompt-input/editor-dom.test.ts b/packages/app/src/components/prompt-input/editor-dom.test.ts deleted file mode 100644 index 3088522a59..0000000000 --- a/packages/app/src/components/prompt-input/editor-dom.test.ts +++ /dev/null @@ -1,99 +0,0 @@ -import { describe, expect, test } from "bun:test" -import { createTextFragment, getCursorPosition, getNodeLength, getTextLength, setCursorPosition } from "./editor-dom" - -describe("prompt-input editor dom", () => { - test("createTextFragment preserves newlines with consecutive br nodes", () => { - const fragment = createTextFragment("foo\n\nbar") - const container = document.createElement("div") - container.appendChild(fragment) - - expect(container.childNodes.length).toBe(4) - expect(container.childNodes[0]?.textContent).toBe("foo") - expect((container.childNodes[1] as HTMLElement).tagName).toBe("BR") - expect((container.childNodes[2] as HTMLElement).tagName).toBe("BR") - expect(container.childNodes[3]?.textContent).toBe("bar") - }) - - test("createTextFragment keeps trailing newline as terminal break", () => { - const fragment = createTextFragment("foo\n") - const container = document.createElement("div") - container.appendChild(fragment) - - expect(container.childNodes.length).toBe(2) - expect(container.childNodes[0]?.textContent).toBe("foo") - expect((container.childNodes[1] as HTMLElement).tagName).toBe("BR") - }) - - test("createTextFragment avoids break-node explosion for large multiline content", () => { - const content = Array.from({ length: 220 }, () => "line").join("\n") - const fragment = createTextFragment(content) - const container = document.createElement("div") - container.appendChild(fragment) - - expect(container.childNodes.length).toBe(1) - expect(container.childNodes[0]?.nodeType).toBe(Node.TEXT_NODE) - expect(container.textContent).toBe(content) - }) - - test("createTextFragment keeps terminal break in large multiline fallback", () => { - const content = `${Array.from({ length: 220 }, () => "line").join("\n")}\n` - const fragment = createTextFragment(content) - const container = document.createElement("div") - container.appendChild(fragment) - - expect(container.childNodes.length).toBe(2) - expect(container.childNodes[0]?.textContent).toBe(content.slice(0, -1)) - expect((container.childNodes[1] as HTMLElement).tagName).toBe("BR") - }) - - test("length helpers treat breaks as one char and ignore zero-width chars", () => { - const container = document.createElement("div") - container.appendChild(document.createTextNode("ab\u200B")) - container.appendChild(document.createElement("br")) - container.appendChild(document.createTextNode("cd")) - - expect(getNodeLength(container.childNodes[0]!)).toBe(2) - expect(getNodeLength(container.childNodes[1]!)).toBe(1) - expect(getTextLength(container)).toBe(5) - }) - - test("setCursorPosition and getCursorPosition round-trip with pills and breaks", () => { - const container = document.createElement("div") - const pill = document.createElement("span") - pill.dataset.type = "file" - pill.textContent = "@file" - container.appendChild(document.createTextNode("ab")) - container.appendChild(pill) - container.appendChild(document.createElement("br")) - container.appendChild(document.createTextNode("cd")) - document.body.appendChild(container) - - setCursorPosition(container, 2) - expect(getCursorPosition(container)).toBe(2) - - setCursorPosition(container, 7) - expect(getCursorPosition(container)).toBe(7) - - setCursorPosition(container, 8) - expect(getCursorPosition(container)).toBe(8) - - container.remove() - }) - - test("setCursorPosition and getCursorPosition round-trip across blank lines", () => { - const container = document.createElement("div") - container.appendChild(document.createTextNode("a")) - container.appendChild(document.createElement("br")) - container.appendChild(document.createElement("br")) - container.appendChild(document.createTextNode("b")) - document.body.appendChild(container) - - setCursorPosition(container, 2) - expect(getCursorPosition(container)).toBe(2) - - setCursorPosition(container, 3) - expect(getCursorPosition(container)).toBe(3) - - container.remove() - }) -}) diff --git a/packages/app/src/components/prompt-input/editor-dom.ts b/packages/app/src/components/prompt-input/editor-dom.ts deleted file mode 100644 index 8575140d7d..0000000000 --- a/packages/app/src/components/prompt-input/editor-dom.ts +++ /dev/null @@ -1,148 +0,0 @@ -const MAX_BREAKS = 200 - -export function createTextFragment(content: string): DocumentFragment { - const fragment = document.createDocumentFragment() - let breaks = 0 - for (const char of content) { - if (char !== "\n") continue - breaks += 1 - if (breaks > MAX_BREAKS) { - const tail = content.endsWith("\n") - const text = tail ? content.slice(0, -1) : content - if (text) fragment.appendChild(document.createTextNode(text)) - if (tail) fragment.appendChild(document.createElement("br")) - return fragment - } - } - - const segments = content.split("\n") - segments.forEach((segment, index) => { - if (segment) { - fragment.appendChild(document.createTextNode(segment)) - } - if (index < segments.length - 1) { - fragment.appendChild(document.createElement("br")) - } - }) - return fragment -} - -export function getNodeLength(node: Node): number { - if (node.nodeType === Node.ELEMENT_NODE && (node as HTMLElement).tagName === "BR") return 1 - return (node.textContent ?? "").replace(/\u200B/g, "").length -} - -export function getTextLength(node: Node): number { - if (node.nodeType === Node.TEXT_NODE) return (node.textContent ?? "").replace(/\u200B/g, "").length - if (node.nodeType === Node.ELEMENT_NODE && (node as HTMLElement).tagName === "BR") return 1 - let length = 0 - for (const child of Array.from(node.childNodes)) { - length += getTextLength(child) - } - return length -} - -export function getCursorPosition(parent: HTMLElement): number { - const selection = window.getSelection() - if (!selection || selection.rangeCount === 0) return 0 - const range = selection.getRangeAt(0) - if (!parent.contains(range.startContainer)) return 0 - const preCaretRange = range.cloneRange() - preCaretRange.selectNodeContents(parent) - preCaretRange.setEnd(range.startContainer, range.startOffset) - return getTextLength(preCaretRange.cloneContents()) -} - -export function setCursorPosition(parent: HTMLElement, position: number) { - let remaining = position - let node = parent.firstChild - while (node) { - const length = getNodeLength(node) - const isText = node.nodeType === Node.TEXT_NODE - const isPill = - node.nodeType === Node.ELEMENT_NODE && - ((node as HTMLElement).dataset.type === "file" || (node as HTMLElement).dataset.type === "agent") - const isBreak = node.nodeType === Node.ELEMENT_NODE && (node as HTMLElement).tagName === "BR" - - if (isText && remaining <= length) { - const range = document.createRange() - const selection = window.getSelection() - range.setStart(node, remaining) - range.collapse(true) - selection?.removeAllRanges() - selection?.addRange(range) - return - } - - if ((isPill || isBreak) && remaining <= length) { - const range = document.createRange() - const selection = window.getSelection() - if (remaining === 0) { - range.setStartBefore(node) - } - if (remaining > 0 && isPill) { - range.setStartAfter(node) - } - if (remaining > 0 && isBreak) { - const next = node.nextSibling - if (next && next.nodeType === Node.TEXT_NODE) { - range.setStart(next, 0) - } - if (!next || next.nodeType !== Node.TEXT_NODE) { - range.setStartAfter(node) - } - } - range.collapse(true) - selection?.removeAllRanges() - selection?.addRange(range) - return - } - - remaining -= length - node = node.nextSibling - } - - const fallbackRange = document.createRange() - const fallbackSelection = window.getSelection() - const last = parent.lastChild - if (last && last.nodeType === Node.TEXT_NODE) { - const len = last.textContent ? last.textContent.length : 0 - fallbackRange.setStart(last, len) - } - if (!last || last.nodeType !== Node.TEXT_NODE) { - fallbackRange.selectNodeContents(parent) - } - fallbackRange.collapse(false) - fallbackSelection?.removeAllRanges() - fallbackSelection?.addRange(fallbackRange) -} - -export function setRangeEdge(parent: HTMLElement, range: Range, edge: "start" | "end", offset: number) { - let remaining = offset - const nodes = Array.from(parent.childNodes) - - for (const node of nodes) { - const length = getNodeLength(node) - const isText = node.nodeType === Node.TEXT_NODE - const isPill = - node.nodeType === Node.ELEMENT_NODE && - ((node as HTMLElement).dataset.type === "file" || (node as HTMLElement).dataset.type === "agent") - const isBreak = node.nodeType === Node.ELEMENT_NODE && (node as HTMLElement).tagName === "BR" - - if (isText && remaining <= length) { - if (edge === "start") range.setStart(node, remaining) - if (edge === "end") range.setEnd(node, remaining) - return - } - - if ((isPill || isBreak) && remaining <= length) { - if (edge === "start" && remaining === 0) range.setStartBefore(node) - if (edge === "start" && remaining > 0) range.setStartAfter(node) - if (edge === "end" && remaining === 0) range.setEndBefore(node) - if (edge === "end" && remaining > 0) range.setEndAfter(node) - return - } - - remaining -= length - } -} diff --git a/packages/app/src/components/prompt-input/files.ts b/packages/app/src/components/prompt-input/files.ts deleted file mode 100644 index eae8af03d9..0000000000 --- a/packages/app/src/components/prompt-input/files.ts +++ /dev/null @@ -1,66 +0,0 @@ -import { ACCEPTED_FILE_TYPES, ACCEPTED_IMAGE_TYPES } from "@/constants/file-picker" - -export { ACCEPTED_FILE_TYPES } - -const IMAGE_MIMES = new Set(ACCEPTED_IMAGE_TYPES) -const IMAGE_EXTS = new Map([ - ["gif", "image/gif"], - ["jpeg", "image/jpeg"], - ["jpg", "image/jpeg"], - ["png", "image/png"], - ["webp", "image/webp"], -]) -const TEXT_MIMES = new Set([ - "application/json", - "application/ld+json", - "application/toml", - "application/x-toml", - "application/x-yaml", - "application/xml", - "application/yaml", -]) - -const SAMPLE = 4096 - -function kind(type: string) { - return type.split(";", 1)[0]?.trim().toLowerCase() ?? "" -} - -function ext(name: string) { - const idx = name.lastIndexOf(".") - if (idx === -1) return "" - return name.slice(idx + 1).toLowerCase() -} - -function textMime(type: string) { - if (!type) return false - if (type.startsWith("text/")) return true - if (TEXT_MIMES.has(type)) return true - if (type.endsWith("+json")) return true - return type.endsWith("+xml") -} - -function textBytes(bytes: Uint8Array) { - if (bytes.length === 0) return true - let count = 0 - for (const byte of bytes) { - if (byte === 0) return false - if (byte < 9 || (byte > 13 && byte < 32)) count += 1 - } - return count / bytes.length <= 0.3 -} - -export async function attachmentMime(file: File) { - const type = kind(file.type) - if (IMAGE_MIMES.has(type)) return type - if (type === "application/pdf") return type - - const suffix = ext(file.name) - const fallback = IMAGE_EXTS.get(suffix) ?? (suffix === "pdf" ? "application/pdf" : undefined) - if ((!type || type === "application/octet-stream") && fallback) return fallback - - if (textMime(type)) return "text/plain" - const bytes = new Uint8Array(await file.slice(0, SAMPLE).arrayBuffer()) - if (!textBytes(bytes)) return - return "text/plain" -} diff --git a/packages/app/src/components/prompt-input/history.test.ts b/packages/app/src/components/prompt-input/history.test.ts deleted file mode 100644 index 5e9c2c66ea..0000000000 --- a/packages/app/src/components/prompt-input/history.test.ts +++ /dev/null @@ -1,153 +0,0 @@ -import { describe, expect, test } from "bun:test" -import type { Prompt } from "@/context/prompt" -import { - canNavigateHistoryAtCursor, - clonePromptParts, - normalizePromptHistoryEntry, - navigatePromptHistory, - prependHistoryEntry, - promptLength, - type PromptHistoryComment, -} from "./history" - -const DEFAULT_PROMPT: Prompt = [{ type: "text", content: "", start: 0, end: 0 }] - -const text = (value: string): Prompt => [{ type: "text", content: value, start: 0, end: value.length }] -const comment = (id: string, value = "note"): PromptHistoryComment => ({ - id, - path: "src/a.ts", - selection: { start: 2, end: 4 }, - comment: value, - time: 1, - origin: "review", - preview: "const a = 1", -}) - -describe("prompt-input history", () => { - test("prependHistoryEntry skips empty prompt and deduplicates consecutive entries", () => { - const first = prependHistoryEntry([], DEFAULT_PROMPT) - expect(first).toEqual([]) - - const commentsOnly = prependHistoryEntry([], DEFAULT_PROMPT, [comment("c1")]) - expect(commentsOnly).toHaveLength(1) - - const withOne = prependHistoryEntry([], text("hello")) - expect(withOne).toHaveLength(1) - - const deduped = prependHistoryEntry(withOne, text("hello")) - expect(deduped).toBe(withOne) - - const dedupedComments = prependHistoryEntry(commentsOnly, DEFAULT_PROMPT, [comment("c1")]) - expect(dedupedComments).toBe(commentsOnly) - }) - - test("navigatePromptHistory restores saved prompt when moving down from newest", () => { - const entries = [text("third"), text("second"), text("first")] - const up = navigatePromptHistory({ - direction: "up", - entries, - historyIndex: -1, - currentPrompt: text("draft"), - currentComments: [comment("draft")], - savedPrompt: null, - }) - expect(up.handled).toBe(true) - if (!up.handled) throw new Error("expected handled") - expect(up.historyIndex).toBe(0) - expect(up.cursor).toBe("start") - expect(up.entry.comments).toEqual([]) - - const down = navigatePromptHistory({ - direction: "down", - entries, - historyIndex: up.historyIndex, - currentPrompt: text("ignored"), - currentComments: [], - savedPrompt: up.savedPrompt, - }) - expect(down.handled).toBe(true) - if (!down.handled) throw new Error("expected handled") - expect(down.historyIndex).toBe(-1) - expect(down.entry.prompt[0]?.type === "text" ? down.entry.prompt[0].content : "").toBe("draft") - expect(down.entry.comments).toEqual([comment("draft")]) - }) - - test("navigatePromptHistory keeps entry comments when moving through history", () => { - const entries = [ - { - prompt: text("with comment"), - comments: [comment("c1")], - }, - ] - - const up = navigatePromptHistory({ - direction: "up", - entries, - historyIndex: -1, - currentPrompt: text("draft"), - currentComments: [], - savedPrompt: null, - }) - - expect(up.handled).toBe(true) - if (!up.handled) throw new Error("expected handled") - expect(up.entry.prompt[0]?.type === "text" ? up.entry.prompt[0].content : "").toBe("with comment") - expect(up.entry.comments).toEqual([comment("c1")]) - }) - - test("normalizePromptHistoryEntry supports legacy prompt arrays", () => { - const entry = normalizePromptHistoryEntry(text("legacy")) - expect(entry.prompt[0]?.type === "text" ? entry.prompt[0].content : "").toBe("legacy") - expect(entry.comments).toEqual([]) - }) - - test("helpers clone prompt and count text content length", () => { - const original: Prompt = [ - { type: "text", content: "one", start: 0, end: 3 }, - { - type: "file", - path: "src/a.ts", - content: "@src/a.ts", - start: 3, - end: 12, - selection: { startLine: 1, startChar: 1, endLine: 2, endChar: 1 }, - }, - { type: "image", id: "1", filename: "img.png", mime: "image/png", dataUrl: "data:image/png;base64,abc" }, - ] - const copy = clonePromptParts(original) - expect(copy).not.toBe(original) - expect(promptLength(copy)).toBe(12) - if (copy[1]?.type !== "file") throw new Error("expected file") - copy[1].selection!.startLine = 9 - if (original[1]?.type !== "file") throw new Error("expected file") - expect(original[1].selection?.startLine).toBe(1) - }) - - test("canNavigateHistoryAtCursor only allows prompt boundaries", () => { - const value = "a\nb\nc" - - expect(canNavigateHistoryAtCursor("up", value, 0)).toBe(false) - expect(canNavigateHistoryAtCursor("down", value, 0)).toBe(false) - - expect(canNavigateHistoryAtCursor("up", value, 2)).toBe(false) - expect(canNavigateHistoryAtCursor("down", value, 2)).toBe(false) - - expect(canNavigateHistoryAtCursor("up", value, 5)).toBe(false) - expect(canNavigateHistoryAtCursor("down", value, 5)).toBe(true) - - expect(canNavigateHistoryAtCursor("up", "abc", 0)).toBe(false) - expect(canNavigateHistoryAtCursor("down", "abc", 3)).toBe(true) - expect(canNavigateHistoryAtCursor("up", "abc", 1)).toBe(false) - expect(canNavigateHistoryAtCursor("down", "abc", 1)).toBe(false) - - expect(canNavigateHistoryAtCursor("up", "", 0)).toBe(true) - expect(canNavigateHistoryAtCursor("down", "", 0)).toBe(true) - - expect(canNavigateHistoryAtCursor("up", "abc", 0, true)).toBe(true) - expect(canNavigateHistoryAtCursor("up", "abc", 3, true)).toBe(true) - expect(canNavigateHistoryAtCursor("down", "abc", 0, true)).toBe(true) - expect(canNavigateHistoryAtCursor("down", "abc", 3, true)).toBe(true) - expect(canNavigateHistoryAtCursor("up", "abc", 1, true)).toBe(false) - expect(canNavigateHistoryAtCursor("down", "abc", 1, true)).toBe(false) - }) -}) diff --git a/packages/app/src/components/prompt-input/history.ts b/packages/app/src/components/prompt-input/history.ts deleted file mode 100644 index 79e8abc0d9..0000000000 --- a/packages/app/src/components/prompt-input/history.ts +++ /dev/null @@ -1,256 +0,0 @@ -import type { Prompt } from "@/context/prompt" -import type { SelectedLineRange } from "@/context/file" - -const DEFAULT_PROMPT: Prompt = [{ type: "text", content: "", start: 0, end: 0 }] - -export const MAX_HISTORY = 100 - -export type PromptHistoryComment = { - id: string - path: string - selection: SelectedLineRange - comment: string - time: number - origin?: "review" | "file" - preview?: string -} - -export type PromptHistoryEntry = { - prompt: Prompt - comments: PromptHistoryComment[] -} - -export type PromptHistoryStoredEntry = Prompt | PromptHistoryEntry - -export function canNavigateHistoryAtCursor(direction: "up" | "down", text: string, cursor: number, inHistory = false) { - const position = Math.max(0, Math.min(cursor, text.length)) - const atStart = position === 0 - const atEnd = position === text.length - if (inHistory) return atStart || atEnd - if (direction === "up") return position === 0 && text.length === 0 - return position === text.length -} - -export function clonePromptParts(prompt: Prompt): Prompt { - return prompt.map((part) => { - if (part.type === "text") return { ...part } - if (part.type === "image") return { ...part } - if (part.type === "agent") return { ...part } - return { - ...part, - selection: part.selection ? { ...part.selection } : undefined, - } - }) -} - -function cloneSelection(selection: SelectedLineRange): SelectedLineRange { - return { - start: selection.start, - end: selection.end, - ...(selection.side ? { side: selection.side } : {}), - ...(selection.endSide ? { endSide: selection.endSide } : {}), - } -} - -export function clonePromptHistoryComments(comments: PromptHistoryComment[]) { - return comments.map((comment) => ({ - ...comment, - selection: cloneSelection(comment.selection), - })) -} - -export function normalizePromptHistoryEntry(entry: PromptHistoryStoredEntry): PromptHistoryEntry { - if (Array.isArray(entry)) { - return { - prompt: clonePromptParts(entry), - comments: [], - } - } - return { - prompt: clonePromptParts(entry.prompt), - comments: clonePromptHistoryComments(entry.comments), - } -} - -export function promptLength(prompt: Prompt) { - return prompt.reduce((len, part) => len + ("content" in part ? part.content.length : 0), 0) -} - -export function prependHistoryEntry( - entries: PromptHistoryStoredEntry[], - prompt: Prompt, - comments: PromptHistoryComment[] = [], - max = MAX_HISTORY, -) { - const text = prompt - .map((part) => ("content" in part ? part.content : "")) - .join("") - .trim() - const hasImages = prompt.some((part) => part.type === "image") - const hasComments = comments.some((comment) => !!comment.comment.trim()) - if (!text && !hasImages && !hasComments) return entries - - const entry = { - prompt: clonePromptParts(prompt), - comments: clonePromptHistoryComments(comments), - } satisfies PromptHistoryEntry - const last = entries[0] - if (last && isPromptEqual(last, entry)) return entries - return [entry, ...entries].slice(0, max) -} - -function isCommentEqual(commentA: PromptHistoryComment, commentB: PromptHistoryComment) { - return ( - commentA.path === commentB.path && - commentA.comment === commentB.comment && - commentA.origin === commentB.origin && - commentA.preview === commentB.preview && - commentA.selection.start === commentB.selection.start && - commentA.selection.end === commentB.selection.end && - commentA.selection.side === commentB.selection.side && - commentA.selection.endSide === commentB.selection.endSide - ) -} - -function isPromptEqual(promptA: PromptHistoryStoredEntry, promptB: PromptHistoryStoredEntry) { - const entryA = normalizePromptHistoryEntry(promptA) - const entryB = normalizePromptHistoryEntry(promptB) - if (entryA.prompt.length !== entryB.prompt.length) return false - for (let i = 0; i < entryA.prompt.length; i++) { - const partA = entryA.prompt[i] - const partB = entryB.prompt[i] - if (partA.type !== partB.type) return false - if (partA.type === "text" && partA.content !== (partB.type === "text" ? partB.content : "")) return false - if (partA.type === "file") { - if (partA.path !== (partB.type === "file" ? partB.path : "")) return false - const a = partA.selection - const b = partB.type === "file" ? partB.selection : undefined - const sameSelection = - (!a && !b) || - (!!a && - !!b && - a.startLine === b.startLine && - a.startChar === b.startChar && - a.endLine === b.endLine && - a.endChar === b.endChar) - if (!sameSelection) return false - } - if (partA.type === "agent" && partA.name !== (partB.type === "agent" ? partB.name : "")) return false - if (partA.type === "image" && partA.id !== (partB.type === "image" ? partB.id : "")) return false - } - if (entryA.comments.length !== entryB.comments.length) return false - for (let i = 0; i < entryA.comments.length; i++) { - const commentA = entryA.comments[i] - const commentB = entryB.comments[i] - if (!commentA || !commentB || !isCommentEqual(commentA, commentB)) return false - } - return true -} - -type HistoryNavInput = { - direction: "up" | "down" - entries: PromptHistoryStoredEntry[] - historyIndex: number - currentPrompt: Prompt - currentComments: PromptHistoryComment[] - savedPrompt: PromptHistoryEntry | null -} - -type HistoryNavResult = - | { - handled: false - historyIndex: number - savedPrompt: PromptHistoryEntry | null - } - | { - handled: true - historyIndex: number - savedPrompt: PromptHistoryEntry | null - entry: PromptHistoryEntry - cursor: "start" | "end" - } - -export function navigatePromptHistory(input: HistoryNavInput): HistoryNavResult { - if (input.direction === "up") { - if (input.entries.length === 0) { - return { - handled: false, - historyIndex: input.historyIndex, - savedPrompt: input.savedPrompt, - } - } - - if (input.historyIndex === -1) { - const entry = normalizePromptHistoryEntry(input.entries[0]) - return { - handled: true, - historyIndex: 0, - savedPrompt: { - prompt: clonePromptParts(input.currentPrompt), - comments: clonePromptHistoryComments(input.currentComments), - }, - entry, - cursor: "start", - } - } - - if (input.historyIndex < input.entries.length - 1) { - const next = input.historyIndex + 1 - const entry = normalizePromptHistoryEntry(input.entries[next]) - return { - handled: true, - historyIndex: next, - savedPrompt: input.savedPrompt, - entry, - cursor: "start", - } - } - - return { - handled: false, - historyIndex: input.historyIndex, - savedPrompt: input.savedPrompt, - } - } - - if (input.historyIndex > 0) { - const next = input.historyIndex - 1 - const entry = normalizePromptHistoryEntry(input.entries[next]) - return { - handled: true, - historyIndex: next, - savedPrompt: input.savedPrompt, - entry, - cursor: "end", - } - } - - if (input.historyIndex === 0) { - if (input.savedPrompt) { - return { - handled: true, - historyIndex: -1, - savedPrompt: null, - entry: input.savedPrompt, - cursor: "end", - } - } - - return { - handled: true, - historyIndex: -1, - savedPrompt: null, - entry: { - prompt: DEFAULT_PROMPT, - comments: [], - }, - cursor: "end", - } - } - - return { - handled: false, - historyIndex: input.historyIndex, - savedPrompt: input.savedPrompt, - } -} diff --git a/packages/app/src/components/prompt-input/image-attachments.tsx b/packages/app/src/components/prompt-input/image-attachments.tsx deleted file mode 100644 index dd8138e5a4..0000000000 --- a/packages/app/src/components/prompt-input/image-attachments.tsx +++ /dev/null @@ -1,61 +0,0 @@ -import { Component, For, Show } from "solid-js" -import { Icon } from "@opencode-ai/ui/icon" -import { Tooltip } from "@opencode-ai/ui/tooltip" -import type { ImageAttachmentPart } from "@/context/prompt" - -type PromptImageAttachmentsProps = { - attachments: ImageAttachmentPart[] - onOpen: (attachment: ImageAttachmentPart) => void - onRemove: (id: string) => void - removeLabel: string -} - -const fallbackClass = "size-16 rounded-md bg-surface-base flex items-center justify-center border border-border-base" -const imageClass = - "size-16 rounded-md object-cover border border-border-base hover:border-border-strong-base transition-colors" -const removeClass = - "absolute -top-1.5 -right-1.5 size-5 rounded-full bg-surface-raised-stronger-non-alpha border border-border-base flex items-center justify-center opacity-0 group-hover:opacity-100 transition-opacity hover:bg-surface-raised-base-hover" -const nameClass = "absolute bottom-0 left-0 right-0 px-1 py-0.5 bg-black/50 rounded-b-md" - -export const PromptImageAttachments: Component = (props) => { - return ( - 0}> -
- - {(attachment) => ( - -
- - -
- } - > - {attachment.filename} props.onOpen(attachment)} - /> - - -
- {attachment.filename} -
-
- - )} - -
- - ) -} diff --git a/packages/app/src/components/prompt-input/paste.ts b/packages/app/src/components/prompt-input/paste.ts deleted file mode 100644 index 6787d50309..0000000000 --- a/packages/app/src/components/prompt-input/paste.ts +++ /dev/null @@ -1,24 +0,0 @@ -const LARGE_PASTE_CHARS = 8000 -const LARGE_PASTE_BREAKS = 120 - -function largePaste(text: string) { - if (text.length >= LARGE_PASTE_CHARS) return true - let breaks = 0 - for (const char of text) { - if (char !== "\n") continue - breaks += 1 - if (breaks >= LARGE_PASTE_BREAKS) return true - } - return false -} - -export function normalizePaste(text: string) { - if (!text.includes("\r")) return text - return text.replace(/\r\n?/g, "\n") -} - -export function pasteMode(text: string) { - if (largePaste(text)) return "manual" - if (text.includes("\n") || text.includes("\r")) return "manual" - return "native" -} diff --git a/packages/app/src/components/prompt-input/placeholder.test.ts b/packages/app/src/components/prompt-input/placeholder.test.ts deleted file mode 100644 index d4caead0d2..0000000000 --- a/packages/app/src/components/prompt-input/placeholder.test.ts +++ /dev/null @@ -1,48 +0,0 @@ -import { describe, expect, test } from "bun:test" -import { promptPlaceholder } from "./placeholder" - -describe("promptPlaceholder", () => { - const t = (key: string, params?: Record) => `${key}${params?.example ? `:${params.example}` : ""}` - - test("returns shell placeholder in shell mode", () => { - const value = promptPlaceholder({ - mode: "shell", - commentCount: 0, - example: "example", - suggest: true, - t, - }) - expect(value).toBe("prompt.placeholder.shell:example") - }) - - test("returns summarize placeholders for comment context", () => { - expect(promptPlaceholder({ mode: "normal", commentCount: 1, example: "example", suggest: true, t })).toBe( - "prompt.placeholder.summarizeComment", - ) - expect(promptPlaceholder({ mode: "normal", commentCount: 2, example: "example", suggest: true, t })).toBe( - "prompt.placeholder.summarizeComments", - ) - }) - - test("returns default placeholder with example when suggestions enabled", () => { - const value = promptPlaceholder({ - mode: "normal", - commentCount: 0, - example: "translated-example", - suggest: true, - t, - }) - expect(value).toBe("prompt.placeholder.normal:translated-example") - }) - - test("returns simple placeholder when suggestions disabled", () => { - const value = promptPlaceholder({ - mode: "normal", - commentCount: 0, - example: "translated-example", - suggest: false, - t, - }) - expect(value).toBe("prompt.placeholder.simple") - }) -}) diff --git a/packages/app/src/components/prompt-input/placeholder.ts b/packages/app/src/components/prompt-input/placeholder.ts deleted file mode 100644 index 6669f13614..0000000000 --- a/packages/app/src/components/prompt-input/placeholder.ts +++ /dev/null @@ -1,15 +0,0 @@ -type PromptPlaceholderInput = { - mode: "normal" | "shell" - commentCount: number - example: string - suggest: boolean - t: (key: string, params?: Record) => string -} - -export function promptPlaceholder(input: PromptPlaceholderInput) { - if (input.mode === "shell") return input.t("prompt.placeholder.shell", { example: input.example }) - if (input.commentCount > 1) return input.t("prompt.placeholder.summarizeComments") - if (input.commentCount === 1) return input.t("prompt.placeholder.summarizeComment") - if (!input.suggest) return input.t("prompt.placeholder.simple") - return input.t("prompt.placeholder.normal", { example: input.example }) -} diff --git a/packages/app/src/components/server/server-row.tsx b/packages/app/src/components/server/server-row.tsx deleted file mode 100644 index d4f68d6306..0000000000 --- a/packages/app/src/components/server/server-row.tsx +++ /dev/null @@ -1,127 +0,0 @@ -import { Tooltip } from "@opencode-ai/ui/tooltip" -import { createResizeObserver } from "@solid-primitives/resize-observer" -import { - children, - createEffect, - createMemo, - createSignal, - type JSXElement, - onMount, - type ParentProps, - Show, -} from "solid-js" -import { useLanguage } from "@/context/language" -import { type ServerConnection, serverName } from "@/context/server" -import type { ServerHealth } from "@/utils/server-health" - -interface ServerRowProps extends ParentProps { - conn: ServerConnection.Any - status?: ServerHealth - class?: string - nameClass?: string - versionClass?: string - dimmed?: boolean - badge?: JSXElement - showCredentials?: boolean -} - -export function ServerRow(props: ServerRowProps) { - const language = useLanguage() - const [truncated, setTruncated] = createSignal(false) - let nameRef: HTMLSpanElement | undefined - let versionRef: HTMLSpanElement | undefined - const name = createMemo(() => serverName(props.conn)) - - const check = () => { - const nameTruncated = nameRef ? nameRef.scrollWidth > nameRef.clientWidth : false - const versionTruncated = versionRef ? versionRef.scrollWidth > versionRef.clientWidth : false - setTruncated(nameTruncated || versionTruncated) - } - - createEffect(() => { - name() - props.conn.http.url - props.status?.version - queueMicrotask(check) - }) - - onMount(() => { - if (typeof ResizeObserver !== "function") return - createResizeObserver([nameRef, versionRef], check) - check() - }) - - const tooltipValue = () => ( - - {serverName(props.conn, true)} - - v{props.status?.version} - - - ) - - const badge = children(() => props.badge) - - return ( - -
-
-
- - {name()} - - - - v{props.status?.version} - - - } - > - {(badge) => badge()} - -
- - {(conn) => ( -
- - {conn().http.username ? ( - {conn().http.username} - ) : ( - {language.t("server.row.noUsername")} - )} - - {conn().http.password && ••••••••} -
- )} -
-
- {props.children} -
-
- ) -} - -export function ServerHealthIndicator(props: { health?: ServerHealth }) { - return ( -
- ) -} diff --git a/packages/app/src/components/session-context-usage.tsx b/packages/app/src/components/session-context-usage.tsx deleted file mode 100644 index 6b7fe4ef7d..0000000000 --- a/packages/app/src/components/session-context-usage.tsx +++ /dev/null @@ -1,124 +0,0 @@ -import { Match, Show, Switch, createMemo } from "solid-js" -import { Tooltip, type TooltipProps } from "@opencode-ai/ui/tooltip" -import { ProgressCircle } from "@opencode-ai/ui/progress-circle" -import { Button } from "@opencode-ai/ui/button" - -import { useFile } from "@/context/file" -import { useLayout } from "@/context/layout" -import { useSync } from "@/context/sync" -import { useLanguage } from "@/context/language" -import { useProviders } from "@/hooks/use-providers" -import { getSessionContextMetrics } from "@/components/session/session-context-metrics" -import { useSessionLayout } from "@/pages/session/session-layout" -import { createSessionTabs } from "@/pages/session/helpers" - -interface SessionContextUsageProps { - variant?: "button" | "indicator" - placement?: TooltipProps["placement"] -} - -function openSessionContext(args: { - view: ReturnType["view"]> - layout: ReturnType - tabs: ReturnType["tabs"]> -}) { - if (!args.view.reviewPanel.opened()) args.view.reviewPanel.open() - if (args.layout.fileTree.opened() && args.layout.fileTree.tab() !== "all") args.layout.fileTree.setTab("all") - void args.tabs.open("context") - args.tabs.setActive("context") -} - -export function SessionContextUsage(props: SessionContextUsageProps) { - const sync = useSync() - const file = useFile() - const layout = useLayout() - const language = useLanguage() - const providers = useProviders() - const { params, tabs, view } = useSessionLayout() - - const variant = createMemo(() => props.variant ?? "button") - const tabState = createSessionTabs({ - tabs, - pathFromTab: file.pathFromTab, - normalizeTab: (tab) => (tab.startsWith("file://") ? file.tab(tab) : tab), - }) - const messages = createMemo(() => (params.id ? (sync.data.message[params.id] ?? []) : [])) - - const usd = createMemo( - () => - new Intl.NumberFormat(language.intl(), { - style: "currency", - currency: "USD", - }), - ) - - const metrics = createMemo(() => getSessionContextMetrics(messages(), providers.all())) - const context = createMemo(() => metrics().context) - const cost = createMemo(() => { - return usd().format(metrics().totalCost) - }) - - const openContext = () => { - if (!params.id) return - - if (tabState.activeTab() === "context") { - tabs().close("context") - return - } - openSessionContext({ - view: view(), - layout, - tabs: tabs(), - }) - } - - const circle = () => ( -
- -
- ) - - const tooltipValue = () => ( -
- - {(ctx) => ( - <> -
- {ctx().total.toLocaleString(language.intl())} - {language.t("context.usage.tokens")} -
-
- {ctx().usage ?? 0}% - {language.t("context.usage.usage")} -
- - )} -
-
- {cost()} - {language.t("context.usage.cost")} -
-
- ) - - return ( - - - - {circle()} - - - - - - - ) -} diff --git a/packages/app/src/components/session/index.ts b/packages/app/src/components/session/index.ts deleted file mode 100644 index 20124b6fde..0000000000 --- a/packages/app/src/components/session/index.ts +++ /dev/null @@ -1,5 +0,0 @@ -export { SessionHeader } from "./session-header" -export { SessionContextTab } from "./session-context-tab" -export { SortableTab, FileVisual } from "./session-sortable-tab" -export { SortableTerminalTab } from "./session-sortable-terminal-tab" -export { NewSessionView } from "./session-new-view" diff --git a/packages/app/src/components/session/session-context-breakdown.test.ts b/packages/app/src/components/session/session-context-breakdown.test.ts deleted file mode 100644 index 36c4384345..0000000000 --- a/packages/app/src/components/session/session-context-breakdown.test.ts +++ /dev/null @@ -1,61 +0,0 @@ -import { describe, expect, test } from "bun:test" -import type { Message, Part } from "@kilocode/sdk/v2/client" -import { estimateSessionContextBreakdown } from "./session-context-breakdown" - -const user = (id: string) => { - return { - id, - role: "user", - time: { created: 1 }, - } as unknown as Message -} - -const assistant = (id: string) => { - return { - id, - role: "assistant", - time: { created: 1 }, - } as unknown as Message -} - -describe("estimateSessionContextBreakdown", () => { - test("estimates tokens and keeps remaining tokens as other", () => { - const messages = [user("u1"), assistant("a1")] - const parts = { - u1: [{ type: "text", text: "hello world" }] as unknown as Part[], - a1: [{ type: "text", text: "assistant response" }] as unknown as Part[], - } - - const output = estimateSessionContextBreakdown({ - messages, - parts, - input: 20, - systemPrompt: "system prompt", - }) - - const map = Object.fromEntries(output.map((segment) => [segment.key, segment.tokens])) - expect(map.system).toBe(4) - expect(map.user).toBe(3) - expect(map.assistant).toBe(5) - expect(map.other).toBe(8) - }) - - test("scales segments when estimates exceed input", () => { - const messages = [user("u1"), assistant("a1")] - const parts = { - u1: [{ type: "text", text: "x".repeat(400) }] as unknown as Part[], - a1: [{ type: "text", text: "y".repeat(400) }] as unknown as Part[], - } - - const output = estimateSessionContextBreakdown({ - messages, - parts, - input: 10, - systemPrompt: "z".repeat(200), - }) - - const total = output.reduce((sum, segment) => sum + segment.tokens, 0) - expect(total).toBeLessThanOrEqual(10) - expect(output.every((segment) => segment.width <= 100)).toBeTrue() - }) -}) diff --git a/packages/app/src/components/session/session-context-breakdown.ts b/packages/app/src/components/session/session-context-breakdown.ts deleted file mode 100644 index 1b11fadcce..0000000000 --- a/packages/app/src/components/session/session-context-breakdown.ts +++ /dev/null @@ -1,132 +0,0 @@ -import type { Message, Part } from "@kilocode/sdk/v2/client" - -export type SessionContextBreakdownKey = "system" | "user" | "assistant" | "tool" | "other" - -export type SessionContextBreakdownSegment = { - key: SessionContextBreakdownKey - tokens: number - width: number - percent: number -} - -const estimateTokens = (chars: number) => Math.ceil(chars / 4) -const toPercent = (tokens: number, input: number) => (tokens / input) * 100 -const toPercentLabel = (tokens: number, input: number) => Math.round(toPercent(tokens, input) * 10) / 10 - -const charsFromUserPart = (part: Part) => { - if (part.type === "text") return part.text.length - if (part.type === "file") return part.source?.text.value.length ?? 0 - if (part.type === "agent") return part.source?.value.length ?? 0 - return 0 -} - -const charsFromAssistantPart = (part: Part) => { - if (part.type === "text") return { assistant: part.text.length, tool: 0 } - if (part.type === "reasoning") return { assistant: part.text.length, tool: 0 } - if (part.type !== "tool") return { assistant: 0, tool: 0 } - - const input = Object.keys(part.state.input).length * 16 - if (part.state.status === "pending") return { assistant: 0, tool: input + part.state.raw.length } - if (part.state.status === "completed") return { assistant: 0, tool: input + part.state.output.length } - if (part.state.status === "error") return { assistant: 0, tool: input + part.state.error.length } - return { assistant: 0, tool: input } -} - -const build = ( - tokens: { system: number; user: number; assistant: number; tool: number; other: number }, - input: number, -) => { - return [ - { - key: "system", - tokens: tokens.system, - }, - { - key: "user", - tokens: tokens.user, - }, - { - key: "assistant", - tokens: tokens.assistant, - }, - { - key: "tool", - tokens: tokens.tool, - }, - { - key: "other", - tokens: tokens.other, - }, - ] - .filter((x) => x.tokens > 0) - .map((x) => ({ - key: x.key, - tokens: x.tokens, - width: toPercent(x.tokens, input), - percent: toPercentLabel(x.tokens, input), - })) as SessionContextBreakdownSegment[] -} - -export function estimateSessionContextBreakdown(args: { - messages: Message[] - parts: Record - input: number - systemPrompt?: string -}) { - if (!args.input) return [] - - const counts = args.messages.reduce( - (acc, msg) => { - const parts = args.parts[msg.id] ?? [] - if (msg.role === "user") { - const user = parts.reduce((sum, part) => sum + charsFromUserPart(part), 0) - return { ...acc, user: acc.user + user } - } - - if (msg.role !== "assistant") return acc - const assistant = parts.reduce( - (sum, part) => { - const next = charsFromAssistantPart(part) - return { - assistant: sum.assistant + next.assistant, - tool: sum.tool + next.tool, - } - }, - { assistant: 0, tool: 0 }, - ) - return { - ...acc, - assistant: acc.assistant + assistant.assistant, - tool: acc.tool + assistant.tool, - } - }, - { - system: args.systemPrompt?.length ?? 0, - user: 0, - assistant: 0, - tool: 0, - }, - ) - - const tokens = { - system: estimateTokens(counts.system), - user: estimateTokens(counts.user), - assistant: estimateTokens(counts.assistant), - tool: estimateTokens(counts.tool), - } - const estimated = tokens.system + tokens.user + tokens.assistant + tokens.tool - - if (estimated <= args.input) { - return build({ ...tokens, other: args.input - estimated }, args.input) - } - - const scale = args.input / estimated - const scaled = { - system: Math.floor(tokens.system * scale), - user: Math.floor(tokens.user * scale), - assistant: Math.floor(tokens.assistant * scale), - tool: Math.floor(tokens.tool * scale), - } - const total = scaled.system + scaled.user + scaled.assistant + scaled.tool - return build({ ...scaled, other: Math.max(0, args.input - total) }, args.input) -} diff --git a/packages/app/src/components/session/session-context-format.ts b/packages/app/src/components/session/session-context-format.ts deleted file mode 100644 index e7c536d584..0000000000 --- a/packages/app/src/components/session/session-context-format.ts +++ /dev/null @@ -1,20 +0,0 @@ -import { DateTime } from "luxon" - -export function createSessionContextFormatter(locale: string) { - return { - number(value: number | null | undefined) { - if (value === undefined) return "—" - if (value === null) return "—" - return value.toLocaleString(locale) - }, - percent(value: number | null | undefined) { - if (value === undefined) return "—" - if (value === null) return "—" - return value.toLocaleString(locale) + "%" - }, - time(value: number | undefined) { - if (!value) return "—" - return DateTime.fromMillis(value).setLocale(locale).toLocaleString(DateTime.DATETIME_MED) - }, - } -} diff --git a/packages/app/src/components/session/session-context-metrics.test.ts b/packages/app/src/components/session/session-context-metrics.test.ts deleted file mode 100644 index be1bbca258..0000000000 --- a/packages/app/src/components/session/session-context-metrics.test.ts +++ /dev/null @@ -1,101 +0,0 @@ -import { describe, expect, test } from "bun:test" -import type { Message } from "@kilocode/sdk/v2/client" -import { getSessionContextMetrics } from "./session-context-metrics" - -const assistant = ( - id: string, - tokens: { input: number; output: number; reasoning: number; read: number; write: number }, - cost: number, - providerID = "openai", - modelID = "gpt-4.1", -) => { - return { - id, - role: "assistant", - providerID, - modelID, - cost, - tokens: { - input: tokens.input, - output: tokens.output, - reasoning: tokens.reasoning, - cache: { - read: tokens.read, - write: tokens.write, - }, - }, - time: { created: 1 }, - } as unknown as Message -} - -const user = (id: string) => { - return { - id, - role: "user", - cost: 0, - time: { created: 1 }, - } as unknown as Message -} - -describe("getSessionContextMetrics", () => { - test("computes totals and usage from latest assistant with tokens", () => { - const messages = [ - user("u1"), - assistant("a1", { input: 0, output: 0, reasoning: 0, read: 0, write: 0 }, 0.5), - assistant("a2", { input: 300, output: 100, reasoning: 50, read: 25, write: 25 }, 1.25), - ] - const providers = [ - { - id: "openai", - name: "OpenAI", - models: { - "gpt-4.1": { - name: "GPT-4.1", - limit: { context: 1000 }, - }, - }, - }, - ] - - const metrics = getSessionContextMetrics(messages, providers) - - expect(metrics.totalCost).toBe(1.75) - expect(metrics.context?.message.id).toBe("a2") - expect(metrics.context?.total).toBe(500) - expect(metrics.context?.usage).toBe(50) - expect(metrics.context?.providerLabel).toBe("OpenAI") - expect(metrics.context?.modelLabel).toBe("GPT-4.1") - }) - - test("preserves fallback labels and null usage when model metadata is missing", () => { - const messages = [assistant("a1", { input: 40, output: 10, reasoning: 0, read: 0, write: 0 }, 0.1, "p-1", "m-1")] - const providers = [{ id: "p-1", models: {} }] - - const metrics = getSessionContextMetrics(messages, providers) - - expect(metrics.context?.providerLabel).toBe("p-1") - expect(metrics.context?.modelLabel).toBe("m-1") - expect(metrics.context?.limit).toBeUndefined() - expect(metrics.context?.usage).toBeNull() - }) - - test("recomputes when message array is mutated in place", () => { - const messages = [assistant("a1", { input: 10, output: 10, reasoning: 10, read: 10, write: 10 }, 0.25)] - const providers = [{ id: "openai", models: {} }] - - const one = getSessionContextMetrics(messages, providers) - messages.push(assistant("a2", { input: 100, output: 20, reasoning: 0, read: 0, write: 0 }, 0.75)) - const two = getSessionContextMetrics(messages, providers) - - expect(one.context?.message.id).toBe("a1") - expect(two.context?.message.id).toBe("a2") - expect(two.totalCost).toBe(1) - }) - - test("returns empty metrics when inputs are undefined", () => { - const metrics = getSessionContextMetrics(undefined, undefined) - - expect(metrics.totalCost).toBe(0) - expect(metrics.context).toBeUndefined() - }) -}) diff --git a/packages/app/src/components/session/session-context-metrics.ts b/packages/app/src/components/session/session-context-metrics.ts deleted file mode 100644 index 4bdcff52eb..0000000000 --- a/packages/app/src/components/session/session-context-metrics.ts +++ /dev/null @@ -1,82 +0,0 @@ -import type { AssistantMessage, Message } from "@kilocode/sdk/v2/client" - -type Provider = { - id: string - name?: string - models: Record -} - -type Model = { - name?: string - limit: { - context: number - } -} - -type Context = { - message: AssistantMessage - provider?: Provider - model?: Model - providerLabel: string - modelLabel: string - limit: number | undefined - input: number - output: number - reasoning: number - cacheRead: number - cacheWrite: number - total: number - usage: number | null -} - -type Metrics = { - totalCost: number - context: Context | undefined -} - -const tokenTotal = (msg: AssistantMessage) => { - return msg.tokens.input + msg.tokens.output + msg.tokens.reasoning + msg.tokens.cache.read + msg.tokens.cache.write -} - -const lastAssistantWithTokens = (messages: Message[]) => { - for (let i = messages.length - 1; i >= 0; i--) { - const msg = messages[i] - if (msg.role !== "assistant") continue - if (tokenTotal(msg) <= 0) continue - return msg - } -} - -const build = (messages: Message[] = [], providers: Provider[] = []): Metrics => { - const totalCost = messages.reduce((sum, msg) => sum + (msg.role === "assistant" ? msg.cost : 0), 0) - const message = lastAssistantWithTokens(messages) - if (!message) return { totalCost, context: undefined } - - const provider = providers.find((item) => item.id === message.providerID) - const model = provider?.models[message.modelID] - const limit = model?.limit.context - const total = tokenTotal(message) - - return { - totalCost, - context: { - message, - provider, - model, - providerLabel: provider?.name ?? message.providerID, - modelLabel: model?.name ?? message.modelID, - limit, - input: message.tokens.input, - output: message.tokens.output, - reasoning: message.tokens.reasoning, - cacheRead: message.tokens.cache.read, - cacheWrite: message.tokens.cache.write, - total, - usage: limit ? Math.round((total / limit) * 100) : null, - }, - } -} - -export function getSessionContextMetrics(messages: Message[] = [], providers: Provider[] = []) { - return build(messages, providers) -} diff --git a/packages/app/src/components/session/session-sortable-terminal-tab.tsx b/packages/app/src/components/session/session-sortable-terminal-tab.tsx deleted file mode 100644 index 2d88ed1806..0000000000 --- a/packages/app/src/components/session/session-sortable-terminal-tab.tsx +++ /dev/null @@ -1,193 +0,0 @@ -import type { JSX } from "solid-js" -import { Show, createEffect, onCleanup } from "solid-js" -import { createStore } from "solid-js/store" -import { createSortable } from "@thisbeyond/solid-dnd" -import { IconButton } from "@opencode-ai/ui/icon-button" -import { Tabs } from "@opencode-ai/ui/tabs" -import { DropdownMenu } from "@opencode-ai/ui/dropdown-menu" -import { Icon } from "@opencode-ai/ui/icon" -import { isDefaultTitle as isDefaultTerminalTitle } from "@/context/terminal-title" -import { useTerminal, type LocalPTY } from "@/context/terminal" -import { useLanguage } from "@/context/language" -import { focusTerminalById } from "@/pages/session/helpers" - -export function SortableTerminalTab(props: { terminal: LocalPTY; onClose?: () => void }): JSX.Element { - const terminal = useTerminal() - const language = useLanguage() - const sortable = createSortable(props.terminal.id) - const [store, setStore] = createStore({ - editing: false, - title: props.terminal.title, - menuOpen: false, - menuPosition: { x: 0, y: 0 }, - blurEnabled: false, - }) - let input: HTMLInputElement | undefined - let blurFrame: number | undefined - let editRequested = false - - const isDefaultTitle = () => { - const number = props.terminal.titleNumber - if (!Number.isFinite(number) || number <= 0) return false - return isDefaultTerminalTitle(props.terminal.title, number) - } - - const label = () => { - language.locale() - if (props.terminal.title && !isDefaultTitle()) return props.terminal.title - - const number = props.terminal.titleNumber - if (Number.isFinite(number) && number > 0) return language.t("terminal.title.numbered", { number }) - if (props.terminal.title) return props.terminal.title - return language.t("terminal.title") - } - - const close = () => { - const count = terminal.all().length - void terminal.close(props.terminal.id) - if (count === 1) { - props.onClose?.() - } - } - - const focus = () => { - if (store.editing) return - if (document.activeElement instanceof HTMLElement) document.activeElement.blur() - focusTerminalById(props.terminal.id) - } - - const edit = (e?: Event) => { - if (e) { - e.stopPropagation() - e.preventDefault() - } - - setStore("blurEnabled", false) - setStore("title", props.terminal.title) - setStore("editing", true) - } - - const save = () => { - if (!store.blurEnabled) return - - const value = store.title.trim() - if (value && value !== props.terminal.title) { - terminal.update({ id: props.terminal.id, title: value }) - } - setStore("editing", false) - } - - const keydown = (e: KeyboardEvent) => { - if (e.key === "Enter") { - e.preventDefault() - save() - return - } - if (e.key === "Escape") { - e.preventDefault() - setStore("editing", false) - } - } - - const menu = (e: MouseEvent) => { - e.preventDefault() - setStore("menuPosition", { x: e.clientX, y: e.clientY }) - setStore("menuOpen", true) - } - - createEffect(() => { - if (!store.editing) return - if (!input) return - input.focus() - input.select() - if (blurFrame !== undefined) cancelAnimationFrame(blurFrame) - blurFrame = requestAnimationFrame(() => { - blurFrame = undefined - setStore("blurEnabled", true) - }) - }) - - onCleanup(() => { - if (blurFrame === undefined) return - cancelAnimationFrame(blurFrame) - }) - - return ( -
-
- e.preventDefault()} - onContextMenu={menu} - class="!shadow-none" - classes={{ - button: "border-0 outline-none focus:outline-none focus-visible:outline-none !shadow-none !ring-0", - }} - closeButton={ - { - e.stopPropagation() - close() - }} - aria-label={language.t("terminal.close")} - /> - } - > - - {label()} - - - -
- setStore("title", e.currentTarget.value)} - onBlur={save} - onKeyDown={keydown} - onMouseDown={(e) => e.stopPropagation()} - class="bg-transparent border-none outline-none text-sm min-w-0 flex-1" - /> -
-
- setStore("menuOpen", open)}> - - { - if (!editRequested) return - e.preventDefault() - editRequested = false - requestAnimationFrame(() => edit()) - }} - > - (editRequested = true)}> - - {language.t("common.rename")} - - - - {language.t("common.close")} - - - - -
-
- ) -} diff --git a/packages/app/src/components/settings-keybinds.tsx b/packages/app/src/components/settings-keybinds.tsx deleted file mode 100644 index 7d2dfaa636..0000000000 --- a/packages/app/src/components/settings-keybinds.tsx +++ /dev/null @@ -1,453 +0,0 @@ -import { Component, For, Show, createMemo, onCleanup, onMount } from "solid-js" -import { createStore } from "solid-js/store" -import { makeEventListener } from "@solid-primitives/event-listener" -import { Button } from "@opencode-ai/ui/button" -import { Icon } from "@opencode-ai/ui/icon" -import { IconButton } from "@opencode-ai/ui/icon-button" -import { TextField } from "@opencode-ai/ui/text-field" -import { showToast } from "@opencode-ai/ui/toast" -import fuzzysort from "fuzzysort" -import { formatKeybind, parseKeybind, useCommand } from "@/context/command" -import { useLanguage } from "@/context/language" -import { useSettings } from "@/context/settings" -import { SettingsList } from "./settings-list" - -const IS_MAC = typeof navigator === "object" && /(Mac|iPod|iPhone|iPad)/.test(navigator.platform) -const PALETTE_ID = "command.palette" -const DEFAULT_PALETTE_KEYBIND = "mod+shift+p" - -type KeybindGroup = "General" | "Session" | "Navigation" | "Model and agent" | "Terminal" | "Prompt" - -type KeybindMeta = { - title: string - group: KeybindGroup -} - -type KeybindMap = Record -type CommandContext = ReturnType - -const GROUPS: KeybindGroup[] = ["General", "Session", "Navigation", "Model and agent", "Terminal", "Prompt"] - -type GroupKey = - | "settings.shortcuts.group.general" - | "settings.shortcuts.group.session" - | "settings.shortcuts.group.navigation" - | "settings.shortcuts.group.modelAndAgent" - | "settings.shortcuts.group.terminal" - | "settings.shortcuts.group.prompt" - -const groupKey: Record = { - General: "settings.shortcuts.group.general", - Session: "settings.shortcuts.group.session", - Navigation: "settings.shortcuts.group.navigation", - "Model and agent": "settings.shortcuts.group.modelAndAgent", - Terminal: "settings.shortcuts.group.terminal", - Prompt: "settings.shortcuts.group.prompt", -} - -function groupFor(id: string): KeybindGroup { - if (id === PALETTE_ID) return "General" - if (id.startsWith("terminal.")) return "Terminal" - if (id.startsWith("model.") || id.startsWith("agent.") || id.startsWith("mcp.")) return "Model and agent" - if (id.startsWith("file.") || id.startsWith("fileTree.")) return "Navigation" - if (id.startsWith("prompt.")) return "Prompt" - if ( - id.startsWith("session.") || - id.startsWith("message.") || - id.startsWith("permissions.") || - id.startsWith("steps.") || - id.startsWith("review.") - ) - return "Session" - - return "General" -} - -function isModifier(key: string) { - return key === "Shift" || key === "Control" || key === "Alt" || key === "Meta" -} - -function normalizeKey(key: string) { - if (key === ",") return "comma" - if (key === "+") return "plus" - if (key === " ") return "space" - return key.toLowerCase() -} - -function recordKeybind(event: KeyboardEvent) { - if (isModifier(event.key)) return - - const parts: string[] = [] - - const mod = IS_MAC ? event.metaKey : event.ctrlKey - if (mod) parts.push("mod") - - if (IS_MAC && event.ctrlKey) parts.push("ctrl") - if (!IS_MAC && event.metaKey) parts.push("meta") - if (event.altKey) parts.push("alt") - if (event.shiftKey) parts.push("shift") - - const key = normalizeKey(event.key) - if (!key) return - parts.push(key) - - return parts.join("+") -} - -function signatures(config: string | undefined) { - if (!config) return [] - const sigs: string[] = [] - - for (const kb of parseKeybind(config)) { - const parts: string[] = [] - if (kb.ctrl) parts.push("ctrl") - if (kb.alt) parts.push("alt") - if (kb.shift) parts.push("shift") - if (kb.meta) parts.push("meta") - if (kb.key) parts.push(kb.key) - if (parts.length === 0) continue - sigs.push(parts.join("+")) - } - - return sigs -} - -function keybinds(value: unknown): KeybindMap { - if (!value || typeof value !== "object" || Array.isArray(value)) return {} - return value as KeybindMap -} - -function listFor(command: CommandContext, map: KeybindMap, palette: string) { - const out = new Map() - out.set(PALETTE_ID, { title: palette, group: "General" }) - - for (const opt of command.catalog) { - if (opt.id.startsWith("suggested.")) continue - out.set(opt.id, { title: opt.title, group: groupFor(opt.id) }) - } - - for (const opt of command.options) { - if (opt.id.startsWith("suggested.")) continue - out.set(opt.id, { title: opt.title, group: groupFor(opt.id) }) - } - - for (const [id, value] of Object.entries(map)) { - if (typeof value !== "string") continue - if (out.has(id)) continue - out.set(id, { title: id, group: groupFor(id) }) - } - - return out -} - -function groupedFor(list: Map) { - const out = new Map() - for (const group of GROUPS) out.set(group, []) - - for (const [id, item] of list) { - const ids = out.get(item.group) - if (!ids) continue - ids.push(id) - } - - for (const group of GROUPS) { - const ids = out.get(group) - if (!ids) continue - ids.sort((a, b) => (list.get(a)?.title ?? "").localeCompare(list.get(b)?.title ?? "")) - } - - return out -} - -function filteredFor( - query: string, - list: Map, - grouped: Map, - keybind: (id: string) => string, -) { - const value = query.toLowerCase().trim() - if (!value) return grouped - - const out = new Map() - for (const group of GROUPS) out.set(group, []) - - const items = Array.from(list.entries()).map(([id, meta]) => ({ - id, - title: meta.title, - group: meta.group, - keybind: keybind(id), - })) - - const results = fuzzysort.go(value, items, { - keys: ["title", "keybind"], - threshold: -10000, - }) - - for (const result of results) { - const ids = out.get(result.obj.group) - if (!ids) continue - ids.push(result.obj.id) - } - - return out -} - -function useKeyCapture(input: { - active: () => string | null - stop: () => void - set: (id: string, keybind: string) => void - used: () => Map - language: ReturnType -}) { - onMount(() => { - const handle = (event: KeyboardEvent) => { - const id = input.active() - if (!id) return - - event.preventDefault() - event.stopPropagation() - event.stopImmediatePropagation() - - if (event.key === "Escape") { - input.stop() - return - } - - const clear = - (event.key === "Backspace" || event.key === "Delete") && - !event.ctrlKey && - !event.metaKey && - !event.altKey && - !event.shiftKey - if (clear) { - input.set(id, "none") - input.stop() - return - } - - const next = recordKeybind(event) - if (!next) return - - const conflicts = new Map() - for (const sig of signatures(next)) { - for (const item of input.used().get(sig) ?? []) { - if (item.id === id) continue - conflicts.set(item.id, item.title) - } - } - - if (conflicts.size > 0) { - showToast({ - title: input.language.t("settings.shortcuts.conflict.title"), - description: input.language.t("settings.shortcuts.conflict.description", { - keybind: formatKeybind(next, input.language.t), - titles: [...conflicts.values()].join(", "), - }), - }) - return - } - - input.set(id, next) - input.stop() - } - - makeEventListener(document, "keydown", handle, { capture: true }) - }) -} - -export const SettingsKeybinds: Component = () => { - const command = useCommand() - const language = useLanguage() - const settings = useSettings() - - const [store, setStore] = createStore({ - active: null as string | null, - filter: "", - }) - - const stop = () => { - if (!store.active) return - setStore("active", null) - command.keybinds(true) - } - - const start = (id: string) => { - if (store.active === id) { - stop() - return - } - - if (store.active) stop() - - setStore("active", id) - command.keybinds(false) - } - - const map = createMemo(() => keybinds(settings.current.keybinds)) - - const hasOverrides = createMemo(() => Object.values(map()).some((x) => typeof x === "string")) - - const resetAll = () => { - stop() - settings.keybinds.resetAll() - showToast({ - title: language.t("settings.shortcuts.reset.toast.title"), - description: language.t("settings.shortcuts.reset.toast.description"), - }) - } - - const list = createMemo(() => { - language.locale() - return listFor(command, map(), language.t("command.palette")) - }) - - const title = (id: string) => list().get(id)?.title ?? "" - - const grouped = createMemo(() => groupedFor(list())) - - const filtered = createMemo(() => { - return filteredFor(store.filter, list(), grouped(), (id) => command.keybind(id) || "") - }) - - const hasResults = createMemo(() => { - for (const group of GROUPS) { - const ids = filtered().get(group) ?? [] - if (ids.length > 0) return true - } - return false - }) - - const used = createMemo(() => { - const map = new Map() - - const add = (key: string, value: { id: string; title: string }) => { - const list = map.get(key) - if (!list) { - map.set(key, [value]) - return - } - list.push(value) - } - - const palette = settings.keybinds.get(PALETTE_ID) ?? DEFAULT_PALETTE_KEYBIND - for (const sig of signatures(palette)) { - add(sig, { id: PALETTE_ID, title: title(PALETTE_ID) }) - } - - const valueFor = (id: string) => { - const custom = settings.keybinds.get(id) - if (typeof custom === "string") return custom - - const live = command.options.find((x) => x.id === id) - if (live?.keybind) return live.keybind - - const meta = command.catalog.find((x) => x.id === id) - return meta?.keybind - } - - for (const id of list().keys()) { - if (id === PALETTE_ID) continue - for (const sig of signatures(valueFor(id))) { - add(sig, { id, title: title(id) }) - } - } - - return map - }) - - const setKeybind = (id: string, keybind: string) => settings.keybinds.set(id, keybind) - - useKeyCapture({ - active: () => store.active, - stop, - set: setKeybind, - used, - language, - }) - - onCleanup(() => { - if (store.active) command.keybinds(true) - }) - - return ( -
-
-
-
-

{language.t("settings.shortcuts.title")}

- -
- -
- - setStore("filter", v)} - placeholder={language.t("settings.shortcuts.search.placeholder")} - spellcheck={false} - autocorrect="off" - autocomplete="off" - autocapitalize="off" - class="flex-1" - /> - - setStore("filter", "")} /> - -
-
-
- -
- - {(group) => ( - 0}> -
-

{language.t(groupKey[group])}

- - - {(id) => ( -
- {title(id)} - -
- )} -
-
-
-
- )} -
- - -
- {language.t("settings.shortcuts.search.empty")} - - "{store.filter}" - -
-
-
-
- ) -} diff --git a/packages/app/src/components/settings-list.tsx b/packages/app/src/components/settings-list.tsx deleted file mode 100644 index bd8e4d7d1f..0000000000 --- a/packages/app/src/components/settings-list.tsx +++ /dev/null @@ -1,5 +0,0 @@ -import { type Component, type JSX } from "solid-js" - -export const SettingsList: Component<{ children: JSX.Element }> = (props) => { - return
{props.children}
-} diff --git a/packages/app/src/components/settings-models.tsx b/packages/app/src/components/settings-models.tsx deleted file mode 100644 index 14667338e9..0000000000 --- a/packages/app/src/components/settings-models.tsx +++ /dev/null @@ -1,137 +0,0 @@ -import { useFilteredList } from "@opencode-ai/ui/hooks" -import { ProviderIcon } from "@opencode-ai/ui/provider-icon" -import { Switch } from "@opencode-ai/ui/switch" -import { Icon } from "@opencode-ai/ui/icon" -import { IconButton } from "@opencode-ai/ui/icon-button" -import { TextField } from "@opencode-ai/ui/text-field" -import { type Component, For, Show } from "solid-js" -import { useLanguage } from "@/context/language" -import { useModels } from "@/context/models" -import { popularProviders } from "@/hooks/use-providers" -import { SettingsList } from "./settings-list" - -type ModelItem = ReturnType["list"]>[number] - -const ListLoadingState: Component<{ label: string }> = (props) => { - return ( -
- {props.label} -
- ) -} - -const ListEmptyState: Component<{ message: string; filter: string }> = (props) => { - return ( -
- {props.message} - - "{props.filter}" - -
- ) -} - -export const SettingsModels: Component = () => { - const language = useLanguage() - const models = useModels() - - const list = useFilteredList({ - items: (_filter) => models.list(), - key: (x) => `${x.provider.id}:${x.id}`, - filterKeys: ["provider.name", "name", "id"], - sortBy: (a, b) => a.name.localeCompare(b.name), - groupBy: (x) => x.provider.id, - sortGroupsBy: (a, b) => { - const aIndex = popularProviders.indexOf(a.category) - const bIndex = popularProviders.indexOf(b.category) - const aPopular = aIndex >= 0 - const bPopular = bIndex >= 0 - - if (aPopular && !bPopular) return -1 - if (!aPopular && bPopular) return 1 - if (aPopular && bPopular) return aIndex - bIndex - - const aName = a.items[0].provider.name - const bName = b.items[0].provider.name - return aName.localeCompare(bName) - }, - }) - - return ( -
-
-
-

{language.t("settings.models.title")}

-
- - - - - -
-
-
- -
- - } - > - 0} - fallback={} - > - - {(group) => ( -
-
- - {group.items[0].provider.name} -
- - - {(item) => { - const key = { providerID: item.provider.id, modelID: item.id } - return ( -
-
- {item.name} -
-
- { - models.setVisibility(key, checked) - }} - hideLabel - > - {item.name} - -
-
- ) - }} -
-
-
- )} -
-
-
-
-
- ) -} diff --git a/packages/app/src/components/settings-providers.tsx b/packages/app/src/components/settings-providers.tsx deleted file mode 100644 index cc69327f80..0000000000 --- a/packages/app/src/components/settings-providers.tsx +++ /dev/null @@ -1,251 +0,0 @@ -import { Button } from "@opencode-ai/ui/button" -import { useDialog } from "@opencode-ai/ui/context/dialog" -import { ProviderIcon } from "@opencode-ai/ui/provider-icon" -import { Tag } from "@opencode-ai/ui/tag" -import { showToast } from "@opencode-ai/ui/toast" -import { popularProviders, useProviders } from "@/hooks/use-providers" -import { createMemo, type Component, For, Show } from "solid-js" -import { useLanguage } from "@/context/language" -import { useGlobalSDK } from "@/context/global-sdk" -import { useGlobalSync } from "@/context/global-sync" -import { DialogConnectProvider } from "./dialog-connect-provider" -import { DialogSelectProvider } from "./dialog-select-provider" -import { DialogCustomProvider } from "./dialog-custom-provider" -import { SettingsList } from "./settings-list" - -type ProviderSource = "env" | "api" | "config" | "custom" -type ProviderItem = ReturnType["connected"]>[number] - -const PROVIDER_NOTES = [ - { match: (id: string) => id === "opencode", key: "dialog.provider.opencode.note" }, - { match: (id: string) => id === "opencode-go", key: "dialog.provider.opencodeGo.tagline" }, - { match: (id: string) => id === "anthropic", key: "dialog.provider.anthropic.note" }, - { match: (id: string) => id.startsWith("github-copilot"), key: "dialog.provider.copilot.note" }, - { match: (id: string) => id === "openai", key: "dialog.provider.openai.note" }, - { match: (id: string) => id === "google", key: "dialog.provider.google.note" }, - { match: (id: string) => id === "openrouter", key: "dialog.provider.openrouter.note" }, - { match: (id: string) => id === "vercel", key: "dialog.provider.vercel.note" }, -] as const - -export const SettingsProviders: Component = () => { - const dialog = useDialog() - const language = useLanguage() - const globalSDK = useGlobalSDK() - const globalSync = useGlobalSync() - const providers = useProviders() - - const connected = createMemo(() => { - return providers - .connected() - .filter((p) => p.id !== "opencode" || Object.values(p.models).find((m) => m.cost?.input)) - }) - - const popular = createMemo(() => { - const connectedIDs = new Set(connected().map((p) => p.id)) - const items = providers - .popular() - .filter((p) => !connectedIDs.has(p.id)) - .slice() - items.sort((a, b) => popularProviders.indexOf(a.id) - popularProviders.indexOf(b.id)) - return items - }) - - const source = (item: ProviderItem): ProviderSource | undefined => { - if (!("source" in item)) return - const value = item.source - if (value === "env" || value === "api" || value === "config" || value === "custom") return value - return - } - - const type = (item: ProviderItem) => { - const current = source(item) - if (current === "env") return language.t("settings.providers.tag.environment") - if (current === "api") return language.t("provider.connect.method.apiKey") - if (current === "config") { - if (isConfigCustom(item.id)) return language.t("settings.providers.tag.custom") - return language.t("settings.providers.tag.config") - } - if (current === "custom") return language.t("settings.providers.tag.custom") - return language.t("settings.providers.tag.other") - } - - const canDisconnect = (item: ProviderItem) => source(item) !== "env" - - const note = (id: string) => PROVIDER_NOTES.find((item) => item.match(id))?.key - - const isConfigCustom = (providerID: string) => { - const provider = globalSync.data.config.provider?.[providerID] - if (!provider) return false - if (provider.npm !== "@ai-sdk/openai-compatible") return false - if (!provider.models || Object.keys(provider.models).length === 0) return false - return true - } - - const disableProvider = async (providerID: string, name: string) => { - const before = globalSync.data.config.disabled_providers ?? [] - const next = before.includes(providerID) ? before : [...before, providerID] - globalSync.set("config", "disabled_providers", next) - - await globalSync - .updateConfig({ disabled_providers: next }) - .then(() => { - showToast({ - variant: "success", - icon: "circle-check", - title: language.t("provider.disconnect.toast.disconnected.title", { provider: name }), - description: language.t("provider.disconnect.toast.disconnected.description", { provider: name }), - }) - }) - .catch((err: unknown) => { - globalSync.set("config", "disabled_providers", before) - const message = err instanceof Error ? err.message : String(err) - showToast({ title: language.t("common.requestFailed"), description: message }) - }) - } - - const disconnect = async (providerID: string, name: string) => { - if (isConfigCustom(providerID)) { - await globalSDK.client.auth.remove({ providerID }).catch(() => undefined) - await disableProvider(providerID, name) - return - } - await globalSDK.client.auth - .remove({ providerID }) - .then(async () => { - await globalSDK.client.global.dispose() - showToast({ - variant: "success", - icon: "circle-check", - title: language.t("provider.disconnect.toast.disconnected.title", { provider: name }), - description: language.t("provider.disconnect.toast.disconnected.description", { provider: name }), - }) - }) - .catch((err: unknown) => { - const message = err instanceof Error ? err.message : String(err) - showToast({ title: language.t("common.requestFailed"), description: message }) - }) - } - - return ( -
-
-
-

{language.t("settings.providers.title")}

-
-
- -
-
-

{language.t("settings.providers.section.connected")}

- - 0} - fallback={ -
- {language.t("settings.providers.connected.empty")} -
- } - > - - {(item) => ( -
-
- - {item.name} - {type(item)} -
- - {language.t("settings.providers.connected.environmentDescription")} - - } - > - - -
- )} -
-
-
-
- -
-

{language.t("settings.providers.section.popular")}

- - - {(item) => ( -
-
-
- - {item.name} - - {language.t("dialog.provider.tag.recommended")} - - - {language.t("dialog.provider.tag.recommended")} - -
- - {(key) => {language.t(key())}} - -
- -
- )} -
- -
-
-
- - {language.t("provider.custom.title")} - {language.t("settings.providers.tag.custom")} -
- - {language.t("settings.providers.custom.description")} - -
- -
-
- - -
-
-
- ) -} diff --git a/packages/app/src/components/status-popover.tsx b/packages/app/src/components/status-popover.tsx deleted file mode 100644 index 6820a940b0..0000000000 --- a/packages/app/src/components/status-popover.tsx +++ /dev/null @@ -1,66 +0,0 @@ -import { Button } from "@opencode-ai/ui/button" -import { Icon } from "@opencode-ai/ui/icon" -import { Popover } from "@opencode-ai/ui/popover" -import { Suspense, createMemo, createSignal, lazy, Show } from "solid-js" -import { useLanguage } from "@/context/language" -import { useServer } from "@/context/server" -import { useSync } from "@/context/sync" - -const Body = lazy(() => import("./status-popover-body").then((x) => ({ default: x.StatusPopoverBody }))) - -export function StatusPopover() { - const language = useLanguage() - const server = useServer() - const sync = useSync() - const [shown, setShown] = createSignal(false) - const ready = createMemo(() => server.healthy() === false || sync.data.mcp_ready) - const healthy = createMemo(() => { - const serverHealthy = server.healthy() === true - const mcp = Object.values(sync.data.mcp ?? {}) - const issue = mcp.some((item) => item.status !== "connected" && item.status !== "disabled") - return serverHealthy && !issue - }) - - return ( - -
- -
-
-
- } - class="[&_[data-slot=popover-body]]:p-0 w-[360px] max-w-[calc(100vw-40px)] bg-transparent border-0 shadow-none rounded-xl" - gutter={4} - placement="bottom-end" - shift={-168} - > - - - } - > - - - -
- ) -} diff --git a/packages/app/src/components/terminal.tsx b/packages/app/src/components/terminal.tsx deleted file mode 100644 index ff5ff9dada..0000000000 --- a/packages/app/src/components/terminal.tsx +++ /dev/null @@ -1,645 +0,0 @@ -import { withAlpha } from "@opencode-ai/ui/theme/color" -import { useTheme } from "@opencode-ai/ui/theme/context" -import { resolveThemeVariant } from "@opencode-ai/ui/theme/resolve" -import type { HexColor } from "@opencode-ai/ui/theme/types" -import { showToast } from "@opencode-ai/ui/toast" -import type { FitAddon, Ghostty, Terminal as Term } from "ghostty-web" -import { type ComponentProps, createEffect, createMemo, onCleanup, onMount, splitProps } from "solid-js" -import { SerializeAddon } from "@/addons/serialize" -import { matchKeybind, parseKeybind } from "@/context/command" -import { useLanguage } from "@/context/language" -import { usePlatform } from "@/context/platform" -import { useSDK } from "@/context/sdk" -import { useServer } from "@/context/server" -import { terminalFontFamily, useSettings } from "@/context/settings" -import type { LocalPTY } from "@/context/terminal" -import { disposeIfDisposable, getHoveredLinkText, setOptionIfSupported } from "@/utils/runtime-adapters" -import { terminalWriter } from "@/utils/terminal-writer" - -const TOGGLE_TERMINAL_ID = "terminal.toggle" -const DEFAULT_TOGGLE_TERMINAL_KEYBIND = "ctrl+`" -export interface TerminalProps extends ComponentProps<"div"> { - pty: LocalPTY - autoFocus?: boolean - onSubmit?: () => void - onCleanup?: (pty: Partial & { id: string }) => void - onConnect?: () => void - onConnectError?: (error: unknown) => void -} - -let shared: Promise<{ mod: typeof import("ghostty-web"); ghostty: Ghostty }> | undefined - -const loadGhostty = () => { - if (shared) return shared - shared = import("ghostty-web") - .then(async (mod) => ({ mod, ghostty: await mod.Ghostty.load() })) - .catch((err) => { - shared = undefined - throw err - }) - return shared -} - -type TerminalColors = { - background: string - foreground: string - cursor: string - selectionBackground: string -} - -const DEFAULT_TERMINAL_COLORS: Record<"light" | "dark", TerminalColors> = { - light: { - background: "#fcfcfc", - foreground: "#211e1e", - cursor: "#211e1e", - selectionBackground: withAlpha("#211e1e", 0.2), - }, - dark: { - background: "#191515", - foreground: "#d4d4d4", - cursor: "#d4d4d4", - selectionBackground: withAlpha("#d4d4d4", 0.25), - }, -} - -const debugTerminal = (...values: unknown[]) => { - if (!import.meta.env.DEV) return - console.debug("[terminal]", ...values) -} - -const errorName = (err: unknown) => { - if (!err || typeof err !== "object") return - if (!("name" in err)) return - const errorName = err.name - return typeof errorName === "string" ? errorName : undefined -} - -const useTerminalUiBindings = (input: { - container: HTMLDivElement - term: Term - cleanups: VoidFunction[] - handlePointerDown: () => void - handleLinkClick: (event: MouseEvent) => void -}) => { - const handleCopy = (event: ClipboardEvent) => { - const selection = input.term.getSelection() - if (!selection) return - - const clipboard = event.clipboardData - if (!clipboard) return - - event.preventDefault() - clipboard.setData("text/plain", selection) - } - - const handlePaste = (event: ClipboardEvent) => { - const clipboard = event.clipboardData - const text = clipboard?.getData("text/plain") ?? clipboard?.getData("text") ?? "" - if (!text) return - - event.preventDefault() - event.stopPropagation() - input.term.paste(text) - } - - const handleTextareaFocus = () => { - input.term.options.cursorBlink = true - } - const handleTextareaBlur = () => { - input.term.options.cursorBlink = false - } - - input.container.addEventListener("copy", handleCopy, true) - input.cleanups.push(() => input.container.removeEventListener("copy", handleCopy, true)) - - input.container.addEventListener("paste", handlePaste, true) - input.cleanups.push(() => input.container.removeEventListener("paste", handlePaste, true)) - - input.container.addEventListener("pointerdown", input.handlePointerDown) - input.cleanups.push(() => input.container.removeEventListener("pointerdown", input.handlePointerDown)) - - input.container.addEventListener("click", input.handleLinkClick, { - capture: true, - }) - input.cleanups.push(() => - input.container.removeEventListener("click", input.handleLinkClick, { - capture: true, - }), - ) - - input.term.textarea?.addEventListener("focus", handleTextareaFocus) - input.term.textarea?.addEventListener("blur", handleTextareaBlur) - input.cleanups.push(() => input.term.textarea?.removeEventListener("focus", handleTextareaFocus)) - input.cleanups.push(() => input.term.textarea?.removeEventListener("blur", handleTextareaBlur)) -} - -const persistTerminal = (input: { - term: Term | undefined - addon: SerializeAddon | undefined - cursor: number - id: string - onCleanup?: (pty: Partial & { id: string }) => void -}) => { - if (!input.addon || !input.onCleanup || !input.term) return - const buffer = (() => { - try { - return input.addon.serialize() - } catch { - debugTerminal("failed to serialize terminal buffer") - return "" - } - })() - - input.onCleanup({ - id: input.id, - buffer, - cursor: input.cursor, - rows: input.term.rows, - cols: input.term.cols, - scrollY: input.term.getViewportY(), - }) -} - -export const Terminal = (props: TerminalProps) => { - const platform = usePlatform() - const sdk = useSDK() - const settings = useSettings() - const theme = useTheme() - const language = useLanguage() - const server = useServer() - const directory = sdk.directory - const client = sdk.client - const url = sdk.url - const auth = server.current?.http - const username = auth?.username ?? "opencode" - const password = auth?.password ?? "" - const sameOrigin = new URL(url, location.href).origin === location.origin - let container!: HTMLDivElement - const [local, others] = splitProps(props, ["pty", "class", "classList", "autoFocus", "onConnect", "onConnectError"]) - const id = local.pty.id - const restore = typeof local.pty.buffer === "string" ? local.pty.buffer : "" - const restoreSize = - restore && - typeof local.pty.cols === "number" && - Number.isSafeInteger(local.pty.cols) && - local.pty.cols > 0 && - typeof local.pty.rows === "number" && - Number.isSafeInteger(local.pty.rows) && - local.pty.rows > 0 - ? { cols: local.pty.cols, rows: local.pty.rows } - : undefined - const scrollY = typeof local.pty.scrollY === "number" ? local.pty.scrollY : undefined - let ws: WebSocket | undefined - let term: Term | undefined - let _ghostty: Ghostty - let serializeAddon: SerializeAddon - let fitAddon: FitAddon - let handleResize: () => void - let fitFrame: number | undefined - let sizeTimer: ReturnType | undefined - let pendingSize: { cols: number; rows: number } | undefined - let lastSize: { cols: number; rows: number } | undefined - let disposed = false - const cleanups: VoidFunction[] = [] - const start = - typeof local.pty.cursor === "number" && Number.isSafeInteger(local.pty.cursor) ? local.pty.cursor : undefined - let cursor = start ?? 0 - let seek = start !== undefined ? start : restore ? -1 : 0 - let output: ReturnType | undefined - let drop: VoidFunction | undefined - let reconn: ReturnType | undefined - let tries = 0 - - const cleanup = () => { - if (!cleanups.length) return - const fns = cleanups.splice(0).reverse() - for (const fn of fns) { - try { - fn() - } catch (err) { - debugTerminal("cleanup failed", err) - } - } - } - - const pushSize = (cols: number, rows: number) => { - return client.pty - .update({ - ptyID: id, - size: { cols, rows }, - }) - .catch((err) => { - debugTerminal("failed to sync terminal size", err) - }) - } - - const getTerminalColors = (): TerminalColors => { - const mode = theme.mode() === "dark" ? "dark" : "light" - const fallback = DEFAULT_TERMINAL_COLORS[mode] - const currentTheme = theme.themes()[theme.themeId()] - if (!currentTheme) return fallback - const variant = mode === "dark" ? currentTheme.dark : currentTheme.light - if (!variant?.seeds && !variant?.palette) return fallback - const resolved = resolveThemeVariant(variant, mode === "dark") - const text = resolved["text-stronger"] ?? fallback.foreground - const background = resolved["background-stronger"] ?? fallback.background - const alpha = mode === "dark" ? 0.25 : 0.2 - const base = text.startsWith("#") ? (text as HexColor) : (fallback.foreground as HexColor) - const selectionBackground = withAlpha(base, alpha) - return { - background, - foreground: text, - cursor: text, - selectionBackground, - } - } - - const terminalColors = createMemo(getTerminalColors) - - const scheduleFit = () => { - if (disposed) return - if (!fitAddon) return - if (fitFrame !== undefined) return - - fitFrame = requestAnimationFrame(() => { - fitFrame = undefined - if (disposed) return - fitAddon.fit() - }) - } - - const scheduleSize = (cols: number, rows: number) => { - if (disposed) return - if (lastSize?.cols === cols && lastSize?.rows === rows) return - - pendingSize = { cols, rows } - - if (!lastSize) { - lastSize = pendingSize - void pushSize(cols, rows) - return - } - - if (sizeTimer !== undefined) return - sizeTimer = setTimeout(() => { - sizeTimer = undefined - const next = pendingSize - if (!next) return - pendingSize = undefined - if (disposed) return - if (lastSize?.cols === next.cols && lastSize?.rows === next.rows) return - lastSize = next - void pushSize(next.cols, next.rows) - }, 100) - } - - createEffect(() => { - const colors = terminalColors() - if (!term) return - setOptionIfSupported(term, "theme", colors) - }) - - createEffect(() => { - const font = terminalFontFamily(settings.appearance.terminalFont()) - if (!term) return - setOptionIfSupported(term, "fontFamily", font) - scheduleFit() - }) - - let zoom = platform.webviewZoom?.() - createEffect(() => { - const next = platform.webviewZoom?.() - if (next === undefined) return - if (next === zoom) return - zoom = next - scheduleFit() - }) - - const focusTerminal = () => { - const t = term - if (!t) return - t.focus() - t.textarea?.focus() - setTimeout(() => t.textarea?.focus(), 0) - } - const handlePointerDown = () => { - const activeElement = document.activeElement - if (activeElement instanceof HTMLElement && activeElement !== container && !container.contains(activeElement)) { - activeElement.blur() - } - focusTerminal() - } - - const handleLinkClick = (event: MouseEvent) => { - if (!event.shiftKey && !event.ctrlKey && !event.metaKey) return - if (event.altKey) return - if (event.button !== 0) return - - const t = term - if (!t) return - - const text = getHoveredLinkText(t) - if (!text) return - - event.preventDefault() - event.stopImmediatePropagation() - platform.openLink(text) - } - - onMount(() => { - const run = async () => { - const loaded = await loadGhostty() - if (disposed) return - - const mod = loaded.mod - const g = loaded.ghostty - - const t = new mod.Terminal({ - cursorBlink: true, - cursorStyle: "bar", - cols: restoreSize?.cols, - rows: restoreSize?.rows, - fontSize: 14, - fontFamily: terminalFontFamily(settings.appearance.terminalFont()), - allowTransparency: false, - convertEol: false, - theme: terminalColors(), - scrollback: 10_000, - ghostty: g, - }) - cleanups.push(() => t.dispose()) - if (disposed) { - cleanup() - return - } - _ghostty = g - term = t - output = terminalWriter((data, done) => - t.write(data, () => { - done?.() - }), - ) - - t.attachCustomKeyEventHandler((event) => { - const key = event.key.toLowerCase() - - if (event.ctrlKey && event.shiftKey && !event.metaKey && key === "c") { - document.execCommand("copy") - return true - } - - // allow for toggle terminal keybinds in parent - const config = settings.keybinds.get(TOGGLE_TERMINAL_ID) ?? DEFAULT_TOGGLE_TERMINAL_KEYBIND - const keybinds = parseKeybind(config) - - return matchKeybind(keybinds, event) - }) - - const fit = new mod.FitAddon() - const serializer = new SerializeAddon() - cleanups.push(() => disposeIfDisposable(fit)) - t.loadAddon(serializer) - t.loadAddon(fit) - fitAddon = fit - serializeAddon = serializer - - t.open(container) - useTerminalUiBindings({ - container, - term: t, - cleanups, - handlePointerDown, - handleLinkClick, - }) - - if (local.autoFocus !== false) focusTerminal() - - if (typeof document !== "undefined" && document.fonts) { - void document.fonts.ready.then(scheduleFit) - } - - const onResize = t.onResize((size) => { - scheduleSize(size.cols, size.rows) - }) - cleanups.push(() => disposeIfDisposable(onResize)) - const onData = t.onData((data) => { - if (ws?.readyState === WebSocket.OPEN) ws.send(data) - }) - cleanups.push(() => disposeIfDisposable(onData)) - const onKey = t.onKey((key) => { - if (key.key == "Enter") { - props.onSubmit?.() - } - }) - cleanups.push(() => disposeIfDisposable(onKey)) - - const startResize = () => { - fit.observeResize() - handleResize = scheduleFit - window.addEventListener("resize", handleResize) - cleanups.push(() => window.removeEventListener("resize", handleResize)) - } - - const write = (data: string) => - new Promise((resolve) => { - if (!output) { - resolve() - return - } - output.push(data) - output.flush(resolve) - }) - - if (restore && restoreSize) { - await write(restore) - fit.fit() - scheduleSize(t.cols, t.rows) - if (scrollY !== undefined) t.scrollToLine(scrollY) - startResize() - } else { - fit.fit() - scheduleSize(t.cols, t.rows) - if (restore) { - await write(restore) - if (scrollY !== undefined) t.scrollToLine(scrollY) - } - startResize() - } - - const once = { value: false } - const decoder = new TextDecoder() - - const fail = (err: unknown) => { - if (disposed) return - if (once.value) return - once.value = true - local.onConnectError?.(err) - } - - const gone = () => - client.pty - .get({ ptyID: id }) - .then(() => false) - .catch((err) => { - if (errorName(err) === "NotFoundError") return true - debugTerminal("failed to inspect terminal session", err) - return false - }) - - const retry = (err: unknown) => { - if (disposed) return - if (reconn !== undefined) return - - const ms = Math.min(250 * 2 ** Math.min(tries, 4), 4_000) - reconn = setTimeout(async () => { - reconn = undefined - if (disposed) return - if (await gone()) { - if (disposed) return - fail(err) - return - } - if (disposed) return - tries += 1 - open() - }, ms) - } - - const open = () => { - if (disposed) return - drop?.() - - const next = new URL(url + `/pty/${id}/connect`) - next.searchParams.set("directory", directory) - next.searchParams.set("cursor", String(seek)) - next.protocol = next.protocol === "https:" ? "wss:" : "ws:" - if (!sameOrigin && password) { - next.searchParams.set("auth_token", btoa(`${username}:${password}`)) - // For same-origin requests, let the browser reuse the page's existing auth. - next.username = username - next.password = password - } - - const socket = new WebSocket(next) - socket.binaryType = "arraybuffer" - ws = socket - - const handleOpen = () => { - if (disposed) return - tries = 0 - local.onConnect?.() - scheduleSize(t.cols, t.rows) - } - - const handleMessage = (event: MessageEvent) => { - if (disposed) return - if (event.data instanceof ArrayBuffer) { - const bytes = new Uint8Array(event.data) - if (bytes[0] !== 0) return - const json = decoder.decode(bytes.subarray(1)) - try { - const meta = JSON.parse(json) as { cursor?: unknown } - const next = meta?.cursor - if (typeof next === "number" && Number.isSafeInteger(next) && next >= 0) { - cursor = next - seek = next - } - } catch (err) { - debugTerminal("invalid websocket control frame", err) - } - return - } - - const data = typeof event.data === "string" ? event.data : "" - if (!data) return - output?.push(data) - cursor += data.length - seek = cursor - } - - const handleError = (error: Event) => { - if (disposed) return - debugTerminal("websocket error", error) - } - - const stop = () => { - socket.removeEventListener("open", handleOpen) - socket.removeEventListener("message", handleMessage) - socket.removeEventListener("error", handleError) - socket.removeEventListener("close", handleClose) - if (ws === socket) ws = undefined - if (drop === stop) drop = undefined - if (socket.readyState !== WebSocket.CLOSED && socket.readyState !== WebSocket.CLOSING) socket.close(1000) - } - - const handleClose = (event: CloseEvent) => { - if (ws === socket) ws = undefined - if (drop === stop) drop = undefined - socket.removeEventListener("open", handleOpen) - socket.removeEventListener("message", handleMessage) - socket.removeEventListener("error", handleError) - socket.removeEventListener("close", handleClose) - if (disposed) return - if (event.code === 1000) return - retry(new Error(language.t("terminal.connectionLost.abnormalClose", { code: event.code }))) - } - - drop = stop - socket.addEventListener("open", handleOpen) - socket.addEventListener("message", handleMessage) - socket.addEventListener("error", handleError) - socket.addEventListener("close", handleClose) - } - - open() - } - - void run().catch((err) => { - if (disposed) return - showToast({ - variant: "error", - title: language.t("terminal.connectionLost.title"), - description: err instanceof Error ? err.message : language.t("terminal.connectionLost.description"), - }) - local.onConnectError?.(err) - }) - }) - - onCleanup(() => { - disposed = true - if (fitFrame !== undefined) cancelAnimationFrame(fitFrame) - if (sizeTimer !== undefined) clearTimeout(sizeTimer) - if (reconn !== undefined) clearTimeout(reconn) - drop?.() - if (ws && ws.readyState !== WebSocket.CLOSED && ws.readyState !== WebSocket.CLOSING) ws.close(1000) - - const finalize = () => { - persistTerminal({ term, addon: serializeAddon, cursor, id, onCleanup: props.onCleanup }) - cleanup() - } - - if (!output) { - finalize() - return - } - - output.flush(finalize) - }) - - return ( -
- ) -} diff --git a/packages/app/src/components/titlebar-history.test.ts b/packages/app/src/components/titlebar-history.test.ts deleted file mode 100644 index 25035d7ccf..0000000000 --- a/packages/app/src/components/titlebar-history.test.ts +++ /dev/null @@ -1,63 +0,0 @@ -import { describe, expect, test } from "bun:test" -import { applyPath, backPath, forwardPath, type TitlebarHistory } from "./titlebar-history" - -function history(): TitlebarHistory { - return { stack: [], index: 0, action: undefined } -} - -describe("titlebar history", () => { - test("append and trim keeps max bounded", () => { - let state = history() - state = applyPath(state, "/", 3) - state = applyPath(state, "/a", 3) - state = applyPath(state, "/b", 3) - state = applyPath(state, "/c", 3) - - expect(state.stack).toEqual(["/a", "/b", "/c"]) - expect(state.stack.length).toBe(3) - expect(state.index).toBe(2) - }) - - test("back and forward indexes stay correct after trimming", () => { - let state = history() - state = applyPath(state, "/", 3) - state = applyPath(state, "/a", 3) - state = applyPath(state, "/b", 3) - state = applyPath(state, "/c", 3) - - expect(state.stack).toEqual(["/a", "/b", "/c"]) - expect(state.index).toBe(2) - - const back = backPath(state) - expect(back?.to).toBe("/b") - expect(back?.state.index).toBe(1) - - const afterBack = applyPath(back!.state, back!.to, 3) - expect(afterBack.stack).toEqual(["/a", "/b", "/c"]) - expect(afterBack.index).toBe(1) - - const forward = forwardPath(afterBack) - expect(forward?.to).toBe("/c") - expect(forward?.state.index).toBe(2) - - const afterForward = applyPath(forward!.state, forward!.to, 3) - expect(afterForward.stack).toEqual(["/a", "/b", "/c"]) - expect(afterForward.index).toBe(2) - }) - - test("action-driven navigation does not push duplicate history entries", () => { - const state: TitlebarHistory = { - stack: ["/", "/a", "/b"], - index: 2, - action: undefined, - } - - const back = backPath(state) - expect(back?.to).toBe("/a") - - const next = applyPath(back!.state, back!.to, 10) - expect(next.stack).toEqual(["/", "/a", "/b"]) - expect(next.index).toBe(1) - expect(next.action).toBeUndefined() - }) -}) diff --git a/packages/app/src/components/titlebar-history.ts b/packages/app/src/components/titlebar-history.ts deleted file mode 100644 index 44dbbfa3a4..0000000000 --- a/packages/app/src/components/titlebar-history.ts +++ /dev/null @@ -1,57 +0,0 @@ -export const MAX_TITLEBAR_HISTORY = 100 - -export type TitlebarAction = "back" | "forward" | undefined - -export type TitlebarHistory = { - stack: string[] - index: number - action: TitlebarAction -} - -export function applyPath(state: TitlebarHistory, current: string, max = MAX_TITLEBAR_HISTORY): TitlebarHistory { - if (!state.stack.length) { - const stack = current === "/" ? ["/"] : ["/", current] - return { stack, index: stack.length - 1, action: undefined } - } - - const active = state.stack[state.index] - if (current === active) { - if (!state.action) return state - return { ...state, action: undefined } - } - - if (state.action) return { ...state, action: undefined } - - return pushPath(state, current, max) -} - -export function pushPath(state: TitlebarHistory, path: string, max = MAX_TITLEBAR_HISTORY): TitlebarHistory { - const stack = state.stack.slice(0, state.index + 1).concat(path) - const next = trimHistory(stack, stack.length - 1, max) - return { ...state, ...next, action: undefined } -} - -export function trimHistory(stack: string[], index: number, max = MAX_TITLEBAR_HISTORY) { - if (stack.length <= max) return { stack, index } - const cut = stack.length - max - return { - stack: stack.slice(cut), - index: Math.max(0, index - cut), - } -} - -export function backPath(state: TitlebarHistory) { - if (state.index <= 0) return - const index = state.index - 1 - const to = state.stack[index] - if (!to) return - return { state: { ...state, index, action: "back" as const }, to } -} - -export function forwardPath(state: TitlebarHistory) { - if (state.index >= state.stack.length - 1) return - const index = state.index + 1 - const to = state.stack[index] - if (!to) return - return { state: { ...state, index, action: "forward" as const }, to } -} diff --git a/packages/app/src/components/titlebar.tsx b/packages/app/src/components/titlebar.tsx deleted file mode 100644 index 29e9828a69..0000000000 --- a/packages/app/src/components/titlebar.tsx +++ /dev/null @@ -1,322 +0,0 @@ -import { createEffect, createMemo, Show, untrack } from "solid-js" -import { createStore } from "solid-js/store" -import { useLocation, useNavigate, useParams } from "@solidjs/router" -import { IconButton } from "@opencode-ai/ui/icon-button" -import { Icon } from "@opencode-ai/ui/icon" -import { Button } from "@opencode-ai/ui/button" -import { Tooltip, TooltipKeybind } from "@opencode-ai/ui/tooltip" -import { useTheme } from "@opencode-ai/ui/theme/context" - -import { useLayout } from "@/context/layout" -import { usePlatform } from "@/context/platform" -import { useCommand } from "@/context/command" -import { useLanguage } from "@/context/language" -import { useSettings } from "@/context/settings" -import { applyPath, backPath, forwardPath } from "./titlebar-history" - -type TauriDesktopWindow = { - startDragging?: () => Promise - toggleMaximize?: () => Promise -} - -type TauriThemeWindow = { - setTheme?: (theme?: "light" | "dark" | null) => Promise -} - -type TauriApi = { - window?: { - getCurrentWindow?: () => TauriDesktopWindow - } - webviewWindow?: { - getCurrentWebviewWindow?: () => TauriThemeWindow - } -} - -const tauriApi = () => (window as unknown as { __TAURI__?: TauriApi }).__TAURI__ -const currentDesktopWindow = () => tauriApi()?.window?.getCurrentWindow?.() -const currentThemeWindow = () => tauriApi()?.webviewWindow?.getCurrentWebviewWindow?.() - -export function Titlebar() { - const layout = useLayout() - const platform = usePlatform() - const command = useCommand() - const language = useLanguage() - const settings = useSettings() - const theme = useTheme() - const navigate = useNavigate() - const location = useLocation() - const params = useParams() - - const mac = createMemo(() => platform.platform === "desktop" && platform.os === "macos") - const windows = createMemo(() => platform.platform === "desktop" && platform.os === "windows") - const web = createMemo(() => platform.platform === "web") - const zoom = () => platform.webviewZoom?.() ?? 1 - const minHeight = () => (mac() ? `${40 / zoom()}px` : undefined) - - const [history, setHistory] = createStore({ - stack: [] as string[], - index: 0, - action: undefined as "back" | "forward" | undefined, - }) - - const path = () => `${location.pathname}${location.search}${location.hash}` - const creating = createMemo(() => { - if (!params.dir) return false - if (params.id) return false - const parts = location.pathname.replace(/\/+$/, "").split("/") - return parts.at(-1) === "session" - }) - - createEffect(() => { - const current = path() - - untrack(() => { - const next = applyPath(history, current) - if (next === history) return - setHistory(next) - }) - }) - - const canBack = createMemo(() => history.index > 0) - const canForward = createMemo(() => history.index < history.stack.length - 1) - const hasProjects = createMemo(() => layout.projects.list().length > 0) - const nav = createMemo(() => import.meta.env.VITE_KILO_CHANNEL !== "beta" || settings.general.showNavigation()) - - const back = () => { - const next = backPath(history) - if (!next) return - setHistory(next.state) - navigate(next.to) - } - - const forward = () => { - const next = forwardPath(history) - if (!next) return - setHistory(next.state) - navigate(next.to) - } - - command.register(() => [ - { - id: "common.goBack", - title: language.t("common.goBack"), - category: language.t("command.category.view"), - keybind: "mod+[", - onSelect: back, - }, - { - id: "common.goForward", - title: language.t("common.goForward"), - category: language.t("command.category.view"), - keybind: "mod+]", - onSelect: forward, - }, - ]) - - const getWin = () => { - if (platform.platform !== "desktop") return - return currentDesktopWindow() - } - - createEffect(() => { - if (platform.platform !== "desktop") return - - const scheme = theme.colorScheme() - const value = scheme === "system" ? null : scheme - - const win = currentThemeWindow() - if (!win?.setTheme) return - - void win.setTheme(value).catch(() => undefined) - }) - - const interactive = (target: EventTarget | null) => { - if (!(target instanceof Element)) return false - - const selector = - "button, a, input, textarea, select, option, [role='button'], [role='menuitem'], [contenteditable='true'], [contenteditable='']" - - return !!target.closest(selector) - } - - const drag = (e: MouseEvent) => { - if (platform.platform !== "desktop") return - if (e.buttons !== 1) return - if (interactive(e.target)) return - - const win = getWin() - if (!win?.startDragging) return - - e.preventDefault() - void win.startDragging().catch(() => undefined) - } - - const maximize = (e: MouseEvent) => { - if (platform.platform !== "desktop") return - if (interactive(e.target)) return - if (e.target instanceof Element && e.target.closest("[data-tauri-decorum-tb]")) return - - const win = getWin() - if (!win?.toggleMaximize) return - - e.preventDefault() - void win.toggleMaximize().catch(() => undefined) - } - - return ( -
-
- -
-
- -
- - -
- -
-
-
- - - - -
- -
-
-
- -
-
- - {!tauriApi() &&
} -
- -
-
- ) -} diff --git a/packages/app/src/constants/file-picker.ts b/packages/app/src/constants/file-picker.ts deleted file mode 100644 index c661bc8f36..0000000000 --- a/packages/app/src/constants/file-picker.ts +++ /dev/null @@ -1,89 +0,0 @@ -export const ACCEPTED_IMAGE_TYPES = ["image/png", "image/jpeg", "image/gif", "image/webp"] - -export const ACCEPTED_FILE_TYPES = [ - ...ACCEPTED_IMAGE_TYPES, - "application/pdf", - "text/*", - "application/json", - "application/ld+json", - "application/toml", - "application/x-toml", - "application/x-yaml", - "application/xml", - "application/yaml", - ".c", - ".cc", - ".cjs", - ".conf", - ".cpp", - ".css", - ".csv", - ".cts", - ".env", - ".go", - ".gql", - ".graphql", - ".h", - ".hh", - ".hpp", - ".htm", - ".html", - ".ini", - ".java", - ".js", - ".json", - ".jsx", - ".log", - ".md", - ".mdx", - ".mjs", - ".mts", - ".py", - ".rb", - ".rs", - ".sass", - ".scss", - ".sh", - ".sql", - ".toml", - ".ts", - ".tsx", - ".txt", - ".xml", - ".yaml", - ".yml", - ".zsh", -] - -const MIME_EXT = new Map([ - ["image/png", "png"], - ["image/jpeg", "jpg"], - ["image/gif", "gif"], - ["image/webp", "webp"], - ["application/pdf", "pdf"], - ["application/json", "json"], - ["application/ld+json", "jsonld"], - ["application/toml", "toml"], - ["application/x-toml", "toml"], - ["application/x-yaml", "yaml"], - ["application/xml", "xml"], - ["application/yaml", "yaml"], -]) - -const TEXT_EXT = ["txt", "text", "md", "markdown", "log", "csv"] - -export const ACCEPTED_FILE_EXTENSIONS = Array.from( - new Set( - ACCEPTED_FILE_TYPES.flatMap((item) => { - if (item.startsWith(".")) return [item.slice(1)] - if (item === "text/*") return TEXT_EXT - const out = MIME_EXT.get(item) - return out ? [out] : [] - }), - ), -).sort() - -export function filePickerFilters(ext?: string[]) { - if (!ext || ext.length === 0) return undefined - return [{ name: "Files", extensions: ext }] -} diff --git a/packages/app/src/context/command-keybind.test.ts b/packages/app/src/context/command-keybind.test.ts deleted file mode 100644 index c8e2dbb5d0..0000000000 --- a/packages/app/src/context/command-keybind.test.ts +++ /dev/null @@ -1,69 +0,0 @@ -import { describe, expect, test } from "bun:test" -import { formatKeybind, matchKeybind, parseKeybind } from "./command" - -describe("command keybind helpers", () => { - test("parseKeybind handles aliases and multiple combos", () => { - const keybinds = parseKeybind("control+option+k, mod+shift+comma") - - expect(keybinds).toHaveLength(2) - expect(keybinds[0]).toEqual({ - key: "k", - ctrl: true, - meta: false, - shift: false, - alt: true, - }) - expect(keybinds[1]?.shift).toBe(true) - expect(keybinds[1]?.key).toBe("comma") - expect(Boolean(keybinds[1]?.ctrl || keybinds[1]?.meta)).toBe(true) - }) - - test("parseKeybind treats none and empty as disabled", () => { - expect(parseKeybind("none")).toEqual([]) - expect(parseKeybind("")).toEqual([]) - }) - - test("matchKeybind normalizes punctuation keys", () => { - const keybinds = parseKeybind("ctrl+comma, shift+plus, meta+space") - - expect(matchKeybind(keybinds, new KeyboardEvent("keydown", { key: ",", ctrlKey: true }))).toBe(true) - expect(matchKeybind(keybinds, new KeyboardEvent("keydown", { key: "+", shiftKey: true }))).toBe(true) - expect(matchKeybind(keybinds, new KeyboardEvent("keydown", { key: " ", metaKey: true }))).toBe(true) - expect(matchKeybind(keybinds, new KeyboardEvent("keydown", { key: ",", ctrlKey: true, altKey: true }))).toBe(false) - }) - - test("matchKeybind supports bracket keys", () => { - const keybinds = parseKeybind("mod+alt+[, mod+alt+]") - const prev = keybinds[0] - const next = keybinds[1] - - expect( - matchKeybind( - keybinds, - new KeyboardEvent("keydown", { key: "[", ctrlKey: prev?.ctrl, metaKey: prev?.meta, altKey: true }), - ), - ).toBe(true) - expect( - matchKeybind( - keybinds, - new KeyboardEvent("keydown", { key: "]", ctrlKey: next?.ctrl, metaKey: next?.meta, altKey: true }), - ), - ).toBe(true) - }) - - test("formatKeybind returns human readable output", () => { - const display = formatKeybind("ctrl+alt+arrowup") - - expect(display).toContain("↑") - expect(display.includes("Ctrl") || display.includes("⌃")).toBe(true) - expect(display.includes("Alt") || display.includes("⌥")).toBe(true) - expect(formatKeybind("none")).toBe("") - }) - - test("formatKeybind prefers the first combo", () => { - const display = formatKeybind("mod+k,mod+p") - - expect(display.includes("K") || display.includes("k")).toBe(true) - expect(display.includes("P") || display.includes("p")).toBe(false) - }) -}) diff --git a/packages/app/src/context/command.test.ts b/packages/app/src/context/command.test.ts deleted file mode 100644 index 2b956287c5..0000000000 --- a/packages/app/src/context/command.test.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { describe, expect, test } from "bun:test" -import { upsertCommandRegistration } from "./command" - -describe("upsertCommandRegistration", () => { - test("replaces keyed registrations", () => { - const one = () => [{ id: "one", title: "One" }] - const two = () => [{ id: "two", title: "Two" }] - - const next = upsertCommandRegistration([{ key: "layout", options: one }], { key: "layout", options: two }) - - expect(next).toHaveLength(1) - expect(next[0]?.options).toBe(two) - }) - - test("keeps unkeyed registrations additive", () => { - const one = () => [{ id: "one", title: "One" }] - const two = () => [{ id: "two", title: "Two" }] - - const next = upsertCommandRegistration([{ options: one }], { options: two }) - - expect(next).toHaveLength(2) - expect(next[0]?.options).toBe(two) - expect(next[1]?.options).toBe(one) - }) -}) diff --git a/packages/app/src/context/command.tsx b/packages/app/src/context/command.tsx deleted file mode 100644 index d2238828c6..0000000000 --- a/packages/app/src/context/command.tsx +++ /dev/null @@ -1,434 +0,0 @@ -import { createSimpleContext } from "@opencode-ai/ui/context" -import { useDialog } from "@opencode-ai/ui/context/dialog" -import { type Accessor, createEffect, createMemo, onCleanup, onMount } from "solid-js" -import { createStore } from "solid-js/store" -import { makeEventListener } from "@solid-primitives/event-listener" -import { useLanguage } from "@/context/language" -import { useSettings } from "@/context/settings" -import { dict as en } from "@/i18n/en" -import { Persist, persisted } from "@/utils/persist" - -const IS_MAC = typeof navigator === "object" && /(Mac|iPod|iPhone|iPad)/.test(navigator.platform) - -const PALETTE_ID = "command.palette" -const DEFAULT_PALETTE_KEYBIND = "mod+shift+p" -const SUGGESTED_PREFIX = "suggested." -const EDITABLE_KEYBIND_IDS = new Set(["terminal.toggle", "terminal.new", "file.attach"]) - -type KeyLabel = - | "common.key.ctrl" - | "common.key.alt" - | "common.key.shift" - | "common.key.meta" - | "common.key.space" - | "common.key.backspace" - | "common.key.enter" - | "common.key.tab" - | "common.key.delete" - | "common.key.home" - | "common.key.end" - | "common.key.pageUp" - | "common.key.pageDown" - | "common.key.insert" - | "common.key.esc" - -function keyText(key: KeyLabel, t?: (key: KeyLabel) => string) { - return t ? t(key) : en[key] -} - -function actionId(id: string) { - if (!id.startsWith(SUGGESTED_PREFIX)) return id - return id.slice(SUGGESTED_PREFIX.length) -} - -function normalizeKey(key: string) { - if (key === ",") return "comma" - if (key === "+") return "plus" - if (key === " ") return "space" - return key.toLowerCase() -} - -function signature(key: string, ctrl: boolean, meta: boolean, shift: boolean, alt: boolean) { - const mask = (ctrl ? 1 : 0) | (meta ? 2 : 0) | (shift ? 4 : 0) | (alt ? 8 : 0) - return `${key}:${mask}` -} - -function signatureFromEvent(event: KeyboardEvent) { - return signature(normalizeKey(event.key), event.ctrlKey, event.metaKey, event.shiftKey, event.altKey) -} - -function isAllowedEditableKeybind(id: string | undefined) { - if (!id) return false - return EDITABLE_KEYBIND_IDS.has(actionId(id)) -} - -export type KeybindConfig = string - -export interface Keybind { - key: string - ctrl: boolean - meta: boolean - shift: boolean - alt: boolean -} - -export interface CommandOption { - id: string - title: string - description?: string - category?: string - keybind?: KeybindConfig - slash?: string - suggested?: boolean - disabled?: boolean - onSelect?: (source?: "palette" | "keybind" | "slash") => void - onHighlight?: () => (() => void) | void -} - -type CommandSource = "palette" | "keybind" | "slash" - -export type CommandCatalogItem = { - title: string - description?: string - category?: string - keybind?: KeybindConfig - slash?: string -} - -export type CommandRegistration = { - key?: string - options: Accessor -} - -export function upsertCommandRegistration(registrations: CommandRegistration[], entry: CommandRegistration) { - if (entry.key === undefined) return [entry, ...registrations] - return [entry, ...registrations.filter((x) => x.key !== entry.key)] -} - -export function parseKeybind(config: string): Keybind[] { - if (!config || config === "none") return [] - - return config.split(",").map((combo) => { - const parts = combo.trim().toLowerCase().split("+") - const keybind: Keybind = { - key: "", - ctrl: false, - meta: false, - shift: false, - alt: false, - } - - for (const part of parts) { - switch (part) { - case "ctrl": - case "control": - keybind.ctrl = true - break - case "meta": - case "cmd": - case "command": - keybind.meta = true - break - case "mod": - if (IS_MAC) keybind.meta = true - else keybind.ctrl = true - break - case "alt": - case "option": - keybind.alt = true - break - case "shift": - keybind.shift = true - break - default: - keybind.key = part - break - } - } - - return keybind - }) -} - -export function matchKeybind(keybinds: Keybind[], event: KeyboardEvent): boolean { - const eventKey = normalizeKey(event.key) - - for (const kb of keybinds) { - const keyMatch = kb.key === eventKey - const ctrlMatch = kb.ctrl === (event.ctrlKey || false) - const metaMatch = kb.meta === (event.metaKey || false) - const shiftMatch = kb.shift === (event.shiftKey || false) - const altMatch = kb.alt === (event.altKey || false) - - if (keyMatch && ctrlMatch && metaMatch && shiftMatch && altMatch) { - return true - } - } - - return false -} - -export function formatKeybind(config: string, t?: (key: KeyLabel) => string): string { - if (!config || config === "none") return "" - - const keybinds = parseKeybind(config) - if (keybinds.length === 0) return "" - - const kb = keybinds[0] - const parts: string[] = [] - - if (kb.ctrl) parts.push(IS_MAC ? "⌃" : keyText("common.key.ctrl", t)) - if (kb.alt) parts.push(IS_MAC ? "⌥" : keyText("common.key.alt", t)) - if (kb.shift) parts.push(IS_MAC ? "⇧" : keyText("common.key.shift", t)) - if (kb.meta) parts.push(IS_MAC ? "⌘" : keyText("common.key.meta", t)) - - if (kb.key) { - const keys: Record = { - arrowup: "↑", - arrowdown: "↓", - arrowleft: "←", - arrowright: "→", - comma: ",", - plus: "+", - } - const named: Record = { - backspace: "common.key.backspace", - delete: "common.key.delete", - end: "common.key.end", - enter: "common.key.enter", - esc: "common.key.esc", - escape: "common.key.esc", - home: "common.key.home", - insert: "common.key.insert", - pagedown: "common.key.pageDown", - pageup: "common.key.pageUp", - space: "common.key.space", - tab: "common.key.tab", - } - const key = kb.key.toLowerCase() - const displayKey = - keys[key] ?? - (named[key] - ? keyText(named[key], t) - : key.length === 1 - ? key.toUpperCase() - : key.charAt(0).toUpperCase() + key.slice(1)) - parts.push(displayKey) - } - - return IS_MAC ? parts.join("") : parts.join("+") -} - -function isEditableTarget(target: EventTarget | null) { - if (!(target instanceof HTMLElement)) return false - if (target.isContentEditable) return true - if (target.closest("[contenteditable='true']")) return true - if (target.closest("input, textarea, select")) return true - return false -} - -export const { use: useCommand, provider: CommandProvider } = createSimpleContext({ - name: "Command", - init: () => { - const dialog = useDialog() - const settings = useSettings() - const language = useLanguage() - const [store, setStore] = createStore({ - registrations: [] as CommandRegistration[], - suspendCount: 0, - }) - const warnedDuplicates = new Set() - - type CommandCatalog = Record - const [catalog, setCatalog, _, catalogReady] = persisted( - Persist.global("command.catalog.v1"), - createStore({}), - ) - - const bind = (id: string, def: KeybindConfig | undefined) => { - const custom = settings.keybinds.get(actionId(id)) - const config = custom ?? def - if (!config || config === "none") return - return config - } - - const registered = createMemo(() => { - const seen = new Set() - const all: CommandOption[] = [] - - for (const reg of store.registrations) { - for (const opt of reg.options()) { - if (seen.has(opt.id)) { - if (import.meta.env.DEV && !warnedDuplicates.has(opt.id)) { - warnedDuplicates.add(opt.id) - console.warn(`[command] duplicate command id "${opt.id}" registered; keeping first entry`) - } - continue - } - seen.add(opt.id) - all.push(opt) - } - } - - return all - }) - - createEffect(() => { - if (!catalogReady()) return - - setCatalog( - registered().reduce((acc, opt) => { - const id = actionId(opt.id) - acc[id] = { - title: opt.title, - description: opt.description, - category: opt.category, - keybind: opt.keybind, - slash: opt.slash, - } - return acc - }, {} as CommandCatalog), - ) - }) - - const catalogOptions = createMemo(() => Object.entries(catalog).map(([id, meta]) => ({ id, ...meta }))) - - const options = createMemo(() => { - const resolved = registered().map((opt) => ({ - ...opt, - keybind: bind(opt.id, opt.keybind), - })) - - const suggested = resolved.filter((x) => x.suggested && !x.disabled) - - return [ - ...suggested.map((x) => ({ - ...x, - id: SUGGESTED_PREFIX + x.id, - category: language.t("command.category.suggested"), - })), - ...resolved, - ] - }) - - const suspended = () => store.suspendCount > 0 - - const palette = createMemo(() => { - const config = settings.keybinds.get(PALETTE_ID) ?? DEFAULT_PALETTE_KEYBIND - const keybinds = parseKeybind(config) - return new Set(keybinds.map((kb) => signature(kb.key, kb.ctrl, kb.meta, kb.shift, kb.alt))) - }) - - const keymap = createMemo(() => { - const map = new Map() - for (const option of options()) { - if (option.id.startsWith(SUGGESTED_PREFIX)) continue - if (option.disabled) continue - if (!option.keybind) continue - - const keybinds = parseKeybind(option.keybind) - for (const kb of keybinds) { - if (!kb.key) continue - const sig = signature(kb.key, kb.ctrl, kb.meta, kb.shift, kb.alt) - if (map.has(sig)) continue - map.set(sig, option) - } - } - return map - }) - - const optionMap = createMemo(() => { - const map = new Map() - for (const option of options()) { - map.set(option.id, option) - map.set(actionId(option.id), option) - } - return map - }) - - const run = (id: string, source?: CommandSource) => { - const option = optionMap().get(id) - option?.onSelect?.(source) - } - - const showPalette = () => { - run("file.open", "palette") - } - - const handleKeyDown = (event: KeyboardEvent) => { - if (suspended() || dialog.active) return - - const sig = signatureFromEvent(event) - const isPalette = palette().has(sig) - const option = keymap().get(sig) - const modified = event.ctrlKey || event.metaKey || event.altKey - const isTab = event.key === "Tab" - - if (isEditableTarget(event.target) && !isPalette && !isAllowedEditableKeybind(option?.id) && !modified && !isTab) - return - - if (isPalette) { - event.preventDefault() - showPalette() - return - } - - if (!option) return - event.preventDefault() - option.onSelect?.("keybind") - } - - onMount(() => { - makeEventListener(document, "keydown", handleKeyDown) - }) - - function register(cb: () => CommandOption[]): void - function register(key: string, cb: () => CommandOption[]): void - function register(key: string | (() => CommandOption[]), cb?: () => CommandOption[]) { - const id = typeof key === "string" ? key : undefined - const next = typeof key === "function" ? key : cb - if (!next) return - const options = createMemo(next) - const entry: CommandRegistration = { - key: id, - options, - } - setStore("registrations", (arr) => upsertCommandRegistration(arr, entry)) - onCleanup(() => { - setStore("registrations", (arr) => arr.filter((x) => x !== entry)) - }) - } - - return { - register, - trigger(id: string, source?: CommandSource) { - run(id, source) - }, - keybind(id: string) { - if (id === PALETTE_ID) { - return formatKeybind(settings.keybinds.get(PALETTE_ID) ?? DEFAULT_PALETTE_KEYBIND, language.t) - } - - const base = actionId(id) - const option = options().find((x) => actionId(x.id) === base) - if (option?.keybind) return formatKeybind(option.keybind, language.t) - - const meta = catalog[base] - const config = bind(base, meta?.keybind) - if (!config) return "" - return formatKeybind(config, language.t) - }, - show: showPalette, - keybinds(enabled: boolean) { - setStore("suspendCount", (count) => Math.max(0, count + (enabled ? -1 : 1))) - }, - suspended, - get catalog() { - return catalogOptions() - }, - get options() { - return options() - }, - } - }, -}) diff --git a/packages/app/src/context/comments.test.ts b/packages/app/src/context/comments.test.ts deleted file mode 100644 index 82fa170f2f..0000000000 --- a/packages/app/src/context/comments.test.ts +++ /dev/null @@ -1,186 +0,0 @@ -import { beforeAll, describe, expect, mock, test } from "bun:test" -import { createRoot } from "solid-js" -import type { LineComment } from "./comments" - -let createCommentSessionForTest: typeof import("./comments").createCommentSessionForTest - -beforeAll(async () => { - mock.module("@solidjs/router", () => ({ - useNavigate: () => () => undefined, - useParams: () => ({}), - })) - mock.module("@opencode-ai/ui/context", () => ({ - createSimpleContext: () => ({ - use: () => undefined, - provider: () => undefined, - }), - })) - const mod = await import("./comments") - createCommentSessionForTest = mod.createCommentSessionForTest -}) - -function line(file: string, id: string, time: number): LineComment { - return { - id, - file, - comment: id, - time, - selection: { start: 1, end: 1 }, - } -} - -describe("comments session indexing", () => { - test("keeps file list behavior and aggregate chronological order", () => { - createRoot((dispose) => { - const now = Date.now() - const comments = createCommentSessionForTest({ - "a.ts": [line("a.ts", "a-late", now + 20_000), line("a.ts", "a-early", now + 1_000)], - "b.ts": [line("b.ts", "b-mid", now + 10_000)], - }) - - expect(comments.list("a.ts").map((item) => item.id)).toEqual(["a-late", "a-early"]) - expect(comments.all().map((item) => item.id)).toEqual(["a-early", "b-mid", "a-late"]) - - const next = comments.add({ - file: "b.ts", - comment: "next", - selection: { start: 2, end: 2 }, - }) - - expect(comments.list("b.ts").at(-1)?.id).toBe(next.id) - expect(comments.all().map((item) => item.time)).toEqual( - comments - .all() - .map((item) => item.time) - .slice() - .sort((a, b) => a - b), - ) - - dispose() - }) - }) - - test("remove updates file and aggregate indexes consistently", () => { - createRoot((dispose) => { - const comments = createCommentSessionForTest({ - "a.ts": [line("a.ts", "a1", 10), line("a.ts", "shared", 20)], - "b.ts": [line("b.ts", "shared", 30)], - }) - - comments.setFocus({ file: "a.ts", id: "shared" }) - comments.setActive({ file: "a.ts", id: "shared" }) - comments.remove("a.ts", "shared") - - expect(comments.list("a.ts").map((item) => item.id)).toEqual(["a1"]) - expect( - comments - .all() - .filter((item) => item.id === "shared") - .map((item) => item.file), - ).toEqual(["b.ts"]) - expect(comments.focus()).toBeNull() - expect(comments.active()).toEqual({ file: "a.ts", id: "shared" }) - - dispose() - }) - }) - - test("clear resets file and aggregate indexes plus focus state", () => { - createRoot((dispose) => { - const comments = createCommentSessionForTest({ - "a.ts": [line("a.ts", "a1", 10)], - }) - - const next = comments.add({ - file: "b.ts", - comment: "next", - selection: { start: 2, end: 2 }, - }) - - comments.setActive({ file: "b.ts", id: next.id }) - comments.clear() - - expect(comments.list("a.ts")).toEqual([]) - expect(comments.list("b.ts")).toEqual([]) - expect(comments.all()).toEqual([]) - expect(comments.focus()).toBeNull() - expect(comments.active()).toBeNull() - - dispose() - }) - }) - - test("remove keeps focus when same comment id exists in another file", () => { - createRoot((dispose) => { - const comments = createCommentSessionForTest({ - "a.ts": [line("a.ts", "shared", 10)], - "b.ts": [line("b.ts", "shared", 20)], - }) - - comments.setFocus({ file: "b.ts", id: "shared" }) - comments.remove("a.ts", "shared") - - expect(comments.focus()).toEqual({ file: "b.ts", id: "shared" }) - expect(comments.list("a.ts")).toEqual([]) - expect(comments.list("b.ts").map((item) => item.id)).toEqual(["shared"]) - - dispose() - }) - }) - - test("setFocus and setActive updater callbacks receive current state", () => { - createRoot((dispose) => { - const comments = createCommentSessionForTest() - - comments.setFocus({ file: "a.ts", id: "a1" }) - comments.setFocus((current) => { - expect(current).toEqual({ file: "a.ts", id: "a1" }) - return { file: "b.ts", id: "b1" } - }) - - comments.setActive({ file: "c.ts", id: "c1" }) - comments.setActive((current) => { - expect(current).toEqual({ file: "c.ts", id: "c1" }) - return null - }) - - expect(comments.focus()).toEqual({ file: "b.ts", id: "b1" }) - expect(comments.active()).toBeNull() - - dispose() - }) - }) - - test("update changes only the targeted comment body", () => { - createRoot((dispose) => { - const comments = createCommentSessionForTest({ - "a.ts": [line("a.ts", "a1", 10), line("a.ts", "a2", 20)], - }) - - comments.update("a.ts", "a2", "edited") - - expect(comments.list("a.ts").map((item) => item.comment)).toEqual(["a1", "edited"]) - - dispose() - }) - }) - - test("replace swaps comment state and clears focus state", () => { - createRoot((dispose) => { - const comments = createCommentSessionForTest({ - "a.ts": [line("a.ts", "a1", 10)], - }) - - comments.setFocus({ file: "a.ts", id: "a1" }) - comments.setActive({ file: "a.ts", id: "a1" }) - comments.replace([line("b.ts", "b1", 30)]) - - expect(comments.list("a.ts")).toEqual([]) - expect(comments.list("b.ts").map((item) => item.id)).toEqual(["b1"]) - expect(comments.focus()).toBeNull() - expect(comments.active()).toBeNull() - - dispose() - }) - }) -}) diff --git a/packages/app/src/context/comments.tsx b/packages/app/src/context/comments.tsx deleted file mode 100644 index a97010c0af..0000000000 --- a/packages/app/src/context/comments.tsx +++ /dev/null @@ -1,243 +0,0 @@ -import { batch, createMemo, createRoot, onCleanup } from "solid-js" -import { createStore, reconcile, type SetStoreFunction, type Store } from "solid-js/store" -import { createSimpleContext } from "@opencode-ai/ui/context" -import { useParams } from "@solidjs/router" -import { Persist, persisted } from "@/utils/persist" -import { createScopedCache } from "@/utils/scoped-cache" -import { uuid } from "@/utils/uuid" -import type { SelectedLineRange } from "@/context/file" - -export type LineComment = { - id: string - file: string - selection: SelectedLineRange - comment: string - time: number -} - -type CommentFocus = { file: string; id: string } - -const WORKSPACE_KEY = "__workspace__" -const MAX_COMMENT_SESSIONS = 20 - -function sessionKey(dir: string, id: string | undefined) { - return `${dir}\n${id ?? WORKSPACE_KEY}` -} - -function decodeSessionKey(key: string) { - const split = key.lastIndexOf("\n") - if (split < 0) return { dir: key, id: WORKSPACE_KEY } - return { - dir: key.slice(0, split), - id: key.slice(split + 1), - } -} - -type CommentStore = { - comments: Record -} - -function aggregate(comments: Record) { - return Object.keys(comments) - .flatMap((file) => comments[file] ?? []) - .slice() - .sort((a, b) => a.time - b.time) -} - -function cloneSelection(selection: SelectedLineRange): SelectedLineRange { - const next: SelectedLineRange = { - start: selection.start, - end: selection.end, - } - - if (selection.side) next.side = selection.side - if (selection.endSide) next.endSide = selection.endSide - return next -} - -function cloneComment(comment: LineComment): LineComment { - return { - ...comment, - selection: cloneSelection(comment.selection), - } -} - -function group(comments: LineComment[]) { - return comments.reduce>((acc, comment) => { - const list = acc[comment.file] - const next = cloneComment(comment) - if (list) { - list.push(next) - return acc - } - acc[comment.file] = [next] - return acc - }, {}) -} - -function createCommentSessionState(store: Store, setStore: SetStoreFunction) { - const [state, setState] = createStore({ - focus: null as CommentFocus | null, - active: null as CommentFocus | null, - }) - - const all = () => aggregate(store.comments) - - const setRef = ( - key: "focus" | "active", - value: CommentFocus | null | ((value: CommentFocus | null) => CommentFocus | null), - ) => setState(key, value) - - const setFocus = (value: CommentFocus | null | ((value: CommentFocus | null) => CommentFocus | null)) => - setRef("focus", value) - - const setActive = (value: CommentFocus | null | ((value: CommentFocus | null) => CommentFocus | null)) => - setRef("active", value) - - const list = (file: string) => store.comments[file] ?? [] - - const add = (input: Omit) => { - const next: LineComment = { - id: uuid(), - time: Date.now(), - ...input, - selection: cloneSelection(input.selection), - } - - batch(() => { - setStore("comments", input.file, (items) => [...(items ?? []), next]) - setFocus({ file: input.file, id: next.id }) - }) - - return next - } - - const remove = (file: string, id: string) => { - batch(() => { - setStore("comments", file, (items) => (items ?? []).filter((item) => item.id !== id)) - setFocus((current) => (current?.file === file && current.id === id ? null : current)) - }) - } - - const update = (file: string, id: string, comment: string) => { - setStore("comments", file, (items) => - (items ?? []).map((item) => { - if (item.id !== id) return item - return { ...item, comment } - }), - ) - } - - const replace = (comments: LineComment[]) => { - batch(() => { - setStore("comments", reconcile(group(comments))) - setFocus(null) - setActive(null) - }) - } - - const clear = () => { - batch(() => { - setStore("comments", reconcile({})) - setFocus(null) - setActive(null) - }) - } - - return { - list, - all, - add, - remove, - update, - replace, - clear, - focus: () => state.focus, - setFocus, - clearFocus: () => setRef("focus", null), - active: () => state.active, - setActive, - clearActive: () => setRef("active", null), - } -} - -export function createCommentSessionForTest(comments: Record = {}) { - const [store, setStore] = createStore({ comments }) - return createCommentSessionState(store, setStore) -} - -function createCommentSession(dir: string, id: string | undefined) { - const legacy = `${dir}/comments${id ? "/" + id : ""}.v1` - - const [store, setStore, _, ready] = persisted( - Persist.scoped(dir, id, "comments", [legacy]), - createStore({ - comments: {}, - }), - ) - const session = createCommentSessionState(store, setStore) - - return { - ready, - list: session.list, - all: session.all, - add: session.add, - remove: session.remove, - update: session.update, - replace: session.replace, - clear: session.clear, - focus: session.focus, - setFocus: session.setFocus, - clearFocus: session.clearFocus, - active: session.active, - setActive: session.setActive, - clearActive: session.clearActive, - } -} - -export const { use: useComments, provider: CommentsProvider } = createSimpleContext({ - name: "Comments", - gate: false, - init: () => { - const params = useParams() - const cache = createScopedCache( - (key) => { - const decoded = decodeSessionKey(key) - return createRoot((dispose) => ({ - value: createCommentSession(decoded.dir, decoded.id === WORKSPACE_KEY ? undefined : decoded.id), - dispose, - })) - }, - { - maxEntries: MAX_COMMENT_SESSIONS, - dispose: (entry) => entry.dispose(), - }, - ) - - onCleanup(() => cache.clear()) - - const load = (dir: string, id: string | undefined) => { - const key = sessionKey(dir, id) - return cache.get(key).value - } - - const session = createMemo(() => load(params.dir!, params.id)) - - return { - ready: () => session().ready(), - list: (file: string) => session().list(file), - all: () => session().all(), - add: (input: Omit) => session().add(input), - remove: (file: string, id: string) => session().remove(file, id), - update: (file: string, id: string, comment: string) => session().update(file, id, comment), - replace: (comments: LineComment[]) => session().replace(comments), - clear: () => session().clear(), - focus: () => session().focus(), - setFocus: (focus: CommentFocus | null) => session().setFocus(focus), - clearFocus: () => session().clearFocus(), - active: () => session().active(), - setActive: (active: CommentFocus | null) => session().setActive(active), - clearActive: () => session().clearActive(), - } - }, -}) diff --git a/packages/app/src/context/file-content-eviction-accounting.test.ts b/packages/app/src/context/file-content-eviction-accounting.test.ts deleted file mode 100644 index 4ef5f947c7..0000000000 --- a/packages/app/src/context/file-content-eviction-accounting.test.ts +++ /dev/null @@ -1,65 +0,0 @@ -import { afterEach, describe, expect, test } from "bun:test" -import { - evictContentLru, - getFileContentBytesTotal, - getFileContentEntryCount, - removeFileContentBytes, - resetFileContentLru, - setFileContentBytes, - touchFileContent, -} from "./file/content-cache" - -describe("file content eviction accounting", () => { - afterEach(() => { - resetFileContentLru() - }) - - test("updates byte totals incrementally for set, overwrite, remove, and reset", () => { - setFileContentBytes("a", 10) - setFileContentBytes("b", 15) - expect(getFileContentBytesTotal()).toBe(25) - expect(getFileContentEntryCount()).toBe(2) - - setFileContentBytes("a", 5) - expect(getFileContentBytesTotal()).toBe(20) - expect(getFileContentEntryCount()).toBe(2) - - touchFileContent("a") - expect(getFileContentBytesTotal()).toBe(20) - - removeFileContentBytes("b") - expect(getFileContentBytesTotal()).toBe(5) - expect(getFileContentEntryCount()).toBe(1) - - resetFileContentLru() - expect(getFileContentBytesTotal()).toBe(0) - expect(getFileContentEntryCount()).toBe(0) - }) - - test("evicts by entry cap using LRU order", () => { - for (const i of Array.from({ length: 41 }, (_, n) => n)) { - setFileContentBytes(`f-${i}`, 1) - } - - const evicted: string[] = [] - evictContentLru(undefined, (path) => evicted.push(path)) - - expect(evicted).toEqual(["f-0"]) - expect(getFileContentEntryCount()).toBe(40) - expect(getFileContentBytesTotal()).toBe(40) - }) - - test("evicts by byte cap while preserving protected entries", () => { - const chunk = 8 * 1024 * 1024 - setFileContentBytes("a", chunk) - setFileContentBytes("b", chunk) - setFileContentBytes("c", chunk) - - const evicted: string[] = [] - evictContentLru(new Set(["a"]), (path) => evicted.push(path)) - - expect(evicted).toEqual(["b"]) - expect(getFileContentEntryCount()).toBe(2) - expect(getFileContentBytesTotal()).toBe(chunk * 2) - }) -}) diff --git a/packages/app/src/context/file/content-cache.ts b/packages/app/src/context/file/content-cache.ts deleted file mode 100644 index 98de196ab6..0000000000 --- a/packages/app/src/context/file/content-cache.ts +++ /dev/null @@ -1,88 +0,0 @@ -import type { FileContent } from "@kilocode/sdk/v2" - -const MAX_FILE_CONTENT_ENTRIES = 40 -const MAX_FILE_CONTENT_BYTES = 20 * 1024 * 1024 - -const lru = new Map() -let total = 0 - -export function approxBytes(content: FileContent) { - const patchBytes = - content.patch?.hunks.reduce((sum, hunk) => { - return sum + hunk.lines.reduce((lineSum, line) => lineSum + line.length, 0) - }, 0) ?? 0 - - return (content.content.length + (content.diff?.length ?? 0) + patchBytes) * 2 -} - -function setBytes(path: string, nextBytes: number) { - const prev = lru.get(path) - if (prev !== undefined) total -= prev - lru.delete(path) - lru.set(path, nextBytes) - total += nextBytes -} - -function touch(path: string, bytes?: number) { - const prev = lru.get(path) - if (prev === undefined && bytes === undefined) return - setBytes(path, bytes ?? prev ?? 0) -} - -function remove(path: string) { - const prev = lru.get(path) - if (prev === undefined) return - lru.delete(path) - total -= prev -} - -function reset() { - lru.clear() - total = 0 -} - -export function evictContentLru(keep: Set | undefined, evict: (path: string) => void) { - const set = keep ?? new Set() - - while (lru.size > MAX_FILE_CONTENT_ENTRIES || total > MAX_FILE_CONTENT_BYTES) { - const path = lru.keys().next().value - if (!path) return - - if (set.has(path)) { - touch(path) - if (lru.size <= set.size) return - continue - } - - remove(path) - evict(path) - } -} - -export function resetFileContentLru() { - reset() -} - -export function setFileContentBytes(path: string, bytes: number) { - setBytes(path, bytes) -} - -export function removeFileContentBytes(path: string) { - remove(path) -} - -export function touchFileContent(path: string, bytes?: number) { - touch(path, bytes) -} - -export function getFileContentBytesTotal() { - return total -} - -export function getFileContentEntryCount() { - return lru.size -} - -export function hasFileContent(path: string) { - return lru.has(path) -} diff --git a/packages/app/src/context/file/path.test.ts b/packages/app/src/context/file/path.test.ts deleted file mode 100644 index feef6d466e..0000000000 --- a/packages/app/src/context/file/path.test.ts +++ /dev/null @@ -1,360 +0,0 @@ -import { describe, expect, test } from "bun:test" -import { createPathHelpers, stripQueryAndHash, unquoteGitPath, encodeFilePath } from "./path" - -describe("file path helpers", () => { - test("normalizes file inputs against workspace root", () => { - const path = createPathHelpers(() => "/repo") - expect(path.normalize("file:///repo/src/app.ts?x=1#h")).toBe("src/app.ts") - expect(path.normalize("/repo/src/app.ts")).toBe("src/app.ts") - expect(path.normalize("./src/app.ts")).toBe("src/app.ts") - expect(path.normalizeDir("src/components///")).toBe("src/components") - expect(path.tab("src/app.ts")).toBe("file://src/app.ts") - expect(path.pathFromTab("file://src/app.ts")).toBe("src/app.ts") - expect(path.pathFromTab("other://src/app.ts")).toBeUndefined() - }) - - test("normalizes Windows absolute paths with mixed separators", () => { - const path = createPathHelpers(() => "C:\\repo") - expect(path.normalize("C:\\repo\\src\\app.ts")).toBe("src\\app.ts") - expect(path.normalize("C:/repo/src/app.ts")).toBe("src/app.ts") - expect(path.normalize("file://C:/repo/src/app.ts")).toBe("src/app.ts") - expect(path.normalize("c:\\repo\\src\\app.ts")).toBe("src\\app.ts") - }) - - test("keeps query/hash stripping behavior stable", () => { - expect(stripQueryAndHash("a/b.ts#L12?x=1")).toBe("a/b.ts") - expect(stripQueryAndHash("a/b.ts?x=1#L12")).toBe("a/b.ts") - expect(stripQueryAndHash("a/b.ts")).toBe("a/b.ts") - }) - - test("unquotes git escaped octal path strings", () => { - expect(unquoteGitPath('"a/\\303\\251.txt"')).toBe("a/\u00e9.txt") - expect(unquoteGitPath('"plain\\nname"')).toBe("plain\nname") - expect(unquoteGitPath("a/b/c.ts")).toBe("a/b/c.ts") - }) -}) - -describe("encodeFilePath", () => { - describe("Linux/Unix paths", () => { - test("should handle Linux absolute path", () => { - const linuxPath = "/home/user/project/README.md" - const result = encodeFilePath(linuxPath) - const fileUrl = `file://${result}` - - // Should create a valid URL - expect(() => new URL(fileUrl)).not.toThrow() - expect(result).toBe("/home/user/project/README.md") - - const url = new URL(fileUrl) - expect(url.protocol).toBe("file:") - expect(url.pathname).toBe("/home/user/project/README.md") - }) - - test("should handle Linux path with special characters", () => { - const linuxPath = "/home/user/file#name with spaces.txt" - const result = encodeFilePath(linuxPath) - const fileUrl = `file://${result}` - - expect(() => new URL(fileUrl)).not.toThrow() - expect(result).toBe("/home/user/file%23name%20with%20spaces.txt") - }) - - test("should handle Linux relative path", () => { - const relativePath = "src/components/App.tsx" - const result = encodeFilePath(relativePath) - - expect(result).toBe("src/components/App.tsx") - }) - - test("should handle Linux root directory", () => { - const result = encodeFilePath("/") - expect(result).toBe("/") - }) - - test("should handle Linux path with all special chars", () => { - const path = "/path/to/file#with?special%chars&more.txt" - const result = encodeFilePath(path) - const fileUrl = `file://${result}` - - expect(() => new URL(fileUrl)).not.toThrow() - expect(result).toContain("%23") // # - expect(result).toContain("%3F") // ? - expect(result).toContain("%25") // % - expect(result).toContain("%26") // & - }) - }) - - describe("macOS paths", () => { - test("should handle macOS absolute path", () => { - const macPath = "/Users/kelvin/Projects/opencode/README.md" - const result = encodeFilePath(macPath) - const fileUrl = `file://${result}` - - expect(() => new URL(fileUrl)).not.toThrow() - expect(result).toBe("/Users/kelvin/Projects/opencode/README.md") - }) - - test("should handle macOS path with spaces", () => { - const macPath = "/Users/kelvin/My Documents/file.txt" - const result = encodeFilePath(macPath) - const fileUrl = `file://${result}` - - expect(() => new URL(fileUrl)).not.toThrow() - expect(result).toContain("My%20Documents") - }) - }) - - describe("Windows paths", () => { - test("should handle Windows absolute path with backslashes", () => { - const windowsPath = "D:\\dev\\projects\\opencode\\README.bs.md" - const result = encodeFilePath(windowsPath) - const fileUrl = `file://${result}` - - // Should create a valid, parseable URL - expect(() => new URL(fileUrl)).not.toThrow() - - const url = new URL(fileUrl) - expect(url.protocol).toBe("file:") - expect(url.pathname).toContain("README.bs.md") - expect(result).toBe("/D:/dev/projects/opencode/README.bs.md") - }) - - test("should handle mixed separator path (Windows + Unix)", () => { - // This is what happens in build-request-parts.ts when concatenating paths - const mixedPath = "D:\\dev\\projects\\opencode/README.bs.md" - const result = encodeFilePath(mixedPath) - const fileUrl = `file://${result}` - - expect(() => new URL(fileUrl)).not.toThrow() - expect(result).toBe("/D:/dev/projects/opencode/README.bs.md") - }) - - test("should handle Windows path with spaces", () => { - const windowsPath = "C:\\Program Files\\MyApp\\file with spaces.txt" - const result = encodeFilePath(windowsPath) - const fileUrl = `file://${result}` - - expect(() => new URL(fileUrl)).not.toThrow() - expect(result).toContain("Program%20Files") - expect(result).toContain("file%20with%20spaces.txt") - }) - - test("should handle Windows path with special chars in filename", () => { - const windowsPath = "D:\\projects\\file#name with ?marks.txt" - const result = encodeFilePath(windowsPath) - const fileUrl = `file://${result}` - - expect(() => new URL(fileUrl)).not.toThrow() - expect(result).toContain("file%23name%20with%20%3Fmarks.txt") - }) - - test("should handle Windows root directory", () => { - const windowsPath = "C:\\" - const result = encodeFilePath(windowsPath) - const fileUrl = `file://${result}` - - expect(() => new URL(fileUrl)).not.toThrow() - expect(result).toBe("/C:/") - }) - - test("should handle Windows relative path with backslashes", () => { - const windowsPath = "src\\components\\App.tsx" - const result = encodeFilePath(windowsPath) - - // Relative paths shouldn't get the leading slash - expect(result).toBe("src/components/App.tsx") - }) - - test("should NOT create invalid URL like the bug report", () => { - // This is the exact scenario from bug report by @alexyaroshuk - const windowsPath = "D:\\dev\\projects\\opencode\\README.bs.md" - const result = encodeFilePath(windowsPath) - const fileUrl = `file://${result}` - - // The bug was creating: file://D%3A%5Cdev%5Cprojects%5Copencode/README.bs.md - expect(result).not.toContain("%5C") // Should not have encoded backslashes - expect(result).not.toBe("D%3A%5Cdev%5Cprojects%5Copencode/README.bs.md") - - // Should be valid - expect(() => new URL(fileUrl)).not.toThrow() - }) - - test("should handle lowercase drive letters", () => { - const windowsPath = "c:\\users\\test\\file.txt" - const result = encodeFilePath(windowsPath) - const fileUrl = `file://${result}` - - expect(() => new URL(fileUrl)).not.toThrow() - expect(result).toBe("/c:/users/test/file.txt") - }) - }) - - describe("Cross-platform compatibility", () => { - test("should preserve Unix paths unchanged (except encoding)", () => { - const unixPath = "/usr/local/bin/app" - const result = encodeFilePath(unixPath) - expect(result).toBe("/usr/local/bin/app") - }) - - test("should normalize Windows paths for cross-platform use", () => { - const windowsPath = "C:\\Users\\test\\file.txt" - const result = encodeFilePath(windowsPath) - // Should convert to forward slashes and add leading / - expect(result).not.toContain("\\") - expect(result).toMatch(/^\/[A-Za-z]:\//) - }) - - test("should handle relative paths the same on all platforms", () => { - const unixRelative = "src/app.ts" - const windowsRelative = "src\\app.ts" - - const unixResult = encodeFilePath(unixRelative) - const windowsResult = encodeFilePath(windowsRelative) - - // Both should normalize to forward slashes - expect(unixResult).toBe("src/app.ts") - expect(windowsResult).toBe("src/app.ts") - }) - }) - - describe("Edge cases", () => { - test("should handle empty path", () => { - const result = encodeFilePath("") - expect(result).toBe("") - }) - - test("should handle path with multiple consecutive slashes", () => { - const result = encodeFilePath("//path//to///file.txt") - // Multiple slashes should be preserved (backend handles normalization) - expect(result).toBe("//path//to///file.txt") - }) - - test("should encode Unicode characters", () => { - const unicodePath = "/home/user/文档/README.md" - const result = encodeFilePath(unicodePath) - const fileUrl = `file://${result}` - - expect(() => new URL(fileUrl)).not.toThrow() - // Unicode should be encoded - expect(result).toContain("%E6%96%87%E6%A1%A3") - }) - - test("should handle already normalized Windows path", () => { - // Path that's already been normalized (has / before drive letter) - const alreadyNormalized = "/D:/path/file.txt" - const result = encodeFilePath(alreadyNormalized) - - // Should not add another leading slash - expect(result).toBe("/D:/path/file.txt") - expect(result).not.toContain("//D") - }) - - test("should handle just drive letter", () => { - const justDrive = "D:" - const result = encodeFilePath(justDrive) - const fileUrl = `file://${result}` - - expect(result).toBe("/D:") - expect(() => new URL(fileUrl)).not.toThrow() - }) - - test("should handle Windows path with trailing backslash", () => { - const trailingBackslash = "C:\\Users\\test\\" - const result = encodeFilePath(trailingBackslash) - const fileUrl = `file://${result}` - - expect(() => new URL(fileUrl)).not.toThrow() - expect(result).toBe("/C:/Users/test/") - }) - - test("should handle very long paths", () => { - const longPath = "C:\\Users\\test\\" + "verylongdirectoryname\\".repeat(20) + "file.txt" - const result = encodeFilePath(longPath) - const fileUrl = `file://${result}` - - expect(() => new URL(fileUrl)).not.toThrow() - expect(result).not.toContain("\\") - }) - - test("should handle paths with dots", () => { - const pathWithDots = "C:\\Users\\..\\test\\.\\file.txt" - const result = encodeFilePath(pathWithDots) - const fileUrl = `file://${result}` - - expect(() => new URL(fileUrl)).not.toThrow() - // Dots should be preserved (backend normalizes) - expect(result).toContain("..") - expect(result).toContain("/./") - }) - }) - - describe("Regression tests for PR #12424", () => { - test("should handle file with # in name", () => { - const path = "/path/to/file#name.txt" - const result = encodeFilePath(path) - const fileUrl = `file://${result}` - - expect(() => new URL(fileUrl)).not.toThrow() - expect(result).toBe("/path/to/file%23name.txt") - }) - - test("should handle file with ? in name", () => { - const path = "/path/to/file?name.txt" - const result = encodeFilePath(path) - const fileUrl = `file://${result}` - - expect(() => new URL(fileUrl)).not.toThrow() - expect(result).toBe("/path/to/file%3Fname.txt") - }) - - test("should handle file with % in name", () => { - const path = "/path/to/file%name.txt" - const result = encodeFilePath(path) - const fileUrl = `file://${result}` - - expect(() => new URL(fileUrl)).not.toThrow() - expect(result).toBe("/path/to/file%25name.txt") - }) - }) - - describe("Integration with file:// URL construction", () => { - test("should work with query parameters (Linux)", () => { - const path = "/home/user/file.txt" - const encoded = encodeFilePath(path) - const fileUrl = `file://${encoded}?start=10&end=20` - - const url = new URL(fileUrl) - expect(url.searchParams.get("start")).toBe("10") - expect(url.searchParams.get("end")).toBe("20") - expect(url.pathname).toBe("/home/user/file.txt") - }) - - test("should work with query parameters (Windows)", () => { - const path = "C:\\Users\\test\\file.txt" - const encoded = encodeFilePath(path) - const fileUrl = `file://${encoded}?start=10&end=20` - - const url = new URL(fileUrl) - expect(url.searchParams.get("start")).toBe("10") - expect(url.searchParams.get("end")).toBe("20") - }) - - test("should parse correctly in URL constructor (Linux)", () => { - const path = "/var/log/app.log" - const fileUrl = `file://${encodeFilePath(path)}` - const url = new URL(fileUrl) - - expect(url.protocol).toBe("file:") - expect(url.pathname).toBe("/var/log/app.log") - }) - - test("should parse correctly in URL constructor (Windows)", () => { - const path = "D:\\logs\\app.log" - const fileUrl = `file://${encodeFilePath(path)}` - const url = new URL(fileUrl) - - expect(url.protocol).toBe("file:") - expect(url.pathname).toContain("app.log") - }) - }) -}) diff --git a/packages/app/src/context/file/path.ts b/packages/app/src/context/file/path.ts deleted file mode 100644 index 53f072b6cb..0000000000 --- a/packages/app/src/context/file/path.ts +++ /dev/null @@ -1,151 +0,0 @@ -export function stripFileProtocol(input: string) { - if (!input.startsWith("file://")) return input - return input.slice("file://".length) -} - -export function stripQueryAndHash(input: string) { - const hashIndex = input.indexOf("#") - const queryIndex = input.indexOf("?") - - if (hashIndex !== -1 && queryIndex !== -1) { - return input.slice(0, Math.min(hashIndex, queryIndex)) - } - - if (hashIndex !== -1) return input.slice(0, hashIndex) - if (queryIndex !== -1) return input.slice(0, queryIndex) - return input -} - -export function unquoteGitPath(input: string) { - if (!input.startsWith('"')) return input - if (!input.endsWith('"')) return input - const body = input.slice(1, -1) - const bytes: number[] = [] - - for (let i = 0; i < body.length; i++) { - const char = body[i]! - if (char !== "\\") { - bytes.push(char.charCodeAt(0)) - continue - } - - const next = body[i + 1] - if (!next) { - bytes.push("\\".charCodeAt(0)) - continue - } - - if (next >= "0" && next <= "7") { - const chunk = body.slice(i + 1, i + 4) - const match = chunk.match(/^[0-7]{1,3}/) - if (!match) { - bytes.push(next.charCodeAt(0)) - i++ - continue - } - bytes.push(parseInt(match[0], 8)) - i += match[0].length - continue - } - - const escaped = - next === "n" - ? "\n" - : next === "r" - ? "\r" - : next === "t" - ? "\t" - : next === "b" - ? "\b" - : next === "f" - ? "\f" - : next === "v" - ? "\v" - : next === "\\" || next === '"' - ? next - : undefined - - bytes.push((escaped ?? next).charCodeAt(0)) - i++ - } - - return new TextDecoder().decode(new Uint8Array(bytes)) -} - -export function decodeFilePath(input: string) { - try { - return decodeURIComponent(input) - } catch { - return input - } -} - -export function encodeFilePath(filepath: string): string { - // Normalize Windows paths: convert backslashes to forward slashes - let normalized = filepath.replace(/\\/g, "/") - - // Handle Windows absolute paths (D:/path -> /D:/path for proper file:// URLs) - if (/^[A-Za-z]:/.test(normalized)) { - normalized = "/" + normalized - } - - // Encode each path segment (preserving forward slashes as path separators) - // Keep the colon in Windows drive letters (`/C:/...`) so downstream file URL parsers - // can reliably detect drives. - return normalized - .split("/") - .map((segment, index) => { - if (index === 1 && /^[A-Za-z]:$/.test(segment)) return segment - return encodeURIComponent(segment) - }) - .join("/") -} - -export function createPathHelpers(scope: () => string) { - const normalize = (input: string) => { - const root = scope() - - let path = unquoteGitPath(decodeFilePath(stripQueryAndHash(stripFileProtocol(input)))) - - // Separator-agnostic prefix stripping for Cygwin/native Windows compatibility - // Only case-insensitive on Windows (drive letter or UNC paths) - const windows = /^[A-Za-z]:/.test(root) || root.startsWith("\\\\") - const canonRoot = windows ? root.replace(/\\/g, "/").toLowerCase() : root.replace(/\\/g, "/") - const canonPath = windows ? path.replace(/\\/g, "/").toLowerCase() : path.replace(/\\/g, "/") - if ( - canonPath.startsWith(canonRoot) && - (canonRoot.endsWith("/") || canonPath === canonRoot || canonPath[canonRoot.length] === "/") - ) { - // Slice from original path to preserve native separators - path = path.slice(root.length) - } - - if (path.startsWith("./") || path.startsWith(".\\")) { - path = path.slice(2) - } - - if (path.startsWith("/") || path.startsWith("\\")) { - path = path.slice(1) - } - return path - } - - const tab = (input: string) => { - const path = normalize(input) - return `file://${encodeFilePath(path)}` - } - - const pathFromTab = (tabValue: string) => { - if (!tabValue.startsWith("file://")) return - return normalize(tabValue) - } - - const normalizeDir = (input: string) => normalize(input).replace(/\/+$/, "") - - return { - normalize, - tab, - pathFromTab, - normalizeDir, - } -} diff --git a/packages/app/src/context/file/tree-store.ts b/packages/app/src/context/file/tree-store.ts deleted file mode 100644 index dc4e58f7a2..0000000000 --- a/packages/app/src/context/file/tree-store.ts +++ /dev/null @@ -1,170 +0,0 @@ -import { createStore, produce, reconcile } from "solid-js/store" -import type { FileNode } from "@kilocode/sdk/v2" - -type DirectoryState = { - expanded: boolean - loaded?: boolean - loading?: boolean - error?: string - children?: string[] -} - -type TreeStoreOptions = { - scope: () => string - normalizeDir: (input: string) => string - list: (input: string) => Promise - onError: (message: string) => void -} - -export function createFileTreeStore(options: TreeStoreOptions) { - const [tree, setTree] = createStore<{ - node: Record - dir: Record - }>({ - node: {}, - dir: { "": { expanded: true } }, - }) - - const inflight = new Map>() - - const reset = () => { - inflight.clear() - setTree("node", reconcile({})) - setTree("dir", reconcile({})) - setTree("dir", "", { expanded: true }) - } - - const ensureDir = (path: string) => { - if (tree.dir[path]) return - setTree("dir", path, { expanded: false }) - } - - const listDir = (input: string, opts?: { force?: boolean }) => { - const dir = options.normalizeDir(input) - ensureDir(dir) - - const current = tree.dir[dir] - if (!opts?.force && current?.loaded) return Promise.resolve() - - const pending = inflight.get(dir) - if (pending) return pending - - setTree( - "dir", - dir, - produce((draft) => { - draft.loading = true - draft.error = undefined - }), - ) - - const directory = options.scope() - - const promise = options - .list(dir) - .then((nodes) => { - if (options.scope() !== directory) return - const prevChildren = tree.dir[dir]?.children ?? [] - const nextChildren = nodes.map((node) => node.path) - const nextSet = new Set(nextChildren) - - setTree( - "node", - produce((draft) => { - const removedDirs: string[] = [] - - for (const child of prevChildren) { - if (nextSet.has(child)) continue - const existing = draft[child] - if (existing?.type === "directory") removedDirs.push(child) - delete draft[child] - } - - if (removedDirs.length > 0) { - const keys = Object.keys(draft) - for (const key of keys) { - for (const removed of removedDirs) { - if (!key.startsWith(removed + "/")) continue - delete draft[key] - break - } - } - } - - for (const node of nodes) { - draft[node.path] = node - } - }), - ) - - setTree( - "dir", - dir, - produce((draft) => { - draft.loaded = true - draft.loading = false - draft.children = nextChildren - }), - ) - }) - .catch((e) => { - if (options.scope() !== directory) return - setTree( - "dir", - dir, - produce((draft) => { - draft.loading = false - draft.error = e.message - }), - ) - options.onError(e.message) - }) - .finally(() => { - inflight.delete(dir) - }) - - inflight.set(dir, promise) - return promise - } - - const expandDir = (input: string) => { - const dir = options.normalizeDir(input) - ensureDir(dir) - setTree("dir", dir, "expanded", true) - void listDir(dir) - } - - const collapseDir = (input: string) => { - const dir = options.normalizeDir(input) - ensureDir(dir) - setTree("dir", dir, "expanded", false) - } - - const dirState = (input: string) => { - const dir = options.normalizeDir(input) - return tree.dir[dir] - } - - const children = (input: string) => { - const dir = options.normalizeDir(input) - const ids = tree.dir[dir]?.children - if (!ids) return [] - const out: FileNode[] = [] - for (const id of ids) { - const node = tree.node[id] - if (node) out.push(node) - } - return out - } - - return { - listDir, - expandDir, - collapseDir, - dirState, - children, - node: (path: string) => tree.node[path], - isLoaded: (path: string) => Boolean(tree.dir[path]?.loaded), - reset, - } -} diff --git a/packages/app/src/context/file/types.ts b/packages/app/src/context/file/types.ts deleted file mode 100644 index ee1cc49f31..0000000000 --- a/packages/app/src/context/file/types.ts +++ /dev/null @@ -1,41 +0,0 @@ -import type { FileContent } from "@kilocode/sdk/v2" - -export type FileSelection = { - startLine: number - startChar: number - endLine: number - endChar: number -} - -export type SelectedLineRange = { - start: number - end: number - side?: "additions" | "deletions" - endSide?: "additions" | "deletions" -} - -export type FileViewState = { - scrollTop?: number - scrollLeft?: number - selectedLines?: SelectedLineRange | null -} - -export type FileState = { - path: string - name: string - loaded?: boolean - loading?: boolean - error?: string - content?: FileContent -} - -export function selectionFromLines(range: SelectedLineRange): FileSelection { - const startLine = Math.min(range.start, range.end) - const endLine = Math.max(range.start, range.end) - return { - startLine, - endLine, - startChar: 0, - endChar: 0, - } -} diff --git a/packages/app/src/context/file/view-cache.ts b/packages/app/src/context/file/view-cache.ts deleted file mode 100644 index 4c060174ab..0000000000 --- a/packages/app/src/context/file/view-cache.ts +++ /dev/null @@ -1,146 +0,0 @@ -import { createEffect, createRoot } from "solid-js" -import { createStore, produce } from "solid-js/store" -import { Persist, persisted } from "@/utils/persist" -import { createScopedCache } from "@/utils/scoped-cache" -import type { FileViewState, SelectedLineRange } from "./types" - -const WORKSPACE_KEY = "__workspace__" -const MAX_FILE_VIEW_SESSIONS = 20 -const MAX_VIEW_FILES = 500 - -function normalizeSelectedLines(range: SelectedLineRange): SelectedLineRange { - if (range.start <= range.end) return { ...range } - - const startSide = range.side - const endSide = range.endSide ?? startSide - - return { - ...range, - start: range.end, - end: range.start, - side: endSide, - endSide: startSide !== endSide ? startSide : undefined, - } -} - -function equalSelectedLines(a: SelectedLineRange | null | undefined, b: SelectedLineRange | null | undefined) { - if (!a && !b) return true - if (!a || !b) return false - const left = normalizeSelectedLines(a) - const right = normalizeSelectedLines(b) - return ( - left.start === right.start && left.end === right.end && left.side === right.side && left.endSide === right.endSide - ) -} - -function createViewSession(dir: string, id: string | undefined) { - const legacyViewKey = `${dir}/file${id ? "/" + id : ""}.v1` - - const [view, setView, _, ready] = persisted( - Persist.scoped(dir, id, "file-view", [legacyViewKey]), - createStore<{ - file: Record - }>({ - file: {}, - }), - ) - - const meta = { pruned: false } - - const pruneView = (keep?: string) => { - const keys = Object.keys(view.file) - if (keys.length <= MAX_VIEW_FILES) return - - const drop = keys.filter((key) => key !== keep).slice(0, keys.length - MAX_VIEW_FILES) - if (drop.length === 0) return - - setView( - produce((draft) => { - for (const key of drop) { - delete draft.file[key] - } - }), - ) - } - - createEffect(() => { - if (!ready()) return - if (meta.pruned) return - meta.pruned = true - pruneView() - }) - - const scrollTop = (path: string) => view.file[path]?.scrollTop - const scrollLeft = (path: string) => view.file[path]?.scrollLeft - const selectedLines = (path: string) => view.file[path]?.selectedLines - - const setScrollTop = (path: string, top: number) => { - setView( - produce((draft) => { - const file = draft.file[path] ?? (draft.file[path] = {}) - if (file.scrollTop === top) return - file.scrollTop = top - }), - ) - pruneView(path) - } - - const setScrollLeft = (path: string, left: number) => { - setView( - produce((draft) => { - const file = draft.file[path] ?? (draft.file[path] = {}) - if (file.scrollLeft === left) return - file.scrollLeft = left - }), - ) - pruneView(path) - } - - const setSelectedLines = (path: string, range: SelectedLineRange | null) => { - const next = range ? normalizeSelectedLines(range) : null - setView( - produce((draft) => { - const file = draft.file[path] ?? (draft.file[path] = {}) - if (equalSelectedLines(file.selectedLines, next)) return - file.selectedLines = next - }), - ) - pruneView(path) - } - - return { - ready, - scrollTop, - scrollLeft, - selectedLines, - setScrollTop, - setScrollLeft, - setSelectedLines, - } -} - -export function createFileViewCache() { - const cache = createScopedCache( - (key) => { - const split = key.lastIndexOf("\n") - const dir = split >= 0 ? key.slice(0, split) : key - const id = split >= 0 ? key.slice(split + 1) : WORKSPACE_KEY - return createRoot((dispose) => ({ - value: createViewSession(dir, id === WORKSPACE_KEY ? undefined : id), - dispose, - })) - }, - { - maxEntries: MAX_FILE_VIEW_SESSIONS, - dispose: (entry) => entry.dispose(), - }, - ) - - return { - load: (dir: string, id: string | undefined) => { - const key = `${dir}\n${id ?? WORKSPACE_KEY}` - return cache.get(key).value - }, - clear: () => cache.clear(), - } -} diff --git a/packages/app/src/context/file/watcher.test.ts b/packages/app/src/context/file/watcher.test.ts deleted file mode 100644 index 9536b52536..0000000000 --- a/packages/app/src/context/file/watcher.test.ts +++ /dev/null @@ -1,149 +0,0 @@ -import { describe, expect, test } from "bun:test" -import { invalidateFromWatcher } from "./watcher" - -describe("file watcher invalidation", () => { - test("reloads open files and refreshes loaded parent on add", () => { - const loads: string[] = [] - const refresh: string[] = [] - invalidateFromWatcher( - { - type: "file.watcher.updated", - properties: { - file: "src/new.ts", - event: "add", - }, - }, - { - normalize: (input) => input, - hasFile: (path) => path === "src/new.ts", - loadFile: (path) => loads.push(path), - node: () => undefined, - isDirLoaded: (path) => path === "src", - refreshDir: (path) => refresh.push(path), - }, - ) - - expect(loads).toEqual(["src/new.ts"]) - expect(refresh).toEqual(["src"]) - }) - - test("reloads files that are open in tabs", () => { - const loads: string[] = [] - - invalidateFromWatcher( - { - type: "file.watcher.updated", - properties: { - file: "src/open.ts", - event: "change", - }, - }, - { - normalize: (input) => input, - hasFile: () => false, - isOpen: (path) => path === "src/open.ts", - loadFile: (path) => loads.push(path), - node: () => ({ - path: "src/open.ts", - type: "file", - name: "open.ts", - absolute: "/repo/src/open.ts", - ignored: false, - }), - isDirLoaded: () => false, - refreshDir: () => {}, - }, - ) - - expect(loads).toEqual(["src/open.ts"]) - }) - - test("refreshes only changed loaded directory nodes", () => { - const refresh: string[] = [] - - invalidateFromWatcher( - { - type: "file.watcher.updated", - properties: { - file: "src", - event: "change", - }, - }, - { - normalize: (input) => input, - hasFile: () => false, - loadFile: () => {}, - node: () => ({ path: "src", type: "directory", name: "src", absolute: "/repo/src", ignored: false }), - isDirLoaded: (path) => path === "src", - refreshDir: (path) => refresh.push(path), - }, - ) - - invalidateFromWatcher( - { - type: "file.watcher.updated", - properties: { - file: "src/file.ts", - event: "change", - }, - }, - { - normalize: (input) => input, - hasFile: () => false, - loadFile: () => {}, - node: () => ({ - path: "src/file.ts", - type: "file", - name: "file.ts", - absolute: "/repo/src/file.ts", - ignored: false, - }), - isDirLoaded: () => true, - refreshDir: (path) => refresh.push(path), - }, - ) - - expect(refresh).toEqual(["src"]) - }) - - test("ignores invalid or git watcher updates", () => { - const refresh: string[] = [] - - invalidateFromWatcher( - { - type: "file.watcher.updated", - properties: { - file: ".git/index.lock", - event: "change", - }, - }, - { - normalize: (input) => input, - hasFile: () => true, - loadFile: () => { - throw new Error("should not load") - }, - node: () => undefined, - isDirLoaded: () => true, - refreshDir: (path) => refresh.push(path), - }, - ) - - invalidateFromWatcher( - { - type: "project.updated", - properties: {}, - }, - { - normalize: (input) => input, - hasFile: () => false, - loadFile: () => {}, - node: () => undefined, - isDirLoaded: () => true, - refreshDir: (path) => refresh.push(path), - }, - ) - - expect(refresh).toEqual([]) - }) -}) diff --git a/packages/app/src/context/file/watcher.ts b/packages/app/src/context/file/watcher.ts deleted file mode 100644 index 57bc68edc9..0000000000 --- a/packages/app/src/context/file/watcher.ts +++ /dev/null @@ -1,53 +0,0 @@ -import type { FileNode } from "@kilocode/sdk/v2" - -type WatcherEvent = { - type: string - properties: unknown -} - -type WatcherOps = { - normalize: (input: string) => string - hasFile: (path: string) => boolean - isOpen?: (path: string) => boolean - loadFile: (path: string) => void - node: (path: string) => FileNode | undefined - isDirLoaded: (path: string) => boolean - refreshDir: (path: string) => void -} - -export function invalidateFromWatcher(event: WatcherEvent, ops: WatcherOps) { - if (event.type !== "file.watcher.updated") return - const props = - typeof event.properties === "object" && event.properties ? (event.properties as Record) : undefined - const rawPath = typeof props?.file === "string" ? props.file : undefined - const kind = typeof props?.event === "string" ? props.event : undefined - if (!rawPath) return - if (!kind) return - - const path = ops.normalize(rawPath) - if (!path) return - if (path.startsWith(".git/")) return - - if (ops.hasFile(path) || ops.isOpen?.(path)) { - ops.loadFile(path) - } - - if (kind === "change") { - const dir = (() => { - if (path === "") return "" - const node = ops.node(path) - if (node?.type !== "directory") return - return path - })() - if (dir === undefined) return - if (!ops.isDirLoaded(dir)) return - ops.refreshDir(dir) - return - } - if (kind !== "add" && kind !== "unlink") return - - const parent = path.split("/").slice(0, -1).join("/") - if (!ops.isDirLoaded(parent)) return - - ops.refreshDir(parent) -} diff --git a/packages/app/src/context/global-sdk.tsx b/packages/app/src/context/global-sdk.tsx deleted file mode 100644 index 5c81174426..0000000000 --- a/packages/app/src/context/global-sdk.tsx +++ /dev/null @@ -1,256 +0,0 @@ -import type { Event } from "@kilocode/sdk/v2/client" -import { createSimpleContext } from "@opencode-ai/ui/context" -import { createGlobalEmitter } from "@solid-primitives/event-bus" -import { makeEventListener } from "@solid-primitives/event-listener" -import { batch, onCleanup, onMount } from "solid-js" -import z from "zod" -import { createSdkForServer } from "@/utils/server" -import { useLanguage } from "./language" -import { usePlatform } from "./platform" -import { useServer } from "./server" - -const abortError = z.object({ - name: z.literal("AbortError"), -}) - -export const { use: useGlobalSDK, provider: GlobalSDKProvider } = createSimpleContext({ - name: "GlobalSDK", - init: () => { - const language = useLanguage() - const server = useServer() - const platform = usePlatform() - const abort = new AbortController() - - const eventFetch = (() => { - if (!platform.fetch || !server.current) return - try { - const url = new URL(server.current.http.url) - const loopback = url.hostname === "localhost" || url.hostname === "127.0.0.1" || url.hostname === "::1" - if (url.protocol === "http:" && !loopback) return platform.fetch - } catch { - return - } - })() - - const currentServer = server.current - if (!currentServer) throw new Error(language.t("error.globalSDK.noServerAvailable")) - - const eventSdk = createSdkForServer({ - signal: abort.signal, - fetch: eventFetch, - server: currentServer.http, - }) - const emitter = createGlobalEmitter<{ - [key: string]: Event - }>() - - type Queued = { directory: string; payload: Event } - const FLUSH_FRAME_MS = 16 - const STREAM_YIELD_MS = 8 - const RECONNECT_DELAY_MS = 250 - - let queue: Queued[] = [] - let buffer: Queued[] = [] - const coalesced = new Map() - const staleDeltas = new Set() - let timer: ReturnType | undefined - let last = 0 - - const deltaKey = (directory: string, messageID: string, partID: string) => `${directory}:${messageID}:${partID}` - - const key = (directory: string, payload: Event) => { - if (payload.type === "session.status") return `session.status:${directory}:${payload.properties.sessionID}` - if (payload.type === "lsp.updated") return `lsp.updated:${directory}` - if (payload.type === "message.part.updated") { - const part = payload.properties.part - return `message.part.updated:${directory}:${part.messageID}:${part.id}` - } - } - - const flush = () => { - if (timer) clearTimeout(timer) - timer = undefined - - if (queue.length === 0) return - - const events = queue - const skip = staleDeltas.size > 0 ? new Set(staleDeltas) : undefined - queue = buffer - buffer = events - queue.length = 0 - coalesced.clear() - staleDeltas.clear() - - last = Date.now() - batch(() => { - for (const event of events) { - if (skip && event.payload.type === "message.part.delta") { - const props = event.payload.properties - if (skip.has(deltaKey(event.directory, props.messageID, props.partID))) continue - } - emitter.emit(event.directory, event.payload) - } - }) - - buffer.length = 0 - } - - const schedule = () => { - if (timer) return - const elapsed = Date.now() - last - timer = setTimeout(flush, Math.max(0, FLUSH_FRAME_MS - elapsed)) - } - - let streamErrorLogged = false - const wait = (ms: number) => new Promise((resolve) => setTimeout(resolve, ms)) - const aborted = (error: unknown) => abortError.safeParse(error).success - - let attempt: AbortController | undefined - let run: Promise | undefined - let started = false - const HEARTBEAT_TIMEOUT_MS = 15_000 - let lastEventAt = Date.now() - let heartbeat: ReturnType | undefined - const resetHeartbeat = () => { - lastEventAt = Date.now() - if (heartbeat) clearTimeout(heartbeat) - heartbeat = setTimeout(() => { - attempt?.abort() - }, HEARTBEAT_TIMEOUT_MS) - } - const clearHeartbeat = () => { - if (!heartbeat) return - clearTimeout(heartbeat) - heartbeat = undefined - } - - const start = () => { - if (started) return run - started = true - run = (async () => { - // oxlint-disable-next-line no-unmodified-loop-condition -- `started` is set to false by stop() which also aborts; both flags are checked to allow graceful exit - while (!abort.signal.aborted && started) { - attempt = new AbortController() - lastEventAt = Date.now() - const onAbort = () => { - attempt?.abort() - } - abort.signal.addEventListener("abort", onAbort) - try { - const events = await eventSdk.global.event({ - signal: attempt.signal, - onSseError: (error) => { - if (aborted(error)) return - if (streamErrorLogged) return - streamErrorLogged = true - console.error("[global-sdk] event stream error", { - url: currentServer.http.url, - fetch: eventFetch ? "platform" : "webview", - error, - }) - }, - }) - let yielded = Date.now() - resetHeartbeat() - for await (const event of events.stream) { - resetHeartbeat() - streamErrorLogged = false - const directory = event.directory ?? "global" - if (event.payload.type === "sync") { - continue - } - - const payload = event.payload as Event - - const k = key(directory, payload) - if (k) { - const i = coalesced.get(k) - if (i !== undefined) { - queue[i] = { directory, payload } - if (payload.type === "message.part.updated") { - const part = payload.properties.part - staleDeltas.add(deltaKey(directory, part.messageID, part.id)) - } - continue - } - coalesced.set(k, queue.length) - } - queue.push({ directory, payload }) - schedule() - - if (Date.now() - yielded < STREAM_YIELD_MS) continue - yielded = Date.now() - await wait(0) - } - } catch (error) { - if (!aborted(error) && !streamErrorLogged) { - streamErrorLogged = true - console.error("[global-sdk] event stream failed", { - url: currentServer.http.url, - fetch: eventFetch ? "platform" : "webview", - error, - }) - } - } finally { - abort.signal.removeEventListener("abort", onAbort) - attempt = undefined - clearHeartbeat() - } - - if (abort.signal.aborted || !started) return - await wait(RECONNECT_DELAY_MS) - } - })().finally(() => { - run = undefined - flush() - }) - return run - } - - const stop = () => { - started = false - attempt?.abort() - clearHeartbeat() - } - - onMount(() => { - makeEventListener(document, "visibilitychange", () => { - if (document.visibilityState !== "visible") return - if (!started) return - if (Date.now() - lastEventAt < HEARTBEAT_TIMEOUT_MS) return - attempt?.abort() - }) - }) - - onCleanup(() => { - stop() - abort.abort() - flush() - }) - - const sdk = createSdkForServer({ - server: server.current.http, - fetch: platform.fetch, - throwOnError: true, - }) - - return { - url: currentServer.http.url, - client: sdk, - event: { - on: emitter.on.bind(emitter), - listen: emitter.listen.bind(emitter), - start, - }, - createClient(opts: Omit[0], "server" | "fetch">) { - const s = server.current - if (!s) throw new Error(language.t("error.globalSDK.serverNotAvailable")) - return createSdkForServer({ - server: s.http, - fetch: platform.fetch, - ...opts, - }) - }, - } - }, -}) diff --git a/packages/app/src/context/global-sync.test.ts b/packages/app/src/context/global-sync.test.ts deleted file mode 100644 index 93e9c41755..0000000000 --- a/packages/app/src/context/global-sync.test.ts +++ /dev/null @@ -1,122 +0,0 @@ -import { describe, expect, test } from "bun:test" -import { canDisposeDirectory, pickDirectoriesToEvict } from "./global-sync/eviction" -import { estimateRootSessionTotal, loadRootSessionsWithFallback } from "./global-sync/session-load" - -describe("pickDirectoriesToEvict", () => { - test("keeps pinned stores and evicts idle stores", () => { - const now = 5_000 - const picks = pickDirectoriesToEvict({ - stores: ["a", "b", "c", "d"], - state: new Map([ - ["a", { lastAccessAt: 1_000 }], - ["b", { lastAccessAt: 4_900 }], - ["c", { lastAccessAt: 4_800 }], - ["d", { lastAccessAt: 3_000 }], - ]), - pins: new Set(["a"]), - max: 2, - ttl: 1_500, - now, - }) - - expect(picks).toEqual(["d", "c"]) - }) -}) - -describe("loadRootSessionsWithFallback", () => { - test("uses limited roots query when supported", async () => { - const calls: Array<{ directory: string; roots: true; limit?: number }> = [] - - const result = await loadRootSessionsWithFallback({ - directory: "dir", - limit: 10, - list: async (query) => { - calls.push(query) - return { data: [] } - }, - }) - - expect(result.data).toEqual([]) - expect(result.limited).toBe(true) - expect(calls).toEqual([{ directory: "dir", roots: true, limit: 10 }]) - }) - - test("falls back to full roots query on limited-query failure", async () => { - const calls: Array<{ directory: string; roots: true; limit?: number }> = [] - - const result = await loadRootSessionsWithFallback({ - directory: "dir", - limit: 25, - list: async (query) => { - calls.push(query) - if (query.limit) throw new Error("unsupported") - return { data: [] } - }, - }) - - expect(result.data).toEqual([]) - expect(result.limited).toBe(false) - expect(calls).toEqual([ - { directory: "dir", roots: true, limit: 25 }, - { directory: "dir", roots: true }, - ]) - }) -}) - -describe("estimateRootSessionTotal", () => { - test("keeps exact total for full fetches", () => { - expect(estimateRootSessionTotal({ count: 42, limit: 10, limited: false })).toBe(42) - }) - - test("marks has-more for full-limit limited fetches", () => { - expect(estimateRootSessionTotal({ count: 10, limit: 10, limited: true })).toBe(11) - }) - - test("keeps exact total when limited fetch is under limit", () => { - expect(estimateRootSessionTotal({ count: 9, limit: 10, limited: true })).toBe(9) - }) -}) - -describe("canDisposeDirectory", () => { - test("rejects pinned or inflight directories", () => { - expect( - canDisposeDirectory({ - directory: "dir", - hasStore: true, - pinned: true, - booting: false, - loadingSessions: false, - }), - ).toBe(false) - expect( - canDisposeDirectory({ - directory: "dir", - hasStore: true, - pinned: false, - booting: true, - loadingSessions: false, - }), - ).toBe(false) - expect( - canDisposeDirectory({ - directory: "dir", - hasStore: true, - pinned: false, - booting: false, - loadingSessions: true, - }), - ).toBe(false) - }) - - test("accepts idle unpinned directory store", () => { - expect( - canDisposeDirectory({ - directory: "dir", - hasStore: true, - pinned: false, - booting: false, - loadingSessions: false, - }), - ).toBe(true) - }) -}) diff --git a/packages/app/src/context/global-sync/event-reducer.test.ts b/packages/app/src/context/global-sync/event-reducer.test.ts deleted file mode 100644 index f66e7ede31..0000000000 --- a/packages/app/src/context/global-sync/event-reducer.test.ts +++ /dev/null @@ -1,554 +0,0 @@ -import { describe, expect, test } from "bun:test" -import type { Message, Part, PermissionRequest, Project, QuestionRequest, Session } from "@kilocode/sdk/v2/client" -import { createStore } from "solid-js/store" -import type { State } from "./types" -import { applyDirectoryEvent, applyGlobalEvent, cleanupDroppedSessionCaches } from "./event-reducer" - -const rootSession = (input: { id: string; parentID?: string; archived?: number }) => - ({ - id: input.id, - parentID: input.parentID, - time: { - created: 1, - updated: 1, - archived: input.archived, - }, - }) as Session - -const userMessage = (id: string, sessionID: string) => - ({ - id, - sessionID, - role: "user", - time: { created: 1 }, - agent: "assistant", - model: { providerID: "openai", modelID: "gpt" }, - }) as Message - -const textPart = (id: string, sessionID: string, messageID: string) => - ({ - id, - sessionID, - messageID, - type: "text", - text: id, - }) as Part - -const permissionRequest = (id: string, sessionID: string, title = id) => - ({ - id, - sessionID, - permission: title, - patterns: ["*"], - metadata: {}, - always: [], - }) as PermissionRequest - -const questionRequest = (id: string, sessionID: string, title = id) => - ({ - id, - sessionID, - questions: [ - { - question: title, - header: title, - options: [{ label: title, description: title }], - }, - ], - }) as QuestionRequest - -const baseState = (input: Partial = {}) => - ({ - status: "complete", - agent: [], - command: [], - project: "", - projectMeta: undefined, - icon: undefined, - provider: {} as State["provider"], - config: {} as State["config"], - path: { directory: "/tmp" } as State["path"], - session: [], - sessionTotal: 0, - session_status: {}, - session_diff: {}, - todo: {}, - permission: {}, - question: {}, - mcp: {}, - lsp: [], - vcs: undefined, - limit: 10, - message: {}, - part: {}, - ...input, - }) as State - -describe("applyGlobalEvent", () => { - test("upserts project.updated in sorted position", () => { - const project = [{ id: "a" }, { id: "c" }] as Project[] - let refreshCount = 0 - applyGlobalEvent({ - event: { type: "project.updated", properties: { id: "b" } }, - project, - refresh: () => { - refreshCount += 1 - }, - setGlobalProject(next) { - if (typeof next === "function") next(project) - }, - }) - - expect(project.map((x) => x.id)).toEqual(["a", "b", "c"]) - expect(refreshCount).toBe(0) - }) - - test("handles global.disposed by triggering refresh", () => { - let refreshCount = 0 - applyGlobalEvent({ - event: { type: "global.disposed" }, - project: [], - refresh: () => { - refreshCount += 1 - }, - setGlobalProject() {}, - }) - - expect(refreshCount).toBe(1) - }) - - test("handles server.connected by triggering refresh", () => { - let refreshCount = 0 - applyGlobalEvent({ - event: { type: "server.connected" }, - project: [], - refresh: () => { - refreshCount += 1 - }, - setGlobalProject() {}, - }) - - expect(refreshCount).toBe(1) - }) -}) - -describe("applyDirectoryEvent", () => { - test("inserts root sessions in sorted order and updates sessionTotal", () => { - const [store, setStore] = createStore( - baseState({ - session: [rootSession({ id: "b" })], - sessionTotal: 1, - }), - ) - - applyDirectoryEvent({ - event: { type: "session.created", properties: { info: rootSession({ id: "a" }) } }, - store, - setStore, - push() {}, - directory: "/tmp", - loadLsp() {}, - }) - - expect(store.session.map((x) => x.id)).toEqual(["a", "b"]) - expect(store.sessionTotal).toBe(2) - - applyDirectoryEvent({ - event: { type: "session.created", properties: { info: rootSession({ id: "c", parentID: "a" }) } }, - store, - setStore, - push() {}, - directory: "/tmp", - loadLsp() {}, - }) - - expect(store.sessionTotal).toBe(2) - }) - - test("cleans session caches when archived", () => { - const message = userMessage("msg_1", "ses_1") - const [store, setStore] = createStore( - baseState({ - session: [rootSession({ id: "ses_1" }), rootSession({ id: "ses_2" })], - sessionTotal: 2, - message: { ses_1: [message] }, - part: { [message.id]: [textPart("prt_1", "ses_1", message.id)] }, - session_diff: { ses_1: [] }, - todo: { ses_1: [] }, - permission: { ses_1: [] }, - question: { ses_1: [] }, - session_status: { ses_1: { type: "busy" } }, - }), - ) - - applyDirectoryEvent({ - event: { type: "session.updated", properties: { info: rootSession({ id: "ses_1", archived: 10 }) } }, - store, - setStore, - push() {}, - directory: "/tmp", - loadLsp() {}, - }) - - expect(store.session.map((x) => x.id)).toEqual(["ses_2"]) - expect(store.sessionTotal).toBe(1) - expect(store.message.ses_1).toBeUndefined() - expect(store.part[message.id]).toBeUndefined() - expect(store.session_diff.ses_1).toBeUndefined() - expect(store.todo.ses_1).toBeUndefined() - expect(store.permission.ses_1).toBeUndefined() - expect(store.question.ses_1).toBeUndefined() - expect(store.session_status.ses_1).toBeUndefined() - }) - - test("cleans session caches when deleted and decrements only root totals", () => { - const cases = [ - { info: rootSession({ id: "ses_1" }), expectedTotal: 1 }, - { info: rootSession({ id: "ses_2", parentID: "ses_1" }), expectedTotal: 2 }, - ] - - for (const item of cases) { - const message = userMessage("msg_1", item.info.id) - const [store, setStore] = createStore( - baseState({ - session: [ - rootSession({ id: "ses_1" }), - rootSession({ id: "ses_2", parentID: "ses_1" }), - rootSession({ id: "ses_3" }), - ], - sessionTotal: 2, - message: { [item.info.id]: [message] }, - part: { [message.id]: [textPart("prt_1", item.info.id, message.id)] }, - session_diff: { [item.info.id]: [] }, - todo: { [item.info.id]: [] }, - permission: { [item.info.id]: [] }, - question: { [item.info.id]: [] }, - session_status: { [item.info.id]: { type: "busy" } }, - }), - ) - - applyDirectoryEvent({ - event: { type: "session.deleted", properties: { info: item.info } }, - store, - setStore, - push() {}, - directory: "/tmp", - loadLsp() {}, - }) - - expect(store.session.find((x) => x.id === item.info.id)).toBeUndefined() - expect(store.sessionTotal).toBe(item.expectedTotal) - expect(store.message[item.info.id]).toBeUndefined() - expect(store.part[message.id]).toBeUndefined() - expect(store.session_diff[item.info.id]).toBeUndefined() - expect(store.todo[item.info.id]).toBeUndefined() - expect(store.permission[item.info.id]).toBeUndefined() - expect(store.question[item.info.id]).toBeUndefined() - expect(store.session_status[item.info.id]).toBeUndefined() - } - }) - - test("cleans caches for trimmed sessions on session.created", () => { - const dropped = rootSession({ id: "ses_b" }) - const kept = rootSession({ id: "ses_a" }) - const message = userMessage("msg_1", dropped.id) - const todos: string[] = [] - const [store, setStore] = createStore( - baseState({ - limit: 1, - session: [dropped], - message: { [dropped.id]: [message] }, - part: { [message.id]: [textPart("prt_1", dropped.id, message.id)] }, - session_diff: { [dropped.id]: [] }, - todo: { [dropped.id]: [] }, - permission: { [dropped.id]: [] }, - question: { [dropped.id]: [] }, - session_status: { [dropped.id]: { type: "busy" } }, - }), - ) - - applyDirectoryEvent({ - event: { type: "session.created", properties: { info: kept } }, - store, - setStore, - push() {}, - directory: "/tmp", - loadLsp() {}, - setSessionTodo(sessionID, value) { - if (value !== undefined) return - todos.push(sessionID) - }, - }) - - expect(store.session.map((x) => x.id)).toEqual([kept.id]) - expect(store.message[dropped.id]).toBeUndefined() - expect(store.part[message.id]).toBeUndefined() - expect(store.session_diff[dropped.id]).toBeUndefined() - expect(store.todo[dropped.id]).toBeUndefined() - expect(store.permission[dropped.id]).toBeUndefined() - expect(store.question[dropped.id]).toBeUndefined() - expect(store.session_status[dropped.id]).toBeUndefined() - expect(todos).toEqual([dropped.id]) - }) - - test("cleanupDroppedSessionCaches clears part-only orphan state", () => { - const [store, setStore] = createStore( - baseState({ - session: [rootSession({ id: "ses_keep" })], - part: { msg_1: [textPart("prt_1", "ses_drop", "msg_1")] }, - }), - ) - - cleanupDroppedSessionCaches(store, setStore, store.session) - - expect(store.part.msg_1).toBeUndefined() - }) - - test("upserts and removes messages while clearing orphaned parts", () => { - const sessionID = "ses_1" - const [store, setStore] = createStore( - baseState({ - message: { [sessionID]: [userMessage("msg_1", sessionID), userMessage("msg_3", sessionID)] }, - part: { msg_2: [textPart("prt_1", sessionID, "msg_2")] }, - }), - ) - - applyDirectoryEvent({ - event: { type: "message.updated", properties: { info: userMessage("msg_2", sessionID) } }, - store, - setStore, - push() {}, - directory: "/tmp", - loadLsp() {}, - }) - - expect(store.message[sessionID]?.map((x) => x.id)).toEqual(["msg_1", "msg_2", "msg_3"]) - - applyDirectoryEvent({ - event: { - type: "message.updated", - properties: { - info: { - ...userMessage("msg_2", sessionID), - role: "assistant", - } as Message, - }, - }, - store, - setStore, - push() {}, - directory: "/tmp", - loadLsp() {}, - }) - - expect(store.message[sessionID]?.find((x) => x.id === "msg_2")?.role).toBe("assistant") - - applyDirectoryEvent({ - event: { type: "message.removed", properties: { sessionID, messageID: "msg_2" } }, - store, - setStore, - push() {}, - directory: "/tmp", - loadLsp() {}, - }) - - expect(store.message[sessionID]?.map((x) => x.id)).toEqual(["msg_1", "msg_3"]) - expect(store.part.msg_2).toBeUndefined() - }) - - test("upserts and prunes message parts", () => { - const sessionID = "ses_1" - const messageID = "msg_1" - const [store, setStore] = createStore( - baseState({ - part: { [messageID]: [textPart("prt_1", sessionID, messageID), textPart("prt_3", sessionID, messageID)] }, - }), - ) - - applyDirectoryEvent({ - event: { type: "message.part.updated", properties: { part: textPart("prt_2", sessionID, messageID) } }, - store, - setStore, - push() {}, - directory: "/tmp", - loadLsp() {}, - }) - expect(store.part[messageID]?.map((x) => x.id)).toEqual(["prt_1", "prt_2", "prt_3"]) - - applyDirectoryEvent({ - event: { - type: "message.part.updated", - properties: { - part: { - ...textPart("prt_2", sessionID, messageID), - text: "changed", - } as Part, - }, - }, - store, - setStore, - push() {}, - directory: "/tmp", - loadLsp() {}, - }) - const updated = store.part[messageID]?.find((x) => x.id === "prt_2") - expect(updated?.type).toBe("text") - if (updated?.type === "text") expect(updated.text).toBe("changed") - - applyDirectoryEvent({ - event: { type: "message.part.removed", properties: { messageID, partID: "prt_1" } }, - store, - setStore, - push() {}, - directory: "/tmp", - loadLsp() {}, - }) - applyDirectoryEvent({ - event: { type: "message.part.removed", properties: { messageID, partID: "prt_2" } }, - store, - setStore, - push() {}, - directory: "/tmp", - loadLsp() {}, - }) - applyDirectoryEvent({ - event: { type: "message.part.removed", properties: { messageID, partID: "prt_3" } }, - store, - setStore, - push() {}, - directory: "/tmp", - loadLsp() {}, - }) - - expect(store.part[messageID]).toBeUndefined() - }) - - test("tracks permission and question request lifecycles", () => { - const sessionID = "ses_1" - const [store, setStore] = createStore( - baseState({ - permission: { [sessionID]: [permissionRequest("perm_1", sessionID), permissionRequest("perm_3", sessionID)] }, - question: { [sessionID]: [questionRequest("q_1", sessionID), questionRequest("q_3", sessionID)] }, - }), - ) - - applyDirectoryEvent({ - event: { type: "permission.asked", properties: permissionRequest("perm_2", sessionID) }, - store, - setStore, - push() {}, - directory: "/tmp", - loadLsp() {}, - }) - expect(store.permission[sessionID]?.map((x) => x.id)).toEqual(["perm_1", "perm_2", "perm_3"]) - - applyDirectoryEvent({ - event: { type: "permission.asked", properties: permissionRequest("perm_2", sessionID, "updated") }, - store, - setStore, - push() {}, - directory: "/tmp", - loadLsp() {}, - }) - expect(store.permission[sessionID]?.find((x) => x.id === "perm_2")?.permission).toBe("updated") - - applyDirectoryEvent({ - event: { type: "permission.replied", properties: { sessionID, requestID: "perm_2" } }, - store, - setStore, - push() {}, - directory: "/tmp", - loadLsp() {}, - }) - expect(store.permission[sessionID]?.map((x) => x.id)).toEqual(["perm_1", "perm_3"]) - - applyDirectoryEvent({ - event: { type: "question.asked", properties: questionRequest("q_2", sessionID) }, - store, - setStore, - push() {}, - directory: "/tmp", - loadLsp() {}, - }) - expect(store.question[sessionID]?.map((x) => x.id)).toEqual(["q_1", "q_2", "q_3"]) - - applyDirectoryEvent({ - event: { type: "question.asked", properties: questionRequest("q_2", sessionID, "updated") }, - store, - setStore, - push() {}, - directory: "/tmp", - loadLsp() {}, - }) - expect(store.question[sessionID]?.find((x) => x.id === "q_2")?.questions[0]?.header).toBe("updated") - - applyDirectoryEvent({ - event: { type: "question.rejected", properties: { sessionID, requestID: "q_2" } }, - store, - setStore, - push() {}, - directory: "/tmp", - loadLsp() {}, - }) - expect(store.question[sessionID]?.map((x) => x.id)).toEqual(["q_1", "q_3"]) - }) - - test("updates vcs branch in store and cache", () => { - const [store, setStore] = createStore(baseState({ vcs: { branch: "main", default_branch: "main" } })) - const [cacheStore, setCacheStore] = createStore({ - value: { branch: "main", default_branch: "main" } as State["vcs"], - }) - - applyDirectoryEvent({ - event: { type: "vcs.branch.updated", properties: { branch: "feature/test" } }, - store, - setStore, - push() {}, - directory: "/tmp", - loadLsp() {}, - vcsCache: { - store: cacheStore, - setStore: setCacheStore, - ready: () => true, - }, - }) - - expect(store.vcs).toEqual({ branch: "feature/test", default_branch: "main" }) - expect(cacheStore.value).toEqual({ branch: "feature/test", default_branch: "main" }) - }) - - test("routes disposal and lsp events to side-effect handlers", () => { - const [store, setStore] = createStore(baseState()) - const pushes: string[] = [] - let lspLoads = 0 - - applyDirectoryEvent({ - event: { type: "server.instance.disposed" }, - store, - setStore, - push(directory) { - pushes.push(directory) - }, - directory: "/tmp", - loadLsp() { - lspLoads += 1 - }, - }) - - applyDirectoryEvent({ - event: { type: "lsp.updated" }, - store, - setStore, - push(directory) { - pushes.push(directory) - }, - directory: "/tmp", - loadLsp() { - lspLoads += 1 - }, - }) - - expect(pushes).toEqual(["/tmp"]) - expect(lspLoads).toBe(1) - }) -}) diff --git a/packages/app/src/context/global-sync/eviction.ts b/packages/app/src/context/global-sync/eviction.ts deleted file mode 100644 index 676a6ee17e..0000000000 --- a/packages/app/src/context/global-sync/eviction.ts +++ /dev/null @@ -1,28 +0,0 @@ -import type { DisposeCheck, EvictPlan } from "./types" - -export function pickDirectoriesToEvict(input: EvictPlan) { - const overflow = Math.max(0, input.stores.length - input.max) - let pendingOverflow = overflow - const sorted = input.stores - .filter((dir) => !input.pins.has(dir)) - .slice() - .sort((a, b) => (input.state.get(a)?.lastAccessAt ?? 0) - (input.state.get(b)?.lastAccessAt ?? 0)) - const output: string[] = [] - for (const dir of sorted) { - const last = input.state.get(dir)?.lastAccessAt ?? 0 - const idle = input.now - last >= input.ttl - if (!idle && pendingOverflow <= 0) continue - output.push(dir) - if (pendingOverflow > 0) pendingOverflow -= 1 - } - return output -} - -export function canDisposeDirectory(input: DisposeCheck) { - if (!input.directory) return false - if (!input.hasStore) return false - if (input.pinned) return false - if (input.booting) return false - if (input.loadingSessions) return false - return true -} diff --git a/packages/app/src/context/global-sync/queue.ts b/packages/app/src/context/global-sync/queue.ts deleted file mode 100644 index 5c228dac04..0000000000 --- a/packages/app/src/context/global-sync/queue.ts +++ /dev/null @@ -1,84 +0,0 @@ -type QueueInput = { - paused: () => boolean - bootstrap: () => Promise - bootstrapInstance: (directory: string) => Promise | void -} - -export function createRefreshQueue(input: QueueInput) { - const queued = new Set() - let root = false - let running = false - let timer: ReturnType | undefined - - const tick = () => new Promise((resolve) => setTimeout(resolve, 0)) - - const take = (count: number) => { - if (queued.size === 0) return [] as string[] - const items: string[] = [] - for (const item of queued) { - queued.delete(item) - items.push(item) - if (items.length >= count) break - } - return items - } - - const schedule = () => { - if (timer) return - timer = setTimeout(() => { - timer = undefined - void drain() - }, 0) - } - - const push = (directory: string) => { - if (!directory) return - queued.add(directory) - if (input.paused()) return - schedule() - } - - const refresh = () => { - root = true - if (input.paused()) return - schedule() - } - - async function drain() { - if (running) return - running = true - try { - while (true) { - if (input.paused()) return - if (root) { - root = false - await input.bootstrap() - await tick() - continue - } - const dirs = take(2) - if (dirs.length === 0) return - await Promise.all(dirs.map((dir) => input.bootstrapInstance(dir))) - await tick() - } - } finally { - running = false - // oxlint-disable-next-line no-unsafe-finally -- intentional: early return skips schedule() when paused - if (input.paused()) return - if (root || queued.size) schedule() - } - } - - return { - push, - refresh, - clear(directory: string) { - queued.delete(directory) - }, - dispose() { - if (!timer) return - clearTimeout(timer) - timer = undefined - }, - } -} diff --git a/packages/app/src/context/global-sync/session-cache.test.ts b/packages/app/src/context/global-sync/session-cache.test.ts deleted file mode 100644 index 471c632d80..0000000000 --- a/packages/app/src/context/global-sync/session-cache.test.ts +++ /dev/null @@ -1,102 +0,0 @@ -import { describe, expect, test } from "bun:test" -import type { - Message, - Part, - PermissionRequest, - QuestionRequest, - SessionStatus, - SnapshotFileDiff, - Todo, -} from "@kilocode/sdk/v2/client" -import { dropSessionCaches, pickSessionCacheEvictions } from "./session-cache" - -const msg = (id: string, sessionID: string) => - ({ - id, - sessionID, - role: "user", - time: { created: 1 }, - agent: "assistant", - model: { providerID: "openai", modelID: "gpt" }, - }) as Message - -const part = (id: string, sessionID: string, messageID: string) => - ({ - id, - sessionID, - messageID, - type: "text", - text: id, - }) as Part - -describe("app session cache", () => { - test("dropSessionCaches clears orphaned parts without message rows", () => { - const store: { - session_status: Record - session_diff: Record - todo: Record - message: Record - part: Record - permission: Record - question: Record - } = { - session_status: { ses_1: { type: "busy" } as SessionStatus }, - session_diff: { ses_1: [] }, - todo: { ses_1: [] as Todo[] }, - message: {}, - part: { msg_1: [part("prt_1", "ses_1", "msg_1")] }, - permission: { ses_1: [] as PermissionRequest[] }, - question: { ses_1: [] as QuestionRequest[] }, - } - - dropSessionCaches(store, ["ses_1"]) - - expect(store.message.ses_1).toBeUndefined() - expect(store.part.msg_1).toBeUndefined() - expect(store.todo.ses_1).toBeUndefined() - expect(store.session_diff.ses_1).toBeUndefined() - expect(store.session_status.ses_1).toBeUndefined() - expect(store.permission.ses_1).toBeUndefined() - expect(store.question.ses_1).toBeUndefined() - }) - - test("dropSessionCaches clears message-backed parts", () => { - const m = msg("msg_1", "ses_1") - const store: { - session_status: Record - session_diff: Record - todo: Record - message: Record - part: Record - permission: Record - question: Record - } = { - session_status: {}, - session_diff: {}, - todo: {}, - message: { ses_1: [m] }, - part: { [m.id]: [part("prt_1", "ses_1", m.id)] }, - permission: {}, - question: {}, - } - - dropSessionCaches(store, ["ses_1"]) - - expect(store.message.ses_1).toBeUndefined() - expect(store.part[m.id]).toBeUndefined() - }) - - test("pickSessionCacheEvictions preserves requested sessions", () => { - const seen = new Set(["ses_1", "ses_2", "ses_3"]) - - const stale = pickSessionCacheEvictions({ - seen, - keep: "ses_4", - limit: 2, - preserve: ["ses_1"], - }) - - expect(stale).toEqual(["ses_2", "ses_3"]) - expect([...seen]).toEqual(["ses_1", "ses_4"]) - }) -}) diff --git a/packages/app/src/context/global-sync/session-cache.ts b/packages/app/src/context/global-sync/session-cache.ts deleted file mode 100644 index b578997307..0000000000 --- a/packages/app/src/context/global-sync/session-cache.ts +++ /dev/null @@ -1,62 +0,0 @@ -import type { - Message, - Part, - PermissionRequest, - QuestionRequest, - SessionStatus, - SnapshotFileDiff, - Todo, -} from "@kilocode/sdk/v2/client" - -export const SESSION_CACHE_LIMIT = 40 - -type SessionCache = { - session_status: Record - session_diff: Record - todo: Record - message: Record - part: Record - permission: Record - question: Record -} - -export function dropSessionCaches(store: SessionCache, sessionIDs: Iterable) { - const stale = new Set(Array.from(sessionIDs).filter(Boolean)) - if (stale.size === 0) return - - for (const key of Object.keys(store.part)) { - const parts = store.part[key] - if (!parts?.some((part) => stale.has(part?.sessionID ?? ""))) continue - delete store.part[key] - } - - for (const sessionID of stale) { - delete store.message[sessionID] - delete store.todo[sessionID] - delete store.session_diff[sessionID] - delete store.session_status[sessionID] - delete store.permission[sessionID] - delete store.question[sessionID] - } -} - -export function pickSessionCacheEvictions(input: { - seen: Set - keep: string - limit: number - preserve?: Iterable -}) { - const stale: string[] = [] - const keep = new Set([input.keep, ...Array.from(input.preserve ?? [])]) - if (input.seen.has(input.keep)) input.seen.delete(input.keep) - input.seen.add(input.keep) - for (const id of input.seen) { - if (input.seen.size - stale.length <= input.limit) break - if (keep.has(id)) continue - stale.push(id) - } - for (const id of stale) { - input.seen.delete(id) - } - return stale -} diff --git a/packages/app/src/context/global-sync/session-load.ts b/packages/app/src/context/global-sync/session-load.ts deleted file mode 100644 index 3693dcb460..0000000000 --- a/packages/app/src/context/global-sync/session-load.ts +++ /dev/null @@ -1,25 +0,0 @@ -import type { RootLoadArgs } from "./types" - -export async function loadRootSessionsWithFallback(input: RootLoadArgs) { - try { - const result = await input.list({ directory: input.directory, roots: true, limit: input.limit }) - return { - data: result.data, - limit: input.limit, - limited: true, - } as const - } catch { - const result = await input.list({ directory: input.directory, roots: true }) - return { - data: result.data, - limit: input.limit, - limited: false, - } as const - } -} - -export function estimateRootSessionTotal(input: { count: number; limit: number; limited: boolean }) { - if (!input.limited) return input.count - if (input.count < input.limit) return input.count - return input.count + 1 -} diff --git a/packages/app/src/context/global-sync/session-prefetch.test.ts b/packages/app/src/context/global-sync/session-prefetch.test.ts deleted file mode 100644 index 1ce8334698..0000000000 --- a/packages/app/src/context/global-sync/session-prefetch.test.ts +++ /dev/null @@ -1,96 +0,0 @@ -import { describe, expect, test } from "bun:test" -import { - clearSessionPrefetch, - clearSessionPrefetchDirectory, - getSessionPrefetch, - runSessionPrefetch, - setSessionPrefetch, - shouldSkipSessionPrefetch, -} from "./session-prefetch" - -describe("session prefetch", () => { - test("stores and clears message metadata by directory", () => { - clearSessionPrefetch("/tmp/a", ["ses_1"]) - clearSessionPrefetch("/tmp/b", ["ses_1"]) - - setSessionPrefetch({ - directory: "/tmp/a", - sessionID: "ses_1", - limit: 200, - cursor: "abc", - complete: false, - at: 123, - }) - - expect(getSessionPrefetch("/tmp/a", "ses_1")).toEqual({ limit: 200, cursor: "abc", complete: false, at: 123 }) - expect(getSessionPrefetch("/tmp/b", "ses_1")).toBeUndefined() - - clearSessionPrefetch("/tmp/a", ["ses_1"]) - - expect(getSessionPrefetch("/tmp/a", "ses_1")).toBeUndefined() - }) - - test("dedupes inflight work", async () => { - clearSessionPrefetch("/tmp/c", ["ses_2"]) - - let calls = 0 - const run = () => - runSessionPrefetch({ - directory: "/tmp/c", - sessionID: "ses_2", - task: async () => { - calls += 1 - return { limit: 100, cursor: "next", complete: true, at: 456 } - }, - }) - - const [a, b] = await Promise.all([run(), run()]) - - expect(calls).toBe(1) - expect(a).toEqual({ limit: 100, cursor: "next", complete: true, at: 456 }) - expect(b).toEqual({ limit: 100, cursor: "next", complete: true, at: 456 }) - }) - - test("clears a whole directory", () => { - setSessionPrefetch({ directory: "/tmp/d", sessionID: "ses_1", limit: 10, cursor: "a", complete: true, at: 1 }) - setSessionPrefetch({ directory: "/tmp/d", sessionID: "ses_2", limit: 20, cursor: "b", complete: false, at: 2 }) - setSessionPrefetch({ directory: "/tmp/e", sessionID: "ses_1", limit: 30, cursor: "c", complete: true, at: 3 }) - - clearSessionPrefetchDirectory("/tmp/d") - - expect(getSessionPrefetch("/tmp/d", "ses_1")).toBeUndefined() - expect(getSessionPrefetch("/tmp/d", "ses_2")).toBeUndefined() - expect(getSessionPrefetch("/tmp/e", "ses_1")).toEqual({ limit: 30, cursor: "c", complete: true, at: 3 }) - }) - - test("refreshes stale first-page prefetched history", () => { - expect( - shouldSkipSessionPrefetch({ - message: true, - info: { limit: 200, cursor: "x", complete: false, at: 1 }, - chunk: 200, - now: 1 + 15_001, - }), - ).toBe(false) - }) - - test("keeps deeper or complete history cached", () => { - expect( - shouldSkipSessionPrefetch({ - message: true, - info: { limit: 400, cursor: "x", complete: false, at: 1 }, - chunk: 200, - now: 1 + 15_001, - }), - ).toBe(true) - - expect( - shouldSkipSessionPrefetch({ - message: true, - info: { limit: 120, complete: true, at: 1 }, - chunk: 200, - now: 1 + 15_001, - }), - ).toBe(true) - }) -}) diff --git a/packages/app/src/context/global-sync/session-prefetch.ts b/packages/app/src/context/global-sync/session-prefetch.ts deleted file mode 100644 index 608561f855..0000000000 --- a/packages/app/src/context/global-sync/session-prefetch.ts +++ /dev/null @@ -1,100 +0,0 @@ -const key = (directory: string, sessionID: string) => `${directory}\n${sessionID}` - -export const SESSION_PREFETCH_TTL = 15_000 - -type Meta = { - limit: number - cursor?: string - complete: boolean - at: number -} - -export function shouldSkipSessionPrefetch(input: { message: boolean; info?: Meta; chunk: number; now?: number }) { - if (input.message) { - if (!input.info) return true - if (input.info.complete) return true - if (input.info.limit > input.chunk) return true - } else { - if (!input.info) return false - } - - return (input.now ?? Date.now()) - input.info.at < SESSION_PREFETCH_TTL -} - -const cache = new Map() -const inflight = new Map>() -const rev = new Map() - -const version = (id: string) => rev.get(id) ?? 0 - -export function getSessionPrefetch(directory: string, sessionID: string) { - return cache.get(key(directory, sessionID)) -} - -export function getSessionPrefetchPromise(directory: string, sessionID: string) { - return inflight.get(key(directory, sessionID)) -} - -export function clearSessionPrefetchInflight() { - inflight.clear() -} - -export function isSessionPrefetchCurrent(directory: string, sessionID: string, value: number) { - return version(key(directory, sessionID)) === value -} - -export function runSessionPrefetch(input: { - directory: string - sessionID: string - task: (value: number) => Promise -}) { - const id = key(input.directory, input.sessionID) - const pending = inflight.get(id) - if (pending) return pending - - const value = version(id) - - const promise = input.task(value).finally(() => { - if (inflight.get(id) === promise) inflight.delete(id) - }) - - inflight.set(id, promise) - return promise -} - -export function setSessionPrefetch(input: { - directory: string - sessionID: string - limit: number - cursor?: string - complete: boolean - at?: number -}) { - cache.set(key(input.directory, input.sessionID), { - limit: input.limit, - cursor: input.cursor, - complete: input.complete, - at: input.at ?? Date.now(), - }) -} - -export function clearSessionPrefetch(directory: string, sessionIDs: Iterable) { - for (const sessionID of sessionIDs) { - if (!sessionID) continue - const id = key(directory, sessionID) - rev.set(id, version(id) + 1) - cache.delete(id) - inflight.delete(id) - } -} - -export function clearSessionPrefetchDirectory(directory: string) { - const prefix = `${directory}\n` - const keys = new Set([...cache.keys(), ...inflight.keys()]) - for (const id of keys) { - if (!id.startsWith(prefix)) continue - rev.set(id, version(id) + 1) - cache.delete(id) - inflight.delete(id) - } -} diff --git a/packages/app/src/context/global-sync/session-trim.test.ts b/packages/app/src/context/global-sync/session-trim.test.ts deleted file mode 100644 index da9ef3498f..0000000000 --- a/packages/app/src/context/global-sync/session-trim.test.ts +++ /dev/null @@ -1,59 +0,0 @@ -import { describe, expect, test } from "bun:test" -import type { PermissionRequest, Session } from "@kilocode/sdk/v2/client" -import { trimSessions } from "./session-trim" - -const session = (input: { id: string; parentID?: string; created: number; updated?: number; archived?: number }) => - ({ - id: input.id, - parentID: input.parentID, - time: { - created: input.created, - updated: input.updated, - archived: input.archived, - }, - }) as Session - -describe("trimSessions", () => { - test("keeps base roots and recent roots beyond the limit", () => { - const now = 1_000_000 - const list = [ - session({ id: "a", created: now - 100_000 }), - session({ id: "b", created: now - 90_000 }), - session({ id: "c", created: now - 80_000 }), - session({ id: "d", created: now - 70_000, updated: now - 1_000 }), - session({ id: "e", created: now - 60_000, archived: now - 10 }), - ] - - const result = trimSessions(list, { limit: 2, permission: {}, now }) - expect(result.map((x) => x.id)).toEqual(["a", "b", "c", "d"]) - }) - - test("keeps children when root is kept, permission exists, or child is recent", () => { - const now = 1_000_000 - const list = [ - session({ id: "root-1", created: now - 1000 }), - session({ id: "root-2", created: now - 2000 }), - session({ id: "z-root", created: now - 30_000_000 }), - session({ id: "child-kept-by-root", parentID: "root-1", created: now - 20_000_000 }), - session({ id: "child-kept-by-permission", parentID: "z-root", created: now - 20_000_000 }), - session({ id: "child-kept-by-recency", parentID: "z-root", created: now - 500 }), - session({ id: "child-trimmed", parentID: "z-root", created: now - 20_000_000 }), - ] - - const result = trimSessions(list, { - limit: 2, - permission: { - "child-kept-by-permission": [{ id: "perm-1" } as PermissionRequest], - }, - now, - }) - - expect(result.map((x) => x.id)).toEqual([ - "child-kept-by-permission", - "child-kept-by-recency", - "child-kept-by-root", - "root-1", - "root-2", - ]) - }) -}) diff --git a/packages/app/src/context/global-sync/session-trim.ts b/packages/app/src/context/global-sync/session-trim.ts deleted file mode 100644 index 8aa3140021..0000000000 --- a/packages/app/src/context/global-sync/session-trim.ts +++ /dev/null @@ -1,56 +0,0 @@ -import type { PermissionRequest, Session } from "@kilocode/sdk/v2/client" -import { cmp } from "./utils" -import { SESSION_RECENT_LIMIT, SESSION_RECENT_WINDOW } from "./types" - -export function sessionUpdatedAt(session: Session) { - return session.time.updated ?? session.time.created -} - -export function compareSessionRecent(a: Session, b: Session) { - const aUpdated = sessionUpdatedAt(a) - const bUpdated = sessionUpdatedAt(b) - if (aUpdated !== bUpdated) return bUpdated - aUpdated - return cmp(a.id, b.id) -} - -export function takeRecentSessions(sessions: Session[], limit: number, cutoff: number) { - if (limit <= 0) return [] as Session[] - const selected: Session[] = [] - const seen = new Set() - for (const session of sessions) { - if (!session?.id) continue - if (seen.has(session.id)) continue - seen.add(session.id) - if (sessionUpdatedAt(session) <= cutoff) continue - const index = selected.findIndex((x) => compareSessionRecent(session, x) < 0) - if (index === -1) selected.push(session) - if (index !== -1) selected.splice(index, 0, session) - if (selected.length > limit) selected.pop() - } - return selected -} - -export function trimSessions( - input: Session[], - options: { limit: number; permission: Record; now?: number }, -) { - const limit = Math.max(0, options.limit) - const cutoff = (options.now ?? Date.now()) - SESSION_RECENT_WINDOW - const all = input - .filter((s) => !!s?.id) - .filter((s) => !s.time?.archived) - .sort((a, b) => cmp(a.id, b.id)) - const roots = all.filter((s) => !s.parentID) - const children = all.filter((s) => !!s.parentID) - const base = roots.slice(0, limit) - const recent = takeRecentSessions(roots.slice(limit), SESSION_RECENT_LIMIT, cutoff) - const keepRoots = [...base, ...recent] - const keepRootIds = new Set(keepRoots.map((s) => s.id)) - const keepChildren = children.filter((s) => { - if (s.parentID && keepRootIds.has(s.parentID)) return true - const perms = options.permission[s.id] ?? [] - if (perms.length > 0) return true - return sessionUpdatedAt(s) > cutoff - }) - return [...keepRoots, ...keepChildren].sort((a, b) => cmp(a.id, b.id)) -} diff --git a/packages/app/src/context/global-sync/types.ts b/packages/app/src/context/global-sync/types.ts deleted file mode 100644 index a1a5184268..0000000000 --- a/packages/app/src/context/global-sync/types.ts +++ /dev/null @@ -1,135 +0,0 @@ -import type { - Agent, - Command, - Config, - LspStatus, - McpStatus, - Message, - Part, - Path, - PermissionRequest, - ProviderListResponse, - QuestionRequest, - Session, - SessionStatus, - SnapshotFileDiff, - Todo, - VcsInfo, -} from "@kilocode/sdk/v2/client" -import type { Accessor } from "solid-js" -import type { SetStoreFunction, Store } from "solid-js/store" - -export type ProjectMeta = { - name?: string - icon?: { - override?: string - color?: string - } - commands?: { - start?: string - } -} - -export type State = { - status: "loading" | "partial" | "complete" - agent: Agent[] - command: Command[] - project: string - projectMeta: ProjectMeta | undefined - icon: string | undefined - provider_ready: boolean - provider: ProviderListResponse - config: Config - path: Path - session: Session[] - sessionTotal: number - session_status: { - [sessionID: string]: SessionStatus - } - session_diff: { - [sessionID: string]: SnapshotFileDiff[] - } - todo: { - [sessionID: string]: Todo[] - } - permission: { - [sessionID: string]: PermissionRequest[] - } - question: { - [sessionID: string]: QuestionRequest[] - } - mcp_ready: boolean - mcp: { - [name: string]: McpStatus - } - lsp_ready: boolean - lsp: LspStatus[] - vcs: VcsInfo | undefined - limit: number - message: { - [sessionID: string]: Message[] - } - part: { - [messageID: string]: Part[] - } -} - -export type VcsCache = { - store: Store<{ value: VcsInfo | undefined }> - setStore: SetStoreFunction<{ value: VcsInfo | undefined }> - ready: Accessor -} - -export type MetaCache = { - store: Store<{ value: ProjectMeta | undefined }> - setStore: SetStoreFunction<{ value: ProjectMeta | undefined }> - ready: Accessor -} - -export type IconCache = { - store: Store<{ value: string | undefined }> - setStore: SetStoreFunction<{ value: string | undefined }> - ready: Accessor -} - -export type ChildOptions = { - bootstrap?: boolean -} - -export type DirState = { - lastAccessAt: number -} - -export type EvictPlan = { - stores: string[] - state: Map - pins: Set - max: number - ttl: number - now: number -} - -export type DisposeCheck = { - directory: string - hasStore: boolean - pinned: boolean - booting: boolean - loadingSessions: boolean -} - -export type RootLoadArgs = { - directory: string - limit: number - list: (query: { directory: string; roots: true; limit?: number }) => Promise<{ data?: Session[] }> -} - -export type RootLoadResult = { - data?: Session[] - limit: number - limited: boolean -} - -export const MAX_DIR_STORES = 30 -export const DIR_IDLE_TTL_MS = 20 * 60 * 1000 -export const SESSION_RECENT_WINDOW = 4 * 60 * 60 * 1000 -export const SESSION_RECENT_LIMIT = 50 diff --git a/packages/app/src/context/global-sync/utils.test.ts b/packages/app/src/context/global-sync/utils.test.ts deleted file mode 100644 index 24f1547c27..0000000000 --- a/packages/app/src/context/global-sync/utils.test.ts +++ /dev/null @@ -1,35 +0,0 @@ -import { describe, expect, test } from "bun:test" -import type { Agent } from "@kilocode/sdk/v2/client" -import { normalizeAgentList } from "./utils" - -const agent = (name = "build") => - ({ - name, - mode: "primary", - permission: {}, - options: {}, - }) as Agent - -describe("normalizeAgentList", () => { - test("keeps array payloads", () => { - expect(normalizeAgentList([agent("build"), agent("docs")])).toEqual([agent("build"), agent("docs")]) - }) - - test("wraps a single agent payload", () => { - expect(normalizeAgentList(agent("docs"))).toEqual([agent("docs")]) - }) - - test("extracts agents from keyed objects", () => { - expect( - normalizeAgentList({ - build: agent("build"), - docs: agent("docs"), - }), - ).toEqual([agent("build"), agent("docs")]) - }) - - test("drops invalid payloads", () => { - expect(normalizeAgentList({ name: "AbortError" })).toEqual([]) - expect(normalizeAgentList([{ name: "build" }, agent("docs")])).toEqual([agent("docs")]) - }) -}) diff --git a/packages/app/src/context/global-sync/utils.ts b/packages/app/src/context/global-sync/utils.ts deleted file mode 100644 index e6fd6f64ba..0000000000 --- a/packages/app/src/context/global-sync/utils.ts +++ /dev/null @@ -1,39 +0,0 @@ -import type { Agent, Project, ProviderListResponse } from "@kilocode/sdk/v2/client" - -export const cmp = (a: string, b: string) => (a < b ? -1 : a > b ? 1 : 0) - -function isAgent(input: unknown): input is Agent { - if (!input || typeof input !== "object") return false - const item = input as { name?: unknown; mode?: unknown } - if (typeof item.name !== "string") return false - return item.mode === "subagent" || item.mode === "primary" || item.mode === "all" -} - -export function normalizeAgentList(input: unknown): Agent[] { - if (Array.isArray(input)) return input.filter(isAgent) - if (isAgent(input)) return [input] - if (!input || typeof input !== "object") return [] - return Object.values(input).filter(isAgent) -} - -export function normalizeProviderList(input: ProviderListResponse): ProviderListResponse { - return { - ...input, - all: input.all.map((provider) => ({ - ...provider, - models: Object.fromEntries(Object.entries(provider.models).filter(([, info]) => info.status !== "deprecated")), - })), - } -} - -export function sanitizeProject(project: Project) { - if (!project.icon?.url && !project.icon?.override) return project - return { - ...project, - icon: { - ...project.icon, - url: undefined, - override: undefined, - }, - } -} diff --git a/packages/app/src/context/highlights.tsx b/packages/app/src/context/highlights.tsx deleted file mode 100644 index 11ee613e64..0000000000 --- a/packages/app/src/context/highlights.tsx +++ /dev/null @@ -1,233 +0,0 @@ -import { createEffect, onCleanup } from "solid-js" -import { createStore } from "solid-js/store" -import { createSimpleContext } from "@opencode-ai/ui/context" -import { useDialog } from "@opencode-ai/ui/context/dialog" -import { usePlatform } from "@/context/platform" -import { useSettings } from "@/context/settings" -import { persisted } from "@/utils/persist" -import { DialogReleaseNotes, type Highlight } from "@/components/dialog-release-notes" - -const CHANGELOG_URL = "https://kilo.ai/changelog.json" - -type Store = { - version?: string -} - -type ParsedRelease = { - tag?: string - highlights: Highlight[] -} - -function isRecord(value: unknown): value is Record { - return typeof value === "object" && value !== null && !Array.isArray(value) -} - -function getText(value: unknown): string | undefined { - if (typeof value === "string") { - const text = value.trim() - return text.length > 0 ? text : undefined - } - - if (typeof value === "number") return String(value) - return -} - -function normalizeVersion(value: string | undefined) { - const text = value?.trim() - if (!text) return - return text.startsWith("v") || text.startsWith("V") ? text.slice(1) : text -} - -function parseMedia(value: unknown, alt: string): Highlight["media"] | undefined { - if (!isRecord(value)) return - const type = getText(value.type)?.toLowerCase() - const src = getText(value.src) ?? getText(value.url) - if (!src) return - if (type !== "image" && type !== "video") return - - return { type, src, alt } -} - -function parseHighlight(value: unknown): Highlight | undefined { - if (!isRecord(value)) return - - const title = getText(value.title) - if (!title) return - - const description = getText(value.description) ?? getText(value.shortDescription) - if (!description) return - - const media = parseMedia(value.media, title) - return { title, description, media } -} - -function parseRelease(value: unknown): ParsedRelease | undefined { - if (!isRecord(value)) return - const tag = getText(value.tag) ?? getText(value.tag_name) ?? getText(value.name) - - if (!Array.isArray(value.highlights)) { - return { tag, highlights: [] } - } - - const highlights = value.highlights.flatMap((group) => { - if (!isRecord(group)) return [] - - const source = getText(group.source) - if (!source) return [] - if (!source.toLowerCase().includes("desktop")) return [] - - if (Array.isArray(group.items)) { - return group.items.map((item) => parseHighlight(item)).filter((item): item is Highlight => item !== undefined) - } - - const item = parseHighlight(group) - if (!item) return [] - return [item] - }) - - return { tag, highlights } -} - -function parseChangelog(value: unknown): ParsedRelease[] | undefined { - if (Array.isArray(value)) { - return value.map(parseRelease).filter((release): release is ParsedRelease => release !== undefined) - } - - if (!isRecord(value)) return - if (!Array.isArray(value.releases)) return - - return value.releases.map(parseRelease).filter((release): release is ParsedRelease => release !== undefined) -} - -function sliceHighlights(input: { releases: ParsedRelease[]; current?: string; previous?: string }) { - const current = normalizeVersion(input.current) - const previous = normalizeVersion(input.previous) - const releases = input.releases - - const start = (() => { - if (!current) return 0 - const index = releases.findIndex((release) => normalizeVersion(release.tag) === current) - return index === -1 ? 0 : index - })() - - const end = (() => { - if (!previous) return releases.length - const index = releases.findIndex((release, i) => i >= start && normalizeVersion(release.tag) === previous) - return index === -1 ? releases.length : index - })() - - const highlights = releases.slice(start, end).flatMap((release) => release.highlights) - const seen = new Set() - const unique = highlights.filter((highlight) => { - const key = dedupeKey(highlight) - if (seen.has(key)) return false - seen.add(key) - return true - }) - return unique.slice(0, 5) -} - -function dedupeKey(highlight: Highlight) { - return [highlight.title, highlight.description, highlight.media?.type ?? "", highlight.media?.src ?? ""].join("\n") -} - -function loadReleaseHighlights(value: unknown, current?: string, previous?: string) { - const releases = parseChangelog(value) - if (!releases?.length) return [] - return sliceHighlights({ releases, current, previous }) -} - -export const { use: useHighlights, provider: HighlightsProvider } = createSimpleContext({ - name: "Highlights", - gate: false, - init: () => { - const platform = usePlatform() - const dialog = useDialog() - const settings = useSettings() - const [store, setStore, _, ready] = persisted("highlights.v1", createStore({ version: undefined })) - - const [range, setRange] = createStore({ - from: undefined as string | undefined, - to: undefined as string | undefined, - }) - const state = { started: false } - let timer: ReturnType | undefined - - const clearTimer = () => { - if (timer === undefined) return - clearTimeout(timer) - timer = undefined - } - - const markSeen = () => { - if (!platform.version) return - setStore("version", platform.version) - } - - const start = (previous: string) => { - if (!settings.general.releaseNotes()) { - markSeen() - return - } - - const fetcher = platform.fetch ?? fetch - const controller = new AbortController() - onCleanup(() => { - controller.abort() - clearTimer() - }) - - fetcher(CHANGELOG_URL, { - signal: controller.signal, - headers: { Accept: "application/json" }, - }) - .then((response) => (response.ok ? (response.json() as Promise) : undefined)) - .then((json) => { - if (!json) return - const highlights = loadReleaseHighlights(json, platform.version, previous) - if (controller.signal.aborted) return - - if (highlights.length === 0) { - markSeen() - return - } - - timer = setTimeout(() => { - timer = undefined - markSeen() - dialog.show(() => ) - }, 500) - }) - .catch(() => undefined) - } - - createEffect(() => { - if (state.started) return - if (!ready()) return - if (!settings.ready()) return - if (!platform.version) return - state.started = true - - const previous = store.version - if (!previous) { - setStore("version", platform.version) - return - } - - if (previous === platform.version) return - - setRange({ from: previous, to: platform.version }) - start(previous) - }) - - return { - ready, - from: () => range.from, - to: () => range.to, - get last() { - return store.version - }, - markSeen, - } - }, -}) diff --git a/packages/app/src/context/language.tsx b/packages/app/src/context/language.tsx deleted file mode 100644 index f4cf0877d2..0000000000 --- a/packages/app/src/context/language.tsx +++ /dev/null @@ -1,247 +0,0 @@ -import * as i18n from "@solid-primitives/i18n" -import { createEffect, createMemo, createResource } from "solid-js" -import { createStore } from "solid-js/store" -import { createSimpleContext } from "@opencode-ai/ui/context" -import { Persist, persisted } from "@/utils/persist" -import { dict as en } from "@/i18n/en" -import { dict as uiEn } from "@opencode-ai/ui/i18n/en" - -// kilocode_change start -import { dict as kiloEn } from "@kilocode/kilo-i18n/en" -// kilocode_change end - -export type Locale = - | "en" - | "zh" - | "zht" - | "ko" - | "de" - | "es" - | "fr" - | "da" - | "ja" - | "pl" - | "ru" - | "ar" - | "no" - | "br" - | "th" - | "bs" - | "nl" - | "tr" - -type RawDictionary = typeof en & typeof uiEn & typeof kiloEn // kilocode_change -type Dictionary = i18n.Flatten -type Source = { dict: Record } - -function cookie(locale: Locale) { - return `oc_locale=${encodeURIComponent(locale)}; Path=/; Max-Age=31536000; SameSite=Lax` -} - -const LOCALES: readonly Locale[] = [ - "en", - "zh", - "zht", - "ko", - "de", - "es", - "fr", - "da", - "ja", - "pl", - "ru", - "bs", - "nl", - "ar", - "no", - "br", - "th", - "tr", -] - -const INTL: Record = { - en: "en", - zh: "zh-Hans", - zht: "zh-Hant", - ko: "ko", - de: "de", - es: "es", - fr: "fr", - da: "da", - ja: "ja", - pl: "pl", - ru: "ru", - ar: "ar", - no: "nb-NO", - br: "pt-BR", - th: "th", - bs: "bs", - nl: "nl", - - tr: "tr", -} - -const LABEL_KEY: Record = { - en: "language.en", - zh: "language.zh", - zht: "language.zht", - ko: "language.ko", - de: "language.de", - es: "language.es", - fr: "language.fr", - da: "language.da", - ja: "language.ja", - pl: "language.pl", - ru: "language.ru", - ar: "language.ar", - no: "language.no", - br: "language.br", - th: "language.th", - bs: "language.bs", - nl: "language.nl", - tr: "language.tr", -} - -const base = i18n.flatten({ ...en, ...uiEn, ...kiloEn }) as Dictionary // kilocode_change -const dicts = new Map([["en", base]]) - -const merge = (app: Promise, ui: Promise) => - Promise.all([app, ui]).then(([a, b]) => ({ ...base, ...i18n.flatten({ ...a.dict, ...b.dict }) }) as Dictionary) - -const loaders: Record, () => Promise> = { - zh: () => merge(import("@/i18n/zh"), import("@opencode-ai/ui/i18n/zh")), - zht: () => merge(import("@/i18n/zht"), import("@opencode-ai/ui/i18n/zht")), - ko: () => merge(import("@/i18n/ko"), import("@opencode-ai/ui/i18n/ko")), - de: () => merge(import("@/i18n/de"), import("@opencode-ai/ui/i18n/de")), - es: () => merge(import("@/i18n/es"), import("@opencode-ai/ui/i18n/es")), - fr: () => merge(import("@/i18n/fr"), import("@opencode-ai/ui/i18n/fr")), - da: () => merge(import("@/i18n/da"), import("@opencode-ai/ui/i18n/da")), - ja: () => merge(import("@/i18n/ja"), import("@opencode-ai/ui/i18n/ja")), - pl: () => merge(import("@/i18n/pl"), import("@opencode-ai/ui/i18n/pl")), - ru: () => merge(import("@/i18n/ru"), import("@opencode-ai/ui/i18n/ru")), - ar: () => merge(import("@/i18n/ar"), import("@opencode-ai/ui/i18n/ar")), - no: () => merge(import("@/i18n/no"), import("@opencode-ai/ui/i18n/no")), - br: () => merge(import("@/i18n/br"), import("@opencode-ai/ui/i18n/br")), - th: () => merge(import("@/i18n/th"), import("@opencode-ai/ui/i18n/th")), - bs: () => merge(import("@/i18n/bs"), import("@opencode-ai/ui/i18n/bs")), - nl: () => merge(import("@/i18n/nl"), import("@opencode-ai/ui/i18n/nl")), - tr: () => merge(import("@/i18n/tr"), import("@opencode-ai/ui/i18n/tr")), -} - -function loadDict(locale: Locale) { - const hit = dicts.get(locale) - if (hit) return Promise.resolve(hit) - if (locale === "en") return Promise.resolve(base) - const load = loaders[locale] - return load().then((next: Dictionary) => { - dicts.set(locale, next) - return next - }) -} - -export function loadLocaleDict(locale: Locale) { - return loadDict(locale).then(() => undefined) -} - -const localeMatchers: Array<{ locale: Locale; match: (language: string) => boolean }> = [ - { locale: "en", match: (language) => language.startsWith("en") }, - { locale: "zht", match: (language) => language.startsWith("zh") && language.includes("hant") }, - { locale: "zh", match: (language) => language.startsWith("zh") }, - { locale: "ko", match: (language) => language.startsWith("ko") }, - { locale: "de", match: (language) => language.startsWith("de") }, - { locale: "es", match: (language) => language.startsWith("es") }, - { locale: "fr", match: (language) => language.startsWith("fr") }, - { locale: "da", match: (language) => language.startsWith("da") }, - { locale: "ja", match: (language) => language.startsWith("ja") }, - { locale: "pl", match: (language) => language.startsWith("pl") }, - { locale: "ru", match: (language) => language.startsWith("ru") }, - { locale: "ar", match: (language) => language.startsWith("ar") }, - { - locale: "no", - match: (language) => language.startsWith("no") || language.startsWith("nb") || language.startsWith("nn"), - }, - { locale: "br", match: (language) => language.startsWith("pt") }, - { locale: "th", match: (language) => language.startsWith("th") }, - { locale: "bs", match: (language) => language.startsWith("bs") }, - { locale: "nl", match: (language) => language.startsWith("nl") }, - { locale: "tr", match: (language) => language.startsWith("tr") }, -] - -function detectLocale(): Locale { - if (typeof navigator !== "object") return "en" - - const languages = navigator.languages?.length ? navigator.languages : [navigator.language] - for (const language of languages) { - if (!language) continue - const normalized = language.toLowerCase() - const match = localeMatchers.find((entry) => entry.match(normalized)) - if (match) return match.locale - } - - return "en" -} - -export function normalizeLocale(value: string): Locale { - return LOCALES.includes(value as Locale) ? (value as Locale) : "en" -} - -function readStoredLocale() { - if (typeof localStorage !== "object") return - try { - const raw = localStorage.getItem("opencode.global.dat:language") - if (!raw) return - const next = JSON.parse(raw) as { locale?: string } - if (typeof next?.locale !== "string") return - return normalizeLocale(next.locale) - } catch { - return - } -} - -const warm = readStoredLocale() ?? detectLocale() -if (warm !== "en") void loadDict(warm) - -export const { use: useLanguage, provider: LanguageProvider } = createSimpleContext({ - name: "Language", - init: (props: { locale?: Locale }) => { - const initial = props.locale ?? readStoredLocale() ?? detectLocale() - const [store, setStore, _, ready] = persisted( - Persist.global("language", ["language.v1"]), - createStore({ - locale: initial, - }), - ) - - const locale = createMemo(() => normalizeLocale(store.locale)) - const intl = createMemo(() => INTL[locale()]) - - const [dict] = createResource(locale, loadDict, { - initialValue: dicts.get(initial) ?? base, - }) - - const t = i18n.translator(() => dict() ?? base, i18n.resolveTemplate) as ( - key: keyof Dictionary, - params?: Record, - ) => string - - const label = (value: Locale) => t(LABEL_KEY[value]) - - createEffect(() => { - if (typeof document !== "object") return - document.documentElement.lang = locale() - document.cookie = cookie(locale()) - }) - - return { - ready, - locale, - intl, - locales: LOCALES, - label, - t, - setLocale(next: Locale) { - setStore("locale", normalizeLocale(next)) - }, - } - }, -}) diff --git a/packages/app/src/context/layout-scroll.test.ts b/packages/app/src/context/layout-scroll.test.ts deleted file mode 100644 index 483be150f6..0000000000 --- a/packages/app/src/context/layout-scroll.test.ts +++ /dev/null @@ -1,64 +0,0 @@ -import { describe, expect, test, vi } from "bun:test" -import { createScrollPersistence } from "./layout-scroll" - -describe("createScrollPersistence", () => { - test("debounces persisted scroll writes", () => { - vi.useFakeTimers() - try { - const snapshot = { - session: { - review: { x: 0, y: 0 }, - }, - } as Record> - const writes: Array> = [] - const scroll = createScrollPersistence({ - debounceMs: 10, - getSnapshot: (sessionKey) => snapshot[sessionKey], - onFlush: (sessionKey, next) => { - snapshot[sessionKey] = next - writes.push(next) - }, - }) - - for (const i of Array.from({ length: 30 }, (_, n) => n + 1)) { - scroll.setScroll("session", "review", { x: 0, y: i }) - } - - vi.advanceTimersByTime(9) - expect(writes).toHaveLength(0) - - vi.advanceTimersByTime(1) - - expect(writes).toHaveLength(1) - expect(writes[0]?.review).toEqual({ x: 0, y: 30 }) - - scroll.setScroll("session", "review", { x: 0, y: 30 }) - vi.advanceTimersByTime(20) - - expect(writes).toHaveLength(1) - scroll.dispose() - } finally { - vi.useRealTimers() - } - }) - - test("reseeds empty cache after persisted snapshot loads", () => { - const snapshot = { - session: {}, - } as Record> - - const scroll = createScrollPersistence({ - getSnapshot: (sessionKey) => snapshot[sessionKey], - onFlush: () => {}, - }) - - expect(scroll.scroll("session", "review")).toBeUndefined() - - snapshot.session = { - review: { x: 12, y: 34 }, - } - - expect(scroll.scroll("session", "review")).toEqual({ x: 12, y: 34 }) - scroll.dispose() - }) -}) diff --git a/packages/app/src/context/layout-scroll.ts b/packages/app/src/context/layout-scroll.ts deleted file mode 100644 index ef66eccd90..0000000000 --- a/packages/app/src/context/layout-scroll.ts +++ /dev/null @@ -1,126 +0,0 @@ -import { createStore, produce } from "solid-js/store" - -export type SessionScroll = { - x: number - y: number -} - -type ScrollMap = Record - -type Options = { - debounceMs?: number - getSnapshot: (sessionKey: string) => ScrollMap | undefined - onFlush: (sessionKey: string, scroll: ScrollMap) => void -} - -export function createScrollPersistence(opts: Options) { - const wait = opts.debounceMs ?? 200 - const [cache, setCache] = createStore>({}) - const dirty = new Set() - const timers = new Map>() - - function clone(input?: ScrollMap) { - const out: ScrollMap = {} - if (!input) return out - - for (const key of Object.keys(input)) { - const pos = input[key] - if (!pos) continue - out[key] = { x: pos.x, y: pos.y } - } - - return out - } - - function seed(sessionKey: string) { - const next = clone(opts.getSnapshot(sessionKey)) - const current = cache[sessionKey] - if (!current) { - setCache(sessionKey, next) - return - } - - if (Object.keys(current).length > 0) return - if (Object.keys(next).length === 0) return - setCache(sessionKey, next) - } - - function scroll(sessionKey: string, tab: string) { - seed(sessionKey) - return cache[sessionKey]?.[tab] ?? opts.getSnapshot(sessionKey)?.[tab] - } - - function schedule(sessionKey: string) { - const prev = timers.get(sessionKey) - if (prev) clearTimeout(prev) - timers.set( - sessionKey, - setTimeout(() => flush(sessionKey), wait), - ) - } - - function setScroll(sessionKey: string, tab: string, pos: SessionScroll) { - seed(sessionKey) - - const prev = cache[sessionKey]?.[tab] - if (prev?.x === pos.x && prev?.y === pos.y) return - - setCache(sessionKey, tab, { x: pos.x, y: pos.y }) - dirty.add(sessionKey) - schedule(sessionKey) - } - - function flush(sessionKey: string) { - const timer = timers.get(sessionKey) - if (timer) clearTimeout(timer) - timers.delete(sessionKey) - - if (!dirty.has(sessionKey)) return - dirty.delete(sessionKey) - - opts.onFlush(sessionKey, clone(cache[sessionKey])) - } - - function flushAll() { - const keys = Array.from(dirty) - if (keys.length === 0) return - - for (const key of keys) { - flush(key) - } - } - - function drop(keys: string[]) { - if (keys.length === 0) return - - for (const key of keys) { - const timer = timers.get(key) - if (timer) clearTimeout(timer) - timers.delete(key) - dirty.delete(key) - } - - setCache( - produce((draft) => { - for (const key of keys) { - delete draft[key] - } - }), - ) - } - - function dispose() { - drop(Array.from(timers.keys())) - } - - return { - cache, - drop, - flush, - flushAll, - scroll, - seed, - setScroll, - dispose, - } -} diff --git a/packages/app/src/context/layout.test.ts b/packages/app/src/context/layout.test.ts deleted file mode 100644 index 582d5edbd2..0000000000 --- a/packages/app/src/context/layout.test.ts +++ /dev/null @@ -1,69 +0,0 @@ -import { describe, expect, test } from "bun:test" -import { createRoot, createSignal } from "solid-js" -import { createSessionKeyReader, ensureSessionKey, pruneSessionKeys } from "./layout" - -describe("layout session-key helpers", () => { - test("couples touch and scroll seed in order", () => { - const calls: string[] = [] - const result = ensureSessionKey( - "dir/a", - (key) => calls.push(`touch:${key}`), - (key) => calls.push(`seed:${key}`), - ) - - expect(result).toBe("dir/a") - expect(calls).toEqual(["touch:dir/a", "seed:dir/a"]) - }) - - test("reads dynamic accessor keys lazily", () => { - const seen: string[] = [] - - createRoot((dispose) => { - const [key, setKey] = createSignal("dir/one") - const read = createSessionKeyReader(key, (value) => seen.push(value)) - - expect(read()).toBe("dir/one") - setKey("dir/two") - expect(read()).toBe("dir/two") - - dispose() - }) - - expect(seen).toEqual(["dir/one", "dir/two"]) - }) -}) - -describe("pruneSessionKeys", () => { - test("keeps active key and drops lowest-used keys", () => { - const drop = pruneSessionKeys({ - keep: "k4", - max: 3, - used: new Map([ - ["k1", 1], - ["k2", 2], - ["k3", 3], - ["k4", 4], - ]), - view: ["k1", "k2", "k4"], - tabs: ["k1", "k3", "k4"], - }) - - expect(drop).toEqual(["k1"]) - expect(drop.includes("k4")).toBe(false) - }) - - test("does not prune without keep key", () => { - const drop = pruneSessionKeys({ - keep: undefined, - max: 1, - used: new Map([ - ["k1", 1], - ["k2", 2], - ]), - view: ["k1"], - tabs: ["k2"], - }) - - expect(drop).toEqual([]) - }) -}) diff --git a/packages/app/src/context/model-variant.test.ts b/packages/app/src/context/model-variant.test.ts deleted file mode 100644 index 583bc5c3dc..0000000000 --- a/packages/app/src/context/model-variant.test.ts +++ /dev/null @@ -1,86 +0,0 @@ -import { describe, expect, test } from "bun:test" -import { cycleModelVariant, getConfiguredAgentVariant, resolveModelVariant } from "./model-variant" - -describe("model variant", () => { - test("resolves configured agent variant when model matches", () => { - const value = getConfiguredAgentVariant({ - agent: { - model: { providerID: "openai", modelID: "gpt-5.2" }, - variant: "xhigh", - }, - model: { - providerID: "openai", - modelID: "gpt-5.2", - variants: { low: {}, high: {}, xhigh: {} }, - }, - }) - - expect(value).toBe("xhigh") - }) - - test("ignores configured variant when model does not match", () => { - const value = getConfiguredAgentVariant({ - agent: { - model: { providerID: "openai", modelID: "gpt-5.2" }, - variant: "xhigh", - }, - model: { - providerID: "anthropic", - modelID: "claude-sonnet-4", - variants: { low: {}, high: {}, xhigh: {} }, - }, - }) - - expect(value).toBeUndefined() - }) - - test("prefers selected variant over configured variant", () => { - const value = resolveModelVariant({ - variants: ["low", "high", "xhigh"], - selected: "high", - configured: "xhigh", - }) - - expect(value).toBe("high") - }) - - test("lets an explicit default override the configured variant", () => { - const value = resolveModelVariant({ - variants: ["low", "high", "xhigh"], - selected: null, - configured: "xhigh", - }) - - expect(value).toBeUndefined() - }) - - test("cycles from configured variant to next", () => { - const value = cycleModelVariant({ - variants: ["low", "high", "xhigh"], - selected: undefined, - configured: "high", - }) - - expect(value).toBe("xhigh") - }) - - test("wraps from configured last variant to first", () => { - const value = cycleModelVariant({ - variants: ["low", "high", "xhigh"], - selected: undefined, - configured: "xhigh", - }) - - expect(value).toBe("low") - }) - - test("cycles from an explicit default to the first variant", () => { - const value = cycleModelVariant({ - variants: ["low", "high", "xhigh"], - selected: null, - configured: "xhigh", - }) - - expect(value).toBe("low") - }) -}) diff --git a/packages/app/src/context/model-variant.ts b/packages/app/src/context/model-variant.ts deleted file mode 100644 index 525acbba32..0000000000 --- a/packages/app/src/context/model-variant.ts +++ /dev/null @@ -1,52 +0,0 @@ -type AgentModel = { - providerID: string - modelID: string -} - -type Agent = { - model?: AgentModel - variant?: string -} - -type Model = AgentModel & { - variants?: Record -} - -type VariantInput = { - variants: string[] - selected: string | null | undefined - configured: string | undefined -} - -export function getConfiguredAgentVariant(input: { agent: Agent | undefined; model: Model | undefined }) { - if (!input.agent?.variant) return undefined - if (!input.agent.model) return undefined - if (!input.model?.variants) return undefined - if (input.agent.model.providerID !== input.model.providerID) return undefined - if (input.agent.model.modelID !== input.model.modelID) return undefined - if (!(input.agent.variant in input.model.variants)) return undefined - return input.agent.variant -} - -export function resolveModelVariant(input: VariantInput) { - if (input.selected === null) return undefined - if (input.selected && input.variants.includes(input.selected)) return input.selected - if (input.configured && input.variants.includes(input.configured)) return input.configured - return undefined -} - -export function cycleModelVariant(input: VariantInput) { - if (input.variants.length === 0) return undefined - if (input.selected === null) return input.variants[0] - if (input.selected && input.variants.includes(input.selected)) { - const index = input.variants.indexOf(input.selected) - if (index === input.variants.length - 1) return undefined - return input.variants[index + 1] - } - if (input.configured && input.variants.includes(input.configured)) { - const index = input.variants.indexOf(input.configured) - if (index === input.variants.length - 1) return input.variants[0] - return input.variants[index + 1] - } - return input.variants[0] -} diff --git a/packages/app/src/context/models.tsx b/packages/app/src/context/models.tsx deleted file mode 100644 index 12ec8371ad..0000000000 --- a/packages/app/src/context/models.tsx +++ /dev/null @@ -1,163 +0,0 @@ -import { createMemo } from "solid-js" -import { createStore } from "solid-js/store" -import { DateTime } from "luxon" -import { filter, firstBy, flat, groupBy, mapValues, pipe, uniqueBy, values } from "remeda" -import { createSimpleContext } from "@opencode-ai/ui/context" -import { useProviders } from "@/hooks/use-providers" -import { Persist, persisted } from "@/utils/persist" - -export type ModelKey = { providerID: string; modelID: string } - -type Visibility = "show" | "hide" -type User = ModelKey & { visibility: Visibility; favorite?: boolean } -type Store = { - user: User[] - recent: ModelKey[] - variant?: Record -} - -const RECENT_LIMIT = 5 - -function modelKey(model: ModelKey) { - return `${model.providerID}:${model.modelID}` -} - -export const { use: useModels, provider: ModelsProvider } = createSimpleContext({ - name: "Models", - init: () => { - const providers = useProviders() - - const [store, setStore, _, ready] = persisted( - Persist.global("model", ["model.v1"]), - createStore({ - user: [], - recent: [], - variant: {}, - }), - ) - - const available = createMemo(() => - providers.connected().flatMap((p) => - Object.values(p.models).map((m) => ({ - ...m, - provider: p, - })), - ), - ) - - const release = createMemo( - () => - new Map( - available().map((model) => { - const parsed = DateTime.fromISO(model.release_date) - return [modelKey({ providerID: model.provider.id, modelID: model.id }), parsed] as const - }), - ), - ) - - const latest = createMemo(() => - pipe( - available(), - filter( - (x) => - Math.abs( - (release().get(modelKey({ providerID: x.provider.id, modelID: x.id })) ?? DateTime.invalid("invalid")) - .diffNow() - .as("months"), - ) < 6, - ), - groupBy((x) => x.provider.id), - mapValues((models) => - pipe( - models, - groupBy((x) => x.family), - values(), - (groups) => - groups.flatMap((g) => { - const first = firstBy(g, [(x) => x.release_date, "desc"]) - return first ? [{ modelID: first.id, providerID: first.provider.id }] : [] - }), - ), - ), - values(), - flat(), - ), - ) - - const latestSet = createMemo(() => new Set(latest().map((x) => modelKey(x)))) - - const visibility = createMemo(() => { - const map = new Map() - for (const item of store.user) map.set(`${item.providerID}:${item.modelID}`, item.visibility) - return map - }) - - const list = createMemo(() => - available().map((m) => ({ - ...m, - name: m.name.replace("(latest)", "").trim(), - latest: m.name.includes("(latest)"), - })), - ) - - const find = (key: ModelKey) => list().find((m) => m.id === key.modelID && m.provider.id === key.providerID) - - function update(model: ModelKey, state: Visibility) { - const index = store.user.findIndex((x) => x.modelID === model.modelID && x.providerID === model.providerID) - if (index >= 0) { - setStore("user", index, (current) => ({ ...current, visibility: state })) - return - } - setStore("user", store.user.length, { ...model, visibility: state }) - } - - const visible = (model: ModelKey) => { - const key = modelKey(model) - const state = visibility().get(key) - if (state === "hide") return false - if (state === "show") return true - if (latestSet().has(key)) return true - const date = release().get(key) - if (!date?.isValid) return true - return false - } - - const setVisibility = (model: ModelKey, state: boolean) => { - update(model, state ? "show" : "hide") - } - - const push = (model: ModelKey) => { - const uniq = uniqueBy([model, ...store.recent], (x) => `${x.providerID}:${x.modelID}`) - if (uniq.length > RECENT_LIMIT) uniq.pop() - setStore("recent", uniq) - } - - const variantKey = (model: ModelKey) => `${model.providerID}/${model.modelID}` - const getVariant = (model: ModelKey) => store.variant?.[variantKey(model)] - - const setVariant = (model: ModelKey, value: string | undefined) => { - const key = variantKey(model) - if (!store.variant) { - setStore("variant", { [key]: value }) - return - } - setStore("variant", key, value) - } - - return { - ready, - list, - find, - visible, - setVisibility, - recent: { - list: createMemo(() => store.recent), - push, - }, - variant: { - get: getVariant, - set: setVariant, - }, - } - }, -}) diff --git a/packages/app/src/context/permission.tsx b/packages/app/src/context/permission.tsx deleted file mode 100644 index 6f890da7c9..0000000000 --- a/packages/app/src/context/permission.tsx +++ /dev/null @@ -1,277 +0,0 @@ -import { createEffect, createMemo, onCleanup } from "solid-js" -import { createStore, produce } from "solid-js/store" -import { createSimpleContext } from "@opencode-ai/ui/context" -import type { PermissionRequest } from "@kilocode/sdk/v2/client" -import { Persist, persisted } from "@/utils/persist" -import { useGlobalSDK } from "@/context/global-sdk" -import { useGlobalSync } from "./global-sync" -import { useParams } from "@solidjs/router" -import { decode64 } from "@/utils/base64" -import { - acceptKey, - directoryAcceptKey, - isDirectoryAutoAccepting, - autoRespondsPermission, -} from "./permission-auto-respond" - -type PermissionRespondFn = (input: { - sessionID: string - permissionID: string - response: "once" | "always" | "reject" - directory?: string -}) => void - -function isNonAllowRule(rule: unknown) { - if (!rule) return false - if (typeof rule === "string") return rule !== "allow" - if (typeof rule !== "object") return false - if (Array.isArray(rule)) return false - - for (const action of Object.values(rule)) { - if (action !== "allow") return true - } - - return false -} - -function hasPermissionPromptRules(permission: unknown) { - if (!permission) return false - if (typeof permission === "string") return permission !== "allow" - if (typeof permission !== "object") return false - if (Array.isArray(permission)) return false - - const config = permission as Record - return Object.values(config).some(isNonAllowRule) -} - -export const { use: usePermission, provider: PermissionProvider } = createSimpleContext({ - name: "Permission", - init: () => { - const params = useParams() - const globalSDK = useGlobalSDK() - const globalSync = useGlobalSync() - - const permissionsEnabled = createMemo(() => { - const directory = decode64(params.dir) - if (!directory) return false - const [store] = globalSync.child(directory) - return hasPermissionPromptRules(store.config.permission) - }) - - const [store, setStore, _, ready] = persisted( - { - ...Persist.global("permission", ["permission.v3"]), - migrate(value) { - if (!value || typeof value !== "object" || Array.isArray(value)) return value - - const data = value as Record - if (data.autoAccept) return value - - return { - ...data, - autoAccept: - typeof data.autoAcceptEdits === "object" && data.autoAcceptEdits && !Array.isArray(data.autoAcceptEdits) - ? data.autoAcceptEdits - : {}, - } - }, - }, - createStore({ - autoAccept: {} as Record, - }), - ) - - // When config has permission: "allow", auto-enable directory-level auto-accept - createEffect(() => { - if (!ready()) return - const directory = decode64(params.dir) - if (!directory) return - const [childStore] = globalSync.child(directory) - const perm = childStore.config.permission - if (typeof perm === "string" && perm === "allow") { - const key = directoryAcceptKey(directory) - if (store.autoAccept[key] === undefined) { - setStore( - produce((draft) => { - draft.autoAccept[key] = true - }), - ) - } - } - }) - - const MAX_RESPONDED = 1000 - const RESPONDED_TTL_MS = 60 * 60 * 1000 - const responded = new Map() - const enableVersion = new Map() - - function pruneResponded(now: number) { - for (const [id, ts] of responded) { - if (now - ts < RESPONDED_TTL_MS) break - responded.delete(id) - } - - for (const id of responded.keys()) { - if (responded.size <= MAX_RESPONDED) break - responded.delete(id) - } - } - - const respond: PermissionRespondFn = (input) => { - globalSDK.client.permission.respond(input).catch(() => { - responded.delete(input.permissionID) - }) - } - - function respondOnce(permission: PermissionRequest, directory?: string) { - const now = Date.now() - const hit = responded.has(permission.id) - responded.delete(permission.id) - responded.set(permission.id, now) - pruneResponded(now) - if (hit) return - respond({ - sessionID: permission.sessionID, - permissionID: permission.id, - response: "once", - directory, - }) - } - - function isAutoAccepting(sessionID: string, directory?: string) { - const session = directory ? globalSync.child(directory, { bootstrap: false })[0].session : [] - return autoRespondsPermission(store.autoAccept, session, { sessionID }, directory) - } - - function isAutoAcceptingDirectory(directory: string) { - return isDirectoryAutoAccepting(store.autoAccept, directory) - } - - function shouldAutoRespond(permission: PermissionRequest, directory?: string) { - const session = directory ? globalSync.child(directory, { bootstrap: false })[0].session : [] - return autoRespondsPermission(store.autoAccept, session, permission, directory) - } - - function bumpEnableVersion(sessionID: string, directory?: string) { - const key = acceptKey(sessionID, directory) - const next = (enableVersion.get(key) ?? 0) + 1 - enableVersion.set(key, next) - return next - } - - const unsubscribe = globalSDK.event.listen((e) => { - const event = e.details - if (event?.type !== "permission.asked") return - - const perm = event.properties - if (!shouldAutoRespond(perm, e.name)) return - - respondOnce(perm, e.name) - }) - onCleanup(unsubscribe) - - function enableDirectory(directory: string) { - const key = directoryAcceptKey(directory) - setStore( - produce((draft) => { - draft.autoAccept[key] = true - }), - ) - - globalSDK.client.permission - .list({ directory }) - .then((x) => { - if (!isAutoAcceptingDirectory(directory)) return - for (const perm of x.data ?? []) { - if (!perm?.id) continue - if (!shouldAutoRespond(perm, directory)) continue - respondOnce(perm, directory) - } - }) - .catch(() => undefined) - } - - function disableDirectory(directory: string) { - const key = directoryAcceptKey(directory) - setStore( - produce((draft) => { - draft.autoAccept[key] = false - }), - ) - } - - function enable(sessionID: string, directory: string) { - const key = acceptKey(sessionID, directory) - const version = bumpEnableVersion(sessionID, directory) - setStore( - produce((draft) => { - draft.autoAccept[key] = true - delete draft.autoAccept[sessionID] - }), - ) - - globalSDK.client.permission - .list({ directory }) - .then((x) => { - if (enableVersion.get(key) !== version) return - if (!isAutoAccepting(sessionID, directory)) return - for (const perm of x.data ?? []) { - if (!perm?.id) continue - if (!shouldAutoRespond(perm, directory)) continue - respondOnce(perm, directory) - } - }) - .catch(() => undefined) - } - - function disable(sessionID: string, directory?: string) { - bumpEnableVersion(sessionID, directory) - const key = directory ? acceptKey(sessionID, directory) : sessionID - setStore( - produce((draft) => { - draft.autoAccept[key] = false - if (!directory) return - delete draft.autoAccept[sessionID] - }), - ) - } - - return { - ready, - respond, - autoResponds(permission: PermissionRequest, directory?: string) { - return shouldAutoRespond(permission, directory) - }, - isAutoAccepting, - isAutoAcceptingDirectory, - toggleAutoAccept(sessionID: string, directory: string) { - if (isAutoAccepting(sessionID, directory)) { - disable(sessionID, directory) - return - } - - enable(sessionID, directory) - }, - toggleAutoAcceptDirectory(directory: string) { - if (isAutoAcceptingDirectory(directory)) { - disableDirectory(directory) - return - } - enableDirectory(directory) - }, - enableAutoAccept(sessionID: string, directory: string) { - if (isAutoAccepting(sessionID, directory)) return - enable(sessionID, directory) - }, - disableAutoAccept(sessionID: string, directory?: string) { - disable(sessionID, directory) - }, - permissionsEnabled, - isPermissionAllowAll(directory: string) { - const [childStore] = globalSync.child(directory) - const perm = childStore.config.permission - return typeof perm === "string" && perm === "allow" - }, - } - }, -}) diff --git a/packages/app/src/context/platform.tsx b/packages/app/src/context/platform.tsx deleted file mode 100644 index fd89bf51ba..0000000000 --- a/packages/app/src/context/platform.tsx +++ /dev/null @@ -1,99 +0,0 @@ -import { createSimpleContext } from "@opencode-ai/ui/context" -import type { AsyncStorage, SyncStorage } from "@solid-primitives/storage" -import type { Accessor } from "solid-js" -import { ServerConnection } from "./server" - -type PickerPaths = string | string[] | null -type OpenDirectoryPickerOptions = { title?: string; multiple?: boolean } -type OpenFilePickerOptions = { title?: string; multiple?: boolean; accept?: string[]; extensions?: string[] } -type SaveFilePickerOptions = { title?: string; defaultPath?: string } -type UpdateInfo = { updateAvailable: boolean; version?: string } - -export type Platform = { - /** Platform discriminator */ - platform: "web" | "desktop" - - /** Desktop OS (Tauri only) */ - os?: "macos" | "windows" | "linux" - - /** App version */ - version?: string - - /** Open a URL in the default browser */ - openLink(url: string): void - - /** Open a local path in a local app (desktop only) */ - openPath?(path: string, app?: string): Promise - - /** Restart the app */ - restart(): Promise - - /** Navigate back in history */ - back(): void - - /** Navigate forward in history */ - forward(): void - - /** Send a system notification (optional deep link) */ - notify(title: string, description?: string, href?: string): Promise - - /** Open directory picker dialog (native on Tauri, server-backed on web) */ - openDirectoryPickerDialog?(opts?: OpenDirectoryPickerOptions): Promise - - /** Open native file picker dialog (Tauri only) */ - openFilePickerDialog?(opts?: OpenFilePickerOptions): Promise - - /** Save file picker dialog (Tauri only) */ - saveFilePickerDialog?(opts?: SaveFilePickerOptions): Promise - - /** Storage mechanism, defaults to localStorage */ - storage?: (name?: string) => SyncStorage | AsyncStorage - - /** Check for a downloadable desktop update */ - checkUpdate?(): Promise - - /** Install the downloaded update using the platform restart flow */ - updateAndRestart?(): Promise - - /** Fetch override */ - fetch?: typeof fetch - - /** Get the configured default server URL (platform-specific) */ - getDefaultServer?(): Promise - - /** Set the default server URL to use on app startup (platform-specific) */ - setDefaultServer?(url: ServerConnection.Key | null): Promise | void - - /** Get the configured WSL integration (desktop only) */ - getWslEnabled?(): Promise - - /** Set the configured WSL integration (desktop only) */ - setWslEnabled?(config: boolean): Promise | void - - /** Get the preferred display backend (desktop only) */ - getDisplayBackend?(): Promise | DisplayBackend | null - - /** Set the preferred display backend (desktop only) */ - setDisplayBackend?(backend: DisplayBackend): Promise - - /** Parse markdown to HTML using native parser (desktop only, returns unprocessed code blocks) */ - parseMarkdown?(markdown: string): Promise - - /** Webview zoom level (desktop only) */ - webviewZoom?: Accessor - - /** Check if an editor app exists (desktop only) */ - checkAppExists?(appName: string): Promise - - /** Read image from clipboard (desktop only) */ - readClipboardImage?(): Promise -} - -export type DisplayBackend = "auto" | "wayland" - -export const { use: usePlatform, provider: PlatformProvider } = createSimpleContext({ - name: "Platform", - init: (props: { value: Platform }) => { - return props.value - }, -}) diff --git a/packages/app/src/context/sdk.tsx b/packages/app/src/context/sdk.tsx deleted file mode 100644 index 2eaf4627c0..0000000000 --- a/packages/app/src/context/sdk.tsx +++ /dev/null @@ -1,49 +0,0 @@ -import type { Event } from "@kilocode/sdk/v2/client" -import { createSimpleContext } from "@opencode-ai/ui/context" -import { createGlobalEmitter } from "@solid-primitives/event-bus" -import { type Accessor, createEffect, createMemo, onCleanup } from "solid-js" -import { useGlobalSDK } from "./global-sdk" - -type SDKEventMap = { - [key in Event["type"]]: Extract -} - -export const { use: useSDK, provider: SDKProvider } = createSimpleContext({ - name: "SDK", - init: (props: { directory: Accessor }) => { - const globalSDK = useGlobalSDK() - - const directory = createMemo(props.directory) - const client = createMemo(() => - globalSDK.createClient({ - directory: directory(), - throwOnError: true, - }), - ) - - const emitter = createGlobalEmitter() - - createEffect(() => { - const unsub = globalSDK.event.on(directory(), (event) => { - emitter.emit(event.type, event) - }) - onCleanup(unsub) - }) - - return { - get directory() { - return directory() - }, - get client() { - return client() - }, - event: emitter, - get url() { - return globalSDK.url - }, - createClient(opts: Parameters[0]) { - return globalSDK.createClient(opts) - }, - } - }, -}) diff --git a/packages/app/src/context/server.tsx b/packages/app/src/context/server.tsx deleted file mode 100644 index 1204fba557..0000000000 --- a/packages/app/src/context/server.tsx +++ /dev/null @@ -1,303 +0,0 @@ -import { createSimpleContext } from "@opencode-ai/ui/context" -import { type Accessor, batch, createEffect, createMemo, onCleanup } from "solid-js" -import { createStore } from "solid-js/store" -import { Persist, persisted } from "@/utils/persist" -import { useCheckServerHealth } from "@/utils/server-health" - -type StoredProject = { worktree: string; expanded: boolean } -type StoredServer = string | ServerConnection.HttpBase | ServerConnection.Http -const HEALTH_POLL_INTERVAL_MS = 10_000 - -export function normalizeServerUrl(input: string) { - const trimmed = input.trim() - if (!trimmed) return - const withProtocol = /^https?:\/\//.test(trimmed) ? trimmed : `http://${trimmed}` - return withProtocol.replace(/\/+$/, "") -} - -export function serverName(conn?: ServerConnection.Any, ignoreDisplayName = false) { - if (!conn) return "" - if (conn.displayName && !ignoreDisplayName) return conn.displayName - return conn.http.url.replace(/^https?:\/\//, "").replace(/\/+$/, "") -} - -function projectsKey(key: ServerConnection.Key) { - if (!key) return "" - if (key === "sidecar") return "local" - if (isLocalHost(key)) return "local" - return key -} - -function isLocalHost(url: string) { - const host = url.replace(/^https?:\/\//, "").split(":")[0] - if (host === "localhost" || host === "127.0.0.1") return "local" -} - -export namespace ServerConnection { - type Base = { displayName?: string } - - export type HttpBase = { - url: string - username?: string - password?: string - } - - // Regular web connections - export type Http = { - type: "http" - http: HttpBase - } & Base - - export type Sidecar = { - type: "sidecar" - http: HttpBase - } & ( - | // Regular desktop server - { variant: "base" } - // WSL server (windows only) - | { - variant: "wsl" - distro: string - } - ) & - Base - - // Remote server desktop can SSH into - export type Ssh = { - type: "ssh" - host: string - // SSH client exposes an HTTP server for the app to use as a proxy - http: HttpBase - } & Base - - export type Any = - | Http - // All these are desktop-only - | (Sidecar | Ssh) - - export const key = (conn: Any): Key => { - switch (conn.type) { - case "http": - return Key.make(conn.http.url) - case "sidecar": { - if (conn.variant === "wsl") return Key.make(`wsl:${conn.distro}`) - return Key.make("sidecar") - } - case "ssh": - return Key.make(`ssh:${conn.host}`) - } - } - - export type Key = string & { _brand: "Key" } - export const Key = { make: (v: string) => v as Key } -} - -export const { use: useServer, provider: ServerProvider } = createSimpleContext({ - name: "Server", - init: (props: { - defaultServer: ServerConnection.Key - disableHealthCheck?: boolean - servers?: Array - }) => { - const checkServerHealth = useCheckServerHealth() - - const [store, setStore, _, ready] = persisted( - Persist.global("server", ["server.v3"]), - createStore({ - list: [] as StoredServer[], - projects: {} as Record, - lastProject: {} as Record, - }), - ) - - const url = (x: StoredServer) => (typeof x === "string" ? x : "type" in x ? x.http.url : x.url) - - const allServers = createMemo((): Array => { - const servers = [ - ...(props.servers ?? []), - ...store.list.map((value) => - typeof value === "string" - ? { - type: "http" as const, - http: { url: value }, - } - : value, - ), - ] - - const deduped = new Map( - servers.map((value) => { - const conn: ServerConnection.Any = "type" in value ? value : { type: "http", http: value } - return [ServerConnection.key(conn), conn] - }), - ) - - return [...deduped.values()] - }) - - const [state, setState] = createStore({ - active: props.defaultServer, - healthy: undefined as boolean | undefined, - }) - - const healthy = () => state.healthy - - function startHealthPolling(conn: ServerConnection.Any) { - let alive = true - let busy = false - - const run = () => { - if (busy) return - busy = true - void check(conn) - .then((next) => { - if (!alive) return - setState("healthy", next) - }) - .finally(() => { - busy = false - }) - } - - run() - const interval = setInterval(run, HEALTH_POLL_INTERVAL_MS) - return () => { - alive = false - clearInterval(interval) - } - } - - function setActive(input: ServerConnection.Key) { - if (state.active !== input) setState("active", input) - } - - function add(input: ServerConnection.Http) { - const url_ = normalizeServerUrl(input.http.url) - if (!url_) return - const conn = { ...input, http: { ...input.http, url: url_ } } - return batch(() => { - const existing = store.list.findIndex((x) => url(x) === url_) - if (existing !== -1) { - setStore("list", existing, conn) - } else { - setStore("list", store.list.length, conn) - } - setState("active", ServerConnection.key(conn)) - return conn - }) - } - - function remove(key: ServerConnection.Key) { - const list = store.list.filter((x) => url(x) !== key) - batch(() => { - setStore("list", list) - if (state.active === key) { - const next = list[0] - setState("active", next ? ServerConnection.Key.make(url(next)) : props.defaultServer) - } - }) - } - - const isReady = createMemo(() => ready() && !!state.active) - - const check = (conn: ServerConnection.Any) => checkServerHealth(conn.http).then((x) => x.healthy) - - createEffect(() => { - const current_ = current() - if (!current_) return - - if (props.disableHealthCheck) { - setState("healthy", true) - return - } - setState("healthy", undefined) - onCleanup(startHealthPolling(current_)) - }) - - const origin = createMemo(() => projectsKey(state.active)) - const projectsList = createMemo(() => store.projects[origin()] ?? []) - const current: Accessor = createMemo( - () => allServers().find((s) => ServerConnection.key(s) === state.active) ?? allServers()[0], - ) - const isLocal = createMemo(() => { - const c = current() - return (c?.type === "sidecar" && c.variant === "base") || (c?.type === "http" && isLocalHost(c.http.url)) - }) - - return { - ready: isReady, - healthy, - isLocal, - get key() { - return state.active - }, - get name() { - return serverName(current()) - }, - get list() { - return allServers() - }, - get current() { - return current() - }, - setActive, - add, - remove, - projects: { - list: projectsList, - open(directory: string) { - const key = origin() - if (!key) return - const current = store.projects[key] ?? [] - if (current.find((x) => x.worktree === directory)) return - setStore("projects", key, [{ worktree: directory, expanded: true }, ...current]) - }, - close(directory: string) { - const key = origin() - if (!key) return - const current = store.projects[key] ?? [] - setStore( - "projects", - key, - current.filter((x) => x.worktree !== directory), - ) - }, - expand(directory: string) { - const key = origin() - if (!key) return - const current = store.projects[key] ?? [] - const index = current.findIndex((x) => x.worktree === directory) - if (index !== -1) setStore("projects", key, index, "expanded", true) - }, - collapse(directory: string) { - const key = origin() - if (!key) return - const current = store.projects[key] ?? [] - const index = current.findIndex((x) => x.worktree === directory) - if (index !== -1) setStore("projects", key, index, "expanded", false) - }, - move(directory: string, toIndex: number) { - const key = origin() - if (!key) return - const current = store.projects[key] ?? [] - const fromIndex = current.findIndex((x) => x.worktree === directory) - if (fromIndex === -1 || fromIndex === toIndex) return - const result = [...current] - const [item] = result.splice(fromIndex, 1) - result.splice(toIndex, 0, item) - setStore("projects", key, result) - }, - last() { - const key = origin() - if (!key) return - return store.lastProject[key] - }, - touch(directory: string) { - const key = origin() - if (!key) return - setStore("lastProject", key, directory) - }, - }, - } - }, -}) diff --git a/packages/app/src/context/settings.tsx b/packages/app/src/context/settings.tsx deleted file mode 100644 index be2fb49d7e..0000000000 --- a/packages/app/src/context/settings.tsx +++ /dev/null @@ -1,332 +0,0 @@ -import { createStore, reconcile } from "solid-js/store" -import { createEffect, createMemo } from "solid-js" -import { createSimpleContext } from "@opencode-ai/ui/context" -import { persisted } from "@/utils/persist" - -export interface NotificationSettings { - agent: boolean - permissions: boolean - errors: boolean -} - -export interface SoundSettings { - agentEnabled: boolean - agent: string - permissionsEnabled: boolean - permissions: string - errorsEnabled: boolean - errors: string -} - -export interface Settings { - general: { - autoSave: boolean - releaseNotes: boolean - followup: "queue" | "steer" - showFileTree: boolean - showNavigation: boolean - showSearch: boolean - showStatus: boolean - showTerminal: boolean - showReasoningSummaries: boolean - shellToolPartsExpanded: boolean - editToolPartsExpanded: boolean - showSessionProgressBar: boolean - } - updates: { - startup: boolean - } - appearance: { - fontSize: number - mono: string - sans: string - terminal: string - } - keybinds: Record - permissions: { - autoApprove: boolean - } - notifications: NotificationSettings - sounds: SoundSettings -} - -export const monoDefault = "System Mono" -export const sansDefault = "System Sans" -export const terminalDefault = "JetBrainsMono Nerd Font Mono" - -const monoFallback = - 'ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace' -const sansFallback = 'ui-sans-serif, system-ui, -apple-system, BlinkMacSystemFont, "Segoe UI", sans-serif' -const terminalFallback = - '"JetBrainsMono Nerd Font Mono", ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace' - -const monoBase = monoFallback -const sansBase = sansFallback -const terminalBase = terminalFallback - -function input(font: string | undefined) { - return font ?? "" -} - -function family(font: string) { - if (/^[\w-]+$/.test(font)) return font - return `"${font.replaceAll("\\", "\\\\").replaceAll('"', '\\"')}"` -} - -function stack(font: string | undefined, base: string) { - const value = font?.trim() ?? "" - if (!value) return base - return `${family(value)}, ${base}` -} - -export function monoInput(font: string | undefined) { - return input(font) -} - -export function sansInput(font: string | undefined) { - return input(font) -} - -export function monoFontFamily(font: string | undefined) { - return stack(font, monoBase) -} - -export function sansFontFamily(font: string | undefined) { - return stack(font, sansBase) -} - -export function terminalInput(font: string | undefined) { - return input(font) -} - -export function terminalFontFamily(font: string | undefined) { - return stack(font, terminalBase) -} - -const defaultSettings: Settings = { - general: { - autoSave: true, - releaseNotes: true, - followup: "steer", - showFileTree: false, - showNavigation: false, - showSearch: false, - showStatus: false, - showTerminal: false, - showReasoningSummaries: false, - shellToolPartsExpanded: false, - editToolPartsExpanded: false, - showSessionProgressBar: true, - }, - updates: { - startup: true, - }, - appearance: { - fontSize: 14, - mono: "", - sans: "", - terminal: "", - }, - keybinds: {}, - permissions: { - autoApprove: false, - }, - notifications: { - agent: true, - permissions: true, - errors: false, - }, - sounds: { - agentEnabled: true, - agent: "staplebops-01", - permissionsEnabled: true, - permissions: "staplebops-02", - errorsEnabled: true, - errors: "nope-03", - }, -} - -function withFallback(read: () => T | undefined, fallback: T) { - return createMemo(() => read() ?? fallback) -} - -export const { use: useSettings, provider: SettingsProvider } = createSimpleContext({ - name: "Settings", - init: () => { - const [store, setStore, _, ready] = persisted("settings.v3", createStore(defaultSettings)) - - createEffect(() => { - if (typeof document === "undefined") return - const root = document.documentElement - root.style.setProperty("--font-family-mono", monoFontFamily(store.appearance?.mono)) - root.style.setProperty("--font-family-sans", sansFontFamily(store.appearance?.sans)) - }) - - createEffect(() => { - if (store.general?.followup !== "queue") return - setStore("general", "followup", "steer") - }) - - return { - ready, - get current() { - return store - }, - general: { - autoSave: withFallback(() => store.general?.autoSave, defaultSettings.general.autoSave), - setAutoSave(value: boolean) { - setStore("general", "autoSave", value) - }, - releaseNotes: withFallback(() => store.general?.releaseNotes, defaultSettings.general.releaseNotes), - setReleaseNotes(value: boolean) { - setStore("general", "releaseNotes", value) - }, - followup: withFallback( - () => (store.general?.followup === "queue" ? "steer" : store.general?.followup), - defaultSettings.general.followup, - ), - setFollowup(value: "queue" | "steer") { - setStore("general", "followup", value === "queue" ? "steer" : value) - }, - showFileTree: withFallback(() => store.general?.showFileTree, defaultSettings.general.showFileTree), - setShowFileTree(value: boolean) { - setStore("general", "showFileTree", value) - }, - showNavigation: withFallback(() => store.general?.showNavigation, defaultSettings.general.showNavigation), - setShowNavigation(value: boolean) { - setStore("general", "showNavigation", value) - }, - showSearch: withFallback(() => store.general?.showSearch, defaultSettings.general.showSearch), - setShowSearch(value: boolean) { - setStore("general", "showSearch", value) - }, - showStatus: withFallback(() => store.general?.showStatus, defaultSettings.general.showStatus), - setShowStatus(value: boolean) { - setStore("general", "showStatus", value) - }, - showTerminal: withFallback(() => store.general?.showTerminal, defaultSettings.general.showTerminal), - setShowTerminal(value: boolean) { - setStore("general", "showTerminal", value) - }, - showReasoningSummaries: withFallback( - () => store.general?.showReasoningSummaries, - defaultSettings.general.showReasoningSummaries, - ), - setShowReasoningSummaries(value: boolean) { - setStore("general", "showReasoningSummaries", value) - }, - shellToolPartsExpanded: withFallback( - () => store.general?.shellToolPartsExpanded, - defaultSettings.general.shellToolPartsExpanded, - ), - setShellToolPartsExpanded(value: boolean) { - setStore("general", "shellToolPartsExpanded", value) - }, - editToolPartsExpanded: withFallback( - () => store.general?.editToolPartsExpanded, - defaultSettings.general.editToolPartsExpanded, - ), - setEditToolPartsExpanded(value: boolean) { - setStore("general", "editToolPartsExpanded", value) - }, - showSessionProgressBar: withFallback( - () => store.general?.showSessionProgressBar, - defaultSettings.general.showSessionProgressBar, - ), - setShowSessionProgressBar(value: boolean) { - setStore("general", "showSessionProgressBar", value) - }, - }, - updates: { - startup: withFallback(() => store.updates?.startup, defaultSettings.updates.startup), - setStartup(value: boolean) { - setStore("updates", "startup", value) - }, - }, - appearance: { - fontSize: withFallback(() => store.appearance?.fontSize, defaultSettings.appearance.fontSize), - setFontSize(value: number) { - setStore("appearance", "fontSize", value) - }, - font: withFallback(() => store.appearance?.mono, defaultSettings.appearance.mono), - setFont(value: string) { - setStore("appearance", "mono", value.trim() ? value : "") - }, - uiFont: withFallback(() => store.appearance?.sans, defaultSettings.appearance.sans), - setUIFont(value: string) { - setStore("appearance", "sans", value.trim() ? value : "") - }, - terminalFont: withFallback(() => store.appearance?.terminal, defaultSettings.appearance.terminal), - setTerminalFont(value: string) { - setStore("appearance", "terminal", value.trim() ? value : "") - }, - }, - keybinds: { - get: (action: string) => store.keybinds?.[action], - set(action: string, keybind: string) { - setStore("keybinds", action, keybind) - }, - reset(action: string) { - setStore("keybinds", (current) => { - if (!Object.prototype.hasOwnProperty.call(current, action)) return current - const next = { ...current } - delete next[action] - return next - }) - }, - resetAll() { - setStore("keybinds", reconcile({})) - }, - }, - permissions: { - autoApprove: withFallback(() => store.permissions?.autoApprove, defaultSettings.permissions.autoApprove), - setAutoApprove(value: boolean) { - setStore("permissions", "autoApprove", value) - }, - }, - notifications: { - agent: withFallback(() => store.notifications?.agent, defaultSettings.notifications.agent), - setAgent(value: boolean) { - setStore("notifications", "agent", value) - }, - permissions: withFallback(() => store.notifications?.permissions, defaultSettings.notifications.permissions), - setPermissions(value: boolean) { - setStore("notifications", "permissions", value) - }, - errors: withFallback(() => store.notifications?.errors, defaultSettings.notifications.errors), - setErrors(value: boolean) { - setStore("notifications", "errors", value) - }, - }, - sounds: { - agentEnabled: withFallback(() => store.sounds?.agentEnabled, defaultSettings.sounds.agentEnabled), - setAgentEnabled(value: boolean) { - setStore("sounds", "agentEnabled", value) - }, - agent: withFallback(() => store.sounds?.agent, defaultSettings.sounds.agent), - setAgent(value: string) { - setStore("sounds", "agent", value) - }, - permissionsEnabled: withFallback( - () => store.sounds?.permissionsEnabled, - defaultSettings.sounds.permissionsEnabled, - ), - setPermissionsEnabled(value: boolean) { - setStore("sounds", "permissionsEnabled", value) - }, - permissions: withFallback(() => store.sounds?.permissions, defaultSettings.sounds.permissions), - setPermissions(value: string) { - setStore("sounds", "permissions", value) - }, - errorsEnabled: withFallback(() => store.sounds?.errorsEnabled, defaultSettings.sounds.errorsEnabled), - setErrorsEnabled(value: boolean) { - setStore("sounds", "errorsEnabled", value) - }, - errors: withFallback(() => store.sounds?.errors, defaultSettings.sounds.errors), - setErrors(value: string) { - setStore("sounds", "errors", value) - }, - }, - } - }, -}) diff --git a/packages/app/src/context/sync-optimistic.test.ts b/packages/app/src/context/sync-optimistic.test.ts deleted file mode 100644 index 3c1b8f4644..0000000000 --- a/packages/app/src/context/sync-optimistic.test.ts +++ /dev/null @@ -1,123 +0,0 @@ -import { describe, expect, test } from "bun:test" -import type { Message, Part } from "@kilocode/sdk/v2/client" -import { applyOptimisticAdd, applyOptimisticRemove, mergeOptimisticPage } from "./sync" - -type Text = Extract - -const userMessage = (id: string, sessionID: string): Message => ({ - id, - sessionID, - role: "user", - time: { created: 1 }, - agent: "assistant", - model: { providerID: "openai", modelID: "gpt" }, -}) - -const textPart = (id: string, sessionID: string, messageID: string): Text => ({ - id, - sessionID, - messageID, - type: "text", - text: id, -}) - -describe("sync optimistic reducers", () => { - test("applyOptimisticAdd inserts message in sorted order and stores parts", () => { - const sessionID = "ses_1" - const draft = { - message: { [sessionID]: [userMessage("msg_2", sessionID)] }, - part: {} as Record, - } - - applyOptimisticAdd(draft, { - sessionID, - message: userMessage("msg_1", sessionID), - parts: [textPart("prt_2", sessionID, "msg_1"), textPart("prt_1", sessionID, "msg_1")], - }) - - expect(draft.message[sessionID]?.map((x) => x.id)).toEqual(["msg_1", "msg_2"]) - expect(draft.part.msg_1?.map((x) => x.id)).toEqual(["prt_1", "prt_2"]) - }) - - test("applyOptimisticRemove removes message and part entries", () => { - const sessionID = "ses_1" - const draft = { - message: { [sessionID]: [userMessage("msg_1", sessionID), userMessage("msg_2", sessionID)] }, - part: { - msg_1: [textPart("prt_1", sessionID, "msg_1")], - msg_2: [textPart("prt_2", sessionID, "msg_2")], - } as Record, - } - - applyOptimisticRemove(draft, { sessionID, messageID: "msg_1" }) - - expect(draft.message[sessionID]?.map((x) => x.id)).toEqual(["msg_2"]) - expect(draft.part.msg_1).toBeUndefined() - expect(draft.part.msg_2).toHaveLength(1) - }) - - test("mergeOptimisticPage keeps pending messages in fetched timelines", () => { - const sessionID = "ses_1" - const page = mergeOptimisticPage( - { - session: [userMessage("msg_1", sessionID)], - part: [{ id: "msg_1", part: [textPart("prt_1", sessionID, "msg_1")] }], - complete: true, - }, - [{ message: userMessage("msg_2", sessionID), parts: [textPart("prt_2", sessionID, "msg_2")] }], - ) - - expect(page.session.map((x) => x.id)).toEqual(["msg_1", "msg_2"]) - expect(page.part.find((x) => x.id === "msg_2")?.part.map((x) => x.id)).toEqual(["prt_2"]) - expect(page.confirmed).toEqual([]) - expect(page.complete).toBe(true) - }) - - test("mergeOptimisticPage keeps missing optimistic parts until the server has them", () => { - const sessionID = "ses_1" - const page = mergeOptimisticPage( - { - session: [userMessage("msg_2", sessionID)], - part: [{ id: "msg_2", part: [textPart("prt_2", sessionID, "msg_2")] }], - complete: true, - }, - [ - { - message: userMessage("msg_2", sessionID), - parts: [textPart("prt_1", sessionID, "msg_2"), textPart("prt_2", sessionID, "msg_2")], - }, - ], - ) - - expect(page.part.find((x) => x.id === "msg_2")?.part.map((x) => x.id)).toEqual(["prt_1", "prt_2"]) - expect(page.confirmed).toEqual([]) - }) - - test("mergeOptimisticPage confirms echoed messages once all parts arrive", () => { - const sessionID = "ses_1" - const page = mergeOptimisticPage( - { - session: [userMessage("msg_2", sessionID)], - part: [ - { - id: "msg_2", - part: [{ ...textPart("prt_1", sessionID, "msg_2"), text: "server" }, textPart("prt_2", sessionID, "msg_2")], - }, - ], - complete: true, - }, - [ - { - message: userMessage("msg_2", sessionID), - parts: [textPart("prt_1", sessionID, "msg_2"), textPart("prt_2", sessionID, "msg_2")], - }, - ], - ) - - expect(page.confirmed).toEqual(["msg_2"]) - expect(page.part.find((x) => x.id === "msg_2")?.part).toMatchObject([ - { id: "prt_1", type: "text", text: "server" }, - { id: "prt_2", type: "text", text: "prt_2" }, - ]) - }) -}) diff --git a/packages/app/src/context/terminal-title.ts b/packages/app/src/context/terminal-title.ts deleted file mode 100644 index c8b18f4211..0000000000 --- a/packages/app/src/context/terminal-title.ts +++ /dev/null @@ -1,24 +0,0 @@ -const template = "Terminal {{number}}" - -const numbered = [ - template, - "محطة طرفية {{number}}", - "Терминал {{number}}", - "ターミナル {{number}}", - "터미널 {{number}}", - "เทอร์มินัล {{number}}", - "终端 {{number}}", - "終端機 {{number}}", -] - -export function defaultTitle(number: number) { - return template.replace("{{number}}", String(number)) -} - -export function isDefaultTitle(title: string, number: number) { - return numbered.some((text) => title === text.replace("{{number}}", String(number))) -} - -export function titleNumber(title: string, max: number) { - return Array.from({ length: max }, (_, idx) => idx + 1).find((number) => isDefaultTitle(title, number)) -} diff --git a/packages/app/src/context/terminal.test.ts b/packages/app/src/context/terminal.test.ts deleted file mode 100644 index 6e07e03124..0000000000 --- a/packages/app/src/context/terminal.test.ts +++ /dev/null @@ -1,82 +0,0 @@ -import { beforeAll, describe, expect, mock, test } from "bun:test" - -let getWorkspaceTerminalCacheKey: (dir: string) => string -let getLegacyTerminalStorageKeys: (dir: string, legacySessionID?: string) => string[] -let migrateTerminalState: (value: unknown) => unknown - -beforeAll(async () => { - mock.module("@solidjs/router", () => ({ - useNavigate: () => () => undefined, - useParams: () => ({}), - })) - mock.module("@opencode-ai/ui/context", () => ({ - createSimpleContext: () => ({ - use: () => undefined, - provider: () => undefined, - }), - })) - const mod = await import("./terminal") - getWorkspaceTerminalCacheKey = mod.getWorkspaceTerminalCacheKey - getLegacyTerminalStorageKeys = mod.getLegacyTerminalStorageKeys - migrateTerminalState = mod.migrateTerminalState -}) - -describe("getWorkspaceTerminalCacheKey", () => { - test("uses workspace-only directory cache key", () => { - expect(getWorkspaceTerminalCacheKey("/repo")).toBe("/repo:__workspace__") - }) -}) - -describe("getLegacyTerminalStorageKeys", () => { - test("keeps workspace storage path when no legacy session id", () => { - expect(getLegacyTerminalStorageKeys("/repo")).toEqual(["/repo/terminal.v1"]) - }) - - test("includes legacy session path before workspace path", () => { - expect(getLegacyTerminalStorageKeys("/repo", "session-123")).toEqual([ - "/repo/terminal/session-123.v1", - "/repo/terminal.v1", - ]) - }) -}) - -describe("migrateTerminalState", () => { - test("drops invalid terminals and restores a valid active terminal", () => { - expect( - migrateTerminalState({ - active: "missing", - all: [ - null, - { id: "one", title: "Terminal 2" }, - { id: "one", title: "duplicate", titleNumber: 9 }, - { id: "two", title: "logs", titleNumber: 4, rows: 24, cols: 80 }, - { title: "no-id" }, - ], - }), - ).toEqual({ - active: "one", - all: [ - { id: "one", title: "Terminal 2", titleNumber: 2 }, - { id: "two", title: "logs", titleNumber: 4, rows: 24, cols: 80 }, - ], - }) - }) - - test("keeps a valid active id", () => { - expect( - migrateTerminalState({ - active: "two", - all: [ - { id: "one", title: "Terminal 1" }, - { id: "two", title: "shell", titleNumber: 7 }, - ], - }), - ).toEqual({ - active: "two", - all: [ - { id: "one", title: "Terminal 1", titleNumber: 1 }, - { id: "two", title: "shell", titleNumber: 7 }, - ], - }) - }) -}) diff --git a/packages/app/src/context/terminal.tsx b/packages/app/src/context/terminal.tsx deleted file mode 100644 index 31d2d6e04c..0000000000 --- a/packages/app/src/context/terminal.tsx +++ /dev/null @@ -1,437 +0,0 @@ -import { createStore, produce } from "solid-js/store" -import { createSimpleContext } from "@opencode-ai/ui/context" -import { batch, createEffect, createMemo, createRoot, on, onCleanup } from "solid-js" -import { useParams } from "@solidjs/router" -import { useSDK } from "./sdk" -import type { Platform } from "./platform" -import { defaultTitle, titleNumber } from "./terminal-title" -import { Persist, persisted, removePersisted } from "@/utils/persist" - -export type LocalPTY = { - id: string - title: string - titleNumber: number - rows?: number - cols?: number - buffer?: string - scrollY?: number - cursor?: number -} - -const WORKSPACE_KEY = "__workspace__" -const MAX_TERMINAL_SESSIONS = 20 - -function record(value: unknown): value is Record { - return typeof value === "object" && value !== null && !Array.isArray(value) -} - -function text(value: unknown) { - return typeof value === "string" ? value : undefined -} - -function num(value: unknown) { - return typeof value === "number" && Number.isFinite(value) ? value : undefined -} - -function numberFromTitle(title: string) { - return titleNumber(title, MAX_TERMINAL_SESSIONS) -} - -function pty(value: unknown): LocalPTY | undefined { - if (!record(value)) return - - const id = text(value.id) - if (!id) return - - const title = text(value.title) ?? "" - const number = num(value.titleNumber) - const rows = num(value.rows) - const cols = num(value.cols) - const buffer = text(value.buffer) - const scrollY = num(value.scrollY) - const cursor = num(value.cursor) - - return { - id, - title, - titleNumber: number && number > 0 ? number : (numberFromTitle(title) ?? 0), - ...(rows !== undefined ? { rows } : {}), - ...(cols !== undefined ? { cols } : {}), - ...(buffer !== undefined ? { buffer } : {}), - ...(scrollY !== undefined ? { scrollY } : {}), - ...(cursor !== undefined ? { cursor } : {}), - } -} - -export function migrateTerminalState(value: unknown) { - if (!record(value)) return value - - const seen = new Set() - const all = (Array.isArray(value.all) ? value.all : []).flatMap((item) => { - const next = pty(item) - if (!next || seen.has(next.id)) return [] - seen.add(next.id) - return [next] - }) - - const active = text(value.active) - - return { - active: active && seen.has(active) ? active : all[0]?.id, - all, - } -} - -export function getWorkspaceTerminalCacheKey(dir: string) { - return `${dir}:${WORKSPACE_KEY}` -} - -export function getLegacyTerminalStorageKeys(dir: string, legacySessionID?: string) { - if (!legacySessionID) return [`${dir}/terminal.v1`] - return [`${dir}/terminal/${legacySessionID}.v1`, `${dir}/terminal.v1`] -} - -type TerminalSession = ReturnType - -type TerminalCacheEntry = { - value: TerminalSession - dispose: VoidFunction -} - -const caches = new Set>() - -const trimTerminal = (pty: LocalPTY) => { - if (!pty.buffer && pty.cursor === undefined && pty.scrollY === undefined) return pty - return { - ...pty, - buffer: undefined, - cursor: undefined, - scrollY: undefined, - } -} - -export function clearWorkspaceTerminals(dir: string, sessionIDs?: string[], platform?: Platform) { - const key = getWorkspaceTerminalCacheKey(dir) - for (const cache of caches) { - const entry = cache.get(key) - entry?.value.clear() - } - - void removePersisted(Persist.workspace(dir, "terminal"), platform) - - const legacy = new Set(getLegacyTerminalStorageKeys(dir)) - for (const id of sessionIDs ?? []) { - for (const key of getLegacyTerminalStorageKeys(dir, id)) { - legacy.add(key) - } - } - for (const key of legacy) { - void removePersisted({ key }, platform) - } -} - -function createWorkspaceTerminalSession(sdk: ReturnType, dir: string, legacySessionID?: string) { - const legacy = getLegacyTerminalStorageKeys(dir, legacySessionID) - - const [store, setStore, _, ready] = persisted( - { - ...Persist.workspace(dir, "terminal", legacy), - migrate: migrateTerminalState, - }, - createStore<{ - active?: string - all: LocalPTY[] - }>({ - all: [], - }), - ) - - const pickNextTerminalNumber = () => { - const existingTitleNumbers = new Set( - store.all.flatMap((pty) => { - const direct = Number.isFinite(pty.titleNumber) && pty.titleNumber > 0 ? pty.titleNumber : undefined - if (direct !== undefined) return [direct] - const parsed = numberFromTitle(pty.title) - if (parsed === undefined) return [] - return [parsed] - }), - ) - - return ( - Array.from({ length: existingTitleNumbers.size + 1 }, (_, index) => index + 1).find( - (number) => !existingTitleNumbers.has(number), - ) ?? 1 - ) - } - - const removeExited = (id: string) => { - const all = store.all - const index = all.findIndex((x) => x.id === id) - if (index === -1) return - const active = store.active === id ? (index === 0 ? all[1]?.id : all[0]?.id) : store.active - batch(() => { - setStore("active", active) - setStore( - "all", - produce((draft) => { - draft.splice(index, 1) - }), - ) - }) - } - - const unsub = sdk.event.on("pty.exited", (event: { properties: { id: string } }) => { - removeExited(event.properties.id) - }) - onCleanup(unsub) - - const update = (client: ReturnType["client"], pty: Partial & { id: string }) => { - const index = store.all.findIndex((x) => x.id === pty.id) - const previous = index >= 0 ? store.all[index] : undefined - if (index >= 0) { - setStore("all", index, (item) => ({ ...item, ...pty })) - } - client.pty - .update({ - ptyID: pty.id, - title: pty.title, - size: pty.cols && pty.rows ? { rows: pty.rows, cols: pty.cols } : undefined, - }) - .catch((error: unknown) => { - if (previous) { - const currentIndex = store.all.findIndex((item) => item.id === pty.id) - if (currentIndex >= 0) setStore("all", currentIndex, previous) - } - console.error("Failed to update terminal", error) - }) - } - - const clone = async (client: ReturnType["client"], id: string) => { - const index = store.all.findIndex((x) => x.id === id) - const pty = store.all[index] - if (!pty) return - const next = await client.pty - .create({ - title: pty.title, - }) - .catch((error: unknown) => { - console.error("Failed to clone terminal", error) - return undefined - }) - if (!next?.data) return - - const active = store.active === pty.id - - batch(() => { - setStore("all", index, { - id: next.data.id, - title: next.data.title ?? pty.title, - titleNumber: pty.titleNumber, - buffer: undefined, - cursor: undefined, - scrollY: undefined, - rows: undefined, - cols: undefined, - }) - if (active) { - setStore("active", next.data.id) - } - }) - } - - return { - ready, - all: createMemo(() => store.all), - active: createMemo(() => store.active), - clear() { - batch(() => { - setStore("active", undefined) - setStore("all", []) - }) - }, - new() { - const nextNumber = pickNextTerminalNumber() - - sdk.client.pty - .create({ title: defaultTitle(nextNumber) }) - .then((pty: { data?: { id?: string; title?: string } }) => { - const id = pty.data?.id - if (!id) return - const newTerminal = { - id, - title: pty.data?.title ?? defaultTitle(nextNumber), - titleNumber: nextNumber, - } - setStore("all", store.all.length, newTerminal) - setStore("active", id) - }) - .catch((error: unknown) => { - console.error("Failed to create terminal", error) - }) - }, - update(pty: Partial & { id: string }) { - update(sdk.client, pty) - }, - trim(id: string) { - const index = store.all.findIndex((x) => x.id === id) - if (index === -1) return - setStore("all", index, (pty) => trimTerminal(pty)) - }, - trimAll() { - setStore("all", (all) => { - const next = all.map(trimTerminal) - if (next.every((pty, index) => pty === all[index])) return all - return next - }) - }, - async clone(id: string) { - await clone(sdk.client, id) - }, - bind() { - const client = sdk.client - return { - trim(id: string) { - const index = store.all.findIndex((x) => x.id === id) - if (index === -1) return - setStore("all", index, (pty) => trimTerminal(pty)) - }, - update(pty: Partial & { id: string }) { - update(client, pty) - }, - async clone(id: string) { - await clone(client, id) - }, - } - }, - open(id: string) { - setStore("active", id) - }, - next() { - const index = store.all.findIndex((x) => x.id === store.active) - if (index === -1) return - const nextIndex = (index + 1) % store.all.length - setStore("active", store.all[nextIndex]?.id) - }, - previous() { - const index = store.all.findIndex((x) => x.id === store.active) - if (index === -1) return - const prevIndex = index === 0 ? store.all.length - 1 : index - 1 - setStore("active", store.all[prevIndex]?.id) - }, - async close(id: string) { - const index = store.all.findIndex((f) => f.id === id) - if (index !== -1) { - batch(() => { - if (store.active === id) { - const next = index > 0 ? store.all[index - 1]?.id : store.all[1]?.id - setStore("active", next) - } - setStore( - "all", - produce((all) => { - all.splice(index, 1) - }), - ) - }) - } - - await sdk.client.pty.remove({ ptyID: id }).catch((error: unknown) => { - console.error("Failed to close terminal", error) - }) - }, - move(id: string, to: number) { - const index = store.all.findIndex((f) => f.id === id) - if (index === -1) return - setStore( - "all", - produce((all) => { - all.splice(to, 0, all.splice(index, 1)[0]) - }), - ) - }, - } -} - -export const { use: useTerminal, provider: TerminalProvider } = createSimpleContext({ - name: "Terminal", - gate: false, - init: () => { - const sdk = useSDK() - const params = useParams() - const cache = new Map() - - caches.add(cache) - onCleanup(() => caches.delete(cache)) - - const disposeAll = () => { - for (const entry of cache.values()) { - entry.dispose() - } - cache.clear() - } - - onCleanup(disposeAll) - - const prune = () => { - while (cache.size > MAX_TERMINAL_SESSIONS) { - const first = cache.keys().next().value - if (!first) return - const entry = cache.get(first) - entry?.dispose() - cache.delete(first) - } - } - - const loadWorkspace = (dir: string, legacySessionID?: string) => { - // Terminals are workspace-scoped so tabs persist while switching sessions in the same directory. - const key = getWorkspaceTerminalCacheKey(dir) - const existing = cache.get(key) - if (existing) { - cache.delete(key) - cache.set(key, existing) - return existing.value - } - - const entry = createRoot((dispose) => ({ - value: createWorkspaceTerminalSession(sdk, dir, legacySessionID), - dispose, - })) - - cache.set(key, entry) - prune() - return entry.value - } - - const workspace = createMemo(() => loadWorkspace(params.dir!, params.id)) - - createEffect( - on( - () => ({ dir: params.dir, id: params.id }), - (next, prev) => { - if (!prev?.dir) return - if (next.dir === prev.dir && next.id === prev.id) return - if (next.dir === prev.dir && next.id) return - loadWorkspace(prev.dir, prev.id).trimAll() - }, - { defer: true }, - ), - ) - - return { - ready: () => workspace().ready(), - all: () => workspace().all(), - active: () => workspace().active(), - new: () => workspace().new(), - update: (pty: Partial & { id: string }) => workspace().update(pty), - trim: (id: string) => workspace().trim(id), - trimAll: () => workspace().trimAll(), - clone: (id: string) => workspace().clone(id), - bind: () => workspace(), - open: (id: string) => workspace().open(id), - close: (id: string) => workspace().close(id), - move: (id: string, to: number) => workspace().move(id, to), - next: () => workspace().next(), - previous: () => workspace().previous(), - } - }, -}) diff --git a/packages/app/src/custom-elements.d.ts b/packages/app/src/custom-elements.d.ts deleted file mode 100644 index ce6acfabc3..0000000000 --- a/packages/app/src/custom-elements.d.ts +++ /dev/null @@ -1,3 +0,0 @@ -// kilocode_change start - keep Windows checkouts parseable when symlinks are disabled. -import "../../ui/src/custom-elements" -// kilocode_change end diff --git a/packages/app/src/entry.tsx b/packages/app/src/entry.tsx deleted file mode 100644 index 84b2b799da..0000000000 --- a/packages/app/src/entry.tsx +++ /dev/null @@ -1,144 +0,0 @@ -// @refresh reload - -import { render } from "solid-js/web" -import { AppBaseProviders, AppInterface } from "@/app" -import { type Platform, PlatformProvider } from "@/context/platform" -import { dict as en } from "@/i18n/en" -import { dict as zh } from "@/i18n/zh" -import { handleNotificationClick } from "@/utils/notification-click" -import pkg from "../package.json" -import { ServerConnection } from "./context/server" - -const DEFAULT_SERVER_URL_KEY = "opencode.settings.dat:defaultServerUrl" - -const getLocale = () => { - if (typeof navigator !== "object") return "en" as const - const languages = navigator.languages?.length ? navigator.languages : [navigator.language] - for (const language of languages) { - if (!language) continue - if (language.toLowerCase().startsWith("zh")) return "zh" as const - } - return "en" as const -} - -const getRootNotFoundError = () => { - const key = "error.dev.rootNotFound" as const - const locale = getLocale() - return locale === "zh" ? (zh[key] ?? en[key]) : en[key] -} - -const getStorage = (key: string) => { - if (typeof localStorage === "undefined") return null - try { - return localStorage.getItem(key) - } catch { - return null - } -} - -const setStorage = (key: string, value: string | null) => { - if (typeof localStorage === "undefined") return - try { - if (value !== null) { - localStorage.setItem(key, value) - return - } - localStorage.removeItem(key) - } catch { - return - } -} - -const readDefaultServerUrl = () => getStorage(DEFAULT_SERVER_URL_KEY) -const writeDefaultServerUrl = (url: string | null) => setStorage(DEFAULT_SERVER_URL_KEY, url) - -const notify: Platform["notify"] = async (title, description, href) => { - if (!("Notification" in window)) return - - const permission = - Notification.permission === "default" - ? await Notification.requestPermission().catch(() => "denied") - : Notification.permission - - if (permission !== "granted") return - - const inView = document.visibilityState === "visible" && document.hasFocus() - if (inView) return - - const notification = new Notification(title, { - body: description ?? "", - icon: "https://opencode.ai/favicon-96x96-v3.png", - }) - - notification.onclick = () => { - handleNotificationClick(href) - notification.close() - } -} - -const openLink: Platform["openLink"] = (url) => { - window.open(url, "_blank") -} - -const back: Platform["back"] = () => { - window.history.back() -} - -const forward: Platform["forward"] = () => { - window.history.forward() -} - -const restart: Platform["restart"] = async () => { - window.location.reload() -} - -const root = document.getElementById("root") -if (!(root instanceof HTMLElement) && import.meta.env.DEV) { - throw new Error(getRootNotFoundError()) -} - -const getCurrentUrl = () => { - if (location.hostname.includes("opencode.ai")) return "http://localhost:4096" - if (import.meta.env.DEV) - return `http://${import.meta.env.VITE_KILO_SERVER_HOST ?? "localhost"}:${import.meta.env.VITE_KILO_SERVER_PORT ?? "4096"}` - return location.origin -} - -const getDefaultUrl = () => { - const lsDefault = readDefaultServerUrl() - if (lsDefault) return lsDefault - return getCurrentUrl() -} - -const platform: Platform = { - platform: "web", - version: pkg.version, - openLink, - back, - forward, - restart, - notify, - getDefaultServer: async () => { - const stored = readDefaultServerUrl() - return stored ? ServerConnection.Key.make(stored) : null - }, - setDefaultServer: writeDefaultServerUrl, -} - -if (root instanceof HTMLElement) { - const server: ServerConnection.Http = { type: "http", http: { url: getCurrentUrl() } } - render( - () => ( - - - - - - ), - root, - ) -} diff --git a/packages/app/src/env.d.ts b/packages/app/src/env.d.ts deleted file mode 100644 index 1197383a3c..0000000000 --- a/packages/app/src/env.d.ts +++ /dev/null @@ -1,17 +0,0 @@ -interface ImportMetaEnv { - readonly VITE_KILO_SERVER_HOST: string - readonly VITE_KILO_SERVER_PORT: string - readonly VITE_KILO_CHANNEL?: "dev" | "beta" | "prod" -} - -interface ImportMeta { - readonly env: ImportMetaEnv -} - -export declare module "solid-js" { - namespace JSX { - interface Directives { - sortable: true - } - } -} diff --git a/packages/app/src/hooks/use-providers.ts b/packages/app/src/hooks/use-providers.ts deleted file mode 100644 index 95260fc766..0000000000 --- a/packages/app/src/hooks/use-providers.ts +++ /dev/null @@ -1,37 +0,0 @@ -import { useGlobalSync } from "@/context/global-sync" -import { decode64 } from "@/utils/base64" -import { useParams } from "@solidjs/router" -import { createMemo } from "solid-js" - -// kilocode_change start - Preferred providers list (order determines display priority) -export const popularProviders = ["kilo", "anthropic", "github-copilot", "openai", "google", "openrouter", "vercel"] -// kilocode_change end -const popularProviderSet = new Set(popularProviders) - -export function useProviders() { - const globalSync = useGlobalSync() - const params = useParams() - const dir = createMemo(() => decode64(params.dir) ?? "") - const providers = () => { - if (dir()) { - const [projectStore] = globalSync.child(dir()) - if (projectStore.provider_ready) return projectStore.provider - } - return globalSync.data.provider - } - return { - all: () => providers().all, - default: () => providers().default, - popular: () => providers().all.filter((p) => popularProviderSet.has(p.id)), - connected: () => { - const connected = new Set(providers().connected) - return providers().all.filter((p) => connected.has(p.id)) - }, - paid: () => { - const connected = new Set(providers().connected) - return providers().all.filter( - (p) => connected.has(p.id) && (p.id !== "opencode" || Object.values(p.models).some((m) => m.cost?.input)), - ) - }, - } -} diff --git a/packages/app/src/i18n/ar.ts b/packages/app/src/i18n/ar.ts deleted file mode 100644 index 47f98dbbfe..0000000000 --- a/packages/app/src/i18n/ar.ts +++ /dev/null @@ -1,845 +0,0 @@ -export const dict = { - "command.category.suggested": "مقترح", - "command.category.view": "عرض", - "command.category.project": "مشروع", - "command.category.provider": "موفر", - "command.category.server": "خادم", - "command.category.session": "جلسة", - "command.category.theme": "سمة", - "command.category.language": "لغة", - "command.category.file": "ملف", - "command.category.context": "سياق", - "command.category.terminal": "محطة طرفية", - "command.category.model": "نموذج", - "command.category.mcp": "MCP", - "command.category.agent": "وكيل", - "command.category.permissions": "أذونات", - "command.category.workspace": "مساحة عمل", - "command.category.settings": "إعدادات", - "theme.scheme.system": "نظام", - "theme.scheme.light": "فاتح", - "theme.scheme.dark": "داكن", - "command.sidebar.toggle": "تبديل الشريط الجانبي", - "command.project.open": "فتح مشروع", - "command.provider.connect": "اتصال بموفر", - "command.server.switch": "تبديل الخادم", - "command.settings.open": "فتح الإعدادات", - "command.session.previous": "الجلسة السابقة", - "command.session.next": "الجلسة التالية", - "command.session.previous.unseen": "الجلسة غير المقروءة السابقة", - "command.session.next.unseen": "الجلسة غير المقروءة التالية", - "command.session.archive": "أرشفة الجلسة", - "command.palette": "لوحة الأوامر", - "command.theme.cycle": "تغيير السمة", - "command.theme.set": "استخدام السمة: {{theme}}", - "command.theme.scheme.cycle": "تغيير مخطط الألوان", - "command.theme.scheme.set": "استخدام مخطط الألوان: {{scheme}}", - "command.language.cycle": "تغيير اللغة", - "command.language.set": "استخدام اللغة: {{language}}", - "command.session.new": "جلسة جديدة", - "command.file.open": "فتح ملف", - "command.tab.close": "إغلاق علامة التبويب", - "command.context.addSelection": "إضافة التحديد إلى السياق", - "command.context.addSelection.description": "إضافة الأسطر المحددة من الملف الحالي", - "command.input.focus": "التركيز على حقل الإدخال", - "command.terminal.toggle": "تبديل المحطة الطرفية", - "command.fileTree.toggle": "تبديل شجرة الملفات", - "command.review.toggle": "تبديل المراجعة", - "command.terminal.new": "محطة طرفية جديدة", - "command.terminal.new.description": "إنشاء علامة تبويب جديدة للمحطة الطرفية", - "command.steps.toggle": "تبديل الخطوات", - "command.steps.toggle.description": "إظهار أو إخفاء خطوات الرسالة الحالية", - "command.message.previous": "الرسالة السابقة", - "command.message.previous.description": "انتقل إلى رسالة المستخدم السابقة", - "command.message.next": "الرسالة التالية", - "command.message.next.description": "انتقل إلى رسالة المستخدم التالية", - "command.model.choose": "اختيار نموذج", - "command.model.choose.description": "حدد نموذجًا مختلفًا", - "command.mcp.toggle": "تبديل MCPs", - "command.mcp.toggle.description": "تبديل MCPs", - "command.agent.cycle": "تغيير الوكيل", - "command.agent.cycle.description": "التبديل إلى الوكيل التالي", - "command.agent.cycle.reverse": "تغيير الوكيل للخلف", - "command.agent.cycle.reverse.description": "التبديل إلى الوكيل السابق", - "command.model.variant.cycle": "تغيير جهد التفكير", - "command.model.variant.cycle.description": "التبديل إلى مستوى الجهد التالي", - "command.prompt.mode.shell": "Shell", - "command.prompt.mode.normal": "Prompt", - "command.permissions.autoaccept.enable": "قبول الأذونات تلقائيًا", - "command.permissions.autoaccept.disable": "إيقاف قبول الأذونات تلقائيًا", - "command.workspace.toggle": "تبديل مساحات العمل", - "command.workspace.toggle.description": "تمكين أو تعطيل مساحات العمل المتعددة في الشريط الجانبي", - "command.session.undo": "تراجع", - "command.session.undo.description": "تراجع عن الرسالة الأخيرة", - "command.session.redo": "إعادة", - "command.session.redo.description": "إعادة الرسالة التي تم التراجع عنها", - "command.session.compact": "ضغط الجلسة", - "command.session.compact.description": "تلخيص الجلسة لتقليل حجم السياق", - "command.session.fork": "تشعب من الرسالة", - "command.session.fork.description": "إنشاء جلسة جديدة من رسالة سابقة", - "command.session.share": "مشاركة الجلسة", - "command.session.share.description": "مشاركة هذه الجلسة ونسخ الرابط إلى الحافظة", - "command.session.unshare": "إلغاء مشاركة الجلسة", - "command.session.unshare.description": "إيقاف مشاركة هذه الجلسة", - "palette.search.placeholder": "البحث في الملفات والأوامر والجلسات", - "palette.empty": "لا توجد نتائج", - "palette.group.commands": "الأوامر", - "palette.group.files": "الملفات", - "dialog.provider.search.placeholder": "البحث عن موفرين", - "dialog.provider.empty": "لم يتم العثور على موفرين", - "dialog.provider.group.popular": "شائع", - "dialog.provider.group.other": "آخر", - "dialog.provider.tag.recommended": "موصى به", - "dialog.provider.opencode.note": "نماذج مختارة تتضمن Claude و GPT و Gemini والمزيد", - "dialog.provider.opencode.tagline": "نماذج موثوقة ومحسنة", - "dialog.provider.opencodeGo.tagline": "اشتراك منخفض التكلفة للجميع", - "dialog.provider.anthropic.note": "اتصل باستخدام Claude Pro/Max أو مفتاح API", - "dialog.provider.copilot.note": "اتصل باستخدام Copilot أو مفتاح API", - "dialog.provider.openai.note": "اتصل باستخدام ChatGPT Pro/Plus أو مفتاح API", - "dialog.provider.google.note": "نماذج Gemini لاستجابات سريعة ومنظمة", - "dialog.provider.openrouter.note": "الوصول إلى جميع النماذج المدعومة من موفر واحد", - "dialog.provider.vercel.note": "وصول موحد إلى نماذج الذكاء الاصطناعي مع توجيه ذكي", - "dialog.model.select.title": "تحديد نموذج", - "dialog.model.search.placeholder": "البحث عن نماذج", - "dialog.model.empty": "لا توجد نتائج للنماذج", - "dialog.model.manage": "إدارة النماذج", - "dialog.model.manage.description": "تخصيص النماذج التي تظهر في محدد النماذج.", - "dialog.model.manage.provider.toggle": "تبديل جميع نماذج {{provider}}", - "dialog.model.unpaid.freeModels.title": "نماذج مجانية مقدمة من Kilo", - "dialog.model.unpaid.addMore.title": "إضافة المزيد من النماذج من موفرين مشهورين", - "dialog.provider.viewAll": "عرض المزيد من الموفرين", - "provider.connect.title": "اتصال {{provider}}", - "provider.connect.title.anthropicProMax": "تسجيل الدخول باستخدام Claude Pro/Max", - "provider.connect.selectMethod": "حدد طريقة تسجيل الدخول لـ {{provider}}.", - "provider.connect.method.apiKey": "مفتاح API", - "provider.connect.status.inProgress": "جارٍ التفويض...", - "provider.connect.status.waiting": "في انتظار التفويض...", - "provider.connect.status.failed": "فشل التفويض: {{error}}", - "provider.connect.apiKey.description": - "أدخل مفتاح واجهة برمجة تطبيقات {{provider}} الخاص بك لتوصيل حسابك واستخدام نماذج {{provider}} في Kilo.", - "provider.connect.apiKey.label": "مفتاح واجهة برمجة تطبيقات {{provider}}", - "provider.connect.apiKey.placeholder": "مفتاح API", - "provider.connect.apiKey.required": "مفتاح API مطلوب", - "provider.connect.opencodeZen.line1": - "يمنحك OpenCode Zen الوصول إلى مجموعة مختارة من النماذج الموثوقة والمحسنة لوكلاء البرمجة.", - "provider.connect.opencodeZen.line2": - "باستخدام مفتاح API واحد، ستحصل على إمكانية الوصول إلى نماذج مثل Claude و GPT و Gemini و GLM والمزيد.", - "provider.connect.opencodeZen.visit.prefix": "قم بزيارة ", - "provider.connect.opencodeZen.visit.link": "opencode.ai/zen", - "provider.connect.opencodeZen.visit.suffix": " للحصول على مفتاح API الخاص بك.", - "provider.connect.oauth.code.visit.prefix": "قم بزيارة ", - "provider.connect.oauth.code.visit.link": "هذا الرابط", - "provider.connect.oauth.code.visit.suffix": - " للحصول على رمز التفويض الخاص بك لتوصيل حسابك واستخدام نماذج {{provider}} في Kilo.", - "provider.connect.oauth.code.label": "رمز تفويض {{method}}", - "provider.connect.oauth.code.placeholder": "رمز التفويض", - "provider.connect.oauth.code.required": "رمز التفويض مطلوب", - "provider.connect.oauth.code.invalid": "رمز التفويض غير صالح", - "provider.connect.oauth.auto.visit.prefix": "قم بزيارة ", - "provider.connect.oauth.auto.visit.link": "هذا الرابط", - "provider.connect.oauth.auto.visit.suffix": " وأدخل الرمز أدناه لتوصيل حسابك واستخدام نماذج {{provider}} في Kilo.", - "provider.connect.oauth.auto.confirmationCode": "رمز التأكيد", - "provider.connect.toast.connected.title": "تم توصيل {{provider}}", - "provider.connect.toast.connected.description": "نماذج {{provider}} متاحة الآن للاستخدام.", - "provider.custom.title": "موفر مخصص", - "provider.custom.description.prefix": "تكوين موفر متوافق مع OpenAI. راجع ", - "provider.custom.description.link": "وثائق تكوين الموفر", - "provider.custom.description.suffix": ".", - "provider.custom.field.providerID.label": "معرف الموفر", - "provider.custom.field.providerID.placeholder": "myprovider", - "provider.custom.field.providerID.description": "أحرف صغيرة، أرقام، شرطات، أو شرطات سفلية", - "provider.custom.field.name.label": "اسم العرض", - "provider.custom.field.name.placeholder": "موفر الذكاء الاصطناعي الخاص بي", - "provider.custom.field.baseURL.label": "عنوان URL الأساسي", - "provider.custom.field.baseURL.placeholder": "https://api.myprovider.com/v1", - "provider.custom.field.apiKey.label": "مفتاح API", - "provider.custom.field.apiKey.placeholder": "مفتاح API", - "provider.custom.field.apiKey.description": "اختياري. اتركه فارغًا إذا كنت تدير المصادقة عبر الترويسات.", - "provider.custom.models.label": "النماذج", - "provider.custom.models.id.label": "المعرف", - "provider.custom.models.id.placeholder": "model-id", - "provider.custom.models.name.label": "الاسم", - "provider.custom.models.name.placeholder": "اسم العرض", - "provider.custom.models.remove": "إزالة النموذج", - "provider.custom.models.add": "إضافة نموذج", - "provider.custom.headers.label": "الترويسات (اختياري)", - "provider.custom.headers.key.label": "ترويسة", - "provider.custom.headers.key.placeholder": "Header-Name", - "provider.custom.headers.value.label": "القيمة", - "provider.custom.headers.value.placeholder": "قيمة", - "provider.custom.headers.remove": "إزالة الترويسة", - "provider.custom.headers.add": "إضافة ترويسة", - "provider.custom.error.providerID.required": "معرف الموفر مطلوب", - "provider.custom.error.providerID.format": "استخدم أحرفًا صغيرة، أرقامًا، شرطات، أو شرطات سفلية", - "provider.custom.error.providerID.exists": "معرف الموفر هذا موجود بالفعل", - "provider.custom.error.name.required": "اسم العرض مطلوب", - "provider.custom.error.baseURL.required": "عنوان URL الأساسي مطلوب", - "provider.custom.error.baseURL.format": "يجب أن يبدأ بـ http:// أو https://", - "provider.custom.error.required": "مطلوب", - "provider.custom.error.duplicate": "مكرر", - "provider.disconnect.toast.disconnected.title": "تم فصل {{provider}}", - "provider.disconnect.toast.disconnected.description": "لم تعد نماذج {{provider}} متاحة.", - "model.tag.free": "مجاني", - "model.tag.latest": "الأحدث", - "model.provider.anthropic": "Anthropic", - "model.provider.openai": "OpenAI", - "model.provider.google": "Google", - "model.provider.xai": "xAI", - "model.provider.meta": "Meta", - "model.input.text": "نص", - "model.input.image": "صورة", - "model.input.audio": "صوت", - "model.input.video": "فيديو", - "model.input.pdf": "pdf", - "model.tooltip.allows": "يسمح: {{inputs}}", - "model.tooltip.reasoning.allowed": "يسمح بالاستنتاج", - "model.tooltip.reasoning.none": "بدون استنتاج", - "model.tooltip.context": "حد السياق {{limit}}", - "common.search.placeholder": "بحث", - "common.goBack": "رجوع", - "common.goForward": "انتقل للأمام", - "common.loading": "جارٍ التحميل", - "common.loading.ellipsis": "...", - "common.cancel": "إلغاء", - "common.connect": "اتصال", - "common.disconnect": "قطع الاتصال", - "common.continue": "إرسال", - "common.submit": "إرسال", - "common.save": "حفظ", - "common.saving": "جارٍ الحفظ...", - "common.default": "افتراضي", - "common.attachment": "مرفق", - "prompt.placeholder.shell": "أدخل أمر shell... {{example}}", - "prompt.placeholder.normal": 'اسأل أي شيء... "{{example}}"', - "prompt.placeholder.simple": "اسأل أي شيء...", - "prompt.placeholder.summarizeComments": "لخّص التعليقات…", - "prompt.placeholder.summarizeComment": "لخّص التعليق…", - "prompt.mode.shell": "Shell", - "prompt.mode.normal": "Prompt", - "prompt.mode.shell.exit": "esc للخروج", - "prompt.example.1": "إصلاح TODO في قاعدة التعليمات البرمجية", - "prompt.example.2": "ما هو المكدس التقني لهذا المشروع؟", - "prompt.example.3": "إصلاح الاختبارات المعطلة", - "prompt.example.4": "اشرح كيف تعمل المصادقة", - "prompt.example.5": "البحث عن وإصلاح الثغرات الأمنية", - "prompt.example.6": "إضافة اختبارات وحدة لخدمة المستخدم", - "prompt.example.7": "إعادة هيكلة هذه الدالة لتكون أكثر قابلية للقراءة", - "prompt.example.8": "ماذا يعني هذا الخطأ؟", - "prompt.example.9": "ساعدني في تصحيح هذه المشكلة", - "prompt.example.10": "توليد وثائق API", - "prompt.example.11": "تحسين استعلامات قاعدة البيانات", - "prompt.example.12": "إضافة التحقق من صحة الإدخال", - "prompt.example.13": "إنشاء مكون جديد لـ...", - "prompt.example.14": "كيف أقوم بنشر هذا المشروع؟", - "prompt.example.15": "مراجعة الكود الخاص بي لأفضل الممارسات", - "prompt.example.16": "إضافة معالجة الأخطاء لهذه الدالة", - "prompt.example.17": "اشرح نمط regex هذا", - "prompt.example.18": "تحويل هذا إلى TypeScript", - "prompt.example.19": "إضافة تسجيل الدخول (logging) في جميع أنحاء قاعدة التعليمات البرمجية", - "prompt.example.20": "ما هي التبعيات القديمة؟", - "prompt.example.21": "ساعدني في كتابة برنامج نصي للهجرة", - "prompt.example.22": "تنفيذ التخزين المؤقت لهذه النقطة النهائية", - "prompt.example.23": "إضافة ترقيم الصفحات إلى هذه القائمة", - "prompt.example.24": "إنشاء أمر CLI لـ...", - "prompt.example.25": "كيف تعمل متغيرات البيئة هنا؟", - "prompt.popover.emptyResults": "لا توجد نتائج مطابقة", - "prompt.popover.emptyCommands": "لا توجد أوامر مطابقة", - "prompt.dropzone.label": "أفلت الصور أو ملفات PDF أو الملفات النصية هنا", - "prompt.dropzone.file.label": "أفلت لإشارة @ للملف", - "prompt.slash.badge.custom": "مخصص", - "prompt.slash.badge.skill": "مهارة", - "prompt.slash.badge.mcp": "mcp", - "prompt.context.active": "نشط", - "prompt.context.includeActiveFile": "تضمين الملف النشط", - "prompt.context.removeActiveFile": "إزالة الملف النشط من السياق", - "prompt.context.removeFile": "إزالة الملف من السياق", - "prompt.action.attachFile": "إرفاق ملف", - "prompt.attachment.remove": "إزالة المرفق", - "prompt.action.send": "إرسال", - "prompt.action.stop": "توقف", - "prompt.toast.pasteUnsupported.title": "مرفق غير مدعوم", - "prompt.toast.pasteUnsupported.description": "يمكن إرفاق الصور أو ملفات PDF أو الملفات النصية فقط هنا.", - "prompt.toast.modelAgentRequired.title": "حدد وكيلاً ونموذجاً", - "prompt.toast.modelAgentRequired.description": "اختر وكيلاً ونموذجاً قبل إرسال الموجه.", - "prompt.toast.worktreeCreateFailed.title": "فشل إنشاء شجرة العمل", - "prompt.toast.sessionCreateFailed.title": "فشل إنشاء الجلسة", - "prompt.toast.shellSendFailed.title": "فشل إرسال أمر shell", - "prompt.toast.commandSendFailed.title": "فشل إرسال الأمر", - "prompt.toast.promptSendFailed.title": "فشل إرسال الموجه", - "prompt.toast.promptSendFailed.description": "تعذر استرداد الجلسة", - "dialog.mcp.title": "MCPs", - "dialog.mcp.description": "{{enabled}} من {{total}} مفعل", - "dialog.mcp.empty": "لم يتم تكوين MCPs", - "dialog.lsp.empty": "تم الكشف تلقائيًا عن LSPs من أنواع الملفات", - "dialog.plugins.empty": "الإضافات المكونة في opencode.json", - "mcp.status.connected": "متصل", - "mcp.status.failed": "فشل", - "mcp.status.needs_auth": "يحتاج إلى مصادقة", - "mcp.status.disabled": "معطل", - "dialog.fork.empty": "لا توجد رسائل للتفرع منها", - "dialog.directory.search.placeholder": "البحث في المجلدات", - "dialog.directory.empty": "لم يتم العثور على مجلدات", - "dialog.server.title": "الخوادم", - "dialog.server.description": "تبديل خادم Kilo الذي يتصل به هذا التطبيق.", - "dialog.server.search.placeholder": "البحث في الخوادم", - "dialog.server.empty": "لا توجد خوادم بعد", - "dialog.server.add.title": "إضافة خادم", - "dialog.server.add.url": "عنوان URL للخادم", - "dialog.server.add.placeholder": "http://localhost:4096", - "dialog.server.add.error": "تعذر الاتصال بالخادم", - "dialog.server.add.checking": "جارٍ التحقق...", - "dialog.server.add.button": "إضافة خادم", - "dialog.server.add.name": "اسم الخادم (اختياري)", - "dialog.server.add.namePlaceholder": "Localhost", - "dialog.server.add.username": "اسم المستخدم (اختياري)", - "dialog.server.add.password": "كلمة المرور (اختياري)", - "dialog.server.edit.title": "تحرير الخادم", - "dialog.server.default.title": "الخادم الافتراضي", - "dialog.server.default.description": - "الاتصال بهذا الخادم عند بدء تشغيل التطبيق بدلاً من بدء خادم محلي. يتطلب إعادة التشغيل.", - "dialog.server.default.none": "لم يتم تحديد خادم", - "dialog.server.default.set": "تعيين الخادم الحالي كافتراضي", - "dialog.server.default.clear": "مسح", - "dialog.server.action.remove": "إزالة الخادم", - "dialog.server.menu.edit": "تعديل", - "dialog.server.menu.default": "تعيين كافتراضي", - "dialog.server.menu.defaultRemove": "إزالة الافتراضي", - "dialog.server.menu.delete": "حذف", - "dialog.server.current": "الخادم الحالي", - "dialog.server.status.default": "افتراضي", - "dialog.project.edit.title": "تحرير المشروع", - "dialog.project.edit.name": "الاسم", - "dialog.project.edit.icon": "أيقونة", - "dialog.project.edit.icon.alt": "أيقونة المشروع", - "dialog.project.edit.icon.hint": "انقر أو اسحب صورة", - "dialog.project.edit.icon.recommended": "موصى به: 128x128px", - "dialog.project.edit.color": "لون", - "dialog.project.edit.color.select": "اختر لون {{color}}", - "dialog.project.edit.worktree.startup": "سكريبت بدء تشغيل مساحة العمل", - "dialog.project.edit.worktree.startup.description": "يتم تشغيله بعد إنشاء مساحة عمل جديدة (شجرة عمل).", - "dialog.project.edit.worktree.startup.placeholder": "مثال: bun install", - "context.breakdown.title": "تفصيل السياق", - "context.breakdown.note": 'تفصيل تقريبي لرموز الإدخال. يشمل "أخرى" تعريفات الأدوات والنفقات العامة.', - "context.breakdown.system": "النظام", - "context.breakdown.user": "المستخدم", - "context.breakdown.assistant": "المساعد", - "context.breakdown.tool": "استدعاءات الأداة", - "context.breakdown.other": "أخرى", - "context.systemPrompt.title": "موجه النظام", - "context.rawMessages.title": "الرسائل الخام", - "context.stats.session": "جلسة", - "context.stats.messages": "رسائل", - "context.stats.provider": "موفر", - "context.stats.model": "نموذج", - "context.stats.limit": "حد السياق", - "context.stats.totalTokens": "إجمالي الرموز", - "context.stats.usage": "استخدام", - "context.stats.inputTokens": "رموز الإدخال", - "context.stats.outputTokens": "رموز الإخراج", - "context.stats.reasoningTokens": "رموز الاستنتاج", - "context.stats.cacheTokens": "رموز التخزين المؤقت (قراءة/كتابة)", - "context.stats.userMessages": "رسائل المستخدم", - "context.stats.assistantMessages": "رسائل المساعد", - "context.stats.totalCost": "التكلفة الإجمالية", - "context.stats.sessionCreated": "تم إنشاء الجلسة", - "context.stats.lastActivity": "آخر نشاط", - "context.usage.tokens": "رموز", - "context.usage.usage": "استخدام", - "context.usage.cost": "تكلفة", - "context.usage.clickToView": "انقر لعرض السياق", - "context.usage.view": "عرض استخدام السياق", - "language.en": "English", - "language.zh": "简体中文", - "language.zht": "繁體中文", - "language.ko": "한국어", - "language.de": "Deutsch", - "language.es": "Español", - "language.fr": "Français", - "language.da": "Dansk", - "language.ja": "日本語", - "language.pl": "Polski", - "language.ru": "Русский", - "language.ar": "العربية", - "language.no": "Norsk", - "language.br": "Português (Brasil)", - "language.bs": "Bosanski", - "language.th": "ไทย", - "language.tr": "Türkçe", - "toast.language.title": "لغة", - "toast.language.description": "تم التبديل إلى {{language}}", - "toast.theme.title": "تم تبديل السمة", - "toast.scheme.title": "مخطط الألوان", - "toast.workspace.enabled.title": "تم تمكين مساحات العمل", - "toast.workspace.enabled.description": "الآن يتم عرض عدة worktrees في الشريط الجانبي", - "toast.workspace.disabled.title": "تم تعطيل مساحات العمل", - "toast.workspace.disabled.description": "يتم عرض worktree الرئيسي فقط في الشريط الجانبي", - "toast.permissions.autoaccept.on.title": "يتم قبول الأذونات تلقائيًا", - "toast.permissions.autoaccept.on.description": "ستتم الموافقة على طلبات الأذونات تلقائيًا", - "toast.permissions.autoaccept.off.title": "تم إيقاف قبول الأذونات تلقائيًا", - "toast.permissions.autoaccept.off.description": "ستتطلب طلبات الأذونات موافقة", - "toast.model.none.title": "لم يتم تحديد نموذج", - "toast.model.none.description": "قم بتوصيل موفر لتلخيص هذه الجلسة", - "toast.file.loadFailed.title": "فشل تحميل الملف", - "toast.file.listFailed.title": "فشل سرد الملفات", - "toast.context.noLineSelection.title": "لا يوجد تحديد للأسطر", - "toast.context.noLineSelection.description": "حدد نطاق أسطر في تبويب ملف أولاً.", - "toast.session.share.copyFailed.title": "فشل نسخ عنوان URL إلى الحافظة", - "toast.session.share.success.title": "تمت مشاركة الجلسة", - "toast.session.share.success.description": "تم نسخ عنوان URL للمشاركة إلى الحافظة!", - "toast.session.share.failed.title": "فشل مشاركة الجلسة", - "toast.session.share.failed.description": "حدث خطأ أثناء مشاركة الجلسة", - "toast.session.unshare.success.title": "تم إلغاء مشاركة الجلسة", - "toast.session.unshare.success.description": "تم إلغاء مشاركة الجلسة بنجاح!", - "toast.session.unshare.failed.title": "فشل إلغاء مشاركة الجلسة", - "toast.session.unshare.failed.description": "حدث خطأ أثناء إلغاء مشاركة الجلسة", - "toast.session.listFailed.title": "فشل تحميل الجلسات لـ {{project}}", - "toast.update.title": "تحديث متاح", - "toast.update.description": "نسخة جديدة من Kilo ({{version}}) متاحة الآن للتثبيت.", - "toast.update.action.installRestart": "تثبيت وإعادة تشغيل", - "toast.update.action.notYet": "ليس الآن", - "error.page.title": "حدث خطأ ما", - "error.page.description": "حدث خطأ أثناء تحميل التطبيق.", - "error.page.details.label": "تفاصيل الخطأ", - "error.page.action.restart": "إعادة تشغيل", - "error.page.action.checking": "جارٍ التحقق...", - "error.page.action.checkUpdates": "التحقق من وجود تحديثات", - "error.page.action.updateTo": "تحديث إلى {{version}}", - "error.page.report.prefix": "يرجى الإبلاغ عن هذا الخطأ لفريق Kilo", - "error.page.report.discord": "على Discord", - "error.page.version": "الإصدار: {{version}}", - "error.dev.rootNotFound": - "لم يتم العثور على العنصر الجذري. هل نسيت إضافته إلى index.html؟ أو ربما تمت كتابة سمة id بشكل خاطئ؟", - "error.globalSync.connectFailed": "تعذر الاتصال بالخادم. هل هناك خادم يعمل في `{{url}}`؟", - "directory.error.invalidUrl": "دليل غير صالح في عنوان URL.", - "error.chain.unknown": "خطأ غير معروف", - "error.chain.causedBy": "بسبب:", - "error.chain.apiError": "خطأ API", - "error.chain.status": "الحالة: {{status}}", - "error.chain.retryable": "قابل لإعادة المحاولة: {{retryable}}", - "error.chain.responseBody": "نص الاستجابة:\n{{body}}", - "error.chain.didYouMean": "هل كنت تعني: {{suggestions}}", - "error.chain.modelNotFound": "النموذج غير موجود: {{provider}}/{{model}}", - "error.chain.checkConfig": "تحقق من أسماء الموفر/النموذج في التكوين (opencode.json)", - "error.chain.mcpFailed": 'فشل خادم MCP "{{name}}". لاحظ أن Kilo لا يدعم مصادقة MCP بعد.', - "error.chain.providerAuthFailed": "فشلت مصادقة الموفر ({{provider}}): {{message}}", - "error.chain.providerInitFailed": 'فشل تهيئة الموفر "{{provider}}". تحقق من بيانات الاعتماد والتكوين.', - "error.chain.configJsonInvalid": "ملف التكوين في {{path}} ليس JSON(C) صالحًا", - "error.chain.configJsonInvalidWithMessage": "ملف التكوين في {{path}} ليس JSON(C) صالحًا: {{message}}", - "error.chain.configDirectoryTypo": - 'الدليل "{{dir}}" في {{path}} غير صالح. أعد تسمية الدليل إلى "{{suggestion}}" أو قم بإزالته. هذا خطأ مطبعي شائع.', - "error.chain.configFrontmatterError": "فشل تحليل frontmatter في {{path}}:\n{{message}}", - "error.chain.configInvalid": "ملف التكوين في {{path}} غير صالح", - "error.chain.configInvalidWithMessage": "ملف التكوين في {{path}} غير صالح: {{message}}", - "notification.permission.title": "مطلوب إذن", - "notification.permission.description": "{{sessionTitle}} في {{projectName}} يحتاج إلى إذن", - "notification.question.title": "سؤال", - "notification.question.description": "{{sessionTitle}} في {{projectName}} لديه سؤال", - "notification.action.goToSession": "انتقل إلى الجلسة", - "notification.session.responseReady.title": "الاستجابة جاهزة", - "notification.session.error.title": "خطأ في الجلسة", - "notification.session.error.fallbackDescription": "حدث خطأ", - "home.recentProjects": "المشاريع الحديثة", - "home.empty.title": "لا توجد مشاريع حديثة", - "home.empty.description": "ابدأ بفتح مشروع محلي", - "session.tab.session": "جلسة", - "session.tab.review": "مراجعة", - "session.tab.context": "سياق", - "session.panel.reviewAndFiles": "المراجعة والملفات", - "session.review.filesChanged": "تم تغيير {{count}} ملفات", - "session.review.change.one": "تغيير", - "session.review.change.other": "تغييرات", - "session.review.loadingChanges": "جارٍ تحميل التغييرات...", - "session.review.empty": "لا توجد تغييرات في هذه الجلسة بعد", - "session.review.noChanges": "لا توجد تغييرات", - "session.review.noVcs": "لم يتم اكتشاف نظام التحكم في الإصدار Git، لن يتم عرض التغييرات", - "session.review.noSnapshot": "تم تعطيل تتبع اللقطات في التكوين، لذا فإن تغييرات الجلسة غير متوفرة", - "session.files.selectToOpen": "اختر ملفًا لفتحه", - "session.files.all": "كل الملفات", - "session.files.empty": "لا توجد ملفات", - "session.files.binaryContent": "ملف ثنائي (لا يمكن عرض المحتوى)", - "session.messages.renderEarlier": "عرض الرسائل السابقة", - "session.messages.loadingEarlier": "جارٍ تحميل الرسائل السابقة...", - "session.messages.loadEarlier": "تحميل الرسائل السابقة", - "session.messages.loading": "جارٍ تحميل الرسائل...", - "session.messages.jumpToLatest": "الانتقال إلى الأحدث", - "session.context.addToContext": "إضافة {{selection}} إلى السياق", - "session.todo.title": "المهام", - "session.todo.collapse": "طي", - "session.todo.expand": "توسيع", - "session.followupDock.summary.one": "{{count}} رسالة في الانتظار", - "session.followupDock.summary.other": "{{count}} رسائل في الانتظار", - "session.followupDock.sendNow": "إرسال الآن", - "session.followupDock.edit": "تحرير", - "session.followupDock.collapse": "طي الرسائل المنتظرة", - "session.followupDock.expand": "توسيع الرسائل المنتظرة", - "session.revertDock.summary.one": "{{count}} رسالة تم التراجع عنها", - "session.revertDock.summary.other": "{{count}} رسائل تم التراجع عنها", - "session.revertDock.collapse": "طي الرسائل التي تم التراجع عنها", - "session.revertDock.expand": "توسيع الرسائل التي تم التراجع عنها", - "session.revertDock.restore": "استعادة الرسالة", - "session.new.title": "ابنِ أي شيء", - "session.new.worktree.main": "الفرع الرئيسي", - "session.new.worktree.mainWithBranch": "الفرع الرئيسي ({{branch}})", - "session.new.worktree.create": "إنشاء شجرة عمل جديدة", - "session.new.lastModified": "آخر تعديل", - "session.header.search.placeholder": "بحث {{project}}", - "session.header.searchFiles": "بحث عن الملفات", - "session.header.openIn": "فتح في", - "session.header.open.action": "فتح {{app}}", - "session.header.open.ariaLabel": "فتح في {{app}}", - "session.header.open.menu": "خيارات الفتح", - "session.header.open.copyPath": "نسخ المسار", - "status.popover.trigger": "الحالة", - "status.popover.ariaLabel": "إعدادات الخوادم", - "status.popover.tab.servers": "الخوادم", - "status.popover.tab.mcp": "MCP", - "status.popover.tab.lsp": "LSP", - "status.popover.tab.plugins": "الإضافات", - "status.popover.action.manageServers": "إدارة الخوادم", - "session.share.popover.title": "نشر على الويب", - "session.share.popover.description.shared": "هذه الجلسة عامة على الويب. يمكن لأي شخص لديه الرابط الوصول إليها.", - "session.share.popover.description.unshared": "شارك الجلسة علنًا على الويب. ستكون متاحة لأي شخص لديه الرابط.", - "session.share.action.share": "مشاركة", - "session.share.action.publish": "نشر", - "session.share.action.publishing": "جارٍ النشر...", - "session.share.action.unpublish": "إلغاء النشر", - "session.share.action.unpublishing": "جارٍ إلغاء النشر...", - "session.share.action.view": "عرض", - "session.share.copy.copied": "تم النسخ", - "session.share.copy.copyLink": "نسخ الرابط", - "lsp.tooltip.none": "لا توجد خوادم LSP", - "lsp.label.connected": "{{count}} LSP", - "prompt.loading": "جارٍ تحميل الموجه...", - "terminal.loading": "جارٍ تحميل المحطة الطرفية...", - "terminal.title": "محطة طرفية", - "terminal.title.numbered": "محطة طرفية {{number}}", - "terminal.close": "إغلاق المحطة الطرفية", - "terminal.connectionLost.title": "فقد الاتصال", - "terminal.connectionLost.description": "انقطع اتصال المحطة الطرفية. يمكن أن يحدث هذا عند إعادة تشغيل الخادم.", - "common.closeTab": "إغلاق علامة التبويب", - "common.dismiss": "رفض", - "common.requestFailed": "فشل الطلب", - "common.moreOptions": "مزيد من الخيارات", - "common.learnMore": "اعرف المزيد", - "common.rename": "إعادة تسمية", - "common.reset": "إعادة تعيين", - "common.archive": "أرشفة", - "common.delete": "حذف", - "common.close": "إغلاق", - "common.edit": "تحرير", - "common.loadMore": "تحميل المزيد", - "common.key.esc": "ESC", - "sidebar.menu.toggle": "تبديل القائمة", - "sidebar.nav.projectsAndSessions": "المشاريع والجلسات", - "sidebar.settings": "الإعدادات", - "sidebar.help": "مساعدة", - "sidebar.workspaces.enable": "تمكين مساحات العمل", - "sidebar.workspaces.disable": "تعطيل مساحات العمل", - "sidebar.gettingStarted.title": "البدء", - "sidebar.gettingStarted.line1": "يتضمن Kilo نماذج مجانية حتى تتمكن من البدء فورًا.", - "sidebar.gettingStarted.line2": "قم بتوصيل أي موفر لاستخدام النماذج، بما في ذلك Claude و GPT و Gemini وما إلى ذلك.", - "sidebar.project.recentSessions": "الجلسات الحديثة", - "sidebar.project.viewAllSessions": "عرض جميع الجلسات", - "sidebar.project.clearNotifications": "مسح الإشعارات", - "app.name.desktop": "Kilo Desktop", - "settings.section.desktop": "سطح المكتب", - "settings.section.server": "الخادم", - "settings.tab.general": "عام", - "settings.tab.shortcuts": "اختصارات", - "settings.desktop.section.wsl": "WSL", - "settings.desktop.wsl.title": "تكامل WSL", - "settings.desktop.wsl.description": "تشغيل خادم Kilo داخل WSL على Windows.", - "settings.general.section.appearance": "المظهر", - "settings.general.section.notifications": "إشعارات النظام", - "settings.general.section.updates": "التحديثات", - "settings.general.section.sounds": "المؤثرات الصوتية", - "settings.general.section.feed": "الخلاصة", - "settings.general.section.display": "شاشة العرض", - "settings.general.row.language.title": "اللغة", - "settings.general.row.language.description": "تغيير لغة العرض لـ Kilo", - "settings.general.row.appearance.title": "المظهر", - "settings.general.row.appearance.description": "تخصيص كيفية ظهور Kilo على جهازك", - "settings.general.row.colorScheme.title": "مخطط الألوان", - "settings.general.row.colorScheme.description": "اختر ما إذا كان Kilo يتبع سمة النظام أو الفاتح أو الداكن", - "settings.general.row.theme.title": "السمة", - "settings.general.row.theme.description": "تخصيص سمة Kilo.", - "settings.general.row.font.title": "خط الكود", - "settings.general.row.font.description": "خصّص الخط المستخدم في كتل التعليمات البرمجية", - "settings.general.row.terminalFont.title": "Terminal Font", - "settings.general.row.terminalFont.description": "Customise the font used in the terminal", - "settings.general.row.uiFont.title": "خط الواجهة", - "settings.general.row.uiFont.description": "خصّص الخط المستخدم في الواجهة بأكملها", - "settings.general.row.followup.title": "سلوك المتابعة", - "settings.general.row.followup.description": "اختر ما إذا كانت طلبات المتابعة توجه فورًا أو تنتظر في قائمة انتظار", - "settings.general.row.followup.option.queue": "قائمة انتظار", - "settings.general.row.followup.option.steer": "توجيه", - "settings.general.row.reasoningSummaries.title": "إظهار ملخصات الاستنتاج", - "settings.general.row.reasoningSummaries.description": "عرض ملخصات استنتاج النموذج في الشريط الزمني", - "settings.general.row.shellToolPartsExpanded.title": "توسيع أجزاء أداة shell", - "settings.general.row.shellToolPartsExpanded.description": - "إظهار أجزاء أداة shell موسعة بشكل افتراضي في الشريط الزمني", - "settings.general.row.editToolPartsExpanded.title": "توسيع أجزاء أداة edit", - "settings.general.row.editToolPartsExpanded.description": - "إظهار أجزاء أدوات edit و write و patch موسعة بشكل افتراضي في الشريط الزمني", - "settings.general.row.showSessionProgressBar.title": "إظهار شريط تقدم الجلسة", - "settings.general.row.showSessionProgressBar.description": "عرض شريط التقدم المتحرك أعلى الجلسة أثناء عمل الوكيل", - "settings.general.row.wayland.title": "استخدام Wayland الأصلي", - "settings.general.row.wayland.description": "تعطيل التراجع إلى X11 على Wayland. يتطلب إعادة التشغيل.", - "settings.general.row.wayland.tooltip": - "على Linux مع شاشات بمعدلات تحديث مختلطة، يمكن أن يكون Wayland الأصلي أكثر استقرارًا.", - "settings.general.row.releaseNotes.title": "ملاحظات الإصدار", - "settings.general.row.releaseNotes.description": 'عرض نوافذ "ما الجديد" المنبثقة بعد التحديثات', - "settings.updates.row.startup.title": "التحقق من التحديثات عند بدء التشغيل", - "settings.updates.row.startup.description": "التحقق تلقائيًا من التحديثات عند تشغيل Kilo", - "settings.updates.row.check.title": "التحقق من التحديثات", - "settings.updates.row.check.description": "التحقق يدويًا من التحديثات وتثبيتها إذا كانت متاحة", - "settings.updates.action.checkNow": "تحقق الآن", - "settings.updates.action.checking": "جارٍ التحقق...", - "settings.updates.toast.latest.title": "أنت على آخر إصدار", - "settings.updates.toast.latest.description": "أنت تستخدم أحدث إصدار من Kilo.", - "sound.option.none": "بلا", - "sound.option.alert01": "تنبيه 01", - "sound.option.alert02": "تنبيه 02", - "sound.option.alert03": "تنبيه 03", - "sound.option.alert04": "تنبيه 04", - "sound.option.alert05": "تنبيه 05", - "sound.option.alert06": "تنبيه 06", - "sound.option.alert07": "تنبيه 07", - "sound.option.alert08": "تنبيه 08", - "sound.option.alert09": "تنبيه 09", - "sound.option.alert10": "تنبيه 10", - "sound.option.bipbop01": "بيب بوب 01", - "sound.option.bipbop02": "بيب بوب 02", - "sound.option.bipbop03": "بيب بوب 03", - "sound.option.bipbop04": "بيب بوب 04", - "sound.option.bipbop05": "بيب بوب 05", - "sound.option.bipbop06": "بيب بوب 06", - "sound.option.bipbop07": "بيب بوب 07", - "sound.option.bipbop08": "بيب بوب 08", - "sound.option.bipbop09": "بيب بوب 09", - "sound.option.bipbop10": "بيب بوب 10", - "sound.option.staplebops01": "ستابل بوبس 01", - "sound.option.staplebops02": "ستابل بوبس 02", - "sound.option.staplebops03": "ستابل بوبس 03", - "sound.option.staplebops04": "ستابل بوبس 04", - "sound.option.staplebops05": "ستابل بوبس 05", - "sound.option.staplebops06": "ستابل بوبس 06", - "sound.option.staplebops07": "ستابل بوبس 07", - "sound.option.nope01": "كلا 01", - "sound.option.nope02": "كلا 02", - "sound.option.nope03": "كلا 03", - "sound.option.nope04": "كلا 04", - "sound.option.nope05": "كلا 05", - "sound.option.nope06": "كلا 06", - "sound.option.nope07": "كلا 07", - "sound.option.nope08": "كلا 08", - "sound.option.nope09": "كلا 09", - "sound.option.nope10": "كلا 10", - "sound.option.nope11": "كلا 11", - "sound.option.nope12": "كلا 12", - "sound.option.yup01": "نعم 01", - "sound.option.yup02": "نعم 02", - "sound.option.yup03": "نعم 03", - "sound.option.yup04": "نعم 04", - "sound.option.yup05": "نعم 05", - "sound.option.yup06": "نعم 06", - "settings.general.notifications.agent.title": "وكيل", - "settings.general.notifications.agent.description": "عرض إشعار النظام عندما يكتمل الوكيل أو يحتاج إلى اهتمام", - "settings.general.notifications.permissions.title": "أذونات", - "settings.general.notifications.permissions.description": "عرض إشعار النظام عند الحاجة إلى إذن", - "settings.general.notifications.errors.title": "أخطاء", - "settings.general.notifications.errors.description": "عرض إشعار النظام عند حدوث خطأ", - "settings.general.sounds.agent.title": "وكيل", - "settings.general.sounds.agent.description": "تشغيل صوت عندما يكتمل الوكيل أو يحتاج إلى اهتمام", - "settings.general.sounds.permissions.title": "أذونات", - "settings.general.sounds.permissions.description": "تشغيل صوت عند الحاجة إلى إذن", - "settings.general.sounds.errors.title": "أخطاء", - "settings.general.sounds.errors.description": "تشغيل صوت عند حدوث خطأ", - "settings.shortcuts.title": "اختصارات لوحة المفاتيح", - "settings.shortcuts.reset.button": "إعادة التعيين إلى الافتراضيات", - "settings.shortcuts.reset.toast.title": "تم إعادة تعيين الاختصارات", - "settings.shortcuts.reset.toast.description": "تم إعادة تعيين اختصارات لوحة المفاتيح إلى الافتراضيات.", - "settings.shortcuts.conflict.title": "الاختصار قيد الاستخدام بالفعل", - "settings.shortcuts.conflict.description": "{{keybind}} معين بالفعل لـ {{titles}}.", - "settings.shortcuts.unassigned": "غير معين", - "settings.shortcuts.pressKeys": "اضغط على المفاتيح", - "settings.shortcuts.search.placeholder": "البحث في الاختصارات", - "settings.shortcuts.search.empty": "لم يتم العثور على اختصارات", - "settings.shortcuts.group.general": "عام", - "settings.shortcuts.group.session": "جلسة", - "settings.shortcuts.group.navigation": "تصفح", - "settings.shortcuts.group.modelAndAgent": "النموذج والوكيل", - "settings.shortcuts.group.terminal": "المحطة الطرفية", - "settings.shortcuts.group.prompt": "موجه", - "settings.providers.title": "الموفرون", - "settings.providers.description": "ستكون إعدادات الموفر قابلة للتكوين هنا.", - "settings.providers.section.connected": "الموفرون المتصلون", - "settings.providers.connected.empty": "لا يوجد موفرون متصلون", - "settings.providers.section.popular": "الموفرون الشائعون", - "settings.providers.tag.environment": "البيئة", - "settings.providers.tag.config": "التكوين", - "settings.providers.tag.custom": "مخصص", - "settings.providers.tag.other": "أخرى", - "settings.models.title": "النماذج", - "settings.models.description": "ستكون إعدادات النموذج قابلة للتكوين هنا.", - "settings.agents.title": "الوكلاء", - "settings.agents.description": "ستكون إعدادات الوكيل قابلة للتكوين هنا.", - "settings.commands.title": "الأوامر", - "settings.commands.description": "ستكون إعدادات الأمر قابلة للتكوين هنا.", - "settings.mcp.title": "MCP", - "settings.mcp.description": "ستكون إعدادات MCP قابلة للتكوين هنا.", - "settings.permissions.title": "الأذونات", - "settings.permissions.description": "تحكم في الأدوات التي يمكن للخادم استخدامها بشكل افتراضي.", - "settings.permissions.section.tools": "الأدوات", - "settings.permissions.toast.updateFailed.title": "فشل تحديث الأذونات", - "settings.permissions.action.allow": "سماح", - "settings.permissions.action.ask": "سؤال", - "settings.permissions.action.deny": "رفض", - "settings.permissions.tool.read.title": "قراءة", - "settings.permissions.tool.read.description": "قراءة ملف (يطابق مسار الملف)", - "settings.permissions.tool.edit.title": "تحرير", - "settings.permissions.tool.edit.description": - "تعديل الملفات، بما في ذلك التحرير والكتابة والتصحيحات والتحرير المتعدد", - "settings.permissions.tool.glob.title": "Glob", - "settings.permissions.tool.glob.description": "مطابقة الملفات باستخدام أنماط glob", - "settings.permissions.tool.grep.title": "Grep", - "settings.permissions.tool.grep.description": "البحث في محتويات الملف باستخدام التعبيرات العادية", - "settings.permissions.tool.list.title": "قائمة", - "settings.permissions.tool.list.description": "سرد الملفات داخل دليل", - "settings.permissions.tool.bash.title": "Bash", - "settings.permissions.tool.bash.description": "تشغيل أوامر shell", - "settings.permissions.tool.task.title": "مهمة", - "settings.permissions.tool.task.description": "تشغيل الوكلاء الفرعيين", - "settings.permissions.tool.skill.title": "مهارة", - "settings.permissions.tool.skill.description": "تحميل مهارة بالاسم", - "settings.permissions.tool.lsp.title": "LSP", - "settings.permissions.tool.lsp.description": "تشغيل استعلامات خادم اللغة", - "settings.permissions.tool.todowrite.title": "كتابة المهام", - "settings.permissions.tool.todowrite.description": "تحديث قائمة المهام", - "settings.permissions.tool.webfetch.title": "جلب الويب", - "settings.permissions.tool.webfetch.description": "جلب محتوى من عنوان URL", - "settings.permissions.tool.websearch.title": "بحث الويب", - "settings.permissions.tool.websearch.description": "البحث في الويب", - "settings.permissions.tool.codesearch.title": "بحث الكود", - "settings.permissions.tool.codesearch.description": "البحث عن كود على الويب", - "settings.permissions.tool.external_directory.title": "دليل خارجي", - "settings.permissions.tool.external_directory.description": "الوصول إلى الملفات خارج دليل المشروع", - "settings.permissions.tool.doom_loop.title": "حلقة الموت", - "settings.permissions.tool.doom_loop.description": "اكتشاف استدعاءات الأدوات المتكررة بمدخلات متطابقة", - "session.delete.failed.title": "فشل حذف الجلسة", - "session.delete.title": "حذف الجلسة", - "session.delete.confirm": 'حذف الجلسة "{{name}}"؟', - "session.delete.button": "حذف الجلسة", - "workspace.new": "مساحة عمل جديدة", - "workspace.type.local": "محلي", - "workspace.type.sandbox": "صندوق رمل", - "workspace.create.failed.title": "فشل إنشاء مساحة العمل", - "workspace.delete.failed.title": "فشل حذف مساحة العمل", - "workspace.resetting.title": "إعادة تعيين مساحة العمل", - "workspace.resetting.description": "قد يستغرق هذا دقيقة.", - "workspace.reset.failed.title": "فشل إعادة تعيين مساحة العمل", - "workspace.reset.success.title": "تمت إعادة تعيين مساحة العمل", - "workspace.reset.success.description": "مساحة العمل تطابق الآن الفرع الافتراضي.", - "workspace.error.stillPreparing": "مساحة العمل لا تزال قيد الإعداد", - "workspace.status.checking": "التحقق من التغييرات غير المدمجة...", - "workspace.status.error": "تعذر التحقق من حالة git.", - "workspace.status.clean": "لم يتم اكتشاف تغييرات غير مدمجة.", - "workspace.status.dirty": "تم اكتشاف تغييرات غير مدمجة في مساحة العمل هذه.", - "workspace.delete.title": "حذف مساحة العمل", - "workspace.delete.confirm": 'حذف مساحة العمل "{{name}}"؟', - "workspace.delete.button": "حذف مساحة العمل", - "workspace.reset.title": "إعادة تعيين مساحة العمل", - "workspace.reset.confirm": 'إعادة تعيين مساحة العمل "{{name}}"؟', - "workspace.reset.button": "إعادة تعيين مساحة العمل", - "workspace.reset.archived.none": "لن تتم أرشفة أي جلسات نشطة.", - "workspace.reset.archived.one": "ستتم أرشفة جلسة واحدة.", - "workspace.reset.archived.many": "ستتم أرشفة {{count}} جلسات.", - "workspace.reset.note": "سيؤدي هذا إلى إعادة تعيين مساحة العمل لتتطابق مع الفرع الافتراضي.", - "common.open": "فتح", - "dialog.releaseNotes.action.getStarted": "البدء", - "dialog.releaseNotes.action.next": "التالي", - "dialog.releaseNotes.action.hideFuture": "عدم إظهار هذا في المستقبل", - "dialog.releaseNotes.media.alt": "معاينة الإصدار", - "toast.project.reloadFailed.title": "فشل في إعادة تحميل {{project}}", - "error.server.invalidConfiguration": "تكوين غير صالح", - "common.moreCountSuffix": " (+{{count}} إضافي)", - "common.time.justNow": "الآن", - "common.time.minutesAgo.short": "قبل {{count}} د", - "common.time.hoursAgo.short": "قبل {{count}} س", - "common.time.daysAgo.short": "قبل {{count}} ي", - "settings.providers.connected.environmentDescription": "متصل من متغيرات البيئة الخاصة بك", - "settings.providers.custom.description": "أضف مزود متوافق مع OpenAI بواسطة عنوان URL الأساسي.", - - "app.server.unreachable": "تعذر الوصول إلى {{server}}", - "app.server.retrying": "جاري إعادة المحاولة تلقائيًا...", - "app.server.otherServers": "خوادم أخرى", - "dialog.server.add.usernamePlaceholder": "اسم المستخدم", - "dialog.server.add.passwordPlaceholder": "كلمة المرور", - "server.row.noUsername": "لا يوجد اسم مستخدم", - "session.review.noVcs.createGit.title": "إنشاء مستودع Git", - "session.review.noVcs.createGit.description": "تتبع ومراجعة والتراجع عن التغييرات في هذا المشروع", - "session.review.noVcs.createGit.actionLoading": "جاري إنشاء مستودع Git...", - "session.review.noVcs.createGit.action": "إنشاء مستودع Git", - "session.todo.progress": "تم إكمال {{done}} من {{total}} مهام", - "session.question.progress": "{{current}} من {{total}} أسئلة", - "session.header.open.finder": "Finder", - "session.header.open.fileExplorer": "مستكشف الملفات", - "session.header.open.fileManager": "مدير الملفات", - "session.header.open.app.vscode": "VS Code", - "session.header.open.app.cursor": "Cursor", - "session.header.open.app.zed": "Zed", - "session.header.open.app.textmate": "TextMate", - "session.header.open.app.antigravity": "Antigravity", - "session.header.open.app.terminal": "المحطة الطرفية", - "session.header.open.app.iterm2": "iTerm2", - "session.header.open.app.ghostty": "Ghostty", - "session.header.open.app.warp": "Warp", - "session.header.open.app.xcode": "Xcode", - "session.header.open.app.androidStudio": "Android Studio", - "session.header.open.app.powershell": "PowerShell", - "session.header.open.app.sublimeText": "Sublime Text", - "debugBar.ariaLabel": "تشخيص أداء التطوير", - "debugBar.na": "غير متاح", - "debugBar.nav.label": "NAV", - "debugBar.nav.tip": "آخر انتقال مكتمل للمسار يمس صفحة جلسة، مُقاسًا من بدء التوجيه حتى أول رسم بعد استقراره.", - "debugBar.fps.label": "FPS", - "debugBar.fps.tip": "الإطارات المتجددة في الثانية خلال آخر 5 ثوانٍ.", - "debugBar.frame.label": "FRAME", - "debugBar.frame.tip": "أسوأ وقت للإطار خلال آخر 5 ثوانٍ.", - "debugBar.jank.label": "JANK", - "debugBar.jank.tip": "الإطارات التي تزيد عن 32 مللي ثانية في آخر 5 ثوانٍ.", - "debugBar.long.label": "LONG", - "debugBar.long.tip": "الوقت المحظور وعدد المهام الطويلة في آخر 5 ثوانٍ. أقصى مهمة: {{max}}.", - "debugBar.delay.label": "DELAY", - "debugBar.delay.tip": "أسوأ تأخير إدخال تمت ملاحظته في آخر 5 ثوانٍ.", - "debugBar.inp.label": "INP", - "debugBar.inp.tip": "مدة التفاعل التقريبية خلال آخر 5 ثوانٍ. هذا يشبه INP، وليس Web Vitals INP الرسمي.", - "debugBar.cls.label": "CLS", - "debugBar.cls.tip": "التحول التخطيطي التراكمي لعمر التطبيق الحالي.", - "debugBar.mem.label": "MEM", - "debugBar.mem.tipUnavailable": "كومة JS المستخدمة مقابل حد الكومة. Chromium فقط.", - "debugBar.mem.tip": "كومة JS المستخدمة مقابل حد الكومة. {{used}} من {{limit}}.", - "common.key.ctrl": "Ctrl", - "common.key.alt": "Alt", - "common.key.shift": "Shift", - "common.key.meta": "Meta", - "common.key.space": "Space", - "common.key.backspace": "Backspace", - "common.key.enter": "Enter", - "common.key.tab": "Tab", - "common.key.delete": "Delete", - "common.key.home": "Home", - "common.key.end": "End", - "common.key.pageUp": "Page Up", - "common.key.pageDown": "Page Down", - "common.key.insert": "Insert", - "common.unknown": "غير معروف", - "error.page.circular": "[دائري]", - "error.globalSDK.noServerAvailable": "لا يوجد خادم متاح", - "error.globalSDK.serverNotAvailable": "الخادم غير متاح", - "error.childStore.persistedCacheCreateFailed": "فشل إنشاء ذاكرة التخزين المؤقت الدائمة", - "error.childStore.persistedProjectMetadataCreateFailed": "فشل إنشاء بيانات تعريف المشروع الدائمة", - "error.childStore.persistedProjectIconCreateFailed": "فشل إنشاء أيقونة المشروع الدائمة", - "error.childStore.storeCreateFailed": "فشل إنشاء المخزن", - "terminal.connectionLost.abnormalClose": "تم إغلاق WebSocket بشكل غير طبيعي: {{code}}", -} diff --git a/packages/app/src/i18n/br.ts b/packages/app/src/i18n/br.ts deleted file mode 100644 index 101eb7ce83..0000000000 --- a/packages/app/src/i18n/br.ts +++ /dev/null @@ -1,859 +0,0 @@ -export const dict = { - "command.category.suggested": "Sugerido", - "command.category.view": "Visualizar", - "command.category.project": "Projeto", - "command.category.provider": "Provedor", - "command.category.server": "Servidor", - "command.category.session": "Sessão", - "command.category.theme": "Tema", - "command.category.language": "Idioma", - "command.category.file": "Arquivo", - "command.category.context": "Contexto", - "command.category.terminal": "Terminal", - "command.category.model": "Modelo", - "command.category.mcp": "MCP", - "command.category.agent": "Agente", - "command.category.permissions": "Permissões", - "command.category.workspace": "Espaço de trabalho", - "command.category.settings": "Configurações", - "theme.scheme.system": "Sistema", - "theme.scheme.light": "Claro", - "theme.scheme.dark": "Escuro", - "command.sidebar.toggle": "Alternar barra lateral", - "command.project.open": "Abrir projeto", - "command.provider.connect": "Conectar provedor", - "command.server.switch": "Trocar servidor", - "command.settings.open": "Abrir configurações", - "command.session.previous": "Sessão anterior", - "command.session.next": "Próxima sessão", - "command.session.previous.unseen": "Sessão não lida anterior", - "command.session.next.unseen": "Próxima sessão não lida", - "command.session.archive": "Arquivar sessão", - "command.palette": "Paleta de comandos", - "command.theme.cycle": "Alternar tema", - "command.theme.set": "Usar tema: {{theme}}", - "command.theme.scheme.cycle": "Alternar esquema de cores", - "command.theme.scheme.set": "Usar esquema de cores: {{scheme}}", - "command.language.cycle": "Alternar idioma", - "command.language.set": "Usar idioma: {{language}}", - "command.session.new": "Nova sessão", - "command.file.open": "Abrir arquivo", - "command.tab.close": "Fechar aba", - "command.context.addSelection": "Adicionar seleção ao contexto", - "command.context.addSelection.description": "Adicionar as linhas selecionadas do arquivo atual", - "command.input.focus": "Focar entrada", - "command.terminal.toggle": "Alternar terminal", - "command.fileTree.toggle": "Alternar árvore de arquivos", - "command.review.toggle": "Alternar revisão", - "command.terminal.new": "Novo terminal", - "command.terminal.new.description": "Criar uma nova aba de terminal", - "command.steps.toggle": "Alternar passos", - "command.steps.toggle.description": "Mostrar ou ocultar passos da mensagem atual", - "command.message.previous": "Mensagem anterior", - "command.message.previous.description": "Ir para a mensagem de usuário anterior", - "command.message.next": "Próxima mensagem", - "command.message.next.description": "Ir para a próxima mensagem de usuário", - "command.model.choose": "Escolher modelo", - "command.model.choose.description": "Selecionar um modelo diferente", - "command.mcp.toggle": "Alternar MCPs", - "command.mcp.toggle.description": "Alternar MCPs", - "command.agent.cycle": "Alternar agente", - "command.agent.cycle.description": "Mudar para o próximo agente", - "command.agent.cycle.reverse": "Alternar agente (reverso)", - "command.agent.cycle.reverse.description": "Mudar para o agente anterior", - "command.model.variant.cycle": "Alternar nível de raciocínio", - "command.model.variant.cycle.description": "Mudar para o próximo nível de esforço", - "command.prompt.mode.shell": "Shell", - "command.prompt.mode.normal": "Prompt", - "command.permissions.autoaccept.enable": "Aceitar permissões automaticamente", - "command.permissions.autoaccept.disable": "Parar de aceitar permissões automaticamente", - "command.workspace.toggle": "Alternar espaços de trabalho", - "command.workspace.toggle.description": "Habilitar ou desabilitar múltiplos espaços de trabalho na barra lateral", - "command.session.undo": "Desfazer", - "command.session.undo.description": "Desfazer a última mensagem", - "command.session.redo": "Refazer", - "command.session.redo.description": "Refazer a última mensagem desfeita", - "command.session.compact": "Compactar sessão", - "command.session.compact.description": "Resumir a sessão para reduzir o tamanho do contexto", - "command.session.fork": "Bifurcar da mensagem", - "command.session.fork.description": "Criar uma nova sessão a partir de uma mensagem anterior", - "command.session.share": "Compartilhar sessão", - "command.session.share.description": "Compartilhar esta sessão e copiar a URL para a área de transferência", - "command.session.unshare": "Parar de compartilhar sessão", - "command.session.unshare.description": "Parar de compartilhar esta sessão", - "palette.search.placeholder": "Buscar arquivos, comandos e sessões", - "palette.empty": "Nenhum resultado encontrado", - "palette.group.commands": "Comandos", - "palette.group.files": "Arquivos", - "dialog.provider.search.placeholder": "Buscar provedores", - "dialog.provider.empty": "Nenhum provedor encontrado", - "dialog.provider.group.popular": "Popular", - "dialog.provider.group.other": "Outro", - "dialog.provider.tag.recommended": "Recomendado", - "dialog.provider.opencode.note": "Modelos selecionados incluindo Claude, GPT, Gemini e mais", - "dialog.provider.opencode.tagline": "Modelos otimizados e confiáveis", - "dialog.provider.opencodeGo.tagline": "Assinatura de baixo custo para todos", - "dialog.provider.anthropic.note": "Conectar com Claude Pro/Max ou chave de API", - "dialog.provider.copilot.note": "Conectar com Copilot ou chave de API", - "dialog.provider.openai.note": "Conectar com ChatGPT Pro/Plus ou chave de API", - "dialog.provider.google.note": "Modelos Gemini para respostas rápidas e estruturadas", - "dialog.provider.openrouter.note": "Acesse todos os modelos suportados de um único provedor", - "dialog.provider.vercel.note": "Acesso unificado a modelos de IA com roteamento inteligente", - "dialog.model.select.title": "Selecionar modelo", - "dialog.model.search.placeholder": "Buscar modelos", - "dialog.model.empty": "Nenhum resultado de modelo", - "dialog.model.manage": "Gerenciar modelos", - "dialog.model.manage.description": "Personalizar quais modelos aparecem no seletor de modelos.", - "dialog.model.manage.provider.toggle": "Alternar todos os modelos {{provider}}", - "dialog.model.unpaid.freeModels.title": "Modelos gratuitos fornecidos pelo Kilo", - "dialog.model.unpaid.addMore.title": "Adicionar mais modelos de provedores populares", - "dialog.provider.viewAll": "Ver mais provedores", - "provider.connect.title": "Conectar {{provider}}", - "provider.connect.title.anthropicProMax": "Entrar com Claude Pro/Max", - "provider.connect.selectMethod": "Selecionar método de login para {{provider}}.", - "provider.connect.method.apiKey": "Chave de API", - "provider.connect.status.inProgress": "Autorização em andamento...", - "provider.connect.status.waiting": "Aguardando autorização...", - "provider.connect.status.failed": "Autorização falhou: {{error}}", - "provider.connect.apiKey.description": - "Digite sua chave de API do {{provider}} para conectar sua conta e usar modelos do {{provider}} no Kilo.", - "provider.connect.apiKey.label": "Chave de API do {{provider}}", - "provider.connect.apiKey.placeholder": "Chave de API", - "provider.connect.apiKey.required": "A chave de API é obrigatória", - "provider.connect.opencodeZen.line1": - "OpenCode Zen oferece acesso a um conjunto selecionado de modelos confiáveis otimizados para agentes de código.", - "provider.connect.opencodeZen.line2": - "Com uma única chave de API você terá acesso a modelos como Claude, GPT, Gemini, GLM e mais.", - "provider.connect.opencodeZen.visit.prefix": "Visite ", - "provider.connect.opencodeZen.visit.link": "opencode.ai/zen", - "provider.connect.opencodeZen.visit.suffix": " para obter sua chave de API.", - "provider.connect.oauth.code.visit.prefix": "Visite ", - "provider.connect.oauth.code.visit.link": "este link", - "provider.connect.oauth.code.visit.suffix": - " para obter seu código de autorização e conectar sua conta para usar modelos do {{provider}} no Kilo.", - "provider.connect.oauth.code.label": "Código de autorização {{method}}", - "provider.connect.oauth.code.placeholder": "Código de autorização", - "provider.connect.oauth.code.required": "O código de autorização é obrigatório", - "provider.connect.oauth.code.invalid": "Código de autorização inválido", - "provider.connect.oauth.auto.visit.prefix": "Visite ", - "provider.connect.oauth.auto.visit.link": "este link", - "provider.connect.oauth.auto.visit.suffix": - " e digite o código abaixo para conectar sua conta e usar modelos do {{provider}} no Kilo.", - "provider.connect.oauth.auto.confirmationCode": "Código de confirmação", - "provider.connect.toast.connected.title": "{{provider}} conectado", - "provider.connect.toast.connected.description": "Modelos do {{provider}} agora estão disponíveis para uso.", - "provider.custom.title": "Provedor personalizado", - "provider.custom.description.prefix": "Configure um provedor compatível com OpenAI. Veja a ", - "provider.custom.description.link": "documentação de configuração do provedor", - "provider.custom.description.suffix": ".", - "provider.custom.field.providerID.label": "ID do Provedor", - "provider.custom.field.providerID.placeholder": "meuprovedor", - "provider.custom.field.providerID.description": "Letras minúsculas, números, hifens ou sublinhados", - "provider.custom.field.name.label": "Nome de exibição", - "provider.custom.field.name.placeholder": "Meu Provedor de IA", - "provider.custom.field.baseURL.label": "URL Base", - "provider.custom.field.baseURL.placeholder": "https://api.meuprovedor.com/v1", - "provider.custom.field.apiKey.label": "Chave de API", - "provider.custom.field.apiKey.placeholder": "Chave de API", - "provider.custom.field.apiKey.description": "Opcional. Deixe em branco se gerenciar autenticação via cabeçalhos.", - "provider.custom.models.label": "Modelos", - "provider.custom.models.id.label": "ID", - "provider.custom.models.id.placeholder": "id-do-modelo", - "provider.custom.models.name.label": "Nome", - "provider.custom.models.name.placeholder": "Nome de Exibição", - "provider.custom.models.remove": "Remover modelo", - "provider.custom.models.add": "Adicionar modelo", - "provider.custom.headers.label": "Cabeçalhos (opcional)", - "provider.custom.headers.key.label": "Cabeçalho", - "provider.custom.headers.key.placeholder": "Nome-Do-Cabeçalho", - "provider.custom.headers.value.label": "Valor", - "provider.custom.headers.value.placeholder": "valor", - "provider.custom.headers.remove": "Remover cabeçalho", - "provider.custom.headers.add": "Adicionar cabeçalho", - "provider.custom.error.providerID.required": "ID do Provedor é obrigatório", - "provider.custom.error.providerID.format": "Use letras minúsculas, números, hifens ou sublinhados", - "provider.custom.error.providerID.exists": "Esse ID de provedor já existe", - "provider.custom.error.name.required": "Nome de exibição é obrigatório", - "provider.custom.error.baseURL.required": "URL Base é obrigatória", - "provider.custom.error.baseURL.format": "Deve começar com http:// ou https://", - "provider.custom.error.required": "Obrigatório", - "provider.custom.error.duplicate": "Duplicado", - "provider.disconnect.toast.disconnected.title": "{{provider}} desconectado", - "provider.disconnect.toast.disconnected.description": "Os modelos de {{provider}} não estão mais disponíveis.", - "model.tag.free": "Grátis", - "model.tag.latest": "Mais recente", - "model.provider.anthropic": "Anthropic", - "model.provider.openai": "OpenAI", - "model.provider.google": "Google", - "model.provider.xai": "xAI", - "model.provider.meta": "Meta", - "model.input.text": "texto", - "model.input.image": "imagem", - "model.input.audio": "áudio", - "model.input.video": "vídeo", - "model.input.pdf": "pdf", - "model.tooltip.allows": "Permite: {{inputs}}", - "model.tooltip.reasoning.allowed": "Permite raciocínio", - "model.tooltip.reasoning.none": "Sem raciocínio", - "model.tooltip.context": "Limite de contexto {{limit}}", - "common.search.placeholder": "Buscar", - "common.goBack": "Voltar", - "common.goForward": "Avançar", - "common.loading": "Carregando", - "common.loading.ellipsis": "...", - "common.cancel": "Cancelar", - "common.connect": "Conectar", - "common.disconnect": "Desconectar", - "common.continue": "Enviar", - "common.submit": "Enviar", - "common.save": "Salvar", - "common.saving": "Salvando...", - "common.default": "Padrão", - "common.attachment": "anexo", - "prompt.placeholder.shell": "Digite comando do shell... {{example}}", - "prompt.placeholder.normal": 'Pergunte qualquer coisa... "{{example}}"', - "prompt.placeholder.simple": "Pergunte qualquer coisa...", - "prompt.placeholder.summarizeComments": "Resumir comentários…", - "prompt.placeholder.summarizeComment": "Resumir comentário…", - "prompt.mode.shell": "Shell", - "prompt.mode.normal": "Prompt", - "prompt.mode.shell.exit": "esc para sair", - "prompt.example.1": "Corrigir um TODO no código", - "prompt.example.2": "Qual é a stack tecnológica deste projeto?", - "prompt.example.3": "Corrigir testes quebrados", - "prompt.example.4": "Explicar como funciona a autenticação", - "prompt.example.5": "Encontrar e corrigir vulnerabilidades de segurança", - "prompt.example.6": "Adicionar testes unitários para o serviço de usuário", - "prompt.example.7": "Refatorar esta função para melhor legibilidade", - "prompt.example.8": "O que significa este erro?", - "prompt.example.9": "Me ajude a depurar este problema", - "prompt.example.10": "Gerar documentação da API", - "prompt.example.11": "Otimizar consultas ao banco de dados", - "prompt.example.12": "Adicionar validação de entrada", - "prompt.example.13": "Criar um novo componente para...", - "prompt.example.14": "Como faço o deploy deste projeto?", - "prompt.example.15": "Revisar meu código para boas práticas", - "prompt.example.16": "Adicionar tratamento de erros a esta função", - "prompt.example.17": "Explicar este padrão regex", - "prompt.example.18": "Converter isto para TypeScript", - "prompt.example.19": "Adicionar logging em todo o código", - "prompt.example.20": "Quais dependências estão desatualizadas?", - "prompt.example.21": "Me ajude a escrever um script de migração", - "prompt.example.22": "Implementar cache para este endpoint", - "prompt.example.23": "Adicionar paginação a esta lista", - "prompt.example.24": "Criar um comando CLI para...", - "prompt.example.25": "Como funcionam as variáveis de ambiente aqui?", - "prompt.popover.emptyResults": "Nenhum resultado correspondente", - "prompt.popover.emptyCommands": "Nenhum comando correspondente", - "prompt.dropzone.label": "Arraste imagens, PDFs ou arquivos de texto aqui", - "prompt.dropzone.file.label": "Solte para @mencionar arquivo", - "prompt.slash.badge.custom": "personalizado", - "prompt.slash.badge.skill": "skill", - "prompt.slash.badge.mcp": "mcp", - "prompt.context.active": "ativo", - "prompt.context.includeActiveFile": "Incluir arquivo ativo", - "prompt.context.removeActiveFile": "Remover arquivo ativo do contexto", - "prompt.context.removeFile": "Remover arquivo do contexto", - "prompt.action.attachFile": "Anexar arquivo", - "prompt.attachment.remove": "Remover anexo", - "prompt.action.send": "Enviar", - "prompt.action.stop": "Parar", - "prompt.toast.pasteUnsupported.title": "Anexo não suportado", - "prompt.toast.pasteUnsupported.description": "Apenas imagens, PDFs ou arquivos de texto podem ser anexados aqui.", - "prompt.toast.modelAgentRequired.title": "Selecione um agente e modelo", - "prompt.toast.modelAgentRequired.description": "Escolha um agente e modelo antes de enviar um prompt.", - "prompt.toast.worktreeCreateFailed.title": "Falha ao criar worktree", - "prompt.toast.sessionCreateFailed.title": "Falha ao criar sessão", - "prompt.toast.shellSendFailed.title": "Falha ao enviar comando shell", - "prompt.toast.commandSendFailed.title": "Falha ao enviar comando", - "prompt.toast.promptSendFailed.title": "Falha ao enviar prompt", - "prompt.toast.promptSendFailed.description": "Não foi possível recuperar a sessão", - "dialog.mcp.title": "MCPs", - "dialog.mcp.description": "{{enabled}} of {{total}} habilitados", - "dialog.mcp.empty": "Nenhum MCP configurado", - "dialog.lsp.empty": "LSPs detectados automaticamente pelos tipos de arquivo", - "dialog.plugins.empty": "Plugins configurados em opencode.json", - "mcp.status.connected": "conectado", - "mcp.status.failed": "falhou", - "mcp.status.needs_auth": "precisa de autenticação", - "mcp.status.disabled": "desabilitado", - "dialog.fork.empty": "Nenhuma mensagem para bifurcar", - "dialog.directory.search.placeholder": "Buscar pastas", - "dialog.directory.empty": "Nenhuma pasta encontrada", - "dialog.server.title": "Servidores", - "dialog.server.description": "Trocar para qual servidor Kilo este aplicativo se conecta.", - "dialog.server.search.placeholder": "Buscar servidores", - "dialog.server.empty": "Nenhum servidor ainda", - "dialog.server.add.title": "Adicionar um servidor", - "dialog.server.add.url": "URL do servidor", - "dialog.server.add.placeholder": "http://localhost:4096", - "dialog.server.add.error": "Não foi possível conectar ao servidor", - "dialog.server.add.checking": "Verificando...", - "dialog.server.add.button": "Adicionar", - "dialog.server.add.name": "Nome do servidor (opcional)", - "dialog.server.add.namePlaceholder": "Localhost", - "dialog.server.add.username": "Nome de usuário (opcional)", - "dialog.server.add.password": "Senha (opcional)", - "dialog.server.edit.title": "Editar servidor", - "dialog.server.default.title": "Servidor padrão", - "dialog.server.default.description": - "Conectar a este servidor na inicialização do aplicativo ao invés de iniciar um servidor local. Requer reinicialização.", - "dialog.server.default.none": "Nenhum servidor selecionado", - "dialog.server.default.set": "Definir servidor atual como padrão", - "dialog.server.default.clear": "Limpar", - "dialog.server.action.remove": "Remover servidor", - "dialog.server.menu.edit": "Editar", - "dialog.server.menu.default": "Definir como padrão", - "dialog.server.menu.defaultRemove": "Remover padrão", - "dialog.server.menu.delete": "Excluir", - "dialog.server.current": "Servidor atual", - "dialog.server.status.default": "Padrão", - "dialog.project.edit.title": "Editar projeto", - "dialog.project.edit.name": "Nome", - "dialog.project.edit.icon": "Ícone", - "dialog.project.edit.icon.alt": "Ícone do projeto", - "dialog.project.edit.icon.hint": "Clique ou arraste uma imagem", - "dialog.project.edit.icon.recommended": "Recomendado: 128x128px", - "dialog.project.edit.color": "Cor", - "dialog.project.edit.color.select": "Selecionar cor {{color}}", - "dialog.project.edit.worktree.startup": "Script de inicialização do espaço de trabalho", - "dialog.project.edit.worktree.startup.description": "Executa após criar um novo espaço de trabalho (worktree).", - "dialog.project.edit.worktree.startup.placeholder": "ex: bun install", - "context.breakdown.title": "Detalhamento do Contexto", - "context.breakdown.note": - 'Detalhamento aproximado dos tokens de entrada. "Outros" inclui definições de ferramentas e overhead.', - "context.breakdown.system": "Sistema", - "context.breakdown.user": "Usuário", - "context.breakdown.assistant": "Assistente", - "context.breakdown.tool": "Chamadas de Ferramentas", - "context.breakdown.other": "Outros", - "context.systemPrompt.title": "Prompt do Sistema", - "context.rawMessages.title": "Mensagens brutas", - "context.stats.session": "Sessão", - "context.stats.messages": "Mensagens", - "context.stats.provider": "Provedor", - "context.stats.model": "Modelo", - "context.stats.limit": "Limite de Contexto", - "context.stats.totalTokens": "Total de Tokens", - "context.stats.usage": "Uso", - "context.stats.inputTokens": "Tokens de Entrada", - "context.stats.outputTokens": "Tokens de Saída", - "context.stats.reasoningTokens": "Tokens de Raciocínio", - "context.stats.cacheTokens": "Tokens de Cache (leitura/escrita)", - "context.stats.userMessages": "Mensagens de Usuário", - "context.stats.assistantMessages": "Mensagens do Assistente", - "context.stats.totalCost": "Custo Total", - "context.stats.sessionCreated": "Sessão Criada", - "context.stats.lastActivity": "Última Atividade", - "context.usage.tokens": "Tokens", - "context.usage.usage": "Uso", - "context.usage.cost": "Custo", - "context.usage.clickToView": "Clique para ver o contexto", - "context.usage.view": "Ver uso do contexto", - "language.en": "English", - "language.zh": "简体中文", - "language.zht": "繁體中文", - "language.ko": "한국어", - "language.de": "Deutsch", - "language.es": "Español", - "language.fr": "Français", - "language.da": "Dansk", - "language.ja": "日本語", - "language.pl": "Polski", - "language.ru": "Русский", - "language.ar": "العربية", - "language.no": "Norsk", - "language.br": "Português (Brasil)", - "language.bs": "Bosanski", - "language.th": "ไทย", - "language.tr": "Türkçe", - "toast.language.title": "Idioma", - "toast.language.description": "Alterado para {{language}}", - "toast.theme.title": "Tema alterado", - "toast.scheme.title": "Esquema de cores", - "toast.workspace.enabled.title": "Espaços de trabalho ativados", - "toast.workspace.enabled.description": "Várias worktrees agora são exibidas na barra lateral", - "toast.workspace.disabled.title": "Espaços de trabalho desativados", - "toast.workspace.disabled.description": "Apenas a worktree principal é exibida na barra lateral", - "toast.permissions.autoaccept.on.title": "Aceitando permissões automaticamente", - "toast.permissions.autoaccept.on.description": "Solicitações de permissão serão aprovadas automaticamente", - "toast.permissions.autoaccept.off.title": "Parou de aceitar permissões automaticamente", - "toast.permissions.autoaccept.off.description": "Solicitações de permissão exigirão aprovação", - "toast.model.none.title": "Nenhum modelo selecionado", - "toast.model.none.description": "Conecte um provedor para resumir esta sessão", - "toast.file.loadFailed.title": "Falha ao carregar arquivo", - "toast.file.listFailed.title": "Falha ao listar arquivos", - "toast.context.noLineSelection.title": "Nenhuma seleção de linhas", - "toast.context.noLineSelection.description": "Selecione primeiro um intervalo de linhas em uma aba de arquivo.", - "toast.session.share.copyFailed.title": "Falha ao copiar URL para a área de transferência", - "toast.session.share.success.title": "Sessão compartilhada", - "toast.session.share.success.description": "URL compartilhada copiada para a área de transferência!", - "toast.session.share.failed.title": "Falha ao compartilhar sessão", - "toast.session.share.failed.description": "Ocorreu um erro ao compartilhar a sessão", - "toast.session.unshare.success.title": "Sessão não compartilhada", - "toast.session.unshare.success.description": "Sessão deixou de ser compartilhada com sucesso!", - "toast.session.unshare.failed.title": "Falha ao parar de compartilhar sessão", - "toast.session.unshare.failed.description": "Ocorreu um erro ao parar de compartilhar a sessão", - "toast.session.listFailed.title": "Falha ao carregar sessões para {{project}}", - "toast.update.title": "Atualização disponível", - "toast.update.description": "Uma nova versão do Kilo ({{version}}) está disponível para instalação.", - "toast.update.action.installRestart": "Instalar e reiniciar", - "toast.update.action.notYet": "Agora não", - "error.page.title": "Algo deu errado", - "error.page.description": "Ocorreu um erro ao carregar a aplicação.", - "error.page.details.label": "Detalhes do Erro", - "error.page.action.restart": "Reiniciar", - "error.page.action.checking": "Verificando...", - "error.page.action.checkUpdates": "Verificar atualizações", - "error.page.action.updateTo": "Atualizar para {{version}}", - "error.page.report.prefix": "Por favor, reporte este erro para a equipe do Kilo", - "error.page.report.discord": "no Discord", - "error.page.version": "Versão: {{version}}", - "error.dev.rootNotFound": - "Elemento raiz não encontrado. Você esqueceu de adicioná-lo ao seu index.html? Ou talvez o atributo id foi escrito incorretamente?", - "error.globalSync.connectFailed": "Não foi possível conectar ao servidor. Há um servidor executando em `{{url}}`?", - "directory.error.invalidUrl": "Diretório inválido na URL.", - "error.chain.unknown": "Erro desconhecido", - "error.chain.causedBy": "Causado por:", - "error.chain.apiError": "Erro de API", - "error.chain.status": "Status: {{status}}", - "error.chain.retryable": "Pode tentar novamente: {{retryable}}", - "error.chain.responseBody": "Corpo da resposta:\n{{body}}", - "error.chain.didYouMean": "Você quis dizer: {{suggestions}}", - "error.chain.modelNotFound": "Modelo não encontrado: {{provider}}/{{model}}", - "error.chain.checkConfig": "Verifique os nomes de provedor/modelo na sua configuração (opencode.json)", - "error.chain.mcpFailed": 'Servidor MCP "{{name}}" falhou. Nota: Kilo ainda não suporta autenticação MCP.', - "error.chain.providerAuthFailed": "Autenticação do provedor falhou ({{provider}}): {{message}}", - "error.chain.providerInitFailed": - 'Falha ao inicializar provedor "{{provider}}". Verifique credenciais e configuração.', - "error.chain.configJsonInvalid": "Arquivo de configuração em {{path}} não é um JSON(C) válido", - "error.chain.configJsonInvalidWithMessage": - "Arquivo de configuração em {{path}} não é um JSON(C) válido: {{message}}", - "error.chain.configDirectoryTypo": - 'Diretório "{{dir}}" em {{path}} não é válido. Renomeie o diretório para "{{suggestion}}" ou remova-o. Este é um erro de digitação comum.', - "error.chain.configFrontmatterError": "Falha ao analisar frontmatter em {{path}}:\n{{message}}", - "error.chain.configInvalid": "Arquivo de configuração em {{path}} é inválido", - "error.chain.configInvalidWithMessage": "Arquivo de configuração em {{path}} é inválido: {{message}}", - "notification.permission.title": "Permissão necessária", - "notification.permission.description": "{{sessionTitle}} em {{projectName}} precisa de permissão", - "notification.question.title": "Pergunta", - "notification.question.description": "{{sessionTitle}} em {{projectName}} tem uma pergunta", - "notification.action.goToSession": "Ir para sessão", - "notification.session.responseReady.title": "Resposta pronta", - "notification.session.error.title": "Erro na sessão", - "notification.session.error.fallbackDescription": "Ocorreu um erro", - "home.recentProjects": "Projetos recentes", - "home.empty.title": "Nenhum projeto recente", - "home.empty.description": "Comece abrindo um projeto local", - "session.tab.session": "Sessão", - "session.tab.review": "Revisão", - "session.tab.context": "Contexto", - "session.panel.reviewAndFiles": "Revisão e arquivos", - "session.review.filesChanged": "{{count}} Arquivos Alterados", - "session.review.change.one": "Alteração", - "session.review.change.other": "Alterações", - "session.review.loadingChanges": "Carregando alterações...", - "session.review.empty": "Nenhuma alteração nesta sessão ainda", - "session.review.noVcs": "Nenhum Sistema de Controle de Versão Git detectado, alterações não exibidas", - "session.review.noSnapshot": - "O rastreamento de snapshot está desabilitado na configuração, então as alterações da sessão estão indisponíveis", - "session.review.noChanges": "Sem alterações", - "session.files.selectToOpen": "Selecione um arquivo para abrir", - "session.files.all": "Todos os arquivos", - "session.files.empty": "Nenhum arquivo", - "session.files.binaryContent": "Arquivo binário (conteúdo não pode ser exibido)", - "session.messages.renderEarlier": "Renderizar mensagens anteriores", - "session.messages.loadingEarlier": "Carregando mensagens anteriores...", - "session.messages.loadEarlier": "Carregar mensagens anteriores", - "session.messages.loading": "Carregando mensagens...", - "session.messages.jumpToLatest": "Ir para a mais recente", - "session.context.addToContext": "Adicionar {{selection}} ao contexto", - "session.todo.title": "Tarefas", - "session.todo.collapse": "Recolher", - "session.todo.expand": "Expandir", - "session.followupDock.summary.one": "{{count}} mensagem na fila", - "session.followupDock.summary.other": "{{count}} mensagens na fila", - "session.followupDock.sendNow": "Enviar agora", - "session.followupDock.edit": "Editar", - "session.followupDock.collapse": "Recolher mensagens na fila", - "session.followupDock.expand": "Expandir mensagens na fila", - "session.revertDock.summary.one": "{{count}} mensagem revertida", - "session.revertDock.summary.other": "{{count}} mensagens revertidas", - "session.revertDock.collapse": "Recolher mensagens revertidas", - "session.revertDock.expand": "Expandir mensagens revertidas", - "session.revertDock.restore": "Restaurar mensagem", - "session.new.title": "Crie qualquer coisa", - "session.new.worktree.main": "Branch principal", - "session.new.worktree.mainWithBranch": "Branch principal ({{branch}})", - "session.new.worktree.create": "Criar novo worktree", - "session.new.lastModified": "Última modificação", - "session.header.search.placeholder": "Buscar {{project}}", - "session.header.searchFiles": "Buscar arquivos", - "session.header.openIn": "Abrir em", - "session.header.open.action": "Abrir {{app}}", - "session.header.open.ariaLabel": "Abrir em {{app}}", - "session.header.open.menu": "Opções de abertura", - "session.header.open.copyPath": "Copiar caminho", - "status.popover.trigger": "Status", - "status.popover.ariaLabel": "Configurações de servidores", - "status.popover.tab.servers": "Servidores", - "status.popover.tab.mcp": "MCP", - "status.popover.tab.lsp": "LSP", - "status.popover.tab.plugins": "Plugins", - "status.popover.action.manageServers": "Gerenciar servidores", - "session.share.popover.title": "Publicar na web", - "session.share.popover.description.shared": - "Esta sessão é pública na web. Está acessível para qualquer pessoa com o link.", - "session.share.popover.description.unshared": - "Compartilhar sessão publicamente na web. Estará acessível para qualquer pessoa com o link.", - "session.share.action.share": "Compartilhar", - "session.share.action.publish": "Publicar", - "session.share.action.publishing": "Publicando...", - "session.share.action.unpublish": "Cancelar publicação", - "session.share.action.unpublishing": "Cancelando publicação...", - "session.share.action.view": "Ver", - "session.share.copy.copied": "Copiado", - "session.share.copy.copyLink": "Copiar link", - "lsp.tooltip.none": "Nenhum servidor LSP", - "lsp.label.connected": "{{count}} LSP", - "prompt.loading": "Carregando prompt...", - "terminal.loading": "Carregando terminal...", - "terminal.title": "Terminal", - "terminal.title.numbered": "Terminal {{number}}", - "terminal.close": "Fechar terminal", - "terminal.connectionLost.title": "Conexão Perdida", - "terminal.connectionLost.description": - "A conexão do terminal foi interrompida. Isso pode acontecer quando o servidor reinicia.", - "common.closeTab": "Fechar aba", - "common.dismiss": "Descartar", - "common.requestFailed": "Requisição falhou", - "common.moreOptions": "Mais opções", - "common.learnMore": "Saiba mais", - "common.rename": "Renomear", - "common.reset": "Redefinir", - "common.archive": "Arquivar", - "common.delete": "Excluir", - "common.close": "Fechar", - "common.edit": "Editar", - "common.loadMore": "Carregar mais", - "common.key.esc": "ESC", - "sidebar.menu.toggle": "Alternar menu", - "sidebar.nav.projectsAndSessions": "Projetos e sessões", - "sidebar.settings": "Configurações", - "sidebar.help": "Ajuda", - "sidebar.workspaces.enable": "Habilitar espaços de trabalho", - "sidebar.workspaces.disable": "Desabilitar espaços de trabalho", - "sidebar.gettingStarted.title": "Começando", - "sidebar.gettingStarted.line1": "Kilo inclui modelos gratuitos para você começar imediatamente.", - "sidebar.gettingStarted.line2": "Conecte qualquer provedor para usar modelos, incluindo Claude, GPT, Gemini etc.", - "sidebar.project.recentSessions": "Sessões recentes", - "sidebar.project.viewAllSessions": "Ver todas as sessões", - "sidebar.project.clearNotifications": "Limpar notificações", - "app.name.desktop": "Kilo Desktop", - "settings.section.desktop": "Desktop", - "settings.section.server": "Servidor", - "settings.tab.general": "Geral", - "settings.tab.shortcuts": "Atalhos", - "settings.desktop.section.wsl": "WSL", - "settings.desktop.wsl.title": "WSL integration", - "settings.desktop.wsl.description": "Executar o servidor Kilo dentro do WSL no Windows.", - "settings.general.section.appearance": "Aparência", - "settings.general.section.notifications": "Notificações do sistema", - "settings.general.section.updates": "Atualizações", - "settings.general.section.sounds": "Efeitos sonoros", - "settings.general.section.feed": "Feed", - "settings.general.section.display": "Tela", - "settings.general.row.language.title": "Idioma", - "settings.general.row.language.description": "Alterar o idioma de exibição do Kilo", - "settings.general.row.appearance.title": "Aparência", - "settings.general.row.appearance.description": "Personalize como o Kilo aparece no seu dispositivo", - "settings.general.row.colorScheme.title": "Esquema de cores", - "settings.general.row.colorScheme.description": "Escolha se o Kilo segue o tema do sistema, claro ou escuro", - "settings.general.row.theme.title": "Tema", - "settings.general.row.theme.description": "Personalize como o Kilo é tematizado.", - "settings.general.row.font.title": "Fonte de código", - "settings.general.row.font.description": "Personalize a fonte usada em blocos de código", - "settings.general.row.terminalFont.title": "Terminal Font", - "settings.general.row.terminalFont.description": "Customise the font used in the terminal", - "settings.general.row.uiFont.title": "Fonte da interface", - "settings.general.row.uiFont.description": "Personalize a fonte usada em toda a interface", - "settings.general.row.followup.title": "Comportamento de acompanhamento", - "settings.general.row.followup.description": - "Escolha se os prompts de acompanhamento orientam imediatamente ou esperam na fila", - "settings.general.row.followup.option.queue": "Fila", - "settings.general.row.followup.option.steer": "Orientar", - "settings.general.row.reasoningSummaries.title": "Mostrar resumos de raciocínio", - "settings.general.row.reasoningSummaries.description": "Exibir resumos de raciocínio do modelo na linha do tempo", - "settings.general.row.shellToolPartsExpanded.title": "Expandir partes da ferramenta shell", - "settings.general.row.shellToolPartsExpanded.description": - "Mostrar partes da ferramenta shell expandidas por padrão na linha do tempo", - "settings.general.row.editToolPartsExpanded.title": "Expandir partes da ferramenta de edição", - "settings.general.row.editToolPartsExpanded.description": - "Mostrar partes das ferramentas de edição, escrita e patch expandidas por padrão na linha do tempo", - "settings.general.row.showSessionProgressBar.title": "Mostrar barra de progresso da sessão", - "settings.general.row.showSessionProgressBar.description": - "Exibir a barra de progresso animada no topo da sessão quando o agente estiver trabalhando", - "settings.general.row.wayland.title": "Usar Wayland nativo", - "settings.general.row.wayland.description": "Desabilitar fallback X11 no Wayland. Requer reinicialização.", - "settings.general.row.wayland.tooltip": - "No Linux com monitores de taxas de atualização mistas, Wayland nativo pode ser mais estável.", - "settings.general.row.releaseNotes.title": "Notas da versão", - "settings.general.row.releaseNotes.description": 'Mostrar pop-ups de "Novidades" após atualizações', - "settings.updates.row.startup.title": "Verificar atualizações ao iniciar", - "settings.updates.row.startup.description": "Verificar atualizações automaticamente quando o Kilo iniciar", - "settings.updates.row.check.title": "Verificar atualizações", - "settings.updates.row.check.description": "Verificar atualizações manualmente e instalar se houver", - "settings.updates.action.checkNow": "Verificar agora", - "settings.updates.action.checking": "Verificando...", - "settings.updates.toast.latest.title": "Você está atualizado", - "settings.updates.toast.latest.description": "Você está usando a versão mais recente do Kilo.", - "sound.option.none": "Nenhum", - "sound.option.alert01": "Alerta 01", - "sound.option.alert02": "Alerta 02", - "sound.option.alert03": "Alerta 03", - "sound.option.alert04": "Alerta 04", - "sound.option.alert05": "Alerta 05", - "sound.option.alert06": "Alerta 06", - "sound.option.alert07": "Alerta 07", - "sound.option.alert08": "Alerta 08", - "sound.option.alert09": "Alerta 09", - "sound.option.alert10": "Alerta 10", - "sound.option.bipbop01": "Bip-bop 01", - "sound.option.bipbop02": "Bip-bop 02", - "sound.option.bipbop03": "Bip-bop 03", - "sound.option.bipbop04": "Bip-bop 04", - "sound.option.bipbop05": "Bip-bop 05", - "sound.option.bipbop06": "Bip-bop 06", - "sound.option.bipbop07": "Bip-bop 07", - "sound.option.bipbop08": "Bip-bop 08", - "sound.option.bipbop09": "Bip-bop 09", - "sound.option.bipbop10": "Bip-bop 10", - "sound.option.staplebops01": "Staplebops 01", - "sound.option.staplebops02": "Staplebops 02", - "sound.option.staplebops03": "Staplebops 03", - "sound.option.staplebops04": "Staplebops 04", - "sound.option.staplebops05": "Staplebops 05", - "sound.option.staplebops06": "Staplebops 06", - "sound.option.staplebops07": "Staplebops 07", - "sound.option.nope01": "Não 01", - "sound.option.nope02": "Não 02", - "sound.option.nope03": "Não 03", - "sound.option.nope04": "Não 04", - "sound.option.nope05": "Não 05", - "sound.option.nope06": "Não 06", - "sound.option.nope07": "Não 07", - "sound.option.nope08": "Não 08", - "sound.option.nope09": "Não 09", - "sound.option.nope10": "Não 10", - "sound.option.nope11": "Não 11", - "sound.option.nope12": "Não 12", - "sound.option.yup01": "Sim 01", - "sound.option.yup02": "Sim 02", - "sound.option.yup03": "Sim 03", - "sound.option.yup04": "Sim 04", - "sound.option.yup05": "Sim 05", - "sound.option.yup06": "Sim 06", - "settings.general.notifications.agent.title": "Agente", - "settings.general.notifications.agent.description": - "Mostrar notificação do sistema quando o agente estiver completo ou precisar de atenção", - "settings.general.notifications.permissions.title": "Permissões", - "settings.general.notifications.permissions.description": - "Mostrar notificação do sistema quando uma permissão for necessária", - "settings.general.notifications.errors.title": "Erros", - "settings.general.notifications.errors.description": "Mostrar notificação do sistema quando ocorrer um erro", - "settings.general.sounds.agent.title": "Agente", - "settings.general.sounds.agent.description": "Reproduzir som quando o agente estiver completo ou precisar de atenção", - "settings.general.sounds.permissions.title": "Permissões", - "settings.general.sounds.permissions.description": "Reproduzir som quando uma permissão for necessária", - "settings.general.sounds.errors.title": "Erros", - "settings.general.sounds.errors.description": "Reproduzir som quando ocorrer um erro", - "settings.shortcuts.title": "Atalhos de teclado", - "settings.shortcuts.reset.button": "Redefinir para padrões", - "settings.shortcuts.reset.toast.title": "Atalhos redefinidos", - "settings.shortcuts.reset.toast.description": "Atalhos de teclado foram redefinidos para os padrões.", - "settings.shortcuts.conflict.title": "Atalho já em uso", - "settings.shortcuts.conflict.description": "{{keybind}} já está atribuído a {{titles}}.", - "settings.shortcuts.unassigned": "Não atribuído", - "settings.shortcuts.pressKeys": "Pressione teclas", - "settings.shortcuts.search.placeholder": "Buscar atalhos", - "settings.shortcuts.search.empty": "Nenhum atalho encontrado", - "settings.shortcuts.group.general": "Geral", - "settings.shortcuts.group.session": "Sessão", - "settings.shortcuts.group.navigation": "Navegação", - "settings.shortcuts.group.modelAndAgent": "Modelo e agente", - "settings.shortcuts.group.terminal": "Terminal", - "settings.shortcuts.group.prompt": "Prompt", - "settings.providers.title": "Provedores", - "settings.providers.description": "Configurações de provedores estarão disponíveis aqui.", - "settings.providers.section.connected": "Provedores conectados", - "settings.providers.connected.empty": "Nenhum provedor conectado", - "settings.providers.section.popular": "Provedores populares", - "settings.providers.tag.environment": "Ambiente", - "settings.providers.tag.config": "Configuração", - "settings.providers.tag.custom": "Personalizado", - "settings.providers.tag.other": "Outro", - "settings.models.title": "Modelos", - "settings.models.description": "Configurações de modelos estarão disponíveis aqui.", - "settings.agents.title": "Agentes", - "settings.agents.description": "Configurações de agentes estarão disponíveis aqui.", - "settings.commands.title": "Comandos", - "settings.commands.description": "Configurações de comandos estarão disponíveis aqui.", - "settings.mcp.title": "MCP", - "settings.mcp.description": "Configurações de MCP estarão disponíveis aqui.", - "settings.permissions.title": "Permissões", - "settings.permissions.description": "Controle quais ferramentas o servidor pode usar por padrão.", - "settings.permissions.section.tools": "Ferramentas", - "settings.permissions.toast.updateFailed.title": "Falha ao atualizar permissões", - "settings.permissions.action.allow": "Permitir", - "settings.permissions.action.ask": "Perguntar", - "settings.permissions.action.deny": "Negar", - "settings.permissions.tool.read.title": "Ler", - "settings.permissions.tool.read.description": "Ler um arquivo (corresponde ao caminho do arquivo)", - "settings.permissions.tool.edit.title": "Editar", - "settings.permissions.tool.edit.description": - "Modificar arquivos, incluindo edições, escritas, patches e multi-edições", - "settings.permissions.tool.glob.title": "Glob", - "settings.permissions.tool.glob.description": "Corresponder arquivos usando padrões glob", - "settings.permissions.tool.grep.title": "Grep", - "settings.permissions.tool.grep.description": "Buscar conteúdo de arquivos usando expressões regulares", - "settings.permissions.tool.list.title": "Listar", - "settings.permissions.tool.list.description": "Listar arquivos dentro de um diretório", - "settings.permissions.tool.bash.title": "Bash", - "settings.permissions.tool.bash.description": "Executar comandos shell", - "settings.permissions.tool.task.title": "Tarefa", - "settings.permissions.tool.task.description": "Lançar sub-agentes", - "settings.permissions.tool.skill.title": "Habilidade", - "settings.permissions.tool.skill.description": "Carregar uma habilidade por nome", - "settings.permissions.tool.lsp.title": "LSP", - "settings.permissions.tool.lsp.description": "Executar consultas de servidor de linguagem", - "settings.permissions.tool.todowrite.title": "Escrever Tarefas", - "settings.permissions.tool.todowrite.description": "Atualizar a lista de tarefas", - "settings.permissions.tool.webfetch.title": "Buscar Web", - "settings.permissions.tool.webfetch.description": "Buscar conteúdo de uma URL", - "settings.permissions.tool.websearch.title": "Pesquisa Web", - "settings.permissions.tool.websearch.description": "Pesquisar na web", - "settings.permissions.tool.codesearch.title": "Pesquisa de Código", - "settings.permissions.tool.codesearch.description": "Pesquisar código na web", - "settings.permissions.tool.external_directory.title": "Diretório Externo", - "settings.permissions.tool.external_directory.description": "Acessar arquivos fora do diretório do projeto", - "settings.permissions.tool.doom_loop.title": "Loop Infinito", - "settings.permissions.tool.doom_loop.description": "Detectar chamadas de ferramentas repetidas com entrada idêntica", - "session.delete.failed.title": "Falha ao excluir sessão", - "session.delete.title": "Excluir sessão", - "session.delete.confirm": 'Excluir sessão "{{name}}"?', - "session.delete.button": "Excluir sessão", - "workspace.new": "Novo espaço de trabalho", - "workspace.type.local": "local", - "workspace.type.sandbox": "sandbox", - "workspace.create.failed.title": "Falha ao criar espaço de trabalho", - "workspace.delete.failed.title": "Falha ao excluir espaço de trabalho", - "workspace.resetting.title": "Redefinindo espaço de trabalho", - "workspace.resetting.description": "Isso pode levar um minuto.", - "workspace.reset.failed.title": "Falha ao redefinir espaço de trabalho", - "workspace.reset.success.title": "Espaço de trabalho redefinido", - "workspace.reset.success.description": "Espaço de trabalho agora corresponde ao branch padrão.", - "workspace.error.stillPreparing": "O espaço de trabalho ainda está sendo preparado", - "workspace.status.checking": "Verificando alterações não mescladas...", - "workspace.status.error": "Não foi possível verificar o status do git.", - "workspace.status.clean": "Nenhuma alteração não mesclada detectada.", - "workspace.status.dirty": "Alterações não mescladas detectadas neste espaço de trabalho.", - "workspace.delete.title": "Excluir espaço de trabalho", - "workspace.delete.confirm": 'Excluir espaço de trabalho "{{name}}"?', - "workspace.delete.button": "Excluir espaço de trabalho", - "workspace.reset.title": "Redefinir espaço de trabalho", - "workspace.reset.confirm": 'Redefinir espaço de trabalho "{{name}}"?', - "workspace.reset.button": "Redefinir espaço de trabalho", - "workspace.reset.archived.none": "Nenhuma sessão ativa será arquivada.", - "workspace.reset.archived.one": "1 sessão será arquivada.", - "workspace.reset.archived.many": "{{count}} sessões serão arquivadas.", - "workspace.reset.note": "Isso redefinirá o espaço de trabalho para corresponder ao branch padrão.", - "common.open": "Abrir", - "dialog.releaseNotes.action.getStarted": "Começar", - "dialog.releaseNotes.action.next": "Próximo", - "dialog.releaseNotes.action.hideFuture": "Não mostrar isso no futuro", - "dialog.releaseNotes.media.alt": "Prévia do lançamento", - "toast.project.reloadFailed.title": "Falha ao recarregar {{project}}", - "error.server.invalidConfiguration": "Configuração inválida", - "common.moreCountSuffix": " (+{{count}} mais)", - "common.time.justNow": "Agora mesmo", - "common.time.minutesAgo.short": "{{count}}m atrás", - "common.time.hoursAgo.short": "{{count}}h atrás", - "common.time.daysAgo.short": "{{count}}d atrás", - "settings.providers.connected.environmentDescription": "Conectado a partir de suas variáveis de ambiente", - "settings.providers.custom.description": "Adicionar um provedor compatível com a OpenAI através do URL base.", - - "app.server.unreachable": "Não foi possível conectar a {{server}}", - "app.server.retrying": "Tentando novamente automaticamente...", - "app.server.otherServers": "Outros servidores", - "dialog.server.add.usernamePlaceholder": "nome de usuário", - "dialog.server.add.passwordPlaceholder": "senha", - "server.row.noUsername": "sem nome de usuário", - "session.review.noVcs.createGit.title": "Criar um repositório Git", - "session.review.noVcs.createGit.description": "Rastreie, revise e desfaça alterações neste projeto", - "session.review.noVcs.createGit.actionLoading": "Criando repositório Git...", - "session.review.noVcs.createGit.action": "Criar repositório Git", - "session.todo.progress": "{{done}} de {{total}} tarefas concluídas", - "session.question.progress": "{{current}} de {{total}} perguntas", - "session.header.open.finder": "Finder", - "session.header.open.fileExplorer": "Explorador de Arquivos", - "session.header.open.fileManager": "Gerenciador de Arquivos", - "session.header.open.app.vscode": "VS Code", - "session.header.open.app.cursor": "Cursor", - "session.header.open.app.zed": "Zed", - "session.header.open.app.textmate": "TextMate", - "session.header.open.app.antigravity": "Antigravity", - "session.header.open.app.terminal": "Terminal", - "session.header.open.app.iterm2": "iTerm2", - "session.header.open.app.ghostty": "Ghostty", - "session.header.open.app.warp": "Warp", - "session.header.open.app.xcode": "Xcode", - "session.header.open.app.androidStudio": "Android Studio", - "session.header.open.app.powershell": "PowerShell", - "session.header.open.app.sublimeText": "Sublime Text", - "debugBar.ariaLabel": "Diagnóstico de desempenho de desenvolvimento", - "debugBar.na": "n/a", - "debugBar.nav.label": "NAV", - "debugBar.nav.tip": - "Última transição de rota concluída tocando em uma página de sessão, medida desde o início do roteador até a primeira pintura após o estabelecimento.", - "debugBar.fps.label": "FPS", - "debugBar.fps.tip": "Quadros por segundo nos últimos 5 segundos.", - "debugBar.frame.label": "FRAME", - "debugBar.frame.tip": "Pior tempo de quadro nos últimos 5 segundos.", - "debugBar.jank.label": "JANK", - "debugBar.jank.tip": "Quadros acima de 32ms nos últimos 5 segundos.", - "debugBar.long.label": "LONG", - "debugBar.long.tip": "Tempo bloqueado e contagem de tarefas longas nos últimos 5 segundos. Tarefa máx: {{max}}.", - "debugBar.delay.label": "DELAY", - "debugBar.delay.tip": "Pior atraso de entrada observado nos últimos 5 segundos.", - "debugBar.inp.label": "INP", - "debugBar.inp.tip": - "Duração aproximada da interação nos últimos 5 segundos. Isso é semelhante ao INP, não o INP oficial do Web Vitals.", - "debugBar.cls.label": "CLS", - "debugBar.cls.tip": "Mudança cumulativa de layout para o tempo de vida atual do aplicativo.", - "debugBar.mem.label": "MEM", - "debugBar.mem.tipUnavailable": "Heap JS usado vs limite de heap. Apenas Chromium.", - "debugBar.mem.tip": "Heap JS usado vs limite de heap. {{used}} de {{limit}}.", - "common.key.ctrl": "Ctrl", - "common.key.alt": "Alt", - "common.key.shift": "Shift", - "common.key.meta": "Meta", - "common.key.space": "Espaço", - "common.key.backspace": "Backspace", - "common.key.enter": "Enter", - "common.key.tab": "Tab", - "common.key.delete": "Delete", - "common.key.home": "Home", - "common.key.end": "End", - "common.key.pageUp": "Page Up", - "common.key.pageDown": "Page Down", - "common.key.insert": "Insert", - "common.unknown": "desconhecido", - "error.page.circular": "[Circular]", - "error.globalSDK.noServerAvailable": "Nenhum servidor disponível", - "error.globalSDK.serverNotAvailable": "Servidor indisponível", - "error.childStore.persistedCacheCreateFailed": "Falha ao criar cache persistente", - "error.childStore.persistedProjectMetadataCreateFailed": "Falha ao criar metadados de projeto persistentes", - "error.childStore.persistedProjectIconCreateFailed": "Falha ao criar ícone de projeto persistente", - "error.childStore.storeCreateFailed": "Falha ao criar armazenamento", - "terminal.connectionLost.abnormalClose": "WebSocket fechado anormalmente: {{code}}", -} diff --git a/packages/app/src/i18n/bs.ts b/packages/app/src/i18n/bs.ts deleted file mode 100644 index a52b04c437..0000000000 --- a/packages/app/src/i18n/bs.ts +++ /dev/null @@ -1,935 +0,0 @@ -export const dict = { - "command.category.suggested": "Predloženo", - "command.category.view": "Prikaz", - "command.category.project": "Projekat", - "command.category.provider": "Provajder", - "command.category.server": "Server", - "command.category.session": "Sesija", - "command.category.theme": "Tema", - "command.category.language": "Jezik", - "command.category.file": "Datoteka", - "command.category.context": "Kontekst", - "command.category.terminal": "Terminal", - "command.category.model": "Model", - "command.category.mcp": "MCP", - "command.category.agent": "Agent", - "command.category.permissions": "Dozvole", - "command.category.workspace": "Radni prostor", - "command.category.settings": "Postavke", - - "theme.scheme.system": "Sistem", - "theme.scheme.light": "Svijetlo", - "theme.scheme.dark": "Tamno", - - "command.sidebar.toggle": "Prikaži/sakrij bočnu traku", - "command.project.open": "Otvori projekat", - "command.provider.connect": "Poveži provajdera", - "command.server.switch": "Promijeni server", - "command.settings.open": "Otvori postavke", - "command.session.previous": "Prethodna sesija", - "command.session.next": "Sljedeća sesija", - "command.session.previous.unseen": "Prethodna nepročitana sesija", - "command.session.next.unseen": "Sljedeća nepročitana sesija", - "command.session.archive": "Arhiviraj sesiju", - - "command.palette": "Paleta komandi", - - "command.theme.cycle": "Promijeni temu", - "command.theme.set": "Koristi temu: {{theme}}", - "command.theme.scheme.cycle": "Promijeni šemu boja", - "command.theme.scheme.set": "Koristi šemu boja: {{scheme}}", - - "command.language.cycle": "Promijeni jezik", - "command.language.set": "Koristi jezik: {{language}}", - - "command.session.new": "Nova sesija", - "command.file.open": "Otvori datoteku", - "command.tab.close": "Zatvori karticu", - "command.context.addSelection": "Dodaj odabir u kontekst", - "command.context.addSelection.description": "Dodaj odabrane linije iz trenutne datoteke", - "command.input.focus": "Fokusiraj polje za unos", - "command.terminal.toggle": "Prikaži/sakrij terminal", - "command.fileTree.toggle": "Prikaži/sakrij stablo datoteka", - "command.review.toggle": "Prikaži/sakrij pregled", - "command.terminal.new": "Novi terminal", - "command.terminal.new.description": "Kreiraj novu karticu terminala", - "command.steps.toggle": "Prikaži/sakrij korake", - "command.steps.toggle.description": "Prikaži ili sakrij korake za trenutnu poruku", - "command.message.previous": "Prethodna poruka", - "command.message.previous.description": "Idi na prethodnu korisničku poruku", - "command.message.next": "Sljedeća poruka", - "command.message.next.description": "Idi na sljedeću korisničku poruku", - "command.model.choose": "Odaberi model", - "command.model.choose.description": "Odaberi drugi model", - "command.mcp.toggle": "Prikaži/sakrij MCP-ove", - "command.mcp.toggle.description": "Prikaži/sakrij MCP-ove", - "command.agent.cycle": "Promijeni agenta", - "command.agent.cycle.description": "Prebaci na sljedećeg agenta", - "command.agent.cycle.reverse": "Promijeni agenta unazad", - "command.agent.cycle.reverse.description": "Prebaci na prethodnog agenta", - "command.model.variant.cycle": "Promijeni nivo razmišljanja", - "command.model.variant.cycle.description": "Prebaci na sljedeći nivo", - "command.prompt.mode.shell": "Shell", - "command.prompt.mode.normal": "Prompt", - "command.permissions.autoaccept.enable": "Automatski prihvati dozvole", - "command.permissions.autoaccept.disable": "Zaustavi automatsko prihvatanje dozvola", - "command.workspace.toggle": "Prikaži/sakrij radne prostore", - "command.workspace.toggle.description": "Omogući ili onemogući više radnih prostora u bočnoj traci", - "command.session.undo": "Poništi", - "command.session.undo.description": "Poništi posljednju poruku", - "command.session.redo": "Vrati", - "command.session.redo.description": "Vrati posljednju poništenu poruku", - "command.session.compact": "Sažmi sesiju", - "command.session.compact.description": "Sažmi sesiju kako bi se smanjio kontekst", - "command.session.fork": "Fork iz poruke", - "command.session.fork.description": "Kreiraj novu sesiju iz prethodne poruke", - "command.session.share": "Podijeli sesiju", - "command.session.share.description": "Podijeli ovu sesiju i kopiraj URL u međuspremnik", - "command.session.unshare": "Ukini dijeljenje sesije", - "command.session.unshare.description": "Zaustavi dijeljenje ove sesije", - - "palette.search.placeholder": "Pretraži datoteke, komande i sesije", - "palette.empty": "Nema rezultata", - "palette.group.commands": "Komande", - "palette.group.files": "Datoteke", - - "dialog.provider.search.placeholder": "Pretraži provajdere", - "dialog.provider.empty": "Nema pronađenih provajdera", - "dialog.provider.group.popular": "Popularno", - "dialog.provider.group.other": "Ostalo", - "dialog.provider.tag.recommended": "Preporučeno", - "dialog.provider.opencode.note": "Kurirani modeli uključujući Claude, GPT, Gemini i druge", - "dialog.provider.opencode.tagline": "Pouzdani optimizovani modeli", - "dialog.provider.opencodeGo.tagline": "Povoljna pretplata za sve", - "dialog.provider.anthropic.note": "Direktan pristup Claude modelima, uključujući Pro i Max", - "dialog.provider.copilot.note": "AI modeli za pomoć pri kodiranju putem GitHub Copilot", - "dialog.provider.openai.note": "GPT modeli za brze, sposobne opšte AI zadatke", - "dialog.provider.google.note": "Gemini modeli za brze, strukturirane odgovore", - "dialog.provider.openrouter.note": "Pristup svim podržanim modelima preko jednog provajdera", - "dialog.provider.vercel.note": "Jedinstven pristup AI modelima uz pametno rutiranje", - - "dialog.model.select.title": "Odaberi model", - "dialog.model.search.placeholder": "Pretraži modele", - "dialog.model.empty": "Nema rezultata za modele", - "dialog.model.manage": "Upravljaj modelima", - "dialog.model.manage.description": "Prilagodi koji se modeli prikazuju u izborniku modela.", - "dialog.model.manage.provider.toggle": "Uključi/isključi sve {{provider}} modele", - - "dialog.model.unpaid.freeModels.title": "Besplatni modeli koje obezbjeđuje Kilo", - "dialog.model.unpaid.addMore.title": "Dodaj još modela od popularnih provajdera", - - "dialog.provider.viewAll": "Prikaži više provajdera", - - "provider.connect.title": "Poveži {{provider}}", - "provider.connect.title.anthropicProMax": "Prijavi se putem Claude Pro/Max", - "provider.connect.selectMethod": "Odaberi način prijave za {{provider}}.", - "provider.connect.method.apiKey": "API ključ", - "provider.connect.status.inProgress": "Autorizacija je u toku...", - "provider.connect.status.waiting": "Čekanje na autorizaciju...", - "provider.connect.status.failed": "Autorizacija nije uspjela: {{error}}", - "provider.connect.apiKey.description": - "Unesi svoj {{provider}} API ključ da povežeš račun i koristiš {{provider}} modele u Kilo-u.", - "provider.connect.apiKey.label": "{{provider}} API ključ", - "provider.connect.apiKey.placeholder": "API ključ", - "provider.connect.apiKey.required": "API ključ je obavezan", - "provider.connect.opencodeZen.line1": - "OpenCode Zen ti daje pristup kuriranom skupu pouzdanih, optimizovanih modela za coding agente.", - "provider.connect.opencodeZen.line2": - "Sa jednim API ključem dobijaš pristup modelima kao što su Claude, GPT, Gemini, GLM i drugi.", - "provider.connect.opencodeZen.visit.prefix": "Posjeti ", - "provider.connect.opencodeZen.visit.link": "opencode.ai/zen", - "provider.connect.opencodeZen.visit.suffix": " da preuzmeš svoj API ključ.", - "provider.connect.oauth.code.visit.prefix": "Posjeti ", - "provider.connect.oauth.code.visit.link": "ovaj link", - "provider.connect.oauth.code.visit.suffix": - " da preuzmeš autorizacijski kod i povežeš račun te koristiš {{provider}} modele u Kilo-u.", - "provider.connect.oauth.code.label": "{{method}} autorizacijski kod", - "provider.connect.oauth.code.placeholder": "Autorizacijski kod", - "provider.connect.oauth.code.required": "Autorizacijski kod je obavezan", - "provider.connect.oauth.code.invalid": "Nevažeći autorizacijski kod", - "provider.connect.oauth.auto.visit.prefix": "Posjeti ", - "provider.connect.oauth.auto.visit.link": "ovaj link", - "provider.connect.oauth.auto.visit.suffix": - " i unesi kod ispod da povežeš račun i koristiš {{provider}} modele u Kilo-u.", - "provider.connect.oauth.auto.confirmationCode": "Kod za potvrdu", - "provider.connect.toast.connected.title": "{{provider}} povezan", - "provider.connect.toast.connected.description": "{{provider}} modeli su sada dostupni za korištenje.", - - "provider.custom.title": "Prilagođeni provajder", - "provider.custom.description.prefix": "Konfiguriši OpenAI-kompatibilnog provajdera. Pogledaj ", - "provider.custom.description.link": "dokumentaciju za konfiguraciju provajdera", - "provider.custom.description.suffix": ".", - "provider.custom.field.providerID.label": "ID provajdera", - "provider.custom.field.providerID.placeholder": "mojprovajder", - "provider.custom.field.providerID.description": "Mala slova, brojevi, crtice ili donje crte", - "provider.custom.field.name.label": "Prikazano ime", - "provider.custom.field.name.placeholder": "Moj AI Provajder", - "provider.custom.field.baseURL.label": "Bazni URL", - "provider.custom.field.baseURL.placeholder": "https://api.mojprovajder.com/v1", - "provider.custom.field.apiKey.label": "API ključ", - "provider.custom.field.apiKey.placeholder": "API ključ", - "provider.custom.field.apiKey.description": - "Opcionalno. Ostavi prazno ako upravljaš autentifikacijom putem zaglavlja.", - "provider.custom.models.label": "Modeli", - "provider.custom.models.id.label": "ID", - "provider.custom.models.id.placeholder": "model-id", - "provider.custom.models.name.label": "Ime", - "provider.custom.models.name.placeholder": "Prikazano ime", - "provider.custom.models.remove": "Ukloni model", - "provider.custom.models.add": "Dodaj model", - "provider.custom.headers.label": "Zaglavlja (opcionalno)", - "provider.custom.headers.key.label": "Zaglavlje", - "provider.custom.headers.key.placeholder": "Ime-Zaglavlja", - "provider.custom.headers.value.label": "Vrijednost", - "provider.custom.headers.value.placeholder": "vrijednost", - "provider.custom.headers.remove": "Ukloni zaglavlje", - "provider.custom.headers.add": "Dodaj zaglavlje", - "provider.custom.error.providerID.required": "ID provajdera je obavezan", - "provider.custom.error.providerID.format": "Koristi mala slova, brojeve, crtice ili donje crte", - "provider.custom.error.providerID.exists": "Taj ID provajdera već postoji", - "provider.custom.error.name.required": "Prikazano ime je obavezno", - "provider.custom.error.baseURL.required": "Bazni URL je obavezan", - "provider.custom.error.baseURL.format": "Mora početi sa http:// ili https://", - "provider.custom.error.required": "Obavezno", - "provider.custom.error.duplicate": "Duplikat", - - "provider.disconnect.toast.disconnected.title": "{{provider}} odspojen", - "provider.disconnect.toast.disconnected.description": "{{provider}} modeli više nisu dostupni.", - - "model.tag.free": "Besplatno", - "model.tag.latest": "Najnovije", - "model.provider.anthropic": "Anthropic", - "model.provider.openai": "OpenAI", - "model.provider.google": "Google", - "model.provider.xai": "xAI", - "model.provider.meta": "Meta", - "model.input.text": "tekst", - "model.input.image": "slika", - "model.input.audio": "zvuk", - "model.input.video": "video", - "model.input.pdf": "pdf", - "model.tooltip.allows": "Podržava: {{inputs}}", - "model.tooltip.reasoning.allowed": "Podržava rasuđivanje", - "model.tooltip.reasoning.none": "Bez rasuđivanja", - "model.tooltip.context": "Limit konteksta {{limit}}", - - "common.search.placeholder": "Pretraži", - "common.goBack": "Nazad", - "common.goForward": "Naprijed", - "common.loading": "Učitavanje", - "common.loading.ellipsis": "...", - "common.cancel": "Otkaži", - "common.connect": "Poveži", - "common.disconnect": "Prekini vezu", - "common.continue": "Pošalji", - "common.submit": "Pošalji", - "common.save": "Sačuvaj", - "common.saving": "Čuvanje...", - "common.default": "Podrazumijevano", - "common.attachment": "prilog", - - "prompt.placeholder.shell": "Unesi shell naredbu... {{example}}", - "prompt.placeholder.normal": 'Pitaj bilo šta... "{{example}}"', - "prompt.placeholder.simple": "Pitaj bilo šta...", - "prompt.placeholder.summarizeComments": "Sažmi komentare…", - "prompt.placeholder.summarizeComment": "Sažmi komentar…", - "prompt.mode.shell": "Shell", - "prompt.mode.normal": "Prompt", - "prompt.mode.shell.exit": "esc za izlaz", - - "prompt.example.1": "Popravi TODO u bazi koda", - "prompt.example.2": "Koji je tehnološki stack ovog projekta?", - "prompt.example.3": "Popravi pokvarene testove", - "prompt.example.4": "Objasni kako radi autentifikacija", - "prompt.example.5": "Pronađi i popravi sigurnosne ranjivosti", - "prompt.example.6": "Dodaj jedinične testove za servis korisnika", - "prompt.example.7": "Refaktoriši ovu funkciju da bude čitljivija", - "prompt.example.8": "Šta znači ova greška?", - "prompt.example.9": "Pomozi mi da otklonim ovu grešku", - "prompt.example.10": "Generiši API dokumentaciju", - "prompt.example.11": "Optimizuj upite prema bazi podataka", - "prompt.example.12": "Dodaj validaciju ulaza", - "prompt.example.13": "Napravi novu komponentu za...", - "prompt.example.14": "Kako da deployam ovaj projekat?", - "prompt.example.15": "Pregledaj moj kod prema najboljim praksama", - "prompt.example.16": "Dodaj obradu grešaka u ovu funkciju", - "prompt.example.17": "Objasni ovaj regex obrazac", - "prompt.example.18": "Pretvori ovo u TypeScript", - "prompt.example.19": "Dodaj logovanje kroz cijelu bazu koda", - "prompt.example.20": "Koje su zavisnosti zastarjele?", - "prompt.example.21": "Pomozi mi da napišem migracijsku skriptu", - "prompt.example.22": "Implementiraj keširanje za ovaj endpoint", - "prompt.example.23": "Dodaj paginaciju u ovu listu", - "prompt.example.24": "Napravi CLI komandu za...", - "prompt.example.25": "Kako ovdje rade varijable okruženja?", - - "prompt.popover.emptyResults": "Nema rezultata", - "prompt.popover.emptyCommands": "Nema komandi", - "prompt.dropzone.label": "Ovdje prevucite slike, PDF-ove ili tekstualne datoteke", - "prompt.dropzone.file.label": "Spusti za @spominjanje datoteke", - "prompt.slash.badge.custom": "prilagođeno", - "prompt.slash.badge.skill": "skill", - "prompt.slash.badge.mcp": "mcp", - "prompt.context.active": "aktivno", - "prompt.context.includeActiveFile": "Uključi aktivnu datoteku", - "prompt.context.removeActiveFile": "Ukloni aktivnu datoteku iz konteksta", - "prompt.context.removeFile": "Ukloni datoteku iz konteksta", - "prompt.action.attachFile": "Priloži datoteku", - "prompt.attachment.remove": "Ukloni prilog", - "prompt.action.send": "Pošalji", - "prompt.action.stop": "Zaustavi", - - "prompt.toast.pasteUnsupported.title": "Nepodržan prilog", - "prompt.toast.pasteUnsupported.description": "Ovdje se mogu priložiti samo slike, PDF-ovi ili tekstualne datoteke.", - "prompt.toast.modelAgentRequired.title": "Odaberi agenta i model", - "prompt.toast.modelAgentRequired.description": "Odaberi agenta i model prije slanja upita.", - "prompt.toast.worktreeCreateFailed.title": "Neuspješno kreiranje worktree-a", - "prompt.toast.sessionCreateFailed.title": "Neuspješno kreiranje sesije", - "prompt.toast.shellSendFailed.title": "Neuspješno slanje shell naredbe", - "prompt.toast.commandSendFailed.title": "Neuspješno slanje komande", - "prompt.toast.promptSendFailed.title": "Neuspješno slanje upita", - "prompt.toast.promptSendFailed.description": "Nije moguće dohvatiti sesiju", - - "dialog.mcp.title": "MCP-ovi", - "dialog.mcp.description": "{{enabled}} od {{total}} omogućeno", - "dialog.mcp.empty": "Nema konfigurisnih MCP-ova", - - "dialog.lsp.empty": "LSP-ovi se automatski otkrivaju prema tipu datoteke", - "dialog.plugins.empty": "Plugini su konfigurisani u opencode.json", - - "mcp.status.connected": "povezano", - "mcp.status.failed": "neuspjelo", - "mcp.status.needs_auth": "potrebna autentifikacija", - "mcp.status.disabled": "onemogućeno", - - "dialog.fork.empty": "Nema poruka za fork", - - "dialog.directory.search.placeholder": "Pretraži foldere", - "dialog.directory.empty": "Nema pronađenih foldera", - - "dialog.server.title": "Serveri", - "dialog.server.description": "Promijeni na koji se Kilo server ova aplikacija povezuje.", - "dialog.server.search.placeholder": "Pretraži servere", - "dialog.server.empty": "Još nema servera", - "dialog.server.add.title": "Dodaj server", - "dialog.server.add.url": "URL servera", - "dialog.server.add.placeholder": "http://localhost:4096", - "dialog.server.add.error": "Nije moguće povezati se na server", - "dialog.server.add.checking": "Provjera...", - "dialog.server.add.button": "Dodaj server", - "dialog.server.add.name": "Ime servera (opcionalno)", - "dialog.server.add.namePlaceholder": "Localhost", - "dialog.server.add.username": "Korisničko ime (opcionalno)", - "dialog.server.add.password": "Lozinka (opcionalno)", - "dialog.server.edit.title": "Uredi server", - "dialog.server.default.title": "Podrazumijevani server", - "dialog.server.default.description": - "Poveži se na ovaj server pri pokretanju aplikacije umjesto pokretanja lokalnog servera. Potreban je restart.", - "dialog.server.default.none": "Nije odabran server", - "dialog.server.default.set": "Postavi trenutni server kao podrazumijevani", - "dialog.server.default.clear": "Očisti", - "dialog.server.action.remove": "Ukloni server", - - "dialog.server.menu.edit": "Uredi", - "dialog.server.menu.default": "Postavi kao podrazumijevano", - "dialog.server.menu.defaultRemove": "Ukloni podrazumijevano", - "dialog.server.menu.delete": "Izbriši", - "dialog.server.current": "Trenutni server", - "dialog.server.status.default": "Podrazumijevano", - - "dialog.project.edit.title": "Uredi projekat", - "dialog.project.edit.name": "Naziv", - "dialog.project.edit.icon": "Ikonica", - "dialog.project.edit.icon.alt": "Ikonica projekta", - "dialog.project.edit.icon.hint": "Klikni ili prevuci sliku", - "dialog.project.edit.icon.recommended": "Preporučeno: 128x128px", - "dialog.project.edit.color": "Boja", - "dialog.project.edit.color.select": "Odaberi boju {{color}}", - "dialog.project.edit.worktree.startup": "Skripta za pokretanje radnog prostora", - "dialog.project.edit.worktree.startup.description": "Pokreće se nakon kreiranja novog radnog prostora (worktree).", - "dialog.project.edit.worktree.startup.placeholder": "npr. bun install", - - "context.breakdown.title": "Razlaganje konteksta", - "context.breakdown.note": - 'Približna raspodjela ulaznih tokena. "Ostalo" uključuje definicije alata i dodatni overhead.', - "context.breakdown.system": "Sistem", - "context.breakdown.user": "Korisnik", - "context.breakdown.assistant": "Asistent", - "context.breakdown.tool": "Pozivi alata", - "context.breakdown.other": "Ostalo", - - "context.systemPrompt.title": "Sistemski prompt", - "context.rawMessages.title": "Sirove poruke", - - "context.stats.session": "Sesija", - "context.stats.messages": "Poruke", - "context.stats.provider": "Provajder", - "context.stats.model": "Model", - "context.stats.limit": "Limit konteksta", - "context.stats.totalTokens": "Ukupno tokena", - "context.stats.usage": "Korištenje", - "context.stats.inputTokens": "Ulazni tokeni", - "context.stats.outputTokens": "Izlazni tokeni", - "context.stats.reasoningTokens": "Tokeni za rasuđivanje", - "context.stats.cacheTokens": "Cache tokeni (čitanje/pisanje)", - "context.stats.userMessages": "Korisničke poruke", - "context.stats.assistantMessages": "Poruke asistenta", - "context.stats.totalCost": "Ukupni trošak", - "context.stats.sessionCreated": "Sesija kreirana", - "context.stats.lastActivity": "Posljednja aktivnost", - - "context.usage.tokens": "Tokeni", - "context.usage.usage": "Korištenje", - "context.usage.cost": "Trošak", - "context.usage.clickToView": "Klikni da vidiš kontekst", - "context.usage.view": "Prikaži korištenje konteksta", - - "language.en": "English", - "language.zh": "简体中文", - "language.zht": "繁體中文", - "language.ko": "한국어", - "language.de": "Deutsch", - "language.es": "Español", - "language.fr": "Français", - "language.da": "Dansk", - "language.ja": "日本語", - "language.pl": "Polski", - "language.ru": "Русский", - "language.ar": "العربية", - "language.no": "Norsk", - "language.br": "Português (Brasil)", - "language.bs": "Bosanski", - "language.th": "ไทย", - "language.tr": "Türkçe", - - "toast.language.title": "Jezik", - "toast.language.description": "Prebačeno na {{language}}", - - "toast.theme.title": "Tema promijenjena", - "toast.scheme.title": "Šema boja", - - "toast.workspace.enabled.title": "Radni prostori omogućeni", - "toast.workspace.enabled.description": "Više worktree-ova se sada prikazuje u bočnoj traci", - "toast.workspace.disabled.title": "Radni prostori onemogućeni", - "toast.workspace.disabled.description": "Samo glavni worktree se prikazuje u bočnoj traci", - - "toast.permissions.autoaccept.on.title": "Automatsko prihvatanje dozvola", - "toast.permissions.autoaccept.on.description": "Zahtjevi za dozvole će biti automatski odobreni", - "toast.permissions.autoaccept.off.title": "Zaustavljeno automatsko prihvatanje dozvola", - "toast.permissions.autoaccept.off.description": "Zahtjevi za dozvole će zahtijevati odobrenje", - - "toast.model.none.title": "Nije odabran model", - "toast.model.none.description": "Poveži provajdera da sažmeš ovu sesiju", - - "toast.file.loadFailed.title": "Neuspjelo učitavanje datoteke", - "toast.file.listFailed.title": "Neuspješno listanje datoteka", - - "toast.context.noLineSelection.title": "Nema odabranih linija", - "toast.context.noLineSelection.description": "Prvo odaberi raspon linija u kartici datoteke.", - - "toast.session.share.copyFailed.title": "Neuspjelo kopiranje URL-a u međuspremnik", - "toast.session.share.success.title": "Sesija podijeljena", - "toast.session.share.success.description": "URL za dijeljenje je kopiran u međuspremnik!", - "toast.session.share.failed.title": "Neuspjelo dijeljenje sesije", - "toast.session.share.failed.description": "Došlo je do greške prilikom dijeljenja sesije", - - "toast.session.unshare.success.title": "Dijeljenje sesije ukinuto", - "toast.session.unshare.success.description": "Dijeljenje sesije je uspješno ukinuto!", - "toast.session.unshare.failed.title": "Neuspjelo ukidanje dijeljenja", - "toast.session.unshare.failed.description": "Došlo je do greške prilikom ukidanja dijeljenja", - - "toast.session.listFailed.title": "Neuspjelo učitavanje sesija za {{project}}", - - "toast.update.title": "Dostupno ažuriranje", - "toast.update.description": "Nova verzija Kilo-a ({{version}}) je dostupna za instalaciju.", - "toast.update.action.installRestart": "Instaliraj i restartuj", - "toast.update.action.notYet": "Ne još", - - "error.page.title": "Nešto je pošlo po zlu", - "error.page.description": "Došlo je do greške prilikom učitavanja aplikacije.", - "error.page.details.label": "Detalji greške", - "error.page.action.restart": "Restartuj", - "error.page.action.checking": "Provjera...", - "error.page.action.checkUpdates": "Provjeri ažuriranja", - "error.page.action.updateTo": "Ažuriraj na {{version}}", - "error.page.report.prefix": "Molimo prijavi ovu grešku Kilo timu", - "error.page.report.discord": "na Discordu", - "error.page.version": "Verzija: {{version}}", - - "error.dev.rootNotFound": - "Korijenski element nije pronađen. Da li si zaboravio da ga dodaš u index.html? Ili je možda id atribut pogrešno napisan?", - - "error.globalSync.connectFailed": "Nije moguće povezati se na server. Da li server radi na `{{url}}`?", - "directory.error.invalidUrl": "Nevažeći direktorij u URL-u.", - - "error.chain.unknown": "Nepoznata greška", - "error.chain.causedBy": "Uzrok:", - "error.chain.apiError": "API greška", - "error.chain.status": "Status: {{status}}", - "error.chain.retryable": "Može se ponoviti: {{retryable}}", - "error.chain.responseBody": "Tijelo odgovora:\n{{body}}", - "error.chain.didYouMean": "Da li si mislio: {{suggestions}}", - "error.chain.modelNotFound": "Model nije pronađen: {{provider}}/{{model}}", - "error.chain.checkConfig": "Provjeri konfiguraciju (opencode.json) provider/model names", - "error.chain.mcpFailed": 'MCP server "{{name}}" nije uspio. Napomena: Kilo još ne podržava MCP autentifikaciju.', - "error.chain.providerAuthFailed": "Autentifikacija provajdera nije uspjela ({{provider}}): {{message}}", - "error.chain.providerInitFailed": - 'Neuspjelo inicijalizovanje provajdera "{{provider}}". Provjeri kredencijale i konfiguraciju.', - "error.chain.configJsonInvalid": "Konfiguracijska datoteka na {{path}} nije važeći JSON(C)", - "error.chain.configJsonInvalidWithMessage": "Konfiguracijska datoteka na {{path}} nije važeći JSON(C): {{message}}", - "error.chain.configDirectoryTypo": - 'Direktorij "{{dir}}" u {{path}} nije ispravan. Preimenuj direktorij u "{{suggestion}}" ili ga ukloni. Ovo je česta greška u kucanju.', - "error.chain.configFrontmatterError": "Neuspjelo parsiranje frontmatter-a u {{path}}:\n{{message}}", - "error.chain.configInvalid": "Konfiguracijska datoteka na {{path}} nije ispravna", - "error.chain.configInvalidWithMessage": "Konfiguracijska datoteka na {{path}} nije ispravna: {{message}}", - - "notification.permission.title": "Potrebna dozvola", - "notification.permission.description": "{{sessionTitle}} u {{projectName}} traži dozvolu", - "notification.question.title": "Pitanje", - "notification.question.description": "{{sessionTitle}} u {{projectName}} ima pitanje", - "notification.action.goToSession": "Idi na sesiju", - - "notification.session.responseReady.title": "Odgovor je spreman", - "notification.session.error.title": "Greška sesije", - "notification.session.error.fallbackDescription": "Došlo je do greške", - - "home.recentProjects": "Nedavni projekti", - "home.empty.title": "Nema nedavnih projekata", - "home.empty.description": "Kreni tako što ćeš otvoriti lokalni projekat", - - "session.tab.session": "Sesija", - "session.tab.review": "Pregled", - "session.tab.context": "Kontekst", - "session.panel.reviewAndFiles": "Pregled i datoteke", - "session.review.filesChanged": "Izmijenjeno {{count}} datoteka", - "session.review.change.one": "Izmjena", - "session.review.change.other": "Izmjene", - "session.review.loadingChanges": "Učitavanje izmjena...", - "session.review.empty": "Još nema izmjena u ovoj sesiji", - "session.review.noVcs": "Nije detektovan Git sistem kontrole verzija, promjene se ne prikazuju", - "session.review.noSnapshot": - "Praćenje snimaka (snapshot) je onemogućeno u konfiguraciji, pa promjene sesije nisu dostupne", - "session.review.noChanges": "Nema izmjena", - - "session.files.selectToOpen": "Odaberi datoteku za otvaranje", - "session.files.all": "Sve datoteke", - "session.files.empty": "Nema datoteka", - "session.files.binaryContent": "Binarna datoteka (sadržaj se ne može prikazati)", - - "session.messages.renderEarlier": "Prikaži ranije poruke", - "session.messages.loadingEarlier": "Učitavanje ranijih poruka...", - "session.messages.loadEarlier": "Učitaj ranije poruke", - "session.messages.loading": "Učitavanje poruka...", - "session.messages.jumpToLatest": "Idi na najnovije", - - "session.context.addToContext": "Dodaj {{selection}} u kontekst", - "session.todo.title": "Zadaci", - "session.todo.collapse": "Sažmi", - "session.todo.expand": "Proširi", - "session.followupDock.summary.one": "{{count}} poruka na čekanju", - "session.followupDock.summary.other": "{{count}} poruka na čekanju", - "session.followupDock.sendNow": "Pošalji sada", - "session.followupDock.edit": "Uredi", - "session.followupDock.collapse": "Sažmi poruke na čekanju", - "session.followupDock.expand": "Proširi poruke na čekanju", - "session.revertDock.summary.one": "{{count}} vraćena poruka", - "session.revertDock.summary.other": "{{count}} vraćenih poruka", - "session.revertDock.collapse": "Sažmi vraćene poruke", - "session.revertDock.expand": "Proširi vraćene poruke", - "session.revertDock.restore": "Vrati poruku", - - "session.new.title": "Napravi bilo šta", - "session.new.worktree.main": "Glavna grana", - "session.new.worktree.mainWithBranch": "Glavna grana ({{branch}})", - "session.new.worktree.create": "Kreiraj novi worktree", - "session.new.lastModified": "Posljednja izmjena", - - "session.header.search.placeholder": "Pretraži {{project}}", - "session.header.searchFiles": "Pretraži datoteke", - "session.header.openIn": "Otvori u", - "session.header.open.action": "Otvori {{app}}", - "session.header.open.ariaLabel": "Otvori u {{app}}", - "session.header.open.menu": "Opcije otvaranja", - "session.header.open.copyPath": "Kopiraj putanju", - - "status.popover.trigger": "Status", - "status.popover.ariaLabel": "Konfiguracije servera", - "status.popover.tab.servers": "Serveri", - "status.popover.tab.mcp": "MCP", - "status.popover.tab.lsp": "LSP", - "status.popover.tab.plugins": "Plugini", - "status.popover.action.manageServers": "Upravljaj serverima", - - "session.share.popover.title": "Objavi na webu", - "session.share.popover.description.shared": "Ova sesija je javna na webu. Dostupna je svima koji imaju link.", - "session.share.popover.description.unshared": "Podijeli sesiju javno na webu. Biće dostupna svima koji imaju link.", - "session.share.action.share": "Podijeli", - "session.share.action.publish": "Objavi", - "session.share.action.publishing": "Objavljivanje...", - "session.share.action.unpublish": "Poništi objavu", - "session.share.action.unpublishing": "Poništavanje objave...", - "session.share.action.view": "Prikaži", - "session.share.copy.copied": "Kopirano", - "session.share.copy.copyLink": "Kopiraj link", - - "lsp.tooltip.none": "Nema LSP servera", - "lsp.label.connected": "{{count}} LSP", - - "prompt.loading": "Učitavanje upita...", - "terminal.loading": "Učitavanje terminala...", - "terminal.title": "Terminal", - "terminal.title.numbered": "Terminal {{number}}", - "terminal.close": "Zatvori terminal", - "terminal.connectionLost.title": "Veza prekinuta", - "terminal.connectionLost.description": - "Veza s terminalom je prekinuta. Ovo se može desiti kada se server restartuje.", - - "common.closeTab": "Zatvori karticu", - "common.dismiss": "Odbaci", - "common.requestFailed": "Zahtjev nije uspio", - "common.moreOptions": "Više opcija", - "common.learnMore": "Saznaj više", - "common.rename": "Preimenuj", - "common.reset": "Resetuj", - "common.archive": "Arhiviraj", - "common.delete": "Izbriši", - "common.close": "Zatvori", - "common.edit": "Uredi", - "common.loadMore": "Učitaj još", - "common.key.esc": "ESC", - - "sidebar.menu.toggle": "Prikaži/sakrij meni", - "sidebar.nav.projectsAndSessions": "Projekti i sesije", - "sidebar.settings": "Postavke", - "sidebar.help": "Pomoć", - "sidebar.workspaces.enable": "Omogući radne prostore", - "sidebar.workspaces.disable": "Onemogući radne prostore", - "sidebar.gettingStarted.title": "Početak", - "sidebar.gettingStarted.line1": "Kilo uključuje besplatne modele, tako da možeš odmah početi.", - "sidebar.gettingStarted.line2": "Poveži bilo kojeg provajdera da koristiš modele, npr. Claude, GPT, Gemini itd.", - "sidebar.project.recentSessions": "Nedavne sesije", - "sidebar.project.viewAllSessions": "Prikaži sve sesije", - "sidebar.project.clearNotifications": "Očisti obavijesti", - - "app.name.desktop": "Kilo Desktop", - - "settings.section.desktop": "Desktop", - "settings.section.server": "Server", - "settings.tab.general": "Opšte", - "settings.tab.shortcuts": "Prečice", - "settings.desktop.section.wsl": "WSL", - "settings.desktop.wsl.title": "WSL integracija", - "settings.desktop.wsl.description": "Pokreni Kilo server unutar WSL-a na Windowsu.", - - "settings.general.section.appearance": "Izgled", - "settings.general.section.notifications": "Sistemske obavijesti", - "settings.general.section.updates": "Ažuriranja", - "settings.general.section.sounds": "Zvučni efekti", - "settings.general.section.feed": "Feed", - "settings.general.section.display": "Prikaz", - - "settings.general.row.language.title": "Jezik", - "settings.general.row.language.description": "Promijeni jezik prikaza u Kilo-u", - "settings.general.row.appearance.title": "Izgled", - "settings.general.row.appearance.description": "Prilagodi kako Kilo izgleda na tvom uređaju", - "settings.general.row.colorScheme.title": "Šema boja", - "settings.general.row.colorScheme.description": "Odaberi da li Kilo prati sistemsku, svijetlu ili tamnu temu", - "settings.general.row.theme.title": "Tema", - "settings.general.row.theme.description": "Prilagodi temu Kilo-a.", - "settings.general.row.font.title": "Font za kod", - "settings.general.row.font.description": "Prilagodi font koji se koristi u blokovima koda", - "settings.general.row.terminalFont.title": "Terminal Font", - "settings.general.row.terminalFont.description": "Customise the font used in the terminal", - "settings.general.row.uiFont.title": "UI font", - "settings.general.row.uiFont.description": "Prilagodi font koji se koristi u cijelom interfejsu", - "settings.general.row.followup.title": "Ponašanje nadovezivanja", - "settings.general.row.followup.description": "Odaberi da li upiti nadovezivanja usmjeravaju odmah ili čekaju u redu", - "settings.general.row.followup.option.queue": "Red čekanja", - "settings.general.row.followup.option.steer": "Usmjeri", - "settings.general.row.reasoningSummaries.title": "Prikaži sažetke rasuđivanja", - "settings.general.row.reasoningSummaries.description": "Prikaži sažetke rasuđivanja modela na vremenskoj traci", - - "settings.general.row.shellToolPartsExpanded.title": "Proširi dijelove shell alata", - "settings.general.row.shellToolPartsExpanded.description": - "Prikaži dijelove shell alata podrazumijevano proširene na vremenskoj traci", - "settings.general.row.editToolPartsExpanded.title": "Proširi dijelove alata za uređivanje", - "settings.general.row.editToolPartsExpanded.description": - "Prikaži dijelove alata za uređivanje, pisanje i patch podrazumijevano proširene na vremenskoj traci", - "settings.general.row.showSessionProgressBar.title": "Prikaži traku napretka sesije", - "settings.general.row.showSessionProgressBar.description": - "Prikaži animiranu traku napretka na vrhu sesije kada agent radi", - "settings.general.row.wayland.title": "Koristi nativni Wayland", - "settings.general.row.wayland.description": "Onemogući X11 fallback na Waylandu. Zahtijeva restart.", - "settings.general.row.wayland.tooltip": - "Na Linuxu sa monitorima miješanih stopa osvježavanja, nativni Wayland može biti stabilniji.", - - "settings.general.row.releaseNotes.title": "Bilješke o izdanju", - "settings.general.row.releaseNotes.description": 'Prikaži iskačuće prozore "Šta je novo" nakon ažuriranja', - - "settings.updates.row.startup.title": "Provjeri ažuriranja pri pokretanju", - "settings.updates.row.startup.description": "Automatski provjerava ažuriranja kada se Kilo pokrene", - "settings.updates.row.check.title": "Provjeri ažuriranja", - "settings.updates.row.check.description": "Ručno provjeri ažuriranja i instaliraj ako su dostupna", - "settings.updates.action.checkNow": "Provjeri sada", - "settings.updates.action.checking": "Provjera...", - "settings.updates.toast.latest.title": "Sve je ažurno", - "settings.updates.toast.latest.description": "Koristiš najnoviju verziju Kilo-a.", - "sound.option.none": "Nijedan", - "sound.option.alert01": "Upozorenje 01", - "sound.option.alert02": "Upozorenje 02", - "sound.option.alert03": "Upozorenje 03", - "sound.option.alert04": "Upozorenje 04", - "sound.option.alert05": "Upozorenje 05", - "sound.option.alert06": "Upozorenje 06", - "sound.option.alert07": "Upozorenje 07", - "sound.option.alert08": "Upozorenje 08", - "sound.option.alert09": "Upozorenje 09", - "sound.option.alert10": "Upozorenje 10", - "sound.option.bipbop01": "Bip-bop 01", - "sound.option.bipbop02": "Bip-bop 02", - "sound.option.bipbop03": "Bip-bop 03", - "sound.option.bipbop04": "Bip-bop 04", - "sound.option.bipbop05": "Bip-bop 05", - "sound.option.bipbop06": "Bip-bop 06", - "sound.option.bipbop07": "Bip-bop 07", - "sound.option.bipbop08": "Bip-bop 08", - "sound.option.bipbop09": "Bip-bop 09", - "sound.option.bipbop10": "Bip-bop 10", - "sound.option.staplebops01": "Staplebops 01", - "sound.option.staplebops02": "Staplebops 02", - "sound.option.staplebops03": "Staplebops 03", - "sound.option.staplebops04": "Staplebops 04", - "sound.option.staplebops05": "Staplebops 05", - "sound.option.staplebops06": "Staplebops 06", - "sound.option.staplebops07": "Staplebops 07", - "sound.option.nope01": "Ne 01", - "sound.option.nope02": "Ne 02", - "sound.option.nope03": "Ne 03", - "sound.option.nope04": "Ne 04", - "sound.option.nope05": "Ne 05", - "sound.option.nope06": "Ne 06", - "sound.option.nope07": "Ne 07", - "sound.option.nope08": "Ne 08", - "sound.option.nope09": "Ne 09", - "sound.option.nope10": "Ne 10", - "sound.option.nope11": "Ne 11", - "sound.option.nope12": "Ne 12", - "sound.option.yup01": "Da 01", - "sound.option.yup02": "Da 02", - "sound.option.yup03": "Da 03", - "sound.option.yup04": "Da 04", - "sound.option.yup05": "Da 05", - "sound.option.yup06": "Da 06", - - "settings.general.notifications.agent.title": "Agent", - "settings.general.notifications.agent.description": - "Prikaži sistemsku obavijest kada agent završi ili zahtijeva pažnju", - "settings.general.notifications.permissions.title": "Dozvole", - "settings.general.notifications.permissions.description": "Prikaži sistemsku obavijest kada je potrebna dozvola", - "settings.general.notifications.errors.title": "Greške", - "settings.general.notifications.errors.description": "Prikaži sistemsku obavijest kada dođe do greške", - - "settings.general.sounds.agent.title": "Agent", - "settings.general.sounds.agent.description": "Pusti zvuk kada agent završi ili zahtijeva pažnju", - "settings.general.sounds.permissions.title": "Dozvole", - "settings.general.sounds.permissions.description": "Pusti zvuk kada je potrebna dozvola", - "settings.general.sounds.errors.title": "Greške", - "settings.general.sounds.errors.description": "Pusti zvuk kada dođe do greške", - - "settings.shortcuts.title": "Prečice na tastaturi", - "settings.shortcuts.reset.button": "Vrati na podrazumijevano", - "settings.shortcuts.reset.toast.title": "Prečice resetovane", - "settings.shortcuts.reset.toast.description": "Prečice na tastaturi su vraćene na podrazumijevane.", - "settings.shortcuts.conflict.title": "Prečica je već u upotrebi", - "settings.shortcuts.conflict.description": "{{keybind}} je već dodijeljeno za {{titles}}.", - "settings.shortcuts.unassigned": "Nedodijeljeno", - "settings.shortcuts.pressKeys": "Pritisni tastere", - "settings.shortcuts.search.placeholder": "Pretraži prečice", - "settings.shortcuts.search.empty": "Nema pronađenih prečica", - - "settings.shortcuts.group.general": "Opšte", - "settings.shortcuts.group.session": "Sesija", - "settings.shortcuts.group.navigation": "Navigacija", - "settings.shortcuts.group.modelAndAgent": "Model i agent", - "settings.shortcuts.group.terminal": "Terminal", - "settings.shortcuts.group.prompt": "Upit", - - "settings.providers.title": "Provajderi", - "settings.providers.description": "Postavke provajdera će se ovdje moći podešavati.", - "settings.providers.section.connected": "Povezani provajderi", - "settings.providers.connected.empty": "Nema povezanih provajdera", - "settings.providers.section.popular": "Popularni provajderi", - "settings.providers.tag.environment": "Okruženje", - "settings.providers.tag.config": "Konfiguracija", - "settings.providers.tag.custom": "Prilagođeno", - "settings.providers.tag.other": "Ostalo", - "settings.models.title": "Modeli", - "settings.models.description": "Postavke modela će se ovdje moći podešavati.", - "settings.agents.title": "Agenti", - "settings.agents.description": "Postavke agenata će se ovdje moći podešavati.", - "settings.commands.title": "Komande", - "settings.commands.description": "Postavke komandi će se ovdje moći podešavati.", - "settings.mcp.title": "MCP", - "settings.mcp.description": "MCP postavke će se ovdje moći podešavati.", - - "settings.permissions.title": "Dozvole", - "settings.permissions.description": "Kontroliši koje alate server smije koristiti po defaultu.", - "settings.permissions.section.tools": "Alati", - "settings.permissions.toast.updateFailed.title": "Neuspjelo ažuriranje dozvola", - - "settings.permissions.action.allow": "Dozvoli", - "settings.permissions.action.ask": "Pitaj", - "settings.permissions.action.deny": "Zabrani", - - "settings.permissions.tool.read.title": "Čitanje", - "settings.permissions.tool.read.description": "Čitanje datoteke (podudara se s putanjom datoteke)", - "settings.permissions.tool.edit.title": "Uređivanje", - "settings.permissions.tool.edit.description": - "Mijenjanje datoteka, uključujući izmjene, pisanja, patch-eve i multi-izmjene", - "settings.permissions.tool.glob.title": "Glob", - "settings.permissions.tool.glob.description": "Podudaranje datoteka pomoću glob šablona", - "settings.permissions.tool.grep.title": "Grep", - "settings.permissions.tool.grep.description": "Pretraživanje sadržaja datoteka pomoću regularnih izraza", - "settings.permissions.tool.list.title": "Lista", - "settings.permissions.tool.list.description": "Listanje datoteka unutar direktorija", - "settings.permissions.tool.bash.title": "Bash", - "settings.permissions.tool.bash.description": "Pokretanje shell komandi", - "settings.permissions.tool.task.title": "Zadatak", - "settings.permissions.tool.task.description": "Pokretanje pod-agenta", - "settings.permissions.tool.skill.title": "Vještina", - "settings.permissions.tool.skill.description": "Učitaj vještinu po nazivu", - "settings.permissions.tool.lsp.title": "LSP", - "settings.permissions.tool.lsp.description": "Pokreni upite jezičnog servera", - "settings.permissions.tool.todowrite.title": "Ažuriranje liste zadataka", - "settings.permissions.tool.todowrite.description": "Ažuriraj listu zadataka", - "settings.permissions.tool.webfetch.title": "Web preuzimanje", - "settings.permissions.tool.webfetch.description": "Preuzmi sadržaj sa URL-a", - "settings.permissions.tool.websearch.title": "Web pretraga", - "settings.permissions.tool.websearch.description": "Pretražuj web", - "settings.permissions.tool.codesearch.title": "Pretraga koda", - "settings.permissions.tool.codesearch.description": "Pretraži kod na webu", - "settings.permissions.tool.external_directory.title": "Vanjski direktorij", - "settings.permissions.tool.external_directory.description": "Pristup datotekama izvan direktorija projekta", - "settings.permissions.tool.doom_loop.title": "Beskonačna petlja", - "settings.permissions.tool.doom_loop.description": "Otkriva ponovljene pozive alata sa identičnim unosom", - - "session.delete.failed.title": "Neuspjelo brisanje sesije", - "session.delete.title": "Izbriši sesiju", - "session.delete.confirm": 'Izbriši sesiju "{{name}}"?', - "session.delete.button": "Izbriši sesiju", - - "workspace.new": "Novi radni prostor", - "workspace.type.local": "lokalno", - "workspace.type.sandbox": "sandbox", - "workspace.create.failed.title": "Neuspješno kreiranje radnog prostora", - "workspace.delete.failed.title": "Neuspješno brisanje radnog prostora", - "workspace.resetting.title": "Resetovanje radnog prostora", - "workspace.resetting.description": "Ovo može potrajati minut.", - "workspace.reset.failed.title": "Neuspješno resetovanje radnog prostora", - "workspace.reset.success.title": "Radni prostor resetovan", - "workspace.reset.success.description": "Radni prostor sada odgovara podrazumijevanoj grani.", - "workspace.error.stillPreparing": "Radni prostor se još priprema", - "workspace.status.checking": "Provjera neobjedinjenih promjena...", - "workspace.status.error": "Nije moguće provjeriti git status.", - "workspace.status.clean": "Nisu pronađene neobjedinjene promjene.", - "workspace.status.dirty": "Pronađene su neobjedinjene promjene u ovom radnom prostoru.", - "workspace.delete.title": "Izbriši radni prostor", - "workspace.delete.confirm": 'Izbriši radni prostor "{{name}}"?', - "workspace.delete.button": "Izbriši radni prostor", - "workspace.reset.title": "Resetuj radni prostor", - "workspace.reset.confirm": 'Resetuj radni prostor "{{name}}"?', - "workspace.reset.button": "Resetuj radni prostor", - "workspace.reset.archived.none": "Nijedna aktivna sesija neće biti arhivirana.", - "workspace.reset.archived.one": "1 sesija će biti arhivirana.", - "workspace.reset.archived.many": "Biće arhivirano {{count}} sesija.", - "workspace.reset.note": "Ovo će resetovati radni prostor da odgovara podrazumijevanoj grani.", - "common.open": "Otvori", - "dialog.releaseNotes.action.getStarted": "Započni", - "dialog.releaseNotes.action.next": "Sljedeće", - "dialog.releaseNotes.action.hideFuture": "Ne prikazuj ovo u budućnosti", - "dialog.releaseNotes.media.alt": "Pregled izdanja", - "toast.project.reloadFailed.title": "Nije uspjelo ponovno učitavanje {{project}}", - "error.server.invalidConfiguration": "Nevažeća konfiguracija", - "common.moreCountSuffix": " (+{{count}} više)", - "common.time.justNow": "Upravo sada", - "common.time.minutesAgo.short": "prije {{count}} min", - "common.time.hoursAgo.short": "prije {{count}} h", - "common.time.daysAgo.short": "prije {{count}} d", - "settings.providers.connected.environmentDescription": "Povezano sa vašim varijablama okruženja", - "settings.providers.custom.description": "Dodajte provajdera kompatibilnog s OpenAI putem osnovnog URL-a.", - - "app.server.unreachable": "Nije moguće pristupiti {{server}}", - "app.server.retrying": "Automatski ponovni pokušaj...", - "app.server.otherServers": "Drugi serveri", - "dialog.server.add.usernamePlaceholder": "korisničko ime", - "dialog.server.add.passwordPlaceholder": "lozinka", - "server.row.noUsername": "nema korisničkog imena", - "session.review.noVcs.createGit.title": "Kreiraj Git repozitorij", - "session.review.noVcs.createGit.description": "Pratite, pregledajte i poništite promjene u ovom projektu", - "session.review.noVcs.createGit.actionLoading": "Kreiranje Git repozitorija...", - "session.review.noVcs.createGit.action": "Kreiraj Git repozitorij", - "session.todo.progress": "{{done}} od {{total}} zadataka završeno", - "session.question.progress": "{{current}} od {{total}} pitanja", - "session.header.open.finder": "Finder", - "session.header.open.fileExplorer": "File Explorer", - "session.header.open.fileManager": "File Manager", - "session.header.open.app.vscode": "VS Code", - "session.header.open.app.cursor": "Cursor", - "session.header.open.app.zed": "Zed", - "session.header.open.app.textmate": "TextMate", - "session.header.open.app.antigravity": "Antigravity", - "session.header.open.app.terminal": "Terminal", - "session.header.open.app.iterm2": "iTerm2", - "session.header.open.app.ghostty": "Ghostty", - "session.header.open.app.warp": "Warp", - "session.header.open.app.xcode": "Xcode", - "session.header.open.app.androidStudio": "Android Studio", - "session.header.open.app.powershell": "PowerShell", - "session.header.open.app.sublimeText": "Sublime Text", - "debugBar.ariaLabel": "Dijagnostika performansi razvoja", - "debugBar.na": "n/a", - "debugBar.nav.label": "NAV", - "debugBar.nav.tip": - "Posljednji završeni prelazak rute koji dotiče stranicu sesije, mjeren od početka rutera do prvog iscrtavanja nakon smirivanja.", - "debugBar.fps.label": "FPS", - "debugBar.fps.tip": "Kadrovi u sekundi tokom posljednjih 5 sekundi.", - "debugBar.frame.label": "FRAME", - "debugBar.frame.tip": "Najgore vrijeme kadra u posljednjih 5 sekundi.", - "debugBar.jank.label": "JANK", - "debugBar.jank.tip": "Kadrovi duži od 32ms u posljednjih 5 sekundi.", - "debugBar.long.label": "LONG", - "debugBar.long.tip": "Blokirano vrijeme i broj dugih zadataka u posljednjih 5 sekundi. Maks zadatak: {{max}}.", - "debugBar.delay.label": "DELAY", - "debugBar.delay.tip": "Najgore zabilježeno kašnjenje unosa u posljednjih 5 sekundi.", - "debugBar.inp.label": "INP", - "debugBar.inp.tip": - "Približno trajanje interakcije tokom posljednjih 5 sekundi. Ovo je slično INP-u, nije službeni Web Vitals INP.", - "debugBar.cls.label": "CLS", - "debugBar.cls.tip": "Kumulativni pomak rasporeda za trenutni životni vijek aplikacije.", - "debugBar.mem.label": "MEM", - "debugBar.mem.tipUnavailable": "Korišteni JS heap naspram limita heapa. Samo Chromium.", - "debugBar.mem.tip": "Korišteni JS heap naspram limita heapa. {{used}} od {{limit}}.", - "common.key.ctrl": "Ctrl", - "common.key.alt": "Alt", - "common.key.shift": "Shift", - "common.key.meta": "Meta", - "common.key.space": "Space", - "common.key.backspace": "Backspace", - "common.key.enter": "Enter", - "common.key.tab": "Tab", - "common.key.delete": "Delete", - "common.key.home": "Home", - "common.key.end": "End", - "common.key.pageUp": "Page Up", - "common.key.pageDown": "Page Down", - "common.key.insert": "Insert", - "common.unknown": "nepoznato", - "error.page.circular": "[Kružno]", - "error.globalSDK.noServerAvailable": "Nema dostupnog servera", - "error.globalSDK.serverNotAvailable": "Server nije dostupan", - "error.childStore.persistedCacheCreateFailed": "Nije uspjelo kreiranje trajnog keša", - "error.childStore.persistedProjectMetadataCreateFailed": "Nije uspjelo kreiranje trajnih metapodataka projekta", - "error.childStore.persistedProjectIconCreateFailed": "Nije uspjelo kreiranje trajne ikone projekta", - "error.childStore.storeCreateFailed": "Nije uspjelo kreiranje skladišta", - "terminal.connectionLost.abnormalClose": "WebSocket zatvoren nenormalno: {{code}}", -} diff --git a/packages/app/src/i18n/da.ts b/packages/app/src/i18n/da.ts deleted file mode 100644 index a281c96e42..0000000000 --- a/packages/app/src/i18n/da.ts +++ /dev/null @@ -1,929 +0,0 @@ -export const dict = { - "command.category.suggested": "Foreslået", - "command.category.view": "Vis", - "command.category.project": "Projekt", - "command.category.provider": "Udbyder", - "command.category.server": "Server", - "command.category.session": "Session", - "command.category.theme": "Tema", - "command.category.language": "Sprog", - "command.category.file": "Fil", - "command.category.context": "Kontekst", - "command.category.terminal": "Terminal", - "command.category.model": "Model", - "command.category.mcp": "MCP", - "command.category.agent": "Agent", - "command.category.permissions": "Tilladelser", - "command.category.workspace": "Arbejdsområde", - - "command.category.settings": "Indstillinger", - "theme.scheme.system": "System", - "theme.scheme.light": "Lys", - "theme.scheme.dark": "Mørk", - - "command.sidebar.toggle": "Skift sidebjælke", - "command.project.open": "Åbn projekt", - "command.provider.connect": "Tilslut udbyder", - "command.server.switch": "Skift server", - "command.settings.open": "Åbn indstillinger", - "command.session.previous": "Forrige session", - "command.session.next": "Næste session", - "command.session.previous.unseen": "Forrige ulæste session", - "command.session.next.unseen": "Næste ulæste session", - "command.session.archive": "Arkivér session", - - "command.palette": "Kommandopalette", - - "command.theme.cycle": "Skift tema", - "command.theme.set": "Brug tema: {{theme}}", - "command.theme.scheme.cycle": "Skift farveskema", - "command.theme.scheme.set": "Brug farveskema: {{scheme}}", - - "command.language.cycle": "Skift sprog", - "command.language.set": "Brug sprog: {{language}}", - - "command.session.new": "Ny session", - "command.file.open": "Åbn fil", - "command.tab.close": "Luk fane", - "command.context.addSelection": "Tilføj markering til kontekst", - "command.context.addSelection.description": "Tilføj markerede linjer fra den aktuelle fil", - "command.input.focus": "Fokuser inputfelt", - "command.terminal.toggle": "Skift terminal", - "command.fileTree.toggle": "Skift filtræ", - "command.review.toggle": "Skift gennemgang", - "command.terminal.new": "Ny terminal", - "command.terminal.new.description": "Opret en ny terminalfane", - "command.steps.toggle": "Skift trin", - "command.steps.toggle.description": "Vis eller skjul trin for den aktuelle besked", - "command.message.previous": "Forrige besked", - "command.message.previous.description": "Gå til den forrige brugerbesked", - "command.message.next": "Næste besked", - "command.message.next.description": "Gå til den næste brugerbesked", - "command.model.choose": "Vælg model", - "command.model.choose.description": "Vælg en anden model", - "command.mcp.toggle": "Skift MCP'er", - "command.mcp.toggle.description": "Skift MCP'er", - "command.agent.cycle": "Skift agent", - "command.agent.cycle.description": "Skift til næste agent", - "command.agent.cycle.reverse": "Skift agent baglæns", - "command.agent.cycle.reverse.description": "Skift til forrige agent", - "command.model.variant.cycle": "Skift tænkeindsats", - "command.model.variant.cycle.description": "Skift til næste indsatsniveau", - "command.prompt.mode.shell": "Shell", - "command.prompt.mode.normal": "Prompt", - "command.permissions.autoaccept.enable": "Accepter tilladelser automatisk", - "command.permissions.autoaccept.disable": "Stop med at acceptere tilladelser automatisk", - "command.workspace.toggle": "Skift arbejdsområder", - "command.workspace.toggle.description": "Aktiver eller deaktiver flere arbejdsområder i sidebjælken", - "command.session.undo": "Fortryd", - "command.session.undo.description": "Fortryd den sidste besked", - "command.session.redo": "Omgør", - "command.session.redo.description": "Omgør den sidste fortrudte besked", - "command.session.compact": "Komprimér session", - "command.session.compact.description": "Opsummer sessionen for at reducere kontekststørrelsen", - "command.session.fork": "Forgren fra besked", - "command.session.fork.description": "Opret en ny session fra en tidligere besked", - "command.session.share": "Del session", - "command.session.share.description": "Del denne session og kopier URL'en til udklipsholderen", - "command.session.unshare": "Stop deling af session", - "command.session.unshare.description": "Stop med at dele denne session", - - "palette.search.placeholder": "Søg i filer, kommandoer og sessioner", - "palette.empty": "Ingen resultater fundet", - "palette.group.commands": "Kommandoer", - "palette.group.files": "Filer", - - "dialog.provider.search.placeholder": "Søg udbydere", - "dialog.provider.empty": "Ingen udbydere fundet", - "dialog.provider.group.popular": "Populære", - "dialog.provider.group.other": "Andre", - "dialog.provider.tag.recommended": "Anbefalet", - "dialog.provider.opencode.note": "Udvalgte modeller inklusive Claude, GPT, Gemini og flere", - "dialog.provider.opencode.tagline": "Pålidelige optimerede modeller", - "dialog.provider.opencodeGo.tagline": "Billigt abonnement for alle", - "dialog.provider.anthropic.note": "Direkte adgang til Claude-modeller, inklusive Pro og Max", - "dialog.provider.copilot.note": "AI-modeller til kodningsassistance via GitHub Copilot", - "dialog.provider.openai.note": "GPT-modeller til hurtige, kompetente generelle AI-opgaver", - "dialog.provider.google.note": "Gemini-modeller til hurtige, strukturerede svar", - "dialog.provider.openrouter.note": "Få adgang til alle understøttede modeller fra én udbyder", - "dialog.provider.vercel.note": "Samlet adgang til AI-modeller med smart routing", - - "dialog.model.select.title": "Vælg model", - "dialog.model.search.placeholder": "Søg modeller", - "dialog.model.empty": "Ingen modeller fundet", - "dialog.model.manage": "Administrer modeller", - "dialog.model.manage.description": "Tilpas hvilke modeller der vises i modelvælgeren.", - "dialog.model.manage.provider.toggle": "Skift alle {{provider}}-modeller", - - "dialog.model.unpaid.freeModels.title": "Gratis modeller leveret af Kilo", - "dialog.model.unpaid.addMore.title": "Tilføj flere modeller fra populære udbydere", - - "dialog.provider.viewAll": "Vis flere udbydere", - - "provider.connect.title": "Forbind {{provider}}", - "provider.connect.title.anthropicProMax": "Log ind med Claude Pro/Max", - "provider.connect.selectMethod": "Vælg loginmetode for {{provider}}.", - "provider.connect.method.apiKey": "API-nøgle", - "provider.connect.status.inProgress": "Godkendelse i gang...", - "provider.connect.status.waiting": "Venter på godkendelse...", - "provider.connect.status.failed": "Godkendelse mislykkedes: {{error}}", - "provider.connect.apiKey.description": - "Indtast din {{provider}} API-nøgle for at forbinde din konto og bruge {{provider}} modeller i Kilo.", - "provider.connect.apiKey.label": "{{provider}} API-nøgle", - "provider.connect.apiKey.placeholder": "API-nøgle", - "provider.connect.apiKey.required": "API-nøgle er påkrævet", - "provider.connect.opencodeZen.line1": - "OpenCode Zen giver dig adgang til et udvalg af pålidelige optimerede modeller til kodningsagenter.", - "provider.connect.opencodeZen.line2": - "Med en enkelt API-nøgle får du adgang til modeller som Claude, GPT, Gemini, GLM og flere.", - "provider.connect.opencodeZen.visit.prefix": "Besøg ", - "provider.connect.opencodeZen.visit.link": "opencode.ai/zen", - "provider.connect.opencodeZen.visit.suffix": " for at hente din API-nøgle.", - "provider.connect.oauth.code.visit.prefix": "Besøg ", - "provider.connect.oauth.code.visit.link": "dette link", - "provider.connect.oauth.code.visit.suffix": - " for at hente din godkendelseskode for at forbinde din konto og bruge {{provider}} modeller i Kilo.", - "provider.connect.oauth.code.label": "{{method}} godkendelseskode", - "provider.connect.oauth.code.placeholder": "Godkendelseskode", - "provider.connect.oauth.code.required": "Godkendelseskode er påkrævet", - "provider.connect.oauth.code.invalid": "Ugyldig godkendelseskode", - "provider.connect.oauth.auto.visit.prefix": "Besøg ", - "provider.connect.oauth.auto.visit.link": "dette link", - "provider.connect.oauth.auto.visit.suffix": - " og indtast koden nedenfor for at forbinde din konto og bruge {{provider}} modeller i Kilo.", - "provider.connect.oauth.auto.confirmationCode": "Bekræftelseskode", - "provider.connect.toast.connected.title": "{{provider}} forbundet", - "provider.connect.toast.connected.description": "{{provider}} modeller er nu tilgængelige.", - - "provider.custom.title": "Brugerdefineret udbyder", - "provider.custom.description.prefix": "Konfigurer en OpenAI-kompatibel udbyder. Se ", - "provider.custom.description.link": "dokumentation for udbyderkonfiguration", - "provider.custom.description.suffix": ".", - "provider.custom.field.providerID.label": "Udbyder-ID", - "provider.custom.field.providerID.placeholder": "minudbyder", - "provider.custom.field.providerID.description": "Små bogstaver, tal, bindestreger eller understregninger", - "provider.custom.field.name.label": "Visningsnavn", - "provider.custom.field.name.placeholder": "Min AI-udbyder", - "provider.custom.field.baseURL.label": "Basis-URL", - "provider.custom.field.baseURL.placeholder": "https://api.minudbyder.dk/v1", - "provider.custom.field.apiKey.label": "API-nøgle", - "provider.custom.field.apiKey.placeholder": "API-nøgle", - "provider.custom.field.apiKey.description": "Valgfri. Lad være tom, hvis du administrerer godkendelse via headers.", - "provider.custom.models.label": "Modeller", - "provider.custom.models.id.label": "ID", - "provider.custom.models.id.placeholder": "model-id", - "provider.custom.models.name.label": "Navn", - "provider.custom.models.name.placeholder": "Visningsnavn", - "provider.custom.models.remove": "Fjern model", - "provider.custom.models.add": "Tilføj model", - "provider.custom.headers.label": "Headers (valgfri)", - "provider.custom.headers.key.label": "Header", - "provider.custom.headers.key.placeholder": "Header-Navn", - "provider.custom.headers.value.label": "Værdi", - "provider.custom.headers.value.placeholder": "værdi", - "provider.custom.headers.remove": "Fjern header", - "provider.custom.headers.add": "Tilføj header", - "provider.custom.error.providerID.required": "Udbyder-ID er påkrævet", - "provider.custom.error.providerID.format": "Brug små bogstaver, tal, bindestreger eller understregninger", - "provider.custom.error.providerID.exists": "Dette udbyder-ID findes allerede", - "provider.custom.error.name.required": "Visningsnavn er påkrævet", - "provider.custom.error.baseURL.required": "Basis-URL er påkrævet", - "provider.custom.error.baseURL.format": "Skal starte med http:// eller https://", - "provider.custom.error.required": "Påkrævet", - "provider.custom.error.duplicate": "Duplikeret", - - "provider.disconnect.toast.disconnected.title": "{{provider}} frakoblet", - "provider.disconnect.toast.disconnected.description": "Modeller fra {{provider}} er ikke længere tilgængelige.", - "model.tag.free": "Gratis", - "model.tag.latest": "Nyeste", - - "model.provider.anthropic": "Anthropic", - "model.provider.openai": "OpenAI", - "model.provider.google": "Google", - "model.provider.xai": "xAI", - "model.provider.meta": "Meta", - "model.input.text": "tekst", - "model.input.image": "billede", - "model.input.audio": "lyd", - "model.input.video": "video", - "model.input.pdf": "pdf", - "model.tooltip.allows": "Tillader: {{inputs}}", - "model.tooltip.reasoning.allowed": "Tillader tænkning", - "model.tooltip.reasoning.none": "Ingen tænkning", - "model.tooltip.context": "Kontekstgrænse {{limit}}", - "common.search.placeholder": "Søg", - "common.goBack": "Gå tilbage", - "common.goForward": "Naviger fremad", - "common.loading": "Indlæser", - "common.loading.ellipsis": "...", - "common.cancel": "Annuller", - "common.connect": "Forbind", - "common.disconnect": "Frakobl", - "common.continue": "Indsend", - "common.submit": "Indsend", - "common.save": "Gem", - "common.saving": "Gemmer...", - "common.default": "Standard", - "common.attachment": "vedhæftning", - - "prompt.placeholder.shell": "Indtast shell-kommando... {{example}}", - "prompt.placeholder.normal": 'Spørg om hvad som helst... "{{example}}"', - "prompt.placeholder.simple": "Spørg om hvad som helst...", - "prompt.placeholder.summarizeComments": "Opsummér kommentarer…", - "prompt.placeholder.summarizeComment": "Opsummér kommentar…", - "prompt.mode.shell": "Shell", - "prompt.mode.normal": "Prompt", - "prompt.mode.shell.exit": "esc for at afslutte", - - "prompt.example.1": "Ret en TODO i koden", - "prompt.example.2": "Hvad er teknologistakken for dette projekt?", - "prompt.example.3": "Ret ødelagte tests", - "prompt.example.4": "Forklar hvordan godkendelse fungerer", - "prompt.example.5": "Find og ret sikkerhedshuller", - "prompt.example.6": "Tilføj enhedstests for brugerservice", - "prompt.example.7": "Refaktorer denne funktion så den er mere læsbar", - "prompt.example.8": "Hvad betyder denne fejl?", - "prompt.example.9": "Hjælp mig med at debugge dette problem", - "prompt.example.10": "Generer API-dokumentation", - "prompt.example.11": "Optimer databaseforespørgsler", - "prompt.example.12": "Tilføj validering af input", - "prompt.example.13": "Opret en ny komponent til...", - "prompt.example.14": "Hvordan deployerer jeg dette projekt?", - "prompt.example.15": "Gennemgå min kode for bedste praksis", - "prompt.example.16": "Tilføj fejlhåndtering til denne funktion", - "prompt.example.17": "Forklar dette regex-mønster", - "prompt.example.18": "Konverter dette til TypeScript", - "prompt.example.19": "Tilføj logning i hele koden", - "prompt.example.20": "Hvilke afhængigheder er forældede?", - "prompt.example.21": "Hjælp mig med at skrive et migreringsscript", - "prompt.example.22": "Implementer caching for dette endpoint", - "prompt.example.23": "Tilføj sideinddeling til denne liste", - "prompt.example.24": "Opret en CLI-kommando til...", - "prompt.example.25": "Hvordan fungerer miljøvariabler her?", - - "prompt.popover.emptyResults": "Ingen matchende resultater", - "prompt.popover.emptyCommands": "Ingen matchende kommandoer", - "prompt.dropzone.label": "Slip billeder, PDF'er eller tekstfiler her", - "prompt.dropzone.file.label": "Slip for at @nævne fil", - "prompt.slash.badge.custom": "brugerdefineret", - "prompt.slash.badge.skill": "skill", - "prompt.slash.badge.mcp": "mcp", - "prompt.context.active": "aktiv", - "prompt.context.includeActiveFile": "Inkluder aktiv fil", - "prompt.context.removeActiveFile": "Fjern aktiv fil fra kontekst", - "prompt.context.removeFile": "Fjern fil fra kontekst", - "prompt.action.attachFile": "Vedhæft fil", - "prompt.attachment.remove": "Fjern vedhæftning", - "prompt.action.send": "Send", - "prompt.action.stop": "Stop", - - "prompt.toast.pasteUnsupported.title": "Ikke understøttet vedhæftning", - "prompt.toast.pasteUnsupported.description": "Kun billeder, PDF'er eller tekstfiler kan vedhæftes her.", - "prompt.toast.modelAgentRequired.title": "Vælg en agent og model", - "prompt.toast.modelAgentRequired.description": "Vælg en agent og model før du sender en forespørgsel.", - "prompt.toast.worktreeCreateFailed.title": "Kunne ikke oprette worktree", - "prompt.toast.sessionCreateFailed.title": "Kunne ikke oprette session", - "prompt.toast.shellSendFailed.title": "Kunne ikke sende shell-kommando", - "prompt.toast.commandSendFailed.title": "Kunne ikke sende kommando", - "prompt.toast.promptSendFailed.title": "Kunne ikke sende forespørgsel", - "prompt.toast.promptSendFailed.description": "Kunne ikke hente session", - - "dialog.mcp.title": "MCP'er", - "dialog.mcp.description": "{{enabled}} af {{total}} aktiveret", - "dialog.mcp.empty": "Ingen MCP'er konfigureret", - - "dialog.lsp.empty": "LSP'er registreret automatisk fra filtyper", - "dialog.plugins.empty": "Plugins konfigureret i opencode.json", - - "mcp.status.connected": "forbundet", - "mcp.status.failed": "mislykkedes", - "mcp.status.needs_auth": "kræver godkendelse", - "mcp.status.disabled": "deaktiveret", - - "dialog.fork.empty": "Ingen beskeder at forgrene fra", - - "dialog.directory.search.placeholder": "Søg mapper", - "dialog.directory.empty": "Ingen mapper fundet", - - "dialog.server.title": "Servere", - "dialog.server.description": "Skift hvilken Kilo-server denne app forbinder til.", - "dialog.server.search.placeholder": "Søg servere", - "dialog.server.empty": "Ingen servere endnu", - "dialog.server.add.title": "Tilføj en server", - "dialog.server.add.url": "Server URL", - "dialog.server.add.placeholder": "http://localhost:4096", - "dialog.server.add.error": "Kunne ikke forbinde til server", - "dialog.server.add.checking": "Tjekker...", - "dialog.server.add.button": "Tilføj server", - "dialog.server.add.name": "Servernavn (valgfrit)", - "dialog.server.add.namePlaceholder": "Localhost", - "dialog.server.add.username": "Brugernavn (valgfrit)", - "dialog.server.add.password": "Adgangskode (valgfrit)", - "dialog.server.edit.title": "Rediger server", - "dialog.server.default.title": "Standardserver", - "dialog.server.default.description": - "Forbind til denne server ved start af app i stedet for at starte en lokal server. Kræver genstart.", - "dialog.server.default.none": "Ingen server valgt", - "dialog.server.default.set": "Sæt nuværende server som standard", - "dialog.server.default.clear": "Ryd", - "dialog.server.action.remove": "Fjern server", - - "dialog.server.menu.edit": "Rediger", - "dialog.server.menu.default": "Sæt som standard", - "dialog.server.menu.defaultRemove": "Fjern som standard", - "dialog.server.menu.delete": "Slet", - "dialog.server.current": "Nuværende server", - "dialog.server.status.default": "Standard", - - "dialog.project.edit.title": "Rediger projekt", - "dialog.project.edit.name": "Navn", - "dialog.project.edit.icon": "Ikon", - "dialog.project.edit.icon.alt": "Projektikon", - "dialog.project.edit.icon.hint": "Klik eller træk et billede", - "dialog.project.edit.icon.recommended": "Anbefalet: 128x128px", - "dialog.project.edit.color": "Farve", - "dialog.project.edit.color.select": "Vælg farven {{color}}", - - "dialog.project.edit.worktree.startup": "Opstartsscript for arbejdsområde", - "dialog.project.edit.worktree.startup.description": "Køres efter oprettelse af et nyt arbejdsområde (worktree).", - "dialog.project.edit.worktree.startup.placeholder": "f.eks. bun install", - "context.breakdown.title": "Kontekstfordeling", - "context.breakdown.note": - 'Omtrentlig fordeling af input-tokens. "Andre" inkluderer værktøjsdefinitioner og overhead.', - "context.breakdown.system": "System", - "context.breakdown.user": "Bruger", - "context.breakdown.assistant": "Assistent", - "context.breakdown.tool": "Værktøjskald", - "context.breakdown.other": "Andre", - - "context.systemPrompt.title": "Systemprompt", - "context.rawMessages.title": "Rå beskeder", - - "context.stats.session": "Session", - "context.stats.messages": "Beskeder", - "context.stats.provider": "Udbyder", - "context.stats.model": "Model", - "context.stats.limit": "Kontekstgrænse", - "context.stats.totalTokens": "Samlede tokens", - "context.stats.usage": "Forbrug", - "context.stats.inputTokens": "Input-tokens", - "context.stats.outputTokens": "Output-tokens", - "context.stats.reasoningTokens": "Tænke Tokens", - "context.stats.cacheTokens": "Cache Tokens (læs/skriv)", - "context.stats.userMessages": "Brugerbeskeder", - "context.stats.assistantMessages": "Assistentbeskeder", - "context.stats.totalCost": "Samlede omkostninger", - "context.stats.sessionCreated": "Session oprettet", - "context.stats.lastActivity": "Seneste aktivitet", - - "context.usage.tokens": "Tokens", - "context.usage.usage": "Forbrug", - "context.usage.cost": "Omkostning", - "context.usage.clickToView": "Klik for at se kontekst", - "context.usage.view": "Se kontekstforbrug", - - "language.en": "English", - "language.zh": "简体中文", - "language.zht": "繁體中文", - "language.ko": "한국어", - "language.de": "Deutsch", - "language.es": "Español", - "language.fr": "Français", - "language.da": "Dansk", - "language.ja": "日本語", - "language.pl": "Polski", - "language.ru": "Русский", - "language.ar": "العربية", - "language.no": "Norsk", - "language.br": "Português (Brasil)", - "language.bs": "Bosanski", - "language.th": "ไทย", - "language.tr": "Türkçe", - - "toast.language.title": "Sprog", - "toast.language.description": "Skiftede til {{language}}", - - "toast.theme.title": "Tema skiftet", - "toast.scheme.title": "Farveskema", - - "toast.permissions.autoaccept.on.title": "Accepterer tilladelser automatisk", - "toast.permissions.autoaccept.on.description": "Anmodninger om tilladelse godkendes automatisk", - "toast.permissions.autoaccept.off.title": "Stoppet med at acceptere tilladelser automatisk", - "toast.permissions.autoaccept.off.description": "Anmodninger om tilladelse vil kræve godkendelse", - - "toast.workspace.enabled.title": "Arbejdsområder aktiveret", - "toast.workspace.enabled.description": "Flere worktrees vises nu i sidepanelet", - "toast.workspace.disabled.title": "Arbejdsområder deaktiveret", - "toast.workspace.disabled.description": "Kun hoved-worktree vises i sidepanelet", - - "toast.model.none.title": "Ingen model valgt", - "toast.model.none.description": "Forbind en udbyder for at opsummere denne session", - - "toast.file.loadFailed.title": "Kunne ikke indlæse fil", - - "toast.file.listFailed.title": "Kunne ikke liste filer", - "toast.context.noLineSelection.title": "Ingen linjevalg", - "toast.context.noLineSelection.description": "Vælg først et linjeinterval i en filfane.", - "toast.session.share.copyFailed.title": "Kunne ikke kopiere URL til udklipsholder", - "toast.session.share.success.title": "Session delt", - "toast.session.share.success.description": "Delings-URL kopieret til udklipsholder!", - "toast.session.share.failed.title": "Kunne ikke dele session", - "toast.session.share.failed.description": "Der opstod en fejl under deling af sessionen", - - "toast.session.unshare.success.title": "Deling af session stoppet", - "toast.session.unshare.success.description": "Deling af session blev stoppet!", - "toast.session.unshare.failed.title": "Kunne ikke stoppe deling af session", - "toast.session.unshare.failed.description": "Der opstod en fejl under stop af sessionsdeling", - - "toast.session.listFailed.title": "Kunne ikke indlæse sessioner for {{project}}", - - "toast.update.title": "Opdatering tilgængelig", - "toast.update.description": "En ny version af Kilo ({{version}}) er nu tilgængelig til installation.", - "toast.update.action.installRestart": "Installer og genstart", - "toast.update.action.notYet": "Ikke endnu", - - "error.page.title": "Noget gik galt", - "error.page.description": "Der opstod en fejl under indlæsning af applikationen.", - "error.page.details.label": "Fejldetaljer", - "error.page.action.restart": "Genstart", - "error.page.action.checking": "Tjekker...", - "error.page.action.checkUpdates": "Tjek for opdateringer", - "error.page.action.updateTo": "Opdater til {{version}}", - "error.page.report.prefix": "Rapporter venligst denne fejl til Kilo-teamet", - "error.page.report.discord": "på Discord", - "error.page.version": "Version: {{version}}", - - "error.dev.rootNotFound": - "Rodelement ikke fundet. Har du glemt at tilføje det til din index.html? Eller måske er id-attributten stavet forkert?", - - "error.globalSync.connectFailed": "Kunne ikke forbinde til server. Kører der en server på `{{url}}`?", - "directory.error.invalidUrl": "Ugyldig mappe i URL.", - - "error.chain.unknown": "Ukendt fejl", - "error.chain.causedBy": "Forårsaget af:", - "error.chain.apiError": "API-fejl", - "error.chain.status": "Status: {{status}}", - "error.chain.retryable": "Kan forsøges igen: {{retryable}}", - "error.chain.responseBody": "Svarindhold:\n{{body}}", - "error.chain.didYouMean": "Mente du: {{suggestions}}", - "error.chain.modelNotFound": "Model ikke fundet: {{provider}}/{{model}}", - "error.chain.checkConfig": "Tjek dine konfigurations (opencode.json) udbyder/modelnavne", - "error.chain.mcpFailed": 'MCP-server "{{name}}" fejlede. Bemærk, Kilo understøtter ikke MCP-godkendelse endnu.', - "error.chain.providerAuthFailed": "Udbydergodkendelse mislykkedes ({{provider}}): {{message}}", - "error.chain.providerInitFailed": - 'Kunne ikke initialisere udbyder "{{provider}}". Tjek legitimationsoplysninger og konfiguration.', - "error.chain.configJsonInvalid": "Konfigurationsfil på {{path}} er ikke gyldig JSON(C)", - "error.chain.configJsonInvalidWithMessage": "Konfigurationsfil på {{path}} er ikke gyldig JSON(C): {{message}}", - "error.chain.configDirectoryTypo": - 'Mappe "{{dir}}" i {{path}} er ikke gyldig. Omdøb mappen til "{{suggestion}}" eller fjern den. Dette er en almindelig slåfejl.', - "error.chain.configFrontmatterError": "Kunne ikke parse frontmatter i {{path}}:\n{{message}}", - "error.chain.configInvalid": "Konfigurationsfil på {{path}} er ugyldig", - "error.chain.configInvalidWithMessage": "Konfigurationsfil på {{path}} er ugyldig: {{message}}", - - "notification.permission.title": "Tilladelse påkrævet", - "notification.permission.description": "{{sessionTitle}} i {{projectName}} kræver tilladelse", - "notification.question.title": "Spørgsmål", - "notification.question.description": "{{sessionTitle}} i {{projectName}} har et spørgsmål", - "notification.action.goToSession": "Gå til session", - - "notification.session.responseReady.title": "Svar klar", - "notification.session.error.title": "Sessionsfejl", - "notification.session.error.fallbackDescription": "Der opstod en fejl", - - "home.recentProjects": "Seneste projekter", - "home.empty.title": "Ingen seneste projekter", - "home.empty.description": "Kom i gang ved at åbne et lokalt projekt", - - "session.tab.session": "Session", - "session.tab.review": "Gennemgang", - "session.tab.context": "Kontekst", - "session.panel.reviewAndFiles": "Gennemgang og filer", - "session.review.filesChanged": "{{count}} Filer ændret", - "session.review.change.one": "Ændring", - "session.review.change.other": "Ændringer", - "session.review.loadingChanges": "Indlæser ændringer...", - "session.review.empty": "Ingen ændringer i denne session endnu", - "session.review.noVcs": "Intet Git versionsstyringssystem fundet, ændringer vises ikke", - "session.review.noSnapshot": - "Snapshot-sporing er deaktiveret i konfigurationen, så sessionsændringer er ikke tilgængelige", - "session.review.noChanges": "Ingen ændringer", - "session.files.selectToOpen": "Vælg en fil at åbne", - "session.files.all": "Alle filer", - "session.files.empty": "Ingen filer", - "session.files.binaryContent": "Binær fil (indhold kan ikke vises)", - "session.messages.renderEarlier": "Vis tidligere beskeder", - "session.messages.loadingEarlier": "Indlæser tidligere beskeder...", - "session.messages.loadEarlier": "Indlæs tidligere beskeder", - "session.messages.loading": "Indlæser beskeder...", - - "session.messages.jumpToLatest": "Gå til seneste", - "session.context.addToContext": "Tilføj {{selection}} til kontekst", - "session.todo.title": "Opgaver", - "session.todo.collapse": "Skjul", - "session.todo.expand": "Udvid", - "session.followupDock.summary.one": "{{count}} besked i kø", - "session.followupDock.summary.other": "{{count}} beskeder i kø", - "session.followupDock.sendNow": "Send nu", - "session.followupDock.edit": "Rediger", - "session.followupDock.collapse": "Skjul beskeder i kø", - "session.followupDock.expand": "Udvid beskeder i kø", - "session.revertDock.summary.one": "{{count}} tilbagerullet besked", - "session.revertDock.summary.other": "{{count}} tilbagerullede beskeder", - "session.revertDock.collapse": "Skjul tilbagerullede beskeder", - "session.revertDock.expand": "Udvid tilbagerullede beskeder", - "session.revertDock.restore": "Gendan besked", - - "session.new.title": "Byg hvad som helst", - "session.new.worktree.main": "Hovedgren", - "session.new.worktree.mainWithBranch": "Hovedgren ({{branch}})", - "session.new.worktree.create": "Opret nyt worktree", - "session.new.lastModified": "Sidst ændret", - - "session.header.search.placeholder": "Søg {{project}}", - "session.header.searchFiles": "Søg efter filer", - "session.header.openIn": "Åbn i", - "session.header.open.action": "Åbn {{app}}", - "session.header.open.ariaLabel": "Åbn i {{app}}", - "session.header.open.menu": "Åbningsmuligheder", - "session.header.open.copyPath": "Kopier sti", - - "status.popover.trigger": "Status", - "status.popover.ariaLabel": "Serverkonfigurationer", - "status.popover.tab.servers": "Servere", - "status.popover.tab.mcp": "MCP", - "status.popover.tab.lsp": "LSP", - "status.popover.tab.plugins": "Plugins", - "status.popover.action.manageServers": "Administrer servere", - - "session.share.popover.title": "Udgiv på nettet", - "session.share.popover.description.shared": - "Denne session er offentlig på nettet. Den er tilgængelig for alle med linket.", - "session.share.popover.description.unshared": - "Del session offentligt på nettet. Den vil være tilgængelig for alle med linket.", - "session.share.action.share": "Del", - "session.share.action.publish": "Udgiv", - "session.share.action.publishing": "Udgiver...", - "session.share.action.unpublish": "Afpublicer", - "session.share.action.unpublishing": "Afpublicerer...", - "session.share.action.view": "Vis", - "session.share.copy.copied": "Kopieret", - "session.share.copy.copyLink": "Kopier link", - - "lsp.tooltip.none": "Ingen LSP-servere", - "lsp.label.connected": "{{count}} LSP", - - "prompt.loading": "Indlæser prompt...", - "terminal.loading": "Indlæser terminal...", - "terminal.title": "Terminal", - "terminal.title.numbered": "Terminal {{number}}", - "terminal.close": "Luk terminal", - - "terminal.connectionLost.title": "Forbindelse mistet", - "terminal.connectionLost.description": "Terminalforbindelsen blev afbrudt. Dette kan ske, når serveren genstarter.", - "common.closeTab": "Luk fane", - "common.dismiss": "Afvis", - "common.requestFailed": "Forespørgsel mislykkedes", - "common.moreOptions": "Flere muligheder", - "common.learnMore": "Lær mere", - "common.rename": "Omdøb", - "common.reset": "Nulstil", - "common.archive": "Arkivér", - "common.delete": "Slet", - "common.close": "Luk", - "common.edit": "Rediger", - "common.loadMore": "Indlæs flere", - - "common.key.esc": "ESC", - "sidebar.menu.toggle": "Skift menu", - "sidebar.nav.projectsAndSessions": "Projekter og sessioner", - "sidebar.settings": "Indstillinger", - "sidebar.help": "Hjælp", - "sidebar.workspaces.enable": "Aktiver arbejdsområder", - "sidebar.workspaces.disable": "Deaktiver arbejdsområder", - "sidebar.gettingStarted.title": "Kom i gang", - "sidebar.gettingStarted.line1": "Kilo inkluderer gratis modeller så du kan starte med det samme.", - "sidebar.gettingStarted.line2": "Forbind enhver udbyder for at bruge modeller, inkl. Claude, GPT, Gemini osv.", - "sidebar.project.recentSessions": "Seneste sessioner", - "sidebar.project.viewAllSessions": "Vis alle sessioner", - "sidebar.project.clearNotifications": "Ryd notifikationer", - - "app.name.desktop": "Kilo Desktop", - "settings.section.desktop": "Desktop", - "settings.section.server": "Server", - "settings.tab.general": "Generelt", - "settings.tab.shortcuts": "Genveje", - "settings.desktop.section.wsl": "WSL", - "settings.desktop.wsl.title": "WSL integration", - "settings.desktop.wsl.description": "Kør Kilo-serveren inde i WSL på Windows.", - - "settings.general.section.appearance": "Udseende", - "settings.general.section.notifications": "Systemmeddelelser", - "settings.general.section.updates": "Opdateringer", - "settings.general.section.sounds": "Lydeffekter", - "settings.general.section.feed": "Feed", - "settings.general.section.display": "Skærm", - - "settings.general.row.language.title": "Sprog", - "settings.general.row.language.description": "Ændr visningssproget for Kilo", - "settings.general.row.appearance.title": "Udseende", - "settings.general.row.appearance.description": "Tilpas hvordan Kilo ser ud på din enhed", - "settings.general.row.colorScheme.title": "Farveskema", - "settings.general.row.colorScheme.description": "Vælg om Kilo følger systemets, lyst eller mørkt tema", - "settings.general.row.theme.title": "Tema", - "settings.general.row.theme.description": "Tilpas hvordan Kilo er temabestemt.", - "settings.general.row.font.title": "Kode-skrifttype", - "settings.general.row.font.description": "Tilpas skrifttypen, der bruges i kodeblokke", - "settings.general.row.terminalFont.title": "Terminal Font", - "settings.general.row.terminalFont.description": "Customise the font used in the terminal", - "settings.general.row.uiFont.title": "UI-skrifttype", - "settings.general.row.uiFont.description": "Tilpas skrifttypen, der bruges i hele brugerfladen", - "settings.general.row.followup.title": "Opfølgningsadfærd", - "settings.general.row.followup.description": "Vælg om opfølgende forespørgsler skal styre straks eller vente i kø", - "settings.general.row.followup.option.queue": "Kø", - "settings.general.row.followup.option.steer": "Styr", - "settings.general.row.reasoningSummaries.title": "Vis tænkeoversigter", - "settings.general.row.reasoningSummaries.description": "Vis model tænkeoversigter i tidslinjen", - - "settings.general.row.shellToolPartsExpanded.title": "Udvid shell-værktøjsdele", - "settings.general.row.shellToolPartsExpanded.description": "Vis shell-værktøjsdele udvidet som standard i tidslinjen", - "settings.general.row.editToolPartsExpanded.title": "Udvid edit-værktøjsdele", - "settings.general.row.editToolPartsExpanded.description": - "Vis edit-, write- og patch-værktøjsdele udvidet som standard i tidslinjen", - "settings.general.row.showSessionProgressBar.title": "Vis sessionens fremdriftslinje", - "settings.general.row.showSessionProgressBar.description": - "Vis den animerede fremdriftslinje øverst i sessionen, når agenten arbejder", - "settings.general.row.wayland.title": "Brug native Wayland", - "settings.general.row.wayland.description": "Deaktiver X11-fallback på Wayland. Kræver genstart.", - "settings.general.row.wayland.tooltip": - "På Linux med skærme med blandet opdateringshastighed kan native Wayland være mere stabilt.", - - "settings.general.row.releaseNotes.title": "Udgivelsesnoter", - "settings.general.row.releaseNotes.description": 'Vis "Hvad er nyt"-popups efter opdateringer', - - "settings.updates.row.startup.title": "Tjek for opdateringer ved opstart", - "settings.updates.row.startup.description": "Tjek automatisk for opdateringer, når Kilo starter", - "settings.updates.row.check.title": "Tjek for opdateringer", - "settings.updates.row.check.description": "Tjek manuelt for opdateringer og installer, hvis tilgængelig", - "settings.updates.action.checkNow": "Tjek nu", - "settings.updates.action.checking": "Tjekker...", - "settings.updates.toast.latest.title": "Du er opdateret", - "settings.updates.toast.latest.description": "Du kører den nyeste version af Kilo.", - - "sound.option.none": "Ingen", - "sound.option.alert01": "Alarm 01", - "sound.option.alert02": "Alarm 02", - "sound.option.alert03": "Alarm 03", - "sound.option.alert04": "Alarm 04", - "sound.option.alert05": "Alarm 05", - "sound.option.alert06": "Alarm 06", - "sound.option.alert07": "Alarm 07", - "sound.option.alert08": "Alarm 08", - "sound.option.alert09": "Alarm 09", - "sound.option.alert10": "Alarm 10", - "sound.option.bipbop01": "Bip-bop 01", - "sound.option.bipbop02": "Bip-bop 02", - "sound.option.bipbop03": "Bip-bop 03", - "sound.option.bipbop04": "Bip-bop 04", - "sound.option.bipbop05": "Bip-bop 05", - "sound.option.bipbop06": "Bip-bop 06", - "sound.option.bipbop07": "Bip-bop 07", - "sound.option.bipbop08": "Bip-bop 08", - "sound.option.bipbop09": "Bip-bop 09", - "sound.option.bipbop10": "Bip-bop 10", - "sound.option.staplebops01": "Staplebops 01", - "sound.option.staplebops02": "Staplebops 02", - "sound.option.staplebops03": "Staplebops 03", - "sound.option.staplebops04": "Staplebops 04", - "sound.option.staplebops05": "Staplebops 05", - "sound.option.staplebops06": "Staplebops 06", - "sound.option.staplebops07": "Staplebops 07", - "sound.option.nope01": "Nej 01", - "sound.option.nope02": "Nej 02", - "sound.option.nope03": "Nej 03", - "sound.option.nope04": "Nej 04", - "sound.option.nope05": "Nej 05", - "sound.option.nope06": "Nej 06", - "sound.option.nope07": "Nej 07", - "sound.option.nope08": "Nej 08", - "sound.option.nope09": "Nej 09", - "sound.option.nope10": "Nej 10", - "sound.option.nope11": "Nej 11", - "sound.option.nope12": "Nej 12", - "sound.option.yup01": "Ja 01", - "sound.option.yup02": "Ja 02", - "sound.option.yup03": "Ja 03", - "sound.option.yup04": "Ja 04", - "sound.option.yup05": "Ja 05", - "sound.option.yup06": "Ja 06", - "settings.general.notifications.agent.title": "Agent", - "settings.general.notifications.agent.description": - "Vis systemmeddelelse når agenten er færdig eller kræver opmærksomhed", - "settings.general.notifications.permissions.title": "Tilladelser", - "settings.general.notifications.permissions.description": "Vis systemmeddelelse når en tilladelse er påkrævet", - "settings.general.notifications.errors.title": "Fejl", - "settings.general.notifications.errors.description": "Vis systemmeddelelse når der opstår en fejl", - - "settings.general.sounds.agent.title": "Agent", - "settings.general.sounds.agent.description": "Afspil lyd når agenten er færdig eller kræver opmærksomhed", - "settings.general.sounds.permissions.title": "Tilladelser", - "settings.general.sounds.permissions.description": "Afspil lyd når en tilladelse er påkrævet", - "settings.general.sounds.errors.title": "Fejl", - "settings.general.sounds.errors.description": "Afspil lyd når der opstår en fejl", - - "settings.shortcuts.title": "Tastaturgenveje", - "settings.shortcuts.reset.button": "Nulstil til standard", - "settings.shortcuts.reset.toast.title": "Genveje nulstillet", - "settings.shortcuts.reset.toast.description": "Tastaturgenveje er blevet nulstillet til standard.", - "settings.shortcuts.conflict.title": "Genvej allerede i brug", - "settings.shortcuts.conflict.description": "{{keybind}} er allerede tildelt til {{titles}}.", - "settings.shortcuts.unassigned": "Ikke tildelt", - "settings.shortcuts.pressKeys": "Tryk på taster", - "settings.shortcuts.search.placeholder": "Søg genveje", - "settings.shortcuts.search.empty": "Ingen genveje fundet", - - "settings.shortcuts.group.general": "Generelt", - "settings.shortcuts.group.session": "Session", - "settings.shortcuts.group.navigation": "Navigation", - "settings.shortcuts.group.modelAndAgent": "Model og agent", - "settings.shortcuts.group.terminal": "Terminal", - "settings.shortcuts.group.prompt": "Prompt", - - "settings.providers.title": "Udbydere", - "settings.providers.description": "Udbyderindstillinger vil kunne konfigureres her.", - "settings.providers.section.connected": "Forbundne udbydere", - "settings.providers.connected.empty": "Ingen forbundne udbydere", - "settings.providers.section.popular": "Populære udbydere", - "settings.providers.tag.environment": "Miljø", - "settings.providers.tag.config": "Konfiguration", - "settings.providers.tag.custom": "Brugerdefineret", - "settings.providers.tag.other": "Andet", - "settings.models.title": "Modeller", - "settings.models.description": "Modelindstillinger vil kunne konfigureres her.", - "settings.agents.title": "Agenter", - "settings.agents.description": "Agentindstillinger vil kunne konfigureres her.", - "settings.commands.title": "Kommandoer", - "settings.commands.description": "Kommandoindstillinger vil kunne konfigureres her.", - "settings.mcp.title": "MCP", - "settings.mcp.description": "MCP-indstillinger vil kunne konfigureres her.", - - "settings.permissions.title": "Tilladelser", - "settings.permissions.description": "Styr hvilke værktøjer serveren kan bruge som standard.", - "settings.permissions.section.tools": "Værktøjer", - "settings.permissions.toast.updateFailed.title": "Kunne ikke opdatere tilladelser", - - "settings.permissions.action.allow": "Tillad", - "settings.permissions.action.ask": "Spørg", - "settings.permissions.action.deny": "Afvis", - - "settings.permissions.tool.read.title": "Læs", - "settings.permissions.tool.read.description": "Læsning af en fil (matcher filstien)", - "settings.permissions.tool.edit.title": "Rediger", - "settings.permissions.tool.edit.description": - "Ændre filer, herunder redigeringer, skrivninger, patches og multi-redigeringer", - "settings.permissions.tool.glob.title": "Glob", - "settings.permissions.tool.glob.description": "Match filer ved hjælp af glob-mønstre", - "settings.permissions.tool.grep.title": "Grep", - "settings.permissions.tool.grep.description": "Søg i filindhold ved hjælp af regulære udtryk", - "settings.permissions.tool.list.title": "Liste", - "settings.permissions.tool.list.description": "List filer i en mappe", - "settings.permissions.tool.bash.title": "Bash", - "settings.permissions.tool.bash.description": "Kør shell-kommandoer", - "settings.permissions.tool.task.title": "Opgave", - "settings.permissions.tool.task.description": "Start underagenter", - "settings.permissions.tool.skill.title": "Færdighed", - "settings.permissions.tool.skill.description": "Indlæs en færdighed efter navn", - "settings.permissions.tool.lsp.title": "LSP", - "settings.permissions.tool.lsp.description": "Kør sprogserverforespørgsler", - "settings.permissions.tool.todowrite.title": "Skriv To-do", - "settings.permissions.tool.todowrite.description": "Opdater to-do listen", - "settings.permissions.tool.webfetch.title": "Webhentning", - "settings.permissions.tool.webfetch.description": "Hent indhold fra en URL", - "settings.permissions.tool.websearch.title": "Websøgning", - "settings.permissions.tool.websearch.description": "Søg på nettet", - "settings.permissions.tool.codesearch.title": "Kodesøgning", - "settings.permissions.tool.codesearch.description": "Søg kode på nettet", - "settings.permissions.tool.external_directory.title": "Ekstern mappe", - "settings.permissions.tool.external_directory.description": "Få adgang til filer uden for projektmappen", - "settings.permissions.tool.doom_loop.title": "Doom Loop", - "settings.permissions.tool.doom_loop.description": "Opdag gentagne værktøjskald med identisk input", - - "session.delete.failed.title": "Kunne ikke slette session", - "session.delete.title": "Slet session", - "session.delete.confirm": 'Slet session "{{name}}"?', - "session.delete.button": "Slet session", - - "workspace.new": "Nyt arbejdsområde", - "workspace.type.local": "lokal", - "workspace.type.sandbox": "sandkasse", - "workspace.create.failed.title": "Kunne ikke oprette arbejdsområde", - "workspace.delete.failed.title": "Kunne ikke slette arbejdsområde", - "workspace.resetting.title": "Nulstiller arbejdsområde", - "workspace.resetting.description": "Dette kan tage et minut.", - "workspace.reset.failed.title": "Kunne ikke nulstille arbejdsområde", - "workspace.reset.success.title": "Arbejdsområde nulstillet", - "workspace.reset.success.description": "Arbejdsområdet matcher nu hovedgrenen.", - "workspace.error.stillPreparing": "Arbejdsområdet er stadig ved at blive klargjort", - "workspace.status.checking": "Tjekker for uflettede ændringer...", - "workspace.status.error": "Kunne ikke bekræfte git-status.", - "workspace.status.clean": "Ingen uflettede ændringer fundet.", - "workspace.status.dirty": "Uflettede ændringer fundet i dette arbejdsområde.", - "workspace.delete.title": "Slet arbejdsområde", - "workspace.delete.confirm": 'Slet arbejdsområde "{{name}}"?', - "workspace.delete.button": "Slet arbejdsområde", - "workspace.reset.title": "Nulstil arbejdsområde", - "workspace.reset.confirm": 'Nulstil arbejdsområde "{{name}}"?', - "workspace.reset.button": "Nulstil arbejdsområde", - "workspace.reset.archived.none": "Ingen aktive sessioner vil blive arkiveret.", - "workspace.reset.archived.one": "1 session vil blive arkiveret.", - "workspace.reset.archived.many": "{{count}} sessioner vil blive arkiveret.", - "workspace.reset.note": "Dette vil nulstille arbejdsområdet til at matche hovedgrenen.", - "common.open": "Åbn", - "dialog.releaseNotes.action.getStarted": "Kom i gang", - "dialog.releaseNotes.action.next": "Næste", - "dialog.releaseNotes.action.hideFuture": "Vis ikke disse i fremtiden", - "dialog.releaseNotes.media.alt": "Forhåndsvisning af udgivelse", - "toast.project.reloadFailed.title": "Kunne ikke genindlæse {{project}}", - "error.server.invalidConfiguration": "Ugyldig konfiguration", - "common.moreCountSuffix": " (+{{count}} mere)", - "common.time.justNow": "Lige nu", - "common.time.minutesAgo.short": "{{count}}m siden", - "common.time.hoursAgo.short": "{{count}}t siden", - "common.time.daysAgo.short": "{{count}}d siden", - "settings.providers.connected.environmentDescription": "Tilsluttet fra dine miljøvariabler", - "settings.providers.custom.description": "Tilføj en OpenAI-kompatibel udbyder via basis-URL.", - - "app.server.unreachable": "Kunne ikke nå {{server}}", - "app.server.retrying": "Prøver igen automatisk...", - "app.server.otherServers": "Andre servere", - "dialog.server.add.usernamePlaceholder": "brugernavn", - "dialog.server.add.passwordPlaceholder": "adgangskode", - "server.row.noUsername": "intet brugernavn", - "session.review.noVcs.createGit.title": "Opret et Git-repository", - "session.review.noVcs.createGit.description": "Spor, gennemgå og fortryd ændringer i dette projekt", - "session.review.noVcs.createGit.actionLoading": "Opretter Git-repository...", - "session.review.noVcs.createGit.action": "Opret Git-repository", - "session.todo.progress": "{{done}} af {{total}} opgaver fuldført", - "session.question.progress": "{{current}} af {{total}} spørgsmål", - "session.header.open.finder": "Finder", - "session.header.open.fileExplorer": "Stifinder", - "session.header.open.fileManager": "Filhåndtering", - "session.header.open.app.vscode": "VS Code", - "session.header.open.app.cursor": "Cursor", - "session.header.open.app.zed": "Zed", - "session.header.open.app.textmate": "TextMate", - "session.header.open.app.antigravity": "Antigravity", - "session.header.open.app.terminal": "Terminal", - "session.header.open.app.iterm2": "iTerm2", - "session.header.open.app.ghostty": "Ghostty", - "session.header.open.app.warp": "Warp", - "session.header.open.app.xcode": "Xcode", - "session.header.open.app.androidStudio": "Android Studio", - "session.header.open.app.powershell": "PowerShell", - "session.header.open.app.sublimeText": "Sublime Text", - "debugBar.ariaLabel": "Udviklingsydelsesdiagnostik", - "debugBar.na": "n/a", - "debugBar.nav.label": "NAV", - "debugBar.nav.tip": - "Sidste gennemførte ruteovergang, der berører en sessionsside, målt fra routerstart til den første optegning efter den falder til ro.", - "debugBar.fps.label": "FPS", - "debugBar.fps.tip": "Rullende billeder pr. sekund over de sidste 5 sekunder.", - "debugBar.frame.label": "FRAME", - "debugBar.frame.tip": "Værste billedtid over de sidste 5 sekunder.", - "debugBar.jank.label": "JANK", - "debugBar.jank.tip": "Billeder over 32ms i de sidste 5 sekunder.", - "debugBar.long.label": "LONG", - "debugBar.long.tip": "Blokeret tid og antal lange opgaver i de sidste 5 sekunder. Maks opgave: {{max}}.", - "debugBar.delay.label": "DELAY", - "debugBar.delay.tip": "Værste observerede inputforsinkelse i de sidste 5 sekunder.", - "debugBar.inp.label": "INP", - "debugBar.inp.tip": - "Omtrentlig interaktionsvarighed over de sidste 5 sekunder. Dette er INP-lignende, ikke den officielle Web Vitals INP.", - "debugBar.cls.label": "CLS", - "debugBar.cls.tip": "Kumulativt layoutskift for den nuværende app-levetid.", - "debugBar.mem.label": "MEM", - "debugBar.mem.tipUnavailable": "Brugt JS-heap vs heap-grænse. Kun Chromium.", - "debugBar.mem.tip": "Brugt JS-heap vs heap-grænse. {{used}} af {{limit}}.", - "common.key.ctrl": "Ctrl", - "common.key.alt": "Alt", - "common.key.shift": "Shift", - "common.key.meta": "Meta", - "common.key.space": "Mellemrum", - "common.key.backspace": "Backspace", - "common.key.enter": "Enter", - "common.key.tab": "Tab", - "common.key.delete": "Delete", - "common.key.home": "Home", - "common.key.end": "End", - "common.key.pageUp": "Page Up", - "common.key.pageDown": "Page Down", - "common.key.insert": "Insert", - "common.unknown": "ukendt", - "error.page.circular": "[Cirkulær]", - "error.globalSDK.noServerAvailable": "Ingen server tilgængelig", - "error.globalSDK.serverNotAvailable": "Server ikke tilgængelig", - "error.childStore.persistedCacheCreateFailed": "Kunne ikke oprette vedvarende cache", - "error.childStore.persistedProjectMetadataCreateFailed": "Kunne ikke oprette vedvarende projektmetadata", - "error.childStore.persistedProjectIconCreateFailed": "Kunne ikke oprette vedvarende projektikon", - "error.childStore.storeCreateFailed": "Kunne ikke oprette lager", - "terminal.connectionLost.abnormalClose": "WebSocket lukkede unormalt: {{code}}", -} diff --git a/packages/app/src/i18n/de.ts b/packages/app/src/i18n/de.ts deleted file mode 100644 index 34ce312aea..0000000000 --- a/packages/app/src/i18n/de.ts +++ /dev/null @@ -1,870 +0,0 @@ -import { dict as en } from "./en" - -type Keys = keyof typeof en - -export const dict = { - "command.category.suggested": "Vorgeschlagen", - "command.category.view": "Ansicht", - "command.category.project": "Projekt", - "command.category.provider": "Anbieter", - "command.category.server": "Server", - "command.category.session": "Sitzung", - "command.category.theme": "Thema", - "command.category.language": "Sprache", - "command.category.file": "Datei", - "command.category.context": "Kontext", - "command.category.terminal": "Terminal", - "command.category.model": "Modell", - "command.category.mcp": "MCP", - "command.category.agent": "Agent", - "command.category.permissions": "Berechtigungen", - "command.category.workspace": "Arbeitsbereich", - "command.category.settings": "Einstellungen", - "theme.scheme.system": "System", - "theme.scheme.light": "Hell", - "theme.scheme.dark": "Dunkel", - "command.sidebar.toggle": "Seitenleiste umschalten", - "command.project.open": "Projekt öffnen", - "command.provider.connect": "Anbieter verbinden", - "command.server.switch": "Server wechseln", - "command.settings.open": "Einstellungen öffnen", - "command.session.previous": "Vorherige Sitzung", - "command.session.next": "Nächste Sitzung", - "command.session.previous.unseen": "Vorherige ungelesene Sitzung", - "command.session.next.unseen": "Nächste ungelesene Sitzung", - "command.session.archive": "Sitzung archivieren", - "command.palette": "Befehlspalette", - "command.theme.cycle": "Thema wechseln", - "command.theme.set": "Thema verwenden: {{theme}}", - "command.theme.scheme.cycle": "Farbschema wechseln", - "command.theme.scheme.set": "Farbschema verwenden: {{scheme}}", - "command.language.cycle": "Sprache wechseln", - "command.language.set": "Sprache verwenden: {{language}}", - "command.session.new": "Neue Sitzung", - "command.file.open": "Datei öffnen", - "command.tab.close": "Tab schließen", - "command.context.addSelection": "Auswahl zum Kontext hinzufügen", - "command.context.addSelection.description": "Ausgewählte Zeilen aus der aktuellen Datei hinzufügen", - "command.input.focus": "Eingabefeld fokussieren", - "command.terminal.toggle": "Terminal umschalten", - "command.fileTree.toggle": "Dateibaum umschalten", - "command.review.toggle": "Überprüfung umschalten", - "command.terminal.new": "Neues Terminal", - "command.terminal.new.description": "Neuen Terminal-Tab erstellen", - "command.steps.toggle": "Schritte umschalten", - "command.steps.toggle.description": "Schritte für die aktuelle Nachricht anzeigen oder ausblenden", - "command.message.previous": "Vorherige Nachricht", - "command.message.previous.description": "Zur vorherigen Benutzernachricht gehen", - "command.message.next": "Nächste Nachricht", - "command.message.next.description": "Zur nächsten Benutzernachricht gehen", - "command.model.choose": "Modell wählen", - "command.model.choose.description": "Ein anderes Modell auswählen", - "command.mcp.toggle": "MCPs umschalten", - "command.mcp.toggle.description": "MCPs umschalten", - "command.agent.cycle": "Agent wechseln", - "command.agent.cycle.description": "Zum nächsten Agenten wechseln", - "command.agent.cycle.reverse": "Agent rückwärts wechseln", - "command.agent.cycle.reverse.description": "Zum vorherigen Agenten wechseln", - "command.model.variant.cycle": "Denkaufwand wechseln", - "command.model.variant.cycle.description": "Zum nächsten Aufwandslevel wechseln", - "command.prompt.mode.shell": "Shell", - "command.prompt.mode.normal": "Prompt", - "command.permissions.autoaccept.enable": "Berechtigungen automatisch akzeptieren", - "command.permissions.autoaccept.disable": "Automatische Akzeptanz von Berechtigungen stoppen", - "command.workspace.toggle": "Arbeitsbereiche umschalten", - "command.workspace.toggle.description": "Mehrere Arbeitsbereiche in der Seitenleiste aktivieren oder deaktivieren", - "command.session.undo": "Rückgängig", - "command.session.undo.description": "Letzte Nachricht rückgängig machen", - "command.session.redo": "Wiederherstellen", - "command.session.redo.description": "Letzte rückgängig gemachte Nachricht wiederherstellen", - "command.session.compact": "Sitzung komprimieren", - "command.session.compact.description": "Sitzung zusammenfassen, um die Kontextgröße zu reduzieren", - "command.session.fork": "Von Nachricht abzweigen", - "command.session.fork.description": "Neue Sitzung aus einer früheren Nachricht erstellen", - "command.session.share": "Sitzung teilen", - "command.session.share.description": "Diese Sitzung teilen und URL in die Zwischenablage kopieren", - "command.session.unshare": "Teilen der Sitzung aufheben", - "command.session.unshare.description": "Teilen dieser Sitzung beenden", - "palette.search.placeholder": "Dateien, Befehle und Sitzungen durchsuchen", - "palette.empty": "Keine Ergebnisse gefunden", - "palette.group.commands": "Befehle", - "palette.group.files": "Dateien", - "dialog.provider.search.placeholder": "Anbieter durchsuchen", - "dialog.provider.empty": "Keine Anbieter gefunden", - "dialog.provider.group.popular": "Beliebt", - "dialog.provider.group.other": "Andere", - "dialog.provider.tag.recommended": "Empfohlen", - "dialog.provider.opencode.note": "Kuratierte Modelle inklusive Claude, GPT, Gemini und mehr", - "dialog.provider.opencode.tagline": "Zuverlässige, optimierte Modelle", - "dialog.provider.opencodeGo.tagline": "Kostengünstiges Abo für alle", - "dialog.provider.anthropic.note": "Mit Claude Pro/Max oder API-Schlüssel verbinden", - "dialog.provider.copilot.note": "Mit Copilot oder API-Schlüssel verbinden", - "dialog.provider.openai.note": "Mit ChatGPT Pro/Plus oder API-Schlüssel verbinden", - "dialog.provider.google.note": "Gemini-Modelle für schnelle, strukturierte Antworten", - "dialog.provider.openrouter.note": "Zugriff auf alle unterstützten Modelle über einen Anbieter", - "dialog.provider.vercel.note": "Einheitlicher Zugriff auf KI-Modelle mit intelligentem Routing", - "dialog.model.select.title": "Modell auswählen", - "dialog.model.search.placeholder": "Modelle durchsuchen", - "dialog.model.empty": "Keine Modellergebnisse", - "dialog.model.manage": "Modelle verwalten", - "dialog.model.manage.description": "Anpassen, welche Modelle in der Modellauswahl erscheinen.", - "dialog.model.manage.provider.toggle": "Alle {{provider}}-Modelle umschalten", - "dialog.model.unpaid.freeModels.title": "Kostenlose Modelle von Kilo", - "dialog.model.unpaid.addMore.title": "Weitere Modelle von beliebten Anbietern hinzufügen", - "dialog.provider.viewAll": "Mehr Anbieter anzeigen", - "provider.connect.title": "{{provider}} verbinden", - "provider.connect.title.anthropicProMax": "Mit Claude Pro/Max anmelden", - "provider.connect.selectMethod": "Anmeldemethode für {{provider}} auswählen.", - "provider.connect.method.apiKey": "API-Schlüssel", - "provider.connect.status.inProgress": "Autorisierung läuft...", - "provider.connect.status.waiting": "Warten auf Autorisierung...", - "provider.connect.status.failed": "Autorisierung fehlgeschlagen: {{error}}", - "provider.connect.apiKey.description": - "Geben Sie Ihren {{provider}} API-Schlüssel ein, um Ihr Konto zu verbinden und {{provider}} Modelle in Kilo zu nutzen.", - "provider.connect.apiKey.label": "{{provider}} API-Schlüssel", - "provider.connect.apiKey.placeholder": "API-Schlüssel", - "provider.connect.apiKey.required": "API-Schlüssel ist erforderlich", - "provider.connect.opencodeZen.line1": - "OpenCode Zen bietet Ihnen Zugriff auf eine kuratierte Auswahl zuverlässiger, optimierter Modelle für Coding-Agenten.", - "provider.connect.opencodeZen.line2": - "Mit einem einzigen API-Schlüssel erhalten Sie Zugriff auf Modelle wie Claude, GPT, Gemini, GLM und mehr.", - "provider.connect.opencodeZen.visit.prefix": "Besuchen Sie ", - "provider.connect.opencodeZen.visit.link": "opencode.ai/zen", - "provider.connect.opencodeZen.visit.suffix": ", um Ihren API-Schlüssel zu erhalten.", - "provider.connect.oauth.code.visit.prefix": "Besuchen Sie ", - "provider.connect.oauth.code.visit.link": "diesen Link", - "provider.connect.oauth.code.visit.suffix": - ", um Ihren Autorisierungscode zu erhalten, Ihr Konto zu verbinden und {{provider}} Modelle in Kilo zu nutzen.", - "provider.connect.oauth.code.label": "{{method}} Autorisierungscode", - "provider.connect.oauth.code.placeholder": "Autorisierungscode", - "provider.connect.oauth.code.required": "Autorisierungscode ist erforderlich", - "provider.connect.oauth.code.invalid": "Ungültiger Autorisierungscode", - "provider.connect.oauth.auto.visit.prefix": "Besuchen Sie ", - "provider.connect.oauth.auto.visit.link": "diesen Link", - "provider.connect.oauth.auto.visit.suffix": - " und geben Sie den untenstehenden Code ein, um Ihr Konto zu verbinden und {{provider}} Modelle in Kilo zu nutzen.", - "provider.connect.oauth.auto.confirmationCode": "Bestätigungscode", - "provider.connect.toast.connected.title": "{{provider}} verbunden", - "provider.connect.toast.connected.description": "{{provider}} Modelle sind jetzt verfügbar.", - "provider.custom.title": "Benutzerdefinierter Anbieter", - "provider.custom.description.prefix": "Konfigurieren Sie einen OpenAI-kompatiblen Anbieter. Siehe die ", - "provider.custom.description.link": "Anbieter-Konfigurationsdokumente", - "provider.custom.description.suffix": ".", - "provider.custom.field.providerID.label": "Anbieter-ID", - "provider.custom.field.providerID.placeholder": "myprovider", - "provider.custom.field.providerID.description": "Kleinbuchstaben, Zahlen, Bindestriche oder Unterstriche", - "provider.custom.field.name.label": "Anzeigename", - "provider.custom.field.name.placeholder": "Mein KI-Anbieter", - "provider.custom.field.baseURL.label": "Basis-URL", - "provider.custom.field.baseURL.placeholder": "https://api.myprovider.com/v1", - "provider.custom.field.apiKey.label": "API-Schlüssel", - "provider.custom.field.apiKey.placeholder": "API-Schlüssel", - "provider.custom.field.apiKey.description": - "Optional. Leer lassen, wenn Sie die Authentifizierung über Header verwalten.", - "provider.custom.models.label": "Modelle", - "provider.custom.models.id.label": "ID", - "provider.custom.models.id.placeholder": "model-id", - "provider.custom.models.name.label": "Name", - "provider.custom.models.name.placeholder": "Anzeigename", - "provider.custom.models.remove": "Modell entfernen", - "provider.custom.models.add": "Modell hinzufügen", - "provider.custom.headers.label": "Header (optional)", - "provider.custom.headers.key.label": "Header", - "provider.custom.headers.key.placeholder": "Header-Name", - "provider.custom.headers.value.label": "Wert", - "provider.custom.headers.value.placeholder": "wert", - "provider.custom.headers.remove": "Header entfernen", - "provider.custom.headers.add": "Header hinzufügen", - "provider.custom.error.providerID.required": "Anbieter-ID ist erforderlich", - "provider.custom.error.providerID.format": "Verwenden Sie Kleinbuchstaben, Zahlen, Bindestriche oder Unterstriche", - "provider.custom.error.providerID.exists": "Diese Anbieter-ID existiert bereits", - "provider.custom.error.name.required": "Anzeigename ist erforderlich", - "provider.custom.error.baseURL.required": "Basis-URL ist erforderlich", - "provider.custom.error.baseURL.format": "Muss mit http:// oder https:// beginnen", - "provider.custom.error.required": "Erforderlich", - "provider.custom.error.duplicate": "Duplikat", - "provider.disconnect.toast.disconnected.title": "{{provider}} getrennt", - "provider.disconnect.toast.disconnected.description": "Die {{provider}}-Modelle sind nicht mehr verfügbar.", - "model.tag.free": "Kostenlos", - "model.tag.latest": "Neueste", - "model.provider.anthropic": "Anthropic", - "model.provider.openai": "OpenAI", - "model.provider.google": "Google", - "model.provider.xai": "xAI", - "model.provider.meta": "Meta", - "model.input.text": "Text", - "model.input.image": "Bild", - "model.input.audio": "Audio", - "model.input.video": "Video", - "model.input.pdf": "PDF", - "model.tooltip.allows": "Erlaubt: {{inputs}}", - "model.tooltip.reasoning.allowed": "Erlaubt Reasoning", - "model.tooltip.reasoning.none": "Kein Reasoning", - "model.tooltip.context": "Kontextlimit {{limit}}", - "common.search.placeholder": "Suchen", - "common.goBack": "Zurück", - "common.goForward": "Vorwärts navigieren", - "common.loading": "Laden", - "common.loading.ellipsis": "...", - "common.cancel": "Abbrechen", - "common.connect": "Verbinden", - "common.disconnect": "Trennen", - "common.continue": "Absenden", - "common.submit": "Absenden", - "common.save": "Speichern", - "common.saving": "Speichert...", - "common.default": "Standard", - "common.attachment": "Anhang", - "prompt.placeholder.shell": "Shell-Befehl eingeben... {{example}}", - "prompt.placeholder.normal": 'Fragen Sie alles... "{{example}}"', - "prompt.placeholder.simple": "Fragen Sie alles...", - "prompt.placeholder.summarizeComments": "Kommentare zusammenfassen…", - "prompt.placeholder.summarizeComment": "Kommentar zusammenfassen…", - "prompt.mode.shell": "Shell", - "prompt.mode.normal": "Prompt", - "prompt.mode.shell.exit": "esc zum Verlassen", - "prompt.example.1": "Ein TODO in der Codebasis beheben", - "prompt.example.2": "Was ist der Tech-Stack dieses Projekts?", - "prompt.example.3": "Fehlerhafte Tests beheben", - "prompt.example.4": "Erkläre, wie die Authentifizierung funktioniert", - "prompt.example.5": "Sicherheitslücken finden und beheben", - "prompt.example.6": "Unit-Tests für den Benutzerdienst hinzufügen", - "prompt.example.7": "Diese Funktion lesbarer gestalten", - "prompt.example.8": "Was bedeutet dieser Fehler?", - "prompt.example.9": "Hilf mir, dieses Problem zu debuggen", - "prompt.example.10": "API-Dokumentation generieren", - "prompt.example.11": "Datenbankabfragen optimieren", - "prompt.example.12": "Eingabevalidierung hinzufügen", - "prompt.example.13": "Neue Komponente erstellen für...", - "prompt.example.14": "Wie deploye ich dieses Projekt?", - "prompt.example.15": "Meinen Code auf Best Practices überprüfen", - "prompt.example.16": "Fehlerbehandlung zu dieser Funktion hinzufügen", - "prompt.example.17": "Erkläre dieses Regex-Muster", - "prompt.example.18": "Dies in TypeScript konvertieren", - "prompt.example.19": "Logging in der gesamten Codebasis hinzufügen", - "prompt.example.20": "Welche Abhängigkeiten sind veraltet?", - "prompt.example.21": "Hilf mir, ein Migrationsskript zu schreiben", - "prompt.example.22": "Caching für diesen Endpunkt implementieren", - "prompt.example.23": "Paginierung zu dieser Liste hinzufügen", - "prompt.example.24": "CLI-Befehl erstellen für...", - "prompt.example.25": "Wie funktionieren Umgebungsvariablen hier?", - "prompt.popover.emptyResults": "Keine passenden Ergebnisse", - "prompt.popover.emptyCommands": "Keine passenden Befehle", - "prompt.dropzone.label": "Bilder, PDFs oder Textdateien hier ablegen", - "prompt.dropzone.file.label": "Ablegen zum @Erwähnen der Datei", - "prompt.slash.badge.custom": "benutzerdefiniert", - "prompt.slash.badge.skill": "Skill", - "prompt.slash.badge.mcp": "MCP", - "prompt.context.active": "aktiv", - "prompt.context.includeActiveFile": "Aktive Datei einbeziehen", - "prompt.context.removeActiveFile": "Aktive Datei aus dem Kontext entfernen", - "prompt.context.removeFile": "Datei aus dem Kontext entfernen", - "prompt.action.attachFile": "Datei anhängen", - "prompt.attachment.remove": "Anhang entfernen", - "prompt.action.send": "Senden", - "prompt.action.stop": "Stopp", - "prompt.toast.pasteUnsupported.title": "Nicht unterstützter Anhang", - "prompt.toast.pasteUnsupported.description": "Hier können nur Bilder, PDFs oder Textdateien angehängt werden.", - "prompt.toast.modelAgentRequired.title": "Wählen Sie einen Agenten und ein Modell", - "prompt.toast.modelAgentRequired.description": - "Wählen Sie einen Agenten und ein Modell, bevor Sie eine Eingabe senden.", - "prompt.toast.worktreeCreateFailed.title": "Worktree konnte nicht erstellt werden", - "prompt.toast.sessionCreateFailed.title": "Sitzung konnte nicht erstellt werden", - "prompt.toast.shellSendFailed.title": "Shell-Befehl konnte nicht gesendet werden", - "prompt.toast.commandSendFailed.title": "Befehl konnte nicht gesendet werden", - "prompt.toast.promptSendFailed.title": "Eingabe konnte nicht gesendet werden", - "prompt.toast.promptSendFailed.description": "Sitzung konnte nicht abgerufen werden", - "dialog.mcp.title": "MCPs", - "dialog.mcp.description": "{{enabled}} von {{total}} aktiviert", - "dialog.mcp.empty": "Keine MCPs konfiguriert", - "dialog.lsp.empty": "LSPs automatisch nach Dateityp erkannt", - "dialog.plugins.empty": "In opencode.json konfigurierte Plugins", - "mcp.status.connected": "verbunden", - "mcp.status.failed": "fehlgeschlagen", - "mcp.status.needs_auth": "benötigt Authentifizierung", - "mcp.status.disabled": "deaktiviert", - "dialog.fork.empty": "Keine Nachrichten zum Abzweigen vorhanden", - "dialog.directory.search.placeholder": "Ordner durchsuchen", - "dialog.directory.empty": "Keine Ordner gefunden", - "dialog.server.title": "Server", - "dialog.server.description": "Wechseln Sie den Kilo-Server, mit dem sich diese App verbindet.", - "dialog.server.search.placeholder": "Server durchsuchen", - "dialog.server.empty": "Noch keine Server", - "dialog.server.add.title": "Server hinzufügen", - "dialog.server.add.url": "Server-URL", - "dialog.server.add.placeholder": "http://localhost:4096", - "dialog.server.add.error": "Verbindung zum Server fehlgeschlagen", - "dialog.server.add.checking": "Prüfen...", - "dialog.server.add.button": "Server hinzufügen", - "dialog.server.add.name": "Servername (optional)", - "dialog.server.add.namePlaceholder": "Localhost", - "dialog.server.add.username": "Benutzername (optional)", - "dialog.server.add.password": "Passwort (optional)", - "dialog.server.edit.title": "Server bearbeiten", - "dialog.server.default.title": "Standardserver", - "dialog.server.default.description": - "Beim App-Start mit diesem Server verbinden, anstatt einen lokalen Server zu starten. Erfordert Neustart.", - "dialog.server.default.none": "Kein Server ausgewählt", - "dialog.server.default.set": "Aktuellen Server als Standard setzen", - "dialog.server.default.clear": "Löschen", - "dialog.server.action.remove": "Server entfernen", - "dialog.server.menu.edit": "Bearbeiten", - "dialog.server.menu.default": "Als Standard festlegen", - "dialog.server.menu.defaultRemove": "Standard entfernen", - "dialog.server.menu.delete": "Löschen", - "dialog.server.current": "Aktueller Server", - "dialog.server.status.default": "Standard", - "dialog.project.edit.title": "Projekt bearbeiten", - "dialog.project.edit.name": "Name", - "dialog.project.edit.icon": "Icon", - "dialog.project.edit.icon.alt": "Projekt-Icon", - "dialog.project.edit.icon.hint": "Klicken oder Bild ziehen", - "dialog.project.edit.icon.recommended": "Empfohlen: 128x128px", - "dialog.project.edit.color": "Farbe", - "dialog.project.edit.color.select": "{{color}}-Farbe auswählen", - "dialog.project.edit.worktree.startup": "Startup-Skript für Arbeitsbereich", - "dialog.project.edit.worktree.startup.description": - "Wird nach dem Erstellen eines neuen Arbeitsbereichs (Worktree) ausgeführt.", - "dialog.project.edit.worktree.startup.placeholder": "z. B. bun install", - "context.breakdown.title": "Kontext-Aufschlüsselung", - "context.breakdown.note": - 'Ungefähre Aufschlüsselung der Eingabe-Token. "Andere" beinhaltet Werkzeugdefinitionen und Overhead.', - "context.breakdown.system": "System", - "context.breakdown.user": "Benutzer", - "context.breakdown.assistant": "Assistent", - "context.breakdown.tool": "Werkzeugaufrufe", - "context.breakdown.other": "Andere", - "context.systemPrompt.title": "System-Prompt", - "context.rawMessages.title": "Rohdaten der Nachrichten", - "context.stats.session": "Sitzung", - "context.stats.messages": "Nachrichten", - "context.stats.provider": "Anbieter", - "context.stats.model": "Modell", - "context.stats.limit": "Kontextlimit", - "context.stats.totalTokens": "Gesamt-Token", - "context.stats.usage": "Nutzung", - "context.stats.inputTokens": "Eingabe-Token", - "context.stats.outputTokens": "Ausgabe-Token", - "context.stats.reasoningTokens": "Reasoning-Token", - "context.stats.cacheTokens": "Cache-Token (lesen/schreiben)", - "context.stats.userMessages": "Benutzernachrichten", - "context.stats.assistantMessages": "Assistentennachrichten", - "context.stats.totalCost": "Gesamtkosten", - "context.stats.sessionCreated": "Sitzung erstellt", - "context.stats.lastActivity": "Letzte Aktivität", - "context.usage.tokens": "Token", - "context.usage.usage": "Nutzung", - "context.usage.cost": "Kosten", - "context.usage.clickToView": "Klicken, um Kontext anzuzeigen", - "context.usage.view": "Kontextnutzung anzeigen", - "language.en": "English", - "language.zh": "简体中文", - "language.zht": "繁體中文", - "language.ko": "한국어", - "language.de": "Deutsch", - "language.es": "Español", - "language.fr": "Français", - "language.da": "Dansk", - "language.ja": "日本語", - "language.pl": "Polski", - "language.ru": "Русский", - "language.ar": "العربية", - "language.no": "Norsk", - "language.br": "Português (Brasil)", - "language.bs": "Bosanski", - "language.th": "ไทย", - "language.tr": "Türkçe", - "toast.language.title": "Sprache", - "toast.language.description": "Zu {{language}} gewechselt", - "toast.theme.title": "Thema gewechselt", - "toast.scheme.title": "Farbschema", - "toast.workspace.enabled.title": "Arbeitsbereiche aktiviert", - "toast.workspace.enabled.description": "Mehrere Worktrees werden jetzt in der Seitenleiste angezeigt", - "toast.workspace.disabled.title": "Arbeitsbereiche deaktiviert", - "toast.workspace.disabled.description": "Nur der Haupt-Worktree wird in der Seitenleiste angezeigt", - "toast.permissions.autoaccept.on.title": "Berechtigungen werden automatisch akzeptiert", - "toast.permissions.autoaccept.on.description": "Berechtigungsanfragen werden automatisch genehmigt", - "toast.permissions.autoaccept.off.title": "Automatische Akzeptanz von Berechtigungen gestoppt", - "toast.permissions.autoaccept.off.description": "Berechtigungsanfragen erfordern eine Genehmigung", - "toast.model.none.title": "Kein Modell ausgewählt", - "toast.model.none.description": "Verbinden Sie einen Anbieter, um diese Sitzung zusammenzufassen", - "toast.file.loadFailed.title": "Datei konnte nicht geladen werden", - "toast.file.listFailed.title": "Dateien konnten nicht aufgelistet werden", - "toast.context.noLineSelection.title": "Keine Zeilenauswahl", - "toast.context.noLineSelection.description": "Wählen Sie zuerst einen Zeilenbereich in einem Datei-Tab aus.", - "toast.session.share.copyFailed.title": "URL konnte nicht in die Zwischenablage kopiert werden", - "toast.session.share.success.title": "Sitzung geteilt", - "toast.session.share.success.description": "Teilen-URL in die Zwischenablage kopiert!", - "toast.session.share.failed.title": "Sitzung konnte nicht geteilt werden", - "toast.session.share.failed.description": "Beim Teilen der Sitzung ist ein Fehler aufgetreten", - "toast.session.unshare.success.title": "Teilen der Sitzung aufgehoben", - "toast.session.unshare.success.description": "Teilen der Sitzung erfolgreich aufgehoben!", - "toast.session.unshare.failed.title": "Aufheben des Teilens fehlgeschlagen", - "toast.session.unshare.failed.description": "Beim Aufheben des Teilens ist ein Fehler aufgetreten", - "toast.session.listFailed.title": "Sitzungen für {{project}} konnten nicht geladen werden", - "toast.update.title": "Update verfügbar", - "toast.update.description": "Eine neue Version von Kilo ({{version}}) ist zur Installation verfügbar.", - "toast.update.action.installRestart": "Installieren und neu starten", - "toast.update.action.notYet": "Noch nicht", - "error.page.title": "Etwas ist schiefgelaufen", - "error.page.description": "Beim Laden der Anwendung ist ein Fehler aufgetreten.", - "error.page.details.label": "Fehlerdetails", - "error.page.action.restart": "Neustart", - "error.page.action.checking": "Prüfen...", - "error.page.action.checkUpdates": "Nach Updates suchen", - "error.page.action.updateTo": "Auf {{version}} aktualisieren", - "error.page.report.prefix": "Bitte melden Sie diesen Fehler dem Kilo-Team", - "error.page.report.discord": "auf Discord", - "error.page.version": "Version: {{version}}", - "error.dev.rootNotFound": - "Wurzelelement nicht gefunden. Haben Sie vergessen, es in Ihre index.html aufzunehmen? Oder wurde das id-Attribut falsch geschrieben?", - "error.globalSync.connectFailed": "Verbindung zum Server fehlgeschlagen. Läuft ein Server unter `{{url}}`?", - "directory.error.invalidUrl": "Ungültiges Verzeichnis in der URL.", - "error.chain.unknown": "Unbekannter Fehler", - "error.chain.causedBy": "Verursacht durch:", - "error.chain.apiError": "API-Fehler", - "error.chain.status": "Status: {{status}}", - "error.chain.retryable": "Wiederholbar: {{retryable}}", - "error.chain.responseBody": "Antwort-Body:\n{{body}}", - "error.chain.didYouMean": "Meinten Sie: {{suggestions}}", - "error.chain.modelNotFound": "Modell nicht gefunden: {{provider}}/{{model}}", - "error.chain.checkConfig": "Überprüfen Sie Ihre Konfiguration (opencode.json) auf Anbieter-/Modellnamen", - "error.chain.mcpFailed": - 'MCP-Server "{{name}}" fehlgeschlagen. Hinweis: Kilo unterstützt noch keine MCP-Authentifizierung.', - "error.chain.providerAuthFailed": "Anbieter-Authentifizierung fehlgeschlagen ({{provider}}): {{message}}", - "error.chain.providerInitFailed": - 'Anbieter "{{provider}}" konnte nicht initialisiert werden. Überprüfen Sie Anmeldeinformationen und Konfiguration.', - "error.chain.configJsonInvalid": "Konfigurationsdatei unter {{path}} ist kein gültiges JSON(C)", - "error.chain.configJsonInvalidWithMessage": - "Konfigurationsdatei unter {{path}} ist kein gültiges JSON(C): {{message}}", - "error.chain.configDirectoryTypo": - 'Verzeichnis "{{dir}}" in {{path}} ist ungültig. Benennen Sie das Verzeichnis in "{{suggestion}}" um oder entfernen Sie es. Dies ist ein häufiger Tippfehler.', - "error.chain.configFrontmatterError": "Frontmatter in {{path}} konnte nicht geparst werden:\n{{message}}", - "error.chain.configInvalid": "Konfigurationsdatei unter {{path}} ist ungültig", - "error.chain.configInvalidWithMessage": "Konfigurationsdatei unter {{path}} ist ungültig: {{message}}", - "notification.permission.title": "Berechtigung erforderlich", - "notification.permission.description": "{{sessionTitle}} in {{projectName}} benötigt Berechtigung", - "notification.question.title": "Frage", - "notification.question.description": "{{sessionTitle}} in {{projectName}} hat eine Frage", - "notification.action.goToSession": "Zur Sitzung gehen", - "notification.session.responseReady.title": "Antwort bereit", - "notification.session.error.title": "Sitzungsfehler", - "notification.session.error.fallbackDescription": "Ein Fehler ist aufgetreten", - "home.recentProjects": "Letzte Projekte", - "home.empty.title": "Keine letzten Projekte", - "home.empty.description": "Starten Sie, indem Sie ein lokales Projekt öffnen", - "session.tab.session": "Sitzung", - "session.tab.review": "Überprüfung", - "session.tab.context": "Kontext", - "session.panel.reviewAndFiles": "Überprüfung und Dateien", - "session.review.filesChanged": "{{count}} Dateien geändert", - "session.review.change.one": "Änderung", - "session.review.change.other": "Änderungen", - "session.review.loadingChanges": "Lade Änderungen...", - "session.review.empty": "Noch keine Änderungen in dieser Sitzung", - "session.review.noVcs": "Kein Git-Versionskontrollsystem erkannt, Änderungen werden nicht angezeigt", - "session.review.noSnapshot": - "Snapshot-Tracking ist in der Konfiguration deaktiviert, daher sind Sitzungsänderungen nicht verfügbar", - "session.review.noChanges": "Keine Änderungen", - "session.files.selectToOpen": "Datei zum Öffnen auswählen", - "session.files.all": "Alle Dateien", - "session.files.empty": "Keine Dateien", - "session.files.binaryContent": "Binärdatei (Inhalt kann nicht angezeigt werden)", - "session.messages.renderEarlier": "Frühere Nachrichten rendern", - "session.messages.loadingEarlier": "Lade frühere Nachrichten...", - "session.messages.loadEarlier": "Frühere Nachrichten laden", - "session.messages.loading": "Lade Nachrichten...", - "session.messages.jumpToLatest": "Zum neuesten springen", - "session.context.addToContext": "{{selection}} zum Kontext hinzufügen", - "session.todo.title": "Aufgaben", - "session.todo.collapse": "Einklappen", - "session.todo.expand": "Ausklappen", - "session.followupDock.summary.one": "{{count}} Nachricht in der Warteschlange", - "session.followupDock.summary.other": "{{count}} Nachrichten in der Warteschlange", - "session.followupDock.sendNow": "Jetzt senden", - "session.followupDock.edit": "Bearbeiten", - "session.followupDock.collapse": "Warteschlange einklappen", - "session.followupDock.expand": "Warteschlange ausklappen", - "session.revertDock.summary.one": "{{count}} zurückgesetzte Nachricht", - "session.revertDock.summary.other": "{{count}} zurückgesetzte Nachrichten", - "session.revertDock.collapse": "Zurückgesetzte Nachrichten einklappen", - "session.revertDock.expand": "Zurückgesetzte Nachrichten ausklappen", - "session.revertDock.restore": "Nachricht wiederherstellen", - "session.new.title": "Baue, was du willst", - "session.new.worktree.main": "Haupt-Branch", - "session.new.worktree.mainWithBranch": "Haupt-Branch ({{branch}})", - "session.new.worktree.create": "Neuen Worktree erstellen", - "session.new.lastModified": "Zuletzt geändert", - "session.header.search.placeholder": "{{project}} durchsuchen", - "session.header.searchFiles": "Dateien suchen", - "session.header.openIn": "Öffnen in", - "session.header.open.action": "{{app}} öffnen", - "session.header.open.ariaLabel": "In {{app}} öffnen", - "session.header.open.menu": "Öffnen-Optionen", - "session.header.open.copyPath": "Pfad kopieren", - "status.popover.trigger": "Status", - "status.popover.ariaLabel": "Serverkonfigurationen", - "status.popover.tab.servers": "Server", - "status.popover.tab.mcp": "MCP", - "status.popover.tab.lsp": "LSP", - "status.popover.tab.plugins": "Plugins", - "status.popover.action.manageServers": "Server verwalten", - "session.share.popover.title": "Im Web veröffentlichen", - "session.share.popover.description.shared": - "Diese Sitzung ist öffentlich im Web. Sie ist für jeden mit dem Link zugänglich.", - "session.share.popover.description.unshared": - "Sitzung öffentlich im Web teilen. Sie wird für jeden mit dem Link zugänglich sein.", - "session.share.action.share": "Teilen", - "session.share.action.publish": "Veröffentlichen", - "session.share.action.publishing": "Veröffentliche...", - "session.share.action.unpublish": "Veröffentlichung aufheben", - "session.share.action.unpublishing": "Hebe Veröffentlichung auf...", - "session.share.action.view": "Ansehen", - "session.share.copy.copied": "Kopiert", - "session.share.copy.copyLink": "Link kopieren", - "lsp.tooltip.none": "Keine LSP-Server", - "lsp.label.connected": "{{count}} LSP", - "prompt.loading": "Lade Prompt...", - "terminal.loading": "Lade Terminal...", - "terminal.title": "Terminal", - "terminal.title.numbered": "Terminal {{number}}", - "terminal.close": "Terminal schließen", - "terminal.connectionLost.title": "Verbindung verloren", - "terminal.connectionLost.description": - "Die Terminalverbindung wurde unterbrochen. Das kann passieren, wenn der Server neu startet.", - "common.closeTab": "Tab schließen", - "common.dismiss": "Verwerfen", - "common.requestFailed": "Anfrage fehlgeschlagen", - "common.moreOptions": "Weitere Optionen", - "common.learnMore": "Mehr erfahren", - "common.rename": "Umbenennen", - "common.reset": "Zurücksetzen", - "common.archive": "Archivieren", - "common.delete": "Löschen", - "common.close": "Schließen", - "common.edit": "Bearbeiten", - "common.loadMore": "Mehr laden", - "common.key.esc": "ESC", - "sidebar.menu.toggle": "Menü umschalten", - "sidebar.nav.projectsAndSessions": "Projekte und Sitzungen", - "sidebar.settings": "Einstellungen", - "sidebar.help": "Hilfe", - "sidebar.workspaces.enable": "Arbeitsbereiche aktivieren", - "sidebar.workspaces.disable": "Arbeitsbereiche deaktivieren", - "sidebar.gettingStarted.title": "Erste Schritte", - "sidebar.gettingStarted.line1": "Kilo enthält kostenlose Modelle, damit Sie sofort loslegen können.", - "sidebar.gettingStarted.line2": - "Verbinden Sie einen beliebigen Anbieter, um Modelle wie Claude, GPT, Gemini usw. zu nutzen.", - "sidebar.project.recentSessions": "Letzte Sitzungen", - "sidebar.project.viewAllSessions": "Alle Sitzungen anzeigen", - "sidebar.project.clearNotifications": "Benachrichtigungen löschen", - "app.name.desktop": "Kilo Desktop", - "settings.section.desktop": "Desktop", - "settings.section.server": "Server", - "settings.tab.general": "Allgemein", - "settings.tab.shortcuts": "Tastenkombinationen", - "settings.desktop.section.wsl": "WSL", - "settings.desktop.wsl.title": "WSL-Integration", - "settings.desktop.wsl.description": "Kilo-Server innerhalb von WSL unter Windows ausführen.", - "settings.general.section.appearance": "Erscheinungsbild", - "settings.general.section.notifications": "Systembenachrichtigungen", - "settings.general.section.updates": "Updates", - "settings.general.section.sounds": "Soundeffekte", - "settings.general.section.feed": "Feed", - "settings.general.section.display": "Anzeige", - "settings.general.row.language.title": "Sprache", - "settings.general.row.language.description": "Die Anzeigesprache für Kilo ändern", - "settings.general.row.appearance.title": "Erscheinungsbild", - "settings.general.row.appearance.description": "Anpassen, wie Kilo auf Ihrem Gerät aussieht", - "settings.general.row.colorScheme.title": "Farbschema", - "settings.general.row.colorScheme.description": "Wählen Sie, ob Kilo dem System-, hellen oder dunklen Thema folgt", - "settings.general.row.theme.title": "Thema", - "settings.general.row.theme.description": "Das Thema von Kilo anpassen.", - "settings.general.row.font.title": "Code-Schriftart", - "settings.general.row.font.description": "Die in Codeblöcken verwendete Schriftart anpassen", - "settings.general.row.terminalFont.title": "Terminal Font", - "settings.general.row.terminalFont.description": "Customise the font used in the terminal", - "settings.general.row.uiFont.title": "UI-Schriftart", - "settings.general.row.uiFont.description": "Die im gesamten Interface verwendete Schriftart anpassen", - "settings.general.row.followup.title": "Verhalten bei Folgefragen", - "settings.general.row.followup.description": - "Wählen Sie, ob Folgefragen sofort steuern oder in einer Warteschlange warten", - "settings.general.row.followup.option.queue": "Warteschlange", - "settings.general.row.followup.option.steer": "Steuern", - "settings.general.row.reasoningSummaries.title": "Reasoning-Zusammenfassungen anzeigen", - "settings.general.row.reasoningSummaries.description": - "Zusammenfassungen des Modell-Reasonings in der Timeline anzeigen", - "settings.general.row.shellToolPartsExpanded.title": "Shell-Tool-Abschnitte ausklappen", - "settings.general.row.shellToolPartsExpanded.description": - "Shell-Tool-Abschnitte standardmäßig in der Timeline ausgeklappt anzeigen", - "settings.general.row.editToolPartsExpanded.title": "Edit-Tool-Abschnitte ausklappen", - "settings.general.row.editToolPartsExpanded.description": - "Edit-, Write- und Patch-Tool-Abschnitte standardmäßig in der Timeline ausgeklappt anzeigen", - "settings.general.row.showSessionProgressBar.title": "Sitzungsfortschrittsleiste anzeigen", - "settings.general.row.showSessionProgressBar.description": - "Die animierte Fortschrittsleiste oben in der Sitzung anzeigen, wenn der Agent arbeitet", - "settings.general.row.wayland.title": "Natives Wayland verwenden", - "settings.general.row.wayland.description": "X11-Fallback unter Wayland deaktivieren. Erfordert Neustart.", - "settings.general.row.wayland.tooltip": - "Unter Linux mit Monitoren unterschiedlicher Bildwiederholraten kann natives Wayland stabiler sein.", - "settings.general.row.releaseNotes.title": "Versionshinweise", - "settings.general.row.releaseNotes.description": '"Neuigkeiten"-Pop-ups nach Updates anzeigen', - "settings.updates.row.startup.title": "Beim Start nach Updates suchen", - "settings.updates.row.startup.description": "Beim Start von Kilo automatisch nach Updates suchen", - "settings.updates.row.check.title": "Nach Updates suchen", - "settings.updates.row.check.description": "Manuell nach Updates suchen und installieren, wenn verfügbar", - "settings.updates.action.checkNow": "Jetzt prüfen", - "settings.updates.action.checking": "Wird geprüft...", - "settings.updates.toast.latest.title": "Du bist auf dem neuesten Stand", - "settings.updates.toast.latest.description": "Du verwendest die aktuelle Version von Kilo.", - "sound.option.none": "Keine", - "sound.option.alert01": "Alarm 01", - "sound.option.alert02": "Alarm 02", - "sound.option.alert03": "Alarm 03", - "sound.option.alert04": "Alarm 04", - "sound.option.alert05": "Alarm 05", - "sound.option.alert06": "Alarm 06", - "sound.option.alert07": "Alarm 07", - "sound.option.alert08": "Alarm 08", - "sound.option.alert09": "Alarm 09", - "sound.option.alert10": "Alarm 10", - "sound.option.bipbop01": "Bip-bop 01", - "sound.option.bipbop02": "Bip-bop 02", - "sound.option.bipbop03": "Bip-bop 03", - "sound.option.bipbop04": "Bip-bop 04", - "sound.option.bipbop05": "Bip-bop 05", - "sound.option.bipbop06": "Bip-bop 06", - "sound.option.bipbop07": "Bip-bop 07", - "sound.option.bipbop08": "Bip-bop 08", - "sound.option.bipbop09": "Bip-bop 09", - "sound.option.bipbop10": "Bip-bop 10", - "sound.option.staplebops01": "Staplebops 01", - "sound.option.staplebops02": "Staplebops 02", - "sound.option.staplebops03": "Staplebops 03", - "sound.option.staplebops04": "Staplebops 04", - "sound.option.staplebops05": "Staplebops 05", - "sound.option.staplebops06": "Staplebops 06", - "sound.option.staplebops07": "Staplebops 07", - "sound.option.nope01": "Nein 01", - "sound.option.nope02": "Nein 02", - "sound.option.nope03": "Nein 03", - "sound.option.nope04": "Nein 04", - "sound.option.nope05": "Nein 05", - "sound.option.nope06": "Nein 06", - "sound.option.nope07": "Nein 07", - "sound.option.nope08": "Nein 08", - "sound.option.nope09": "Nein 09", - "sound.option.nope10": "Nein 10", - "sound.option.nope11": "Nein 11", - "sound.option.nope12": "Nein 12", - "sound.option.yup01": "Ja 01", - "sound.option.yup02": "Ja 02", - "sound.option.yup03": "Ja 03", - "sound.option.yup04": "Ja 04", - "sound.option.yup05": "Ja 05", - "sound.option.yup06": "Ja 06", - "settings.general.notifications.agent.title": "Agent", - "settings.general.notifications.agent.description": - "Systembenachrichtigung anzeigen, wenn der Agent fertig ist oder Aufmerksamkeit benötigt", - "settings.general.notifications.permissions.title": "Berechtigungen", - "settings.general.notifications.permissions.description": - "Systembenachrichtigung anzeigen, wenn eine Berechtigung erforderlich ist", - "settings.general.notifications.errors.title": "Fehler", - "settings.general.notifications.errors.description": "Systembenachrichtigung anzeigen, wenn ein Fehler auftritt", - "settings.general.sounds.agent.title": "Agent", - "settings.general.sounds.agent.description": "Ton abspielen, wenn der Agent fertig ist oder Aufmerksamkeit benötigt", - "settings.general.sounds.permissions.title": "Berechtigungen", - "settings.general.sounds.permissions.description": "Ton abspielen, wenn eine Berechtigung erforderlich ist", - "settings.general.sounds.errors.title": "Fehler", - "settings.general.sounds.errors.description": "Ton abspielen, wenn ein Fehler auftritt", - "settings.shortcuts.title": "Tastenkombinationen", - "settings.shortcuts.reset.button": "Auf Standard zurücksetzen", - "settings.shortcuts.reset.toast.title": "Tastenkombinationen zurückgesetzt", - "settings.shortcuts.reset.toast.description": "Die Tastenkombinationen wurden auf die Standardwerte zurückgesetzt.", - "settings.shortcuts.conflict.title": "Tastenkombination bereits in Verwendung", - "settings.shortcuts.conflict.description": "{{keybind}} ist bereits {{titles}} zugewiesen.", - "settings.shortcuts.unassigned": "Nicht zugewiesen", - "settings.shortcuts.pressKeys": "Tasten drücken", - "settings.shortcuts.search.placeholder": "Tastenkürzel suchen", - "settings.shortcuts.search.empty": "Keine Tastenkürzel gefunden", - "settings.shortcuts.group.general": "Allgemein", - "settings.shortcuts.group.session": "Sitzung", - "settings.shortcuts.group.navigation": "Navigation", - "settings.shortcuts.group.modelAndAgent": "Modell und Agent", - "settings.shortcuts.group.terminal": "Terminal", - "settings.shortcuts.group.prompt": "Prompt", - "settings.providers.title": "Anbieter", - "settings.providers.description": "Anbietereinstellungen können hier konfiguriert werden.", - "settings.providers.section.connected": "Verbundene Anbieter", - "settings.providers.connected.empty": "Keine verbundenen Anbieter", - "settings.providers.section.popular": "Beliebte Anbieter", - "settings.providers.tag.environment": "Umgebung", - "settings.providers.tag.config": "Konfiguration", - "settings.providers.tag.custom": "Benutzerdefiniert", - "settings.providers.tag.other": "Andere", - "settings.models.title": "Modelle", - "settings.models.description": "Modelleinstellungen können hier konfiguriert werden.", - "settings.agents.title": "Agenten", - "settings.agents.description": "Agenteneinstellungen können hier konfiguriert werden.", - "settings.commands.title": "Befehle", - "settings.commands.description": "Befehlseinstellungen können hier konfiguriert werden.", - "settings.mcp.title": "MCP", - "settings.mcp.description": "MCP-Einstellungen können hier konfiguriert werden.", - "settings.permissions.title": "Berechtigungen", - "settings.permissions.description": "Steuern Sie, welche Tools der Server standardmäßig verwenden darf.", - "settings.permissions.section.tools": "Tools", - "settings.permissions.toast.updateFailed.title": "Berechtigungen konnten nicht aktualisiert werden", - "settings.permissions.action.allow": "Erlauben", - "settings.permissions.action.ask": "Fragen", - "settings.permissions.action.deny": "Verweigern", - "settings.permissions.tool.read.title": "Lesen", - "settings.permissions.tool.read.description": "Lesen einer Datei (stimmt mit dem Dateipfad überein)", - "settings.permissions.tool.edit.title": "Bearbeiten", - "settings.permissions.tool.edit.description": - "Dateien ändern, einschließlich Bearbeitungen, Schreibvorgängen, Patches und Mehrfachbearbeitungen", - "settings.permissions.tool.glob.title": "Glob", - "settings.permissions.tool.glob.description": "Dateien mithilfe von Glob-Mustern abgleichen", - "settings.permissions.tool.grep.title": "Grep", - "settings.permissions.tool.grep.description": "Dateiinhalte mit regulären Ausdrücken durchsuchen", - "settings.permissions.tool.list.title": "Auflisten", - "settings.permissions.tool.list.description": "Dateien in einem Verzeichnis auflisten", - "settings.permissions.tool.bash.title": "Bash", - "settings.permissions.tool.bash.description": "Shell-Befehle ausführen", - "settings.permissions.tool.task.title": "Aufgabe", - "settings.permissions.tool.task.description": "Unteragenten starten", - "settings.permissions.tool.skill.title": "Fähigkeit", - "settings.permissions.tool.skill.description": "Eine Fähigkeit nach Namen laden", - "settings.permissions.tool.lsp.title": "LSP", - "settings.permissions.tool.lsp.description": "Language-Server-Abfragen ausführen", - "settings.permissions.tool.todowrite.title": "Todo schreiben", - "settings.permissions.tool.todowrite.description": "Die Todo-Liste aktualisieren", - "settings.permissions.tool.webfetch.title": "Web-Abruf", - "settings.permissions.tool.webfetch.description": "Inhalt von einer URL abrufen", - "settings.permissions.tool.websearch.title": "Web-Suche", - "settings.permissions.tool.websearch.description": "Das Web durchsuchen", - "settings.permissions.tool.codesearch.title": "Code-Suche", - "settings.permissions.tool.codesearch.description": "Code im Web durchsuchen", - "settings.permissions.tool.external_directory.title": "Externes Verzeichnis", - "settings.permissions.tool.external_directory.description": "Zugriff auf Dateien außerhalb des Projektverzeichnisses", - "settings.permissions.tool.doom_loop.title": "Doom Loop", - "settings.permissions.tool.doom_loop.description": "Wiederholte Tool-Aufrufe mit identischer Eingabe erkennen", - "session.delete.failed.title": "Sitzung konnte nicht gelöscht werden", - "session.delete.title": "Sitzung löschen", - "session.delete.confirm": 'Sitzung "{{name}}" löschen?', - "session.delete.button": "Sitzung löschen", - "workspace.new": "Neuer Arbeitsbereich", - "workspace.type.local": "lokal", - "workspace.type.sandbox": "Sandbox", - "workspace.create.failed.title": "Arbeitsbereich konnte nicht erstellt werden", - "workspace.delete.failed.title": "Arbeitsbereich konnte nicht gelöscht werden", - "workspace.resetting.title": "Arbeitsbereich wird zurückgesetzt", - "workspace.resetting.description": "Dies kann eine Minute dauern.", - "workspace.reset.failed.title": "Arbeitsbereich konnte nicht zurückgesetzt werden", - "workspace.reset.success.title": "Arbeitsbereich zurückgesetzt", - "workspace.reset.success.description": "Der Arbeitsbereich entspricht jetzt dem Standard-Branch.", - "workspace.error.stillPreparing": "Arbeitsbereich wird noch vorbereitet", - "workspace.status.checking": "Suche nach nicht zusammengeführten Änderungen...", - "workspace.status.error": "Git-Status konnte nicht überprüft werden.", - "workspace.status.clean": "Keine nicht zusammengeführten Änderungen erkannt.", - "workspace.status.dirty": "Nicht zusammengeführte Änderungen in diesem Arbeitsbereich erkannt.", - "workspace.delete.title": "Arbeitsbereich löschen", - "workspace.delete.confirm": 'Arbeitsbereich "{{name}}" löschen?', - "workspace.delete.button": "Arbeitsbereich löschen", - "workspace.reset.title": "Arbeitsbereich zurücksetzen", - "workspace.reset.confirm": 'Arbeitsbereich "{{name}}" zurücksetzen?', - "workspace.reset.button": "Arbeitsbereich zurücksetzen", - "workspace.reset.archived.none": "Keine aktiven Sitzungen werden archiviert.", - "workspace.reset.archived.one": "1 Sitzung wird archiviert.", - "workspace.reset.archived.many": "{{count}} Sitzungen werden archiviert.", - "workspace.reset.note": "Dadurch wird der Arbeitsbereich auf den Standard-Branch zurückgesetzt.", - "common.open": "Öffnen", - "dialog.releaseNotes.action.getStarted": "Loslegen", - "dialog.releaseNotes.action.next": "Weiter", - "dialog.releaseNotes.action.hideFuture": "In Zukunft nicht mehr anzeigen", - "dialog.releaseNotes.media.alt": "Vorschau auf die Version", - "toast.project.reloadFailed.title": "Fehler beim Neuladen von {{project}}", - "error.server.invalidConfiguration": "Ungültige Konfiguration", - "common.moreCountSuffix": " (+{{count}} weitere)", - "common.time.justNow": "Gerade eben", - "common.time.minutesAgo.short": "vor {{count}} Min", - "common.time.hoursAgo.short": "vor {{count}} Std", - "common.time.daysAgo.short": "vor {{count}} Tg", - "settings.providers.connected.environmentDescription": "Verbunden aus Ihren Umgebungsvariablen", - "settings.providers.custom.description": "Fügen Sie einen OpenAI-kompatiblen Anbieter per Basis-URL hinzu.", - - "app.server.unreachable": "Konnte {{server}} nicht erreichen", - "app.server.retrying": "Automatische erneute Verbindung...", - "app.server.otherServers": "Andere Server", - "dialog.server.add.usernamePlaceholder": "Benutzername", - "dialog.server.add.passwordPlaceholder": "Passwort", - "server.row.noUsername": "Kein Benutzername", - "session.review.noVcs.createGit.title": "Git-Repository erstellen", - "session.review.noVcs.createGit.description": - "Änderungen in diesem Projekt verfolgen, überprüfen und rückgängig machen", - "session.review.noVcs.createGit.actionLoading": "Git-Repository wird erstellt...", - "session.review.noVcs.createGit.action": "Git-Repository erstellen", - "session.todo.progress": "{{done}} von {{total}} Aufgaben erledigt", - "session.question.progress": "{{current}} von {{total}} Fragen", - "session.header.open.finder": "Finder", - "session.header.open.fileExplorer": "Datei-Explorer", - "session.header.open.fileManager": "Dateimanager", - "session.header.open.app.vscode": "VS Code", - "session.header.open.app.cursor": "Cursor", - "session.header.open.app.zed": "Zed", - "session.header.open.app.textmate": "TextMate", - "session.header.open.app.antigravity": "Antigravity", - "session.header.open.app.terminal": "Terminal", - "session.header.open.app.iterm2": "iTerm2", - "session.header.open.app.ghostty": "Ghostty", - "session.header.open.app.warp": "Warp", - "session.header.open.app.xcode": "Xcode", - "session.header.open.app.androidStudio": "Android Studio", - "session.header.open.app.powershell": "PowerShell", - "session.header.open.app.sublimeText": "Sublime Text", - "debugBar.ariaLabel": "Entwicklungs-Leistungsdiagnose", - "debugBar.na": "n.v.", - "debugBar.nav.label": "NAV", - "debugBar.nav.tip": - "Letzter abgeschlossener Routenübergang, der eine Sitzungsseite berührt, gemessen vom Start des Routers bis zum ersten Rendern nach dem Einschwingen.", - "debugBar.fps.label": "FPS", - "debugBar.fps.tip": "Gleitende Bilder pro Sekunde in den letzten 5 Sekunden.", - "debugBar.frame.label": "FRAME", - "debugBar.frame.tip": "Schlechteste Frame-Zeit in den letzten 5 Sekunden.", - "debugBar.jank.label": "JANK", - "debugBar.jank.tip": "Frames über 32ms in den letzten 5 Sekunden.", - "debugBar.long.label": "LONG", - "debugBar.long.tip": "Blockierte Zeit und Anzahl langer Aufgaben in den letzten 5 Sekunden. Max Aufgabe: {{max}}.", - "debugBar.delay.label": "DELAY", - "debugBar.delay.tip": "Schlechteste beobachtete Eingabeverzögerung in den letzten 5 Sekunden.", - "debugBar.inp.label": "INP", - "debugBar.inp.tip": - "Ungefähre Interaktionsdauer in den letzten 5 Sekunden. Dies ist INP-ähnlich, nicht das offizielle Web Vitals INP.", - "debugBar.cls.label": "CLS", - "debugBar.cls.tip": "Kumulative Layoutverschiebung für die aktuelle App-Lebensdauer.", - "debugBar.mem.label": "MEM", - "debugBar.mem.tipUnavailable": "Verwendeter JS-Heap vs Heap-Limit. Nur Chromium.", - "debugBar.mem.tip": "Verwendeter JS-Heap vs Heap-Limit. {{used}} von {{limit}}.", - "common.key.ctrl": "Strg", - "common.key.alt": "Alt", - "common.key.shift": "Umschalt", - "common.key.meta": "Meta", - "common.key.space": "Leertaste", - "common.key.backspace": "Rücktaste", - "common.key.enter": "Eingabe", - "common.key.tab": "Tab", - "common.key.delete": "Entf", - "common.key.home": "Pos1", - "common.key.end": "Ende", - "common.key.pageUp": "Bild auf", - "common.key.pageDown": "Bild ab", - "common.key.insert": "Einfg", - "common.unknown": "unbekannt", - "error.page.circular": "[Zirkulär]", - "error.globalSDK.noServerAvailable": "Kein Server verfügbar", - "error.globalSDK.serverNotAvailable": "Server nicht verfügbar", - "error.childStore.persistedCacheCreateFailed": "Dauerhafter Cache konnte nicht erstellt werden", - "error.childStore.persistedProjectMetadataCreateFailed": "Dauerhafte Projektmetadaten konnten nicht erstellt werden", - "error.childStore.persistedProjectIconCreateFailed": "Dauerhaftes Projekticon konnte nicht erstellt werden", - "error.childStore.storeCreateFailed": "Speicher konnte nicht erstellt werden", - "terminal.connectionLost.abnormalClose": "WebSocket abnormal geschlossen: {{code}}", -} satisfies Partial> diff --git a/packages/app/src/i18n/es.ts b/packages/app/src/i18n/es.ts deleted file mode 100644 index 9ed93e813e..0000000000 --- a/packages/app/src/i18n/es.ts +++ /dev/null @@ -1,942 +0,0 @@ -export const dict = { - "command.category.suggested": "Sugerido", - "command.category.view": "Ver", - "command.category.project": "Proyecto", - "command.category.provider": "Proveedor", - "command.category.server": "Servidor", - "command.category.session": "Sesión", - "command.category.theme": "Tema", - "command.category.language": "Idioma", - "command.category.file": "Archivo", - "command.category.context": "Contexto", - "command.category.terminal": "Terminal", - "command.category.model": "Modelo", - "command.category.mcp": "MCP", - "command.category.agent": "Agente", - "command.category.permissions": "Permisos", - "command.category.workspace": "Espacio de trabajo", - "command.category.settings": "Ajustes", - - "theme.scheme.system": "Sistema", - "theme.scheme.light": "Claro", - "theme.scheme.dark": "Oscuro", - - "command.sidebar.toggle": "Alternar barra lateral", - "command.project.open": "Abrir proyecto", - "command.provider.connect": "Conectar proveedor", - "command.server.switch": "Cambiar servidor", - "command.settings.open": "Abrir ajustes", - "command.session.previous": "Sesión anterior", - "command.session.next": "Siguiente sesión", - "command.session.previous.unseen": "Sesión no leída anterior", - "command.session.next.unseen": "Siguiente sesión no leída", - "command.session.archive": "Archivar sesión", - - "command.palette": "Paleta de comandos", - - "command.theme.cycle": "Alternar tema", - "command.theme.set": "Usar tema: {{theme}}", - "command.theme.scheme.cycle": "Alternar esquema de color", - "command.theme.scheme.set": "Usar esquema de color: {{scheme}}", - - "command.language.cycle": "Alternar idioma", - "command.language.set": "Usar idioma: {{language}}", - - "command.session.new": "Nueva sesión", - "command.file.open": "Abrir archivo", - "command.tab.close": "Cerrar pestaña", - "command.context.addSelection": "Añadir selección al contexto", - "command.context.addSelection.description": "Añadir las líneas seleccionadas del archivo actual", - "command.input.focus": "Enfocar entrada", - "command.terminal.toggle": "Alternar terminal", - "command.fileTree.toggle": "Alternar árbol de archivos", - "command.review.toggle": "Alternar revisión", - "command.terminal.new": "Nueva terminal", - "command.terminal.new.description": "Crear una nueva pestaña de terminal", - "command.steps.toggle": "Alternar pasos", - "command.steps.toggle.description": "Mostrar u ocultar pasos para el mensaje actual", - "command.message.previous": "Mensaje anterior", - "command.message.previous.description": "Ir al mensaje de usuario anterior", - "command.message.next": "Siguiente mensaje", - "command.message.next.description": "Ir al siguiente mensaje de usuario", - "command.model.choose": "Elegir modelo", - "command.model.choose.description": "Seleccionar un modelo diferente", - "command.mcp.toggle": "Alternar MCPs", - "command.mcp.toggle.description": "Alternar MCPs", - "command.agent.cycle": "Alternar agente", - "command.agent.cycle.description": "Cambiar al siguiente agente", - "command.agent.cycle.reverse": "Alternar agente hacia atrás", - "command.agent.cycle.reverse.description": "Cambiar al agente anterior", - "command.model.variant.cycle": "Alternar esfuerzo de pensamiento", - "command.model.variant.cycle.description": "Cambiar al siguiente nivel de esfuerzo", - "command.prompt.mode.shell": "Shell", - "command.prompt.mode.normal": "Prompt", - "command.permissions.autoaccept.enable": "Aceptar permisos automáticamente", - "command.permissions.autoaccept.disable": "Dejar de aceptar permisos automáticamente", - "command.workspace.toggle": "Alternar espacios de trabajo", - "command.workspace.toggle.description": "Habilitar o deshabilitar múltiples espacios de trabajo en la barra lateral", - "command.session.undo": "Deshacer", - "command.session.undo.description": "Deshacer el último mensaje", - "command.session.redo": "Rehacer", - "command.session.redo.description": "Rehacer el último mensaje deshecho", - "command.session.compact": "Compactar sesión", - "command.session.compact.description": "Resumir la sesión para reducir el tamaño del contexto", - "command.session.fork": "Bifurcar desde mensaje", - "command.session.fork.description": "Crear una nueva sesión desde un mensaje anterior", - "command.session.share": "Compartir sesión", - "command.session.share.description": "Compartir esta sesión y copiar la URL al portapapeles", - "command.session.unshare": "Dejar de compartir sesión", - "command.session.unshare.description": "Dejar de compartir esta sesión", - - "palette.search.placeholder": "Buscar archivos, comandos y sesiones", - "palette.empty": "No se encontraron resultados", - "palette.group.commands": "Comandos", - "palette.group.files": "Archivos", - - "dialog.provider.search.placeholder": "Buscar proveedores", - "dialog.provider.empty": "No se encontraron proveedores", - "dialog.provider.group.popular": "Popular", - "dialog.provider.group.other": "Otro", - "dialog.provider.tag.recommended": "Recomendado", - "dialog.provider.opencode.note": "Modelos seleccionados incluyendo Claude, GPT, Gemini y más", - "dialog.provider.opencode.tagline": "Modelos optimizados y fiables", - "dialog.provider.opencodeGo.tagline": "Suscripción económica para todos", - "dialog.provider.anthropic.note": "Acceso directo a modelos Claude, incluyendo Pro y Max", - "dialog.provider.copilot.note": "Modelos de IA para asistencia de codificación a través de GitHub Copilot", - "dialog.provider.openai.note": "Modelos GPT para tareas de IA generales rápidas y capaces", - "dialog.provider.google.note": "Modelos Gemini para respuestas rápidas y estructuradas", - "dialog.provider.openrouter.note": "Accede a todos los modelos soportados desde un solo proveedor", - "dialog.provider.vercel.note": "Acceso unificado a modelos de IA con enrutamiento inteligente", - - "dialog.model.select.title": "Seleccionar modelo", - "dialog.model.search.placeholder": "Buscar modelos", - "dialog.model.empty": "Sin resultados de modelos", - "dialog.model.manage": "Gestionar modelos", - "dialog.model.manage.description": "Personalizar qué modelos aparecen en el selector de modelos.", - "dialog.model.manage.provider.toggle": "Alternar todos los modelos de {{provider}}", - - "dialog.model.unpaid.freeModels.title": "Modelos gratuitos proporcionados por Kilo", - "dialog.model.unpaid.addMore.title": "Añadir más modelos de proveedores populares", - - "dialog.provider.viewAll": "Ver más proveedores", - - "provider.connect.title": "Conectar {{provider}}", - "provider.connect.title.anthropicProMax": "Iniciar sesión con Claude Pro/Max", - "provider.connect.selectMethod": "Seleccionar método de inicio de sesión para {{provider}}.", - "provider.connect.method.apiKey": "Clave API", - "provider.connect.status.inProgress": "Autorización en progreso...", - "provider.connect.status.waiting": "Esperando autorización...", - "provider.connect.status.failed": "Autorización fallida: {{error}}", - "provider.connect.apiKey.description": - "Introduce tu clave API de {{provider}} para conectar tu cuenta y usar modelos de {{provider}} en Kilo.", - "provider.connect.apiKey.label": "Clave API de {{provider}}", - "provider.connect.apiKey.placeholder": "Clave API", - "provider.connect.apiKey.required": "La clave API es obligatoria", - "provider.connect.opencodeZen.line1": - "OpenCode Zen te da acceso a un conjunto curado de modelos fiables optimizados para agentes de programación.", - "provider.connect.opencodeZen.line2": - "Con una sola clave API obtendrás acceso a modelos como Claude, GPT, Gemini, GLM y más.", - "provider.connect.opencodeZen.visit.prefix": "Visita ", - "provider.connect.opencodeZen.visit.link": "opencode.ai/zen", - "provider.connect.opencodeZen.visit.suffix": " para obtener tu clave API.", - "provider.connect.oauth.code.visit.prefix": "Visita ", - "provider.connect.oauth.code.visit.link": "este enlace", - "provider.connect.oauth.code.visit.suffix": - " para obtener tu código de autorización para conectar tu cuenta y usar modelos de {{provider}} en Kilo.", - "provider.connect.oauth.code.label": "Código de autorización {{method}}", - "provider.connect.oauth.code.placeholder": "Código de autorización", - "provider.connect.oauth.code.required": "El código de autorización es obligatorio", - "provider.connect.oauth.code.invalid": "Código de autorización inválido", - "provider.connect.oauth.auto.visit.prefix": "Visita ", - "provider.connect.oauth.auto.visit.link": "este enlace", - "provider.connect.oauth.auto.visit.suffix": - " e introduce el código a continuación para conectar tu cuenta y usar modelos de {{provider}} en Kilo.", - "provider.connect.oauth.auto.confirmationCode": "Código de confirmación", - "provider.connect.toast.connected.title": "{{provider}} conectado", - "provider.connect.toast.connected.description": "Los modelos de {{provider}} ahora están disponibles para usar.", - - "provider.custom.title": "Proveedor personalizado", - "provider.custom.description.prefix": "Configurar un proveedor compatible con OpenAI. Ver la ", - "provider.custom.description.link": "documentación de configuración del proveedor", - "provider.custom.description.suffix": ".", - "provider.custom.field.providerID.label": "ID del proveedor", - "provider.custom.field.providerID.placeholder": "miproveedor", - "provider.custom.field.providerID.description": "Letras minúsculas, números, guiones o guiones bajos", - "provider.custom.field.name.label": "Nombre para mostrar", - "provider.custom.field.name.placeholder": "Mi Proveedor de IA", - "provider.custom.field.baseURL.label": "URL base", - "provider.custom.field.baseURL.placeholder": "https://api.miproveedor.com/v1", - "provider.custom.field.apiKey.label": "Clave API", - "provider.custom.field.apiKey.placeholder": "Clave API", - "provider.custom.field.apiKey.description": "Opcional. Dejar vacío si gestionas la autenticación mediante cabeceras.", - "provider.custom.models.label": "Modelos", - "provider.custom.models.id.label": "ID", - "provider.custom.models.id.placeholder": "id-modelo", - "provider.custom.models.name.label": "Nombre", - "provider.custom.models.name.placeholder": "Nombre para mostrar", - "provider.custom.models.remove": "Eliminar modelo", - "provider.custom.models.add": "Añadir modelo", - "provider.custom.headers.label": "Cabeceras (opcional)", - "provider.custom.headers.key.label": "Cabecera", - "provider.custom.headers.key.placeholder": "Nombre-Cabecera", - "provider.custom.headers.value.label": "Valor", - "provider.custom.headers.value.placeholder": "valor", - "provider.custom.headers.remove": "Eliminar cabecera", - "provider.custom.headers.add": "Añadir cabecera", - "provider.custom.error.providerID.required": "El ID del proveedor es obligatorio", - "provider.custom.error.providerID.format": "Usa letras minúsculas, números, guiones o guiones bajos", - "provider.custom.error.providerID.exists": "Ese ID de proveedor ya existe", - "provider.custom.error.name.required": "El nombre para mostrar es obligatorio", - "provider.custom.error.baseURL.required": "La URL base es obligatoria", - "provider.custom.error.baseURL.format": "Debe comenzar con http:// o https://", - "provider.custom.error.required": "Obligatorio", - "provider.custom.error.duplicate": "Duplicado", - - "provider.disconnect.toast.disconnected.title": "{{provider}} desconectado", - "provider.disconnect.toast.disconnected.description": "Los modelos de {{provider}} ya no están disponibles.", - - "model.tag.free": "Gratis", - "model.tag.latest": "Último", - "model.provider.anthropic": "Anthropic", - "model.provider.openai": "OpenAI", - "model.provider.google": "Google", - "model.provider.xai": "xAI", - "model.provider.meta": "Meta", - "model.input.text": "texto", - "model.input.image": "imagen", - "model.input.audio": "audio", - "model.input.video": "video", - "model.input.pdf": "pdf", - "model.tooltip.allows": "Permite: {{inputs}}", - "model.tooltip.reasoning.allowed": "Permite razonamiento", - "model.tooltip.reasoning.none": "Sin razonamiento", - "model.tooltip.context": "Límite de contexto {{limit}}", - - "common.search.placeholder": "Buscar", - "common.goBack": "Volver", - "common.goForward": "Avanzar", - "common.loading": "Cargando", - "common.loading.ellipsis": "...", - "common.cancel": "Cancelar", - "common.connect": "Conectar", - "common.disconnect": "Desconectar", - "common.continue": "Enviar", - "common.submit": "Enviar", - "common.save": "Guardar", - "common.saving": "Guardando...", - "common.default": "Predeterminado", - "common.attachment": "adjunto", - - "prompt.placeholder.shell": "Introduce comando de shell... {{example}}", - "prompt.placeholder.normal": 'Pregunta cualquier cosa... "{{example}}"', - "prompt.placeholder.simple": "Pregunta cualquier cosa...", - "prompt.placeholder.summarizeComments": "Resumir comentarios…", - "prompt.placeholder.summarizeComment": "Resumir comentario…", - "prompt.mode.shell": "Shell", - "prompt.mode.normal": "Prompt", - "prompt.mode.shell.exit": "esc para salir", - - "prompt.example.1": "Arreglar un TODO en el código", - "prompt.example.2": "¿Cuál es el stack tecnológico de este proyecto?", - "prompt.example.3": "Arreglar pruebas rotas", - "prompt.example.4": "Explicar cómo funciona la autenticación", - "prompt.example.5": "Encontrar y arreglar vulnerabilidades de seguridad", - "prompt.example.6": "Añadir pruebas unitarias para el servicio de usuario", - "prompt.example.7": "Refactorizar esta función para que sea más legible", - "prompt.example.8": "¿Qué significa este error?", - "prompt.example.9": "Ayúdame a depurar este problema", - "prompt.example.10": "Generar documentación de API", - "prompt.example.11": "Optimizar consultas a la base de datos", - "prompt.example.12": "Añadir validación de entrada", - "prompt.example.13": "Crear un nuevo componente para...", - "prompt.example.14": "¿Cómo despliego este proyecto?", - "prompt.example.15": "Revisar mi código para mejores prácticas", - "prompt.example.16": "Añadir manejo de errores a esta función", - "prompt.example.17": "Explicar este patrón de regex", - "prompt.example.18": "Convertir esto a TypeScript", - "prompt.example.19": "Añadir logging en todo el código", - "prompt.example.20": "¿Qué dependencias están desactualizadas?", - "prompt.example.21": "Ayúdame a escribir un script de migración", - "prompt.example.22": "Implementar caché para este endpoint", - "prompt.example.23": "Añadir paginación a esta lista", - "prompt.example.24": "Crear un comando CLI para...", - "prompt.example.25": "¿Cómo funcionan las variables de entorno aquí?", - - "prompt.popover.emptyResults": "Sin resultados coincidentes", - "prompt.popover.emptyCommands": "Sin comandos coincidentes", - "prompt.dropzone.label": "Suelta imágenes, PDFs o archivos de texto aquí", - "prompt.dropzone.file.label": "Suelta para @mencionar archivo", - "prompt.slash.badge.custom": "personalizado", - "prompt.slash.badge.skill": "skill", - "prompt.slash.badge.mcp": "mcp", - "prompt.context.active": "activo", - "prompt.context.includeActiveFile": "Incluir archivo activo", - "prompt.context.removeActiveFile": "Eliminar archivo activo del contexto", - "prompt.context.removeFile": "Eliminar archivo del contexto", - "prompt.action.attachFile": "Adjuntar archivo", - "prompt.attachment.remove": "Eliminar adjunto", - "prompt.action.send": "Enviar", - "prompt.action.stop": "Detener", - - "prompt.toast.pasteUnsupported.title": "Adjunto no compatible", - "prompt.toast.pasteUnsupported.description": "Solo se pueden adjuntar imágenes, PDFs o archivos de texto aquí.", - "prompt.toast.modelAgentRequired.title": "Selecciona un agente y modelo", - "prompt.toast.modelAgentRequired.description": "Elige un agente y modelo antes de enviar un prompt.", - "prompt.toast.worktreeCreateFailed.title": "Fallo al crear el árbol de trabajo", - "prompt.toast.sessionCreateFailed.title": "Fallo al crear la sesión", - "prompt.toast.shellSendFailed.title": "Fallo al enviar comando de shell", - "prompt.toast.commandSendFailed.title": "Fallo al enviar comando", - "prompt.toast.promptSendFailed.title": "Fallo al enviar prompt", - "prompt.toast.promptSendFailed.description": "No se pudo recuperar la sesión", - - "dialog.mcp.title": "MCPs", - "dialog.mcp.description": "{{enabled}} de {{total}} habilitados", - "dialog.mcp.empty": "No hay MCPs configurados", - - "dialog.lsp.empty": "LSPs detectados automáticamente por tipo de archivo", - "dialog.plugins.empty": "Plugins configurados en opencode.json", - - "mcp.status.connected": "conectado", - "mcp.status.failed": "fallido", - "mcp.status.needs_auth": "necesita auth", - "mcp.status.disabled": "deshabilitado", - - "dialog.fork.empty": "No hay mensajes desde donde bifurcar", - - "dialog.directory.search.placeholder": "Buscar carpetas", - "dialog.directory.empty": "No se encontraron carpetas", - - "dialog.server.title": "Servidores", - "dialog.server.description": "Cambiar a qué servidor de Kilo se conecta esta app.", - "dialog.server.search.placeholder": "Buscar servidores", - "dialog.server.empty": "No hay servidores aún", - "dialog.server.add.title": "Añadir un servidor", - "dialog.server.add.url": "URL del servidor", - "dialog.server.add.placeholder": "http://localhost:4096", - "dialog.server.add.error": "No se pudo conectar al servidor", - "dialog.server.add.checking": "Comprobando...", - "dialog.server.add.button": "Añadir servidor", - "dialog.server.add.name": "Nombre del servidor (opcional)", - "dialog.server.add.namePlaceholder": "Localhost", - "dialog.server.add.username": "Nombre de usuario (opcional)", - "dialog.server.add.password": "Contraseña (opcional)", - "dialog.server.edit.title": "Editar servidor", - "dialog.server.default.title": "Servidor predeterminado", - "dialog.server.default.description": - "Conectar a este servidor al iniciar la app en lugar de iniciar un servidor local. Requiere reinicio.", - "dialog.server.default.none": "Ningún servidor seleccionado", - "dialog.server.default.set": "Establecer servidor actual como predeterminado", - "dialog.server.default.clear": "Limpiar", - "dialog.server.action.remove": "Eliminar servidor", - - "dialog.server.menu.edit": "Editar", - "dialog.server.menu.default": "Establecer como predeterminado", - "dialog.server.menu.defaultRemove": "Quitar predeterminado", - "dialog.server.menu.delete": "Eliminar", - "dialog.server.current": "Servidor actual", - "dialog.server.status.default": "Predeterminado", - - "dialog.project.edit.title": "Editar proyecto", - "dialog.project.edit.name": "Nombre", - "dialog.project.edit.icon": "Icono", - "dialog.project.edit.icon.alt": "Icono del proyecto", - "dialog.project.edit.icon.hint": "Haz clic o arrastra una imagen", - "dialog.project.edit.icon.recommended": "Recomendado: 128x128px", - "dialog.project.edit.color": "Color", - "dialog.project.edit.color.select": "Seleccionar color {{color}}", - "dialog.project.edit.worktree.startup": "Script de inicio del espacio de trabajo", - "dialog.project.edit.worktree.startup.description": - "Se ejecuta después de crear un nuevo espacio de trabajo (árbol de trabajo).", - "dialog.project.edit.worktree.startup.placeholder": "p. ej. bun install", - - "context.breakdown.title": "Desglose de Contexto", - "context.breakdown.note": - 'Desglose aproximado de tokens de entrada. "Otro" incluye definiciones de herramientas y sobrecarga.', - "context.breakdown.system": "Sistema", - "context.breakdown.user": "Usuario", - "context.breakdown.assistant": "Asistente", - "context.breakdown.tool": "Llamadas a herramientas", - "context.breakdown.other": "Otro", - - "context.systemPrompt.title": "Prompt del Sistema", - "context.rawMessages.title": "Mensajes en bruto", - - "context.stats.session": "Sesión", - "context.stats.messages": "Mensajes", - "context.stats.provider": "Proveedor", - "context.stats.model": "Modelo", - "context.stats.limit": "Límite de Contexto", - "context.stats.totalTokens": "Tokens Totales", - "context.stats.usage": "Uso", - "context.stats.inputTokens": "Tokens de Entrada", - "context.stats.outputTokens": "Tokens de Salida", - "context.stats.reasoningTokens": "Tokens de Razonamiento", - "context.stats.cacheTokens": "Tokens de Caché (lectura/escritura)", - "context.stats.userMessages": "Mensajes de Usuario", - "context.stats.assistantMessages": "Mensajes de Asistente", - "context.stats.totalCost": "Costo Total", - "context.stats.sessionCreated": "Sesión Creada", - "context.stats.lastActivity": "Última Actividad", - - "context.usage.tokens": "Tokens", - "context.usage.usage": "Uso", - "context.usage.cost": "Costo", - "context.usage.clickToView": "Haz clic para ver contexto", - "context.usage.view": "Ver uso del contexto", - - "language.en": "English", - "language.zh": "简体中文", - "language.zht": "繁體中文", - "language.ko": "한국어", - "language.de": "Deutsch", - "language.es": "Español", - "language.fr": "Français", - "language.da": "Dansk", - "language.ja": "日本語", - "language.pl": "Polski", - "language.ru": "Русский", - "language.ar": "العربية", - "language.no": "Norsk", - "language.br": "Português (Brasil)", - "language.bs": "Bosanski", - "language.th": "ไทย", - "language.tr": "Türkçe", - - "toast.language.title": "Idioma", - "toast.language.description": "Cambiado a {{language}}", - - "toast.theme.title": "Tema cambiado", - "toast.scheme.title": "Esquema de color", - - "toast.workspace.enabled.title": "Espacios de trabajo habilitados", - "toast.workspace.enabled.description": "Ahora se muestran varios worktrees en la barra lateral", - "toast.workspace.disabled.title": "Espacios de trabajo deshabilitados", - "toast.workspace.disabled.description": "Solo se muestra el worktree principal en la barra lateral", - - "toast.permissions.autoaccept.on.title": "Aceptando permisos automáticamente", - "toast.permissions.autoaccept.on.description": "Las solicitudes de permisos se aprobarán automáticamente", - "toast.permissions.autoaccept.off.title": "Se dejó de aceptar permisos automáticamente", - "toast.permissions.autoaccept.off.description": "Las solicitudes de permisos requerirán aprobación", - - "toast.model.none.title": "Ningún modelo seleccionado", - "toast.model.none.description": "Conecta un proveedor para resumir esta sesión", - - "toast.file.loadFailed.title": "Fallo al cargar archivo", - "toast.file.listFailed.title": "Fallo al listar archivos", - - "toast.context.noLineSelection.title": "Sin selección de líneas", - "toast.context.noLineSelection.description": "Primero selecciona un rango de líneas en una pestaña de archivo.", - - "toast.session.share.copyFailed.title": "Fallo al copiar URL al portapapeles", - "toast.session.share.success.title": "Sesión compartida", - "toast.session.share.success.description": "¡URL compartida copiada al portapapeles!", - "toast.session.share.failed.title": "Fallo al compartir sesión", - "toast.session.share.failed.description": "Ocurrió un error al compartir la sesión", - - "toast.session.unshare.success.title": "Sesión dejó de compartirse", - "toast.session.unshare.success.description": "¡La sesión dejó de compartirse exitosamente!", - "toast.session.unshare.failed.title": "Fallo al dejar de compartir sesión", - "toast.session.unshare.failed.description": "Ocurrió un error al dejar de compartir la sesión", - - "toast.session.listFailed.title": "Fallo al cargar sesiones para {{project}}", - - "toast.update.title": "Actualización disponible", - "toast.update.description": "Una nueva versión de Kilo ({{version}}) está disponible para instalar.", - "toast.update.action.installRestart": "Instalar y reiniciar", - "toast.update.action.notYet": "Todavía no", - - "error.page.title": "Algo salió mal", - "error.page.description": "Ocurrió un error al cargar la aplicación.", - "error.page.details.label": "Detalles del error", - "error.page.action.restart": "Reiniciar", - "error.page.action.checking": "Comprobando...", - "error.page.action.checkUpdates": "Buscar actualizaciones", - "error.page.action.updateTo": "Actualizar a {{version}}", - "error.page.report.prefix": "Por favor reporta este error al equipo de Kilo", - "error.page.report.discord": "en Discord", - "error.page.version": "Versión: {{version}}", - - "error.dev.rootNotFound": - "Elemento raíz no encontrado. ¿Olvidaste añadirlo a tu index.html? ¿O tal vez el atributo id está mal escrito?", - - "error.globalSync.connectFailed": "No se pudo conectar al servidor. ¿Hay un servidor ejecutándose en `{{url}}`?", - "directory.error.invalidUrl": "URL de directorio inválida.", - - "error.chain.unknown": "Error desconocido", - "error.chain.causedBy": "Causado por:", - "error.chain.apiError": "Error de API", - "error.chain.status": "Estado: {{status}}", - "error.chain.retryable": "Reintentable: {{retryable}}", - "error.chain.responseBody": "Cuerpo de la respuesta:\n{{body}}", - "error.chain.didYouMean": "¿Quisiste decir: {{suggestions}}", - "error.chain.modelNotFound": "Modelo no encontrado: {{provider}}/{{model}}", - "error.chain.checkConfig": "Comprueba los nombres de proveedor/modelo en tu configuración (opencode.json)", - "error.chain.mcpFailed": 'El servidor MCP "{{name}}" falló. Nota, Kilo no soporta autenticación MCP todavía.', - "error.chain.providerAuthFailed": "Autenticación de proveedor fallida ({{provider}}): {{message}}", - "error.chain.providerInitFailed": - 'Fallo al inicializar proveedor "{{provider}}". Comprueba credenciales y configuración.', - "error.chain.configJsonInvalid": "El archivo de configuración en {{path}} no es un JSON(C) válido", - "error.chain.configJsonInvalidWithMessage": - "El archivo de configuración en {{path}} no es un JSON(C) válido: {{message}}", - "error.chain.configDirectoryTypo": - 'El directorio "{{dir}}" en {{path}} no es válido. Renombra el directorio a "{{suggestion}}" o elimínalo. Esto es un error tipográfico común.', - "error.chain.configFrontmatterError": "Fallo al analizar frontmatter en {{path}}:\n{{message}}", - "error.chain.configInvalid": "El archivo de configuración en {{path}} es inválido", - "error.chain.configInvalidWithMessage": "El archivo de configuración en {{path}} es inválido: {{message}}", - - "notification.permission.title": "Permiso requerido", - "notification.permission.description": "{{sessionTitle}} en {{projectName}} necesita permiso", - "notification.question.title": "Pregunta", - "notification.question.description": "{{sessionTitle}} en {{projectName}} tiene una pregunta", - "notification.action.goToSession": "Ir a sesión", - - "notification.session.responseReady.title": "Respuesta lista", - "notification.session.error.title": "Error de sesión", - "notification.session.error.fallbackDescription": "Ocurrió un error", - - "home.recentProjects": "Proyectos recientes", - "home.empty.title": "Sin proyectos recientes", - "home.empty.description": "Empieza abriendo un proyecto local", - - "session.tab.session": "Sesión", - "session.tab.review": "Revisión", - "session.tab.context": "Contexto", - "session.panel.reviewAndFiles": "Revisión y archivos", - "session.review.filesChanged": "{{count}} Archivos Cambiados", - "session.review.change.one": "Cambio", - "session.review.change.other": "Cambios", - "session.review.loadingChanges": "Cargando cambios...", - "session.review.empty": "No hay cambios en esta sesión aún", - "session.review.noVcs": "No se detectó Sistema de Control de Versiones Git, los cambios no se muestran", - "session.review.noSnapshot": - "El seguimiento de instantáneas está deshabilitado en la configuración, por lo que los cambios de sesión no están disponibles", - "session.review.noChanges": "Sin cambios", - - "session.files.selectToOpen": "Selecciona un archivo para abrir", - "session.files.all": "Todos los archivos", - "session.files.empty": "Sin archivos", - "session.files.binaryContent": "Archivo binario (el contenido no puede ser mostrado)", - - "session.messages.renderEarlier": "Renderizar mensajes anteriores", - "session.messages.loadingEarlier": "Cargando mensajes anteriores...", - "session.messages.loadEarlier": "Cargar mensajes anteriores", - "session.messages.loading": "Cargando mensajes...", - "session.messages.jumpToLatest": "Ir al último", - - "session.context.addToContext": "Añadir {{selection}} al contexto", - "session.todo.title": "Tareas", - "session.todo.collapse": "Contraer", - "session.todo.expand": "Expandir", - "session.followupDock.summary.one": "{{count}} mensaje en cola", - "session.followupDock.summary.other": "{{count}} mensajes en cola", - "session.followupDock.sendNow": "Enviar ahora", - "session.followupDock.edit": "Editar", - "session.followupDock.collapse": "Contraer mensajes en cola", - "session.followupDock.expand": "Expandir mensajes en cola", - "session.revertDock.summary.one": "{{count}} mensaje revertido", - "session.revertDock.summary.other": "{{count}} mensajes revertidos", - "session.revertDock.collapse": "Contraer mensajes revertidos", - "session.revertDock.expand": "Expandir mensajes revertidos", - "session.revertDock.restore": "Restaurar mensaje", - - "session.new.title": "Construye lo que quieras", - "session.new.worktree.main": "Rama principal", - "session.new.worktree.mainWithBranch": "Rama principal ({{branch}})", - "session.new.worktree.create": "Crear nuevo árbol de trabajo", - "session.new.lastModified": "Última modificación", - - "session.header.search.placeholder": "Buscar {{project}}", - "session.header.searchFiles": "Buscar archivos", - "session.header.openIn": "Abrir en", - "session.header.open.action": "Abrir {{app}}", - "session.header.open.ariaLabel": "Abrir en {{app}}", - "session.header.open.menu": "Opciones de apertura", - "session.header.open.copyPath": "Copiar ruta", - - "status.popover.trigger": "Estado", - "status.popover.ariaLabel": "Configuraciones del servidor", - "status.popover.tab.servers": "Servidores", - "status.popover.tab.mcp": "MCP", - "status.popover.tab.lsp": "LSP", - "status.popover.tab.plugins": "Plugins", - "status.popover.action.manageServers": "Administrar servidores", - - "session.share.popover.title": "Publicar en web", - "session.share.popover.description.shared": - "Esta sesión es pública en la web. Es accesible para cualquiera con el enlace.", - "session.share.popover.description.unshared": - "Compartir sesión públicamente en la web. Será accesible para cualquiera con el enlace.", - "session.share.action.share": "Compartir", - "session.share.action.publish": "Publicar", - "session.share.action.publishing": "Publicando...", - "session.share.action.unpublish": "Despublicar", - "session.share.action.unpublishing": "Despublicando...", - "session.share.action.view": "Ver", - "session.share.copy.copied": "Copiado", - "session.share.copy.copyLink": "Copiar enlace", - - "lsp.tooltip.none": "Sin servidores LSP", - "lsp.label.connected": "{{count}} LSP", - - "prompt.loading": "Cargando prompt...", - "terminal.loading": "Cargando terminal...", - "terminal.title": "Terminal", - "terminal.title.numbered": "Terminal {{number}}", - "terminal.close": "Cerrar terminal", - "terminal.connectionLost.title": "Conexión perdida", - "terminal.connectionLost.description": - "La conexión del terminal se interrumpió. Esto puede ocurrir cuando el servidor se reinicia.", - - "common.closeTab": "Cerrar pestaña", - "common.dismiss": "Descartar", - "common.requestFailed": "Solicitud fallida", - "common.moreOptions": "Más opciones", - "common.learnMore": "Saber más", - "common.rename": "Renombrar", - "common.reset": "Restablecer", - "common.archive": "Archivar", - "common.delete": "Eliminar", - "common.close": "Cerrar", - "common.edit": "Editar", - "common.loadMore": "Cargar más", - "common.key.esc": "ESC", - - "sidebar.menu.toggle": "Alternar menú", - "sidebar.nav.projectsAndSessions": "Proyectos y sesiones", - "sidebar.settings": "Ajustes", - "sidebar.help": "Ayuda", - "sidebar.workspaces.enable": "Habilitar espacios de trabajo", - "sidebar.workspaces.disable": "Deshabilitar espacios de trabajo", - "sidebar.gettingStarted.title": "Empezando", - "sidebar.gettingStarted.line1": "Kilo incluye modelos gratuitos para que puedas empezar inmediatamente.", - "sidebar.gettingStarted.line2": "Conecta cualquier proveedor para usar modelos, inc. Claude, GPT, Gemini etc.", - "sidebar.project.recentSessions": "Sesiones recientes", - "sidebar.project.viewAllSessions": "Ver todas las sesiones", - "sidebar.project.clearNotifications": "Borrar notificaciones", - - "app.name.desktop": "Kilo Desktop", - - "settings.section.desktop": "Escritorio", - "settings.section.server": "Servidor", - "settings.tab.general": "General", - "settings.tab.shortcuts": "Atajos", - "settings.desktop.section.wsl": "WSL", - "settings.desktop.wsl.title": "Integración con WSL", - "settings.desktop.wsl.description": "Ejecutar el servidor Kilo dentro de WSL en Windows.", - - "settings.general.section.appearance": "Apariencia", - "settings.general.section.notifications": "Notificaciones del sistema", - "settings.general.section.updates": "Actualizaciones", - "settings.general.section.sounds": "Efectos de sonido", - "settings.general.section.feed": "Feed", - "settings.general.section.display": "Pantalla", - - "settings.general.row.language.title": "Idioma", - "settings.general.row.language.description": "Cambiar el idioma de visualización para Kilo", - "settings.general.row.appearance.title": "Apariencia", - "settings.general.row.appearance.description": "Personaliza cómo se ve Kilo en tu dispositivo", - "settings.general.row.colorScheme.title": "Esquema de color", - "settings.general.row.colorScheme.description": "Elige si Kilo sigue el tema del sistema, claro u oscuro", - "settings.general.row.theme.title": "Tema", - "settings.general.row.theme.description": "Personaliza el tema de Kilo.", - "settings.general.row.font.title": "Fuente de código", - "settings.general.row.font.description": "Personaliza la fuente usada en bloques de código", - "settings.general.row.terminalFont.title": "Terminal Font", - "settings.general.row.terminalFont.description": "Customise the font used in the terminal", - "settings.general.row.uiFont.title": "Fuente de la interfaz", - "settings.general.row.uiFont.description": "Personaliza la fuente usada en toda la interfaz", - "settings.general.row.followup.title": "Comportamiento de seguimiento", - "settings.general.row.followup.description": - "Elige si los prompts de seguimiento se dirigen inmediatamente o esperan en una cola", - "settings.general.row.followup.option.queue": "Cola", - "settings.general.row.followup.option.steer": "Dirigir", - "settings.general.row.reasoningSummaries.title": "Mostrar resúmenes de razonamiento", - "settings.general.row.reasoningSummaries.description": - "Mostrar resúmenes del razonamiento del modelo en la línea de tiempo", - "settings.general.row.shellToolPartsExpanded.title": "Expandir partes de la herramienta shell", - "settings.general.row.shellToolPartsExpanded.description": - "Mostrar las partes de la herramienta shell expandidas por defecto en la línea de tiempo", - "settings.general.row.editToolPartsExpanded.title": "Expandir partes de la herramienta de edición", - "settings.general.row.editToolPartsExpanded.description": - "Mostrar las partes de las herramientas de edición, escritura y parcheado expandidas por defecto en la línea de tiempo", - "settings.general.row.showSessionProgressBar.title": "Mostrar barra de progreso de la sesión", - "settings.general.row.showSessionProgressBar.description": - "Mostrar la barra de progreso animada en la parte superior de la sesión cuando el agente esté trabajando", - "settings.general.row.wayland.title": "Usar Wayland nativo", - "settings.general.row.wayland.description": "Deshabilitar fallback a X11 en Wayland. Requiere reinicio.", - "settings.general.row.wayland.tooltip": - "En Linux con monitores de frecuencia de actualización mixta, Wayland nativo puede ser más estable.", - - "settings.general.row.releaseNotes.title": "Notas de la versión", - "settings.general.row.releaseNotes.description": - 'Mostrar ventanas emergentes de "Novedades" después de las actualizaciones', - - "settings.updates.row.startup.title": "Buscar actualizaciones al iniciar", - "settings.updates.row.startup.description": "Buscar actualizaciones automáticamente cuando se inicia Kilo", - "settings.updates.row.check.title": "Buscar actualizaciones", - "settings.updates.row.check.description": "Buscar actualizaciones manualmente e instalarlas si hay alguna", - "settings.updates.action.checkNow": "Buscar ahora", - "settings.updates.action.checking": "Buscando...", - "settings.updates.toast.latest.title": "Estás al día", - "settings.updates.toast.latest.description": "Estás usando la última versión de Kilo.", - "sound.option.none": "Ninguno", - "sound.option.alert01": "Alerta 01", - "sound.option.alert02": "Alerta 02", - "sound.option.alert03": "Alerta 03", - "sound.option.alert04": "Alerta 04", - "sound.option.alert05": "Alerta 05", - "sound.option.alert06": "Alerta 06", - "sound.option.alert07": "Alerta 07", - "sound.option.alert08": "Alerta 08", - "sound.option.alert09": "Alerta 09", - "sound.option.alert10": "Alerta 10", - "sound.option.bipbop01": "Bip-bop 01", - "sound.option.bipbop02": "Bip-bop 02", - "sound.option.bipbop03": "Bip-bop 03", - "sound.option.bipbop04": "Bip-bop 04", - "sound.option.bipbop05": "Bip-bop 05", - "sound.option.bipbop06": "Bip-bop 06", - "sound.option.bipbop07": "Bip-bop 07", - "sound.option.bipbop08": "Bip-bop 08", - "sound.option.bipbop09": "Bip-bop 09", - "sound.option.bipbop10": "Bip-bop 10", - "sound.option.staplebops01": "Staplebops 01", - "sound.option.staplebops02": "Staplebops 02", - "sound.option.staplebops03": "Staplebops 03", - "sound.option.staplebops04": "Staplebops 04", - "sound.option.staplebops05": "Staplebops 05", - "sound.option.staplebops06": "Staplebops 06", - "sound.option.staplebops07": "Staplebops 07", - "sound.option.nope01": "No 01", - "sound.option.nope02": "No 02", - "sound.option.nope03": "No 03", - "sound.option.nope04": "No 04", - "sound.option.nope05": "No 05", - "sound.option.nope06": "No 06", - "sound.option.nope07": "No 07", - "sound.option.nope08": "No 08", - "sound.option.nope09": "No 09", - "sound.option.nope10": "No 10", - "sound.option.nope11": "No 11", - "sound.option.nope12": "No 12", - "sound.option.yup01": "Sí 01", - "sound.option.yup02": "Sí 02", - "sound.option.yup03": "Sí 03", - "sound.option.yup04": "Sí 04", - "sound.option.yup05": "Sí 05", - "sound.option.yup06": "Sí 06", - - "settings.general.notifications.agent.title": "Agente", - "settings.general.notifications.agent.description": - "Mostrar notificación del sistema cuando el agente termine o necesite atención", - "settings.general.notifications.permissions.title": "Permisos", - "settings.general.notifications.permissions.description": - "Mostrar notificación del sistema cuando se requiera un permiso", - "settings.general.notifications.errors.title": "Errores", - "settings.general.notifications.errors.description": "Mostrar notificación del sistema cuando ocurra un error", - - "settings.general.sounds.agent.title": "Agente", - "settings.general.sounds.agent.description": "Reproducir sonido cuando el agente termine o necesite atención", - "settings.general.sounds.permissions.title": "Permisos", - "settings.general.sounds.permissions.description": "Reproducir sonido cuando se requiera un permiso", - "settings.general.sounds.errors.title": "Errores", - "settings.general.sounds.errors.description": "Reproducir sonido cuando ocurra un error", - - "settings.shortcuts.title": "Atajos de teclado", - "settings.shortcuts.reset.button": "Restablecer a valores predeterminados", - "settings.shortcuts.reset.toast.title": "Atajos restablecidos", - "settings.shortcuts.reset.toast.description": - "Los atajos de teclado han sido restablecidos a los valores predeterminados.", - "settings.shortcuts.conflict.title": "Atajo ya en uso", - "settings.shortcuts.conflict.description": "{{keybind}} ya está asignado a {{titles}}.", - "settings.shortcuts.unassigned": "Sin asignar", - "settings.shortcuts.pressKeys": "Presiona teclas", - "settings.shortcuts.search.placeholder": "Buscar atajos", - "settings.shortcuts.search.empty": "No se encontraron atajos", - - "settings.shortcuts.group.general": "General", - "settings.shortcuts.group.session": "Sesión", - "settings.shortcuts.group.navigation": "Navegación", - "settings.shortcuts.group.modelAndAgent": "Modelo y agente", - "settings.shortcuts.group.terminal": "Terminal", - "settings.shortcuts.group.prompt": "Prompt", - - "settings.providers.title": "Proveedores", - "settings.providers.description": "La configuración de proveedores estará disponible aquí.", - "settings.providers.section.connected": "Proveedores conectados", - "settings.providers.connected.empty": "No hay proveedores conectados", - "settings.providers.section.popular": "Proveedores populares", - "settings.providers.tag.environment": "Entorno", - "settings.providers.tag.config": "Configuración", - "settings.providers.tag.custom": "Personalizado", - "settings.providers.tag.other": "Otro", - "settings.models.title": "Modelos", - "settings.models.description": "La configuración de modelos estará disponible aquí.", - "settings.agents.title": "Agentes", - "settings.agents.description": "La configuración de agentes estará disponible aquí.", - "settings.commands.title": "Comandos", - "settings.commands.description": "La configuración de comandos estará disponible aquí.", - "settings.mcp.title": "MCP", - "settings.mcp.description": "La configuración de MCP estará disponible aquí.", - - "settings.permissions.title": "Permisos", - "settings.permissions.description": "Controla qué herramientas puede usar el servidor por defecto.", - "settings.permissions.section.tools": "Herramientas", - "settings.permissions.toast.updateFailed.title": "Fallo al actualizar permisos", - - "settings.permissions.action.allow": "Permitir", - "settings.permissions.action.ask": "Preguntar", - "settings.permissions.action.deny": "Denegar", - - "settings.permissions.tool.read.title": "Leer", - "settings.permissions.tool.read.description": "Leer un archivo (coincide con la ruta del archivo)", - "settings.permissions.tool.edit.title": "Editar", - "settings.permissions.tool.edit.description": - "Modificar archivos, incluyendo ediciones, escrituras, parches y multi-ediciones", - "settings.permissions.tool.glob.title": "Glob", - "settings.permissions.tool.glob.description": "Coincidir archivos usando patrones glob", - "settings.permissions.tool.grep.title": "Grep", - "settings.permissions.tool.grep.description": "Buscar contenidos de archivo usando expresiones regulares", - "settings.permissions.tool.list.title": "Listar", - "settings.permissions.tool.list.description": "Listar archivos dentro de un directorio", - "settings.permissions.tool.bash.title": "Bash", - "settings.permissions.tool.bash.description": "Ejecutar comandos de shell", - "settings.permissions.tool.task.title": "Tarea", - "settings.permissions.tool.task.description": "Lanzar sub-agentes", - "settings.permissions.tool.skill.title": "Habilidad", - "settings.permissions.tool.skill.description": "Cargar una habilidad por nombre", - "settings.permissions.tool.lsp.title": "LSP", - "settings.permissions.tool.lsp.description": "Ejecutar consultas de servidor de lenguaje", - "settings.permissions.tool.todowrite.title": "Escribir Todo", - "settings.permissions.tool.todowrite.description": "Actualizar la lista de tareas", - "settings.permissions.tool.webfetch.title": "Web Fetch", - "settings.permissions.tool.webfetch.description": "Obtener contenido de una URL", - "settings.permissions.tool.websearch.title": "Búsqueda Web", - "settings.permissions.tool.websearch.description": "Buscar en la web", - "settings.permissions.tool.codesearch.title": "Búsqueda de Código", - "settings.permissions.tool.codesearch.description": "Buscar código en la web", - "settings.permissions.tool.external_directory.title": "Directorio Externo", - "settings.permissions.tool.external_directory.description": "Acceder a archivos fuera del directorio del proyecto", - "settings.permissions.tool.doom_loop.title": "Bucle Infinito", - "settings.permissions.tool.doom_loop.description": "Detectar llamadas a herramientas repetidas con entrada idéntica", - - "session.delete.failed.title": "Fallo al eliminar sesión", - "session.delete.title": "Eliminar sesión", - "session.delete.confirm": '¿Eliminar sesión "{{name}}"?', - "session.delete.button": "Eliminar sesión", - - "workspace.new": "Nuevo espacio de trabajo", - "workspace.type.local": "local", - "workspace.type.sandbox": "sandbox", - "workspace.create.failed.title": "Fallo al crear espacio de trabajo", - "workspace.delete.failed.title": "Fallo al eliminar espacio de trabajo", - "workspace.resetting.title": "Restableciendo espacio de trabajo", - "workspace.resetting.description": "Esto puede tomar un minuto.", - "workspace.reset.failed.title": "Fallo al restablecer espacio de trabajo", - "workspace.reset.success.title": "Espacio de trabajo restablecido", - "workspace.reset.success.description": "El espacio de trabajo ahora coincide con la rama predeterminada.", - "workspace.error.stillPreparing": "El espacio de trabajo aún se está preparando", - "workspace.status.checking": "Comprobando cambios no fusionados...", - "workspace.status.error": "No se pudo verificar el estado de git.", - "workspace.status.clean": "No se detectaron cambios no fusionados.", - "workspace.status.dirty": "Cambios no fusionados detectados en este espacio de trabajo.", - "workspace.delete.title": "Eliminar espacio de trabajo", - "workspace.delete.confirm": '¿Eliminar espacio de trabajo "{{name}}"?', - "workspace.delete.button": "Eliminar espacio de trabajo", - "workspace.reset.title": "Restablecer espacio de trabajo", - "workspace.reset.confirm": '¿Restablecer espacio de trabajo "{{name}}"?', - "workspace.reset.button": "Restablecer espacio de trabajo", - "workspace.reset.archived.none": "No se archivarán sesiones activas.", - "workspace.reset.archived.one": "1 sesión será archivada.", - "workspace.reset.archived.many": "{{count}} sesiones serán archivadas.", - "workspace.reset.note": "Esto restablecerá el espacio de trabajo para coincidir con la rama predeterminada.", - "common.open": "Abrir", - "dialog.releaseNotes.action.getStarted": "Comenzar", - "dialog.releaseNotes.action.next": "Siguiente", - "dialog.releaseNotes.action.hideFuture": "No mostrar esto en el futuro", - "dialog.releaseNotes.media.alt": "Vista previa de la versión", - "toast.project.reloadFailed.title": "Error al recargar {{project}}", - "error.server.invalidConfiguration": "Configuración inválida", - "common.moreCountSuffix": " (+{{count}} más)", - "common.time.justNow": "Justo ahora", - "common.time.minutesAgo.short": "hace {{count}} min", - "common.time.hoursAgo.short": "hace {{count}} h", - "common.time.daysAgo.short": "hace {{count}} d", - "settings.providers.connected.environmentDescription": "Conectado desde tus variables de entorno", - "settings.providers.custom.description": "Añade un proveedor compatible con OpenAI por su URL base.", - - "app.server.unreachable": "No se pudo conectar con {{server}}", - "app.server.retrying": "Reintentando automáticamente...", - "app.server.otherServers": "Otros servidores", - "dialog.server.add.usernamePlaceholder": "usuario", - "dialog.server.add.passwordPlaceholder": "contraseña", - "server.row.noUsername": "sin usuario", - "session.review.noVcs.createGit.title": "Crear repositorio Git", - "session.review.noVcs.createGit.description": "Rastrea, revisa y deshaz cambios en este proyecto", - "session.review.noVcs.createGit.actionLoading": "Creando repositorio Git...", - "session.review.noVcs.createGit.action": "Crear repositorio Git", - "session.todo.progress": "{{done}} de {{total}} tareas completadas", - "session.question.progress": "{{current}} de {{total}} preguntas", - "session.header.open.finder": "Finder", - "session.header.open.fileExplorer": "Explorador de archivos", - "session.header.open.fileManager": "Gestor de archivos", - "session.header.open.app.vscode": "VS Code", - "session.header.open.app.cursor": "Cursor", - "session.header.open.app.zed": "Zed", - "session.header.open.app.textmate": "TextMate", - "session.header.open.app.antigravity": "Antigravity", - "session.header.open.app.terminal": "Terminal", - "session.header.open.app.iterm2": "iTerm2", - "session.header.open.app.ghostty": "Ghostty", - "session.header.open.app.warp": "Warp", - "session.header.open.app.xcode": "Xcode", - "session.header.open.app.androidStudio": "Android Studio", - "session.header.open.app.powershell": "PowerShell", - "session.header.open.app.sublimeText": "Sublime Text", - "debugBar.ariaLabel": "Diagnóstico de rendimiento de desarrollo", - "debugBar.na": "n/d", - "debugBar.nav.label": "NAV", - "debugBar.nav.tip": - "Última transición de ruta completada tocando una página de sesión, medida desde el inicio del router hasta el primer pintado después de asentarse.", - "debugBar.fps.label": "FPS", - "debugBar.fps.tip": "Cuadros por segundo en los últimos 5 segundos.", - "debugBar.frame.label": "FRAME", - "debugBar.frame.tip": "Peor tiempo de cuadro en los últimos 5 segundos.", - "debugBar.jank.label": "JANK", - "debugBar.jank.tip": "Cuadros superiores a 32ms en los últimos 5 segundos.", - "debugBar.long.label": "LONG", - "debugBar.long.tip": "Tiempo bloqueado y recuento de tareas largas en los últimos 5 segundos. Tarea máx: {{max}}.", - "debugBar.delay.label": "DELAY", - "debugBar.delay.tip": "Peor retraso de entrada observado en los últimos 5 segundos.", - "debugBar.inp.label": "INP", - "debugBar.inp.tip": - "Duración aproximada de la interacción en los últimos 5 segundos. Esto es similar a INP, no el INP oficial de Web Vitals.", - "debugBar.cls.label": "CLS", - "debugBar.cls.tip": "Cambio de diseño acumulativo para la vida útil actual de la aplicación.", - "debugBar.mem.label": "MEM", - "debugBar.mem.tipUnavailable": "Heap JS usado vs límite de heap. Solo Chromium.", - "debugBar.mem.tip": "Heap JS usado vs límite de heap. {{used}} de {{limit}}.", - "common.key.ctrl": "Ctrl", - "common.key.alt": "Alt", - "common.key.shift": "Mayús", - "common.key.meta": "Meta", - "common.key.space": "Espacio", - "common.key.backspace": "Retroceso", - "common.key.enter": "Intro", - "common.key.tab": "Tab", - "common.key.delete": "Supr", - "common.key.home": "Inicio", - "common.key.end": "Fin", - "common.key.pageUp": "RePág", - "common.key.pageDown": "AvPág", - "common.key.insert": "Insert", - "common.unknown": "desconocido", - "error.page.circular": "[Circular]", - "error.globalSDK.noServerAvailable": "Ningún servidor disponible", - "error.globalSDK.serverNotAvailable": "Servidor no disponible", - "error.childStore.persistedCacheCreateFailed": "Error al crear caché persistente", - "error.childStore.persistedProjectMetadataCreateFailed": "Error al crear metadatos de proyecto persistentes", - "error.childStore.persistedProjectIconCreateFailed": "Error al crear icono de proyecto persistente", - "error.childStore.storeCreateFailed": "Error al crear almacén", - "terminal.connectionLost.abnormalClose": "WebSocket cerrado anormalmente: {{code}}", -} diff --git a/packages/app/src/i18n/fr.ts b/packages/app/src/i18n/fr.ts deleted file mode 100644 index 587076d240..0000000000 --- a/packages/app/src/i18n/fr.ts +++ /dev/null @@ -1,868 +0,0 @@ -export const dict = { - "command.category.suggested": "Suggéré", - "command.category.view": "Affichage", - "command.category.project": "Projet", - "command.category.provider": "Fournisseur", - "command.category.server": "Serveur", - "command.category.session": "Session", - "command.category.theme": "Thème", - "command.category.language": "Langue", - "command.category.file": "Fichier", - "command.category.context": "Contexte", - "command.category.terminal": "Terminal", - "command.category.model": "Modèle", - "command.category.mcp": "MCP", - "command.category.agent": "Agent", - "command.category.permissions": "Permissions", - "command.category.workspace": "Espace de travail", - "command.category.settings": "Paramètres", - "theme.scheme.system": "Système", - "theme.scheme.light": "Clair", - "theme.scheme.dark": "Sombre", - "command.sidebar.toggle": "Basculer la barre latérale", - "command.project.open": "Ouvrir un projet", - "command.provider.connect": "Connecter un fournisseur", - "command.server.switch": "Changer de serveur", - "command.settings.open": "Ouvrir les paramètres", - "command.session.previous": "Session précédente", - "command.session.next": "Session suivante", - "command.session.previous.unseen": "Session non lue précédente", - "command.session.next.unseen": "Session non lue suivante", - "command.session.archive": "Archiver la session", - "command.palette": "Palette de commandes", - "command.theme.cycle": "Changer de thème", - "command.theme.set": "Utiliser le thème : {{theme}}", - "command.theme.scheme.cycle": "Changer de schéma de couleurs", - "command.theme.scheme.set": "Utiliser le schéma de couleurs : {{scheme}}", - "command.language.cycle": "Changer de langue", - "command.language.set": "Utiliser la langue : {{language}}", - "command.session.new": "Nouvelle session", - "command.file.open": "Ouvrir un fichier", - "command.tab.close": "Fermer l'onglet", - "command.context.addSelection": "Ajouter la sélection au contexte", - "command.context.addSelection.description": "Ajouter les lignes sélectionnées du fichier actuel", - "command.input.focus": "Focus sur l'entrée", - "command.terminal.toggle": "Basculer le terminal", - "command.fileTree.toggle": "Basculer l'arborescence des fichiers", - "command.review.toggle": "Basculer la revue", - "command.terminal.new": "Nouveau terminal", - "command.terminal.new.description": "Créer un nouvel onglet de terminal", - "command.steps.toggle": "Basculer les étapes", - "command.steps.toggle.description": "Afficher ou masquer les étapes du message actuel", - "command.message.previous": "Message précédent", - "command.message.previous.description": "Aller au message utilisateur précédent", - "command.message.next": "Message suivant", - "command.message.next.description": "Aller au message utilisateur suivant", - "command.model.choose": "Choisir le modèle", - "command.model.choose.description": "Sélectionner un modèle différent", - "command.mcp.toggle": "Basculer MCP", - "command.mcp.toggle.description": "Basculer les MCPs", - "command.agent.cycle": "Changer d'agent", - "command.agent.cycle.description": "Passer à l'agent suivant", - "command.agent.cycle.reverse": "Changer d'agent (inverse)", - "command.agent.cycle.reverse.description": "Passer à l'agent précédent", - "command.model.variant.cycle": "Changer l'effort de réflexion", - "command.model.variant.cycle.description": "Passer au niveau d'effort suivant", - "command.prompt.mode.shell": "Shell", - "command.prompt.mode.normal": "Prompt", - "command.permissions.autoaccept.enable": "Accepter automatiquement les permissions", - "command.permissions.autoaccept.disable": "Arrêter d'accepter automatiquement les permissions", - "command.workspace.toggle": "Basculer les espaces de travail", - "command.workspace.toggle.description": "Activer ou désactiver plusieurs espaces de travail dans la barre latérale", - "command.session.undo": "Annuler", - "command.session.undo.description": "Annuler le dernier message", - "command.session.redo": "Rétablir", - "command.session.redo.description": "Rétablir le dernier message annulé", - "command.session.compact": "Compacter la session", - "command.session.compact.description": "Résumer la session pour réduire la taille du contexte", - "command.session.fork": "Bifurquer à partir du message", - "command.session.fork.description": "Créer une nouvelle session à partir d'un message précédent", - "command.session.share": "Partager la session", - "command.session.share.description": "Partager cette session et copier l'URL dans le presse-papiers", - "command.session.unshare": "Ne plus partager la session", - "command.session.unshare.description": "Arrêter de partager cette session", - "palette.search.placeholder": "Rechercher des fichiers, des commandes et des sessions", - "palette.empty": "Aucun résultat trouvé", - "palette.group.commands": "Commandes", - "palette.group.files": "Fichiers", - "dialog.provider.search.placeholder": "Rechercher des fournisseurs", - "dialog.provider.empty": "Aucun fournisseur trouvé", - "dialog.provider.group.popular": "Populaire", - "dialog.provider.group.other": "Autre", - "dialog.provider.tag.recommended": "Recommandé", - "dialog.provider.opencode.note": "Modèles sélectionnés incluant Claude, GPT, Gemini et plus", - "dialog.provider.opencode.tagline": "Modèles optimisés et fiables", - "dialog.provider.opencodeGo.tagline": "Abonnement abordable pour tous", - "dialog.provider.anthropic.note": "Connectez-vous avec Claude Pro/Max ou une clé API", - "dialog.provider.copilot.note": "Connectez-vous avec Copilot ou une clé API", - "dialog.provider.openai.note": "Connectez-vous avec ChatGPT Pro/Plus ou une clé API", - "dialog.provider.google.note": "Modèles Gemini pour des réponses rapides et structurées", - "dialog.provider.openrouter.note": "Accédez à tous les modèles pris en charge depuis un seul fournisseur", - "dialog.provider.vercel.note": "Accès unifié aux modèles d'IA avec routage intelligent", - "dialog.model.select.title": "Sélectionner un modèle", - "dialog.model.search.placeholder": "Rechercher des modèles", - "dialog.model.empty": "Aucun résultat de modèle", - "dialog.model.manage": "Gérer les modèles", - "dialog.model.manage.description": "Personnalisez les modèles qui apparaissent dans le sélecteur.", - "dialog.model.manage.provider.toggle": "Basculer tous les modèles {{provider}}", - "dialog.model.unpaid.freeModels.title": "Modèles gratuits fournis par Kilo", - "dialog.model.unpaid.addMore.title": "Ajouter plus de modèles de fournisseurs populaires", - "dialog.provider.viewAll": "Voir plus de fournisseurs", - "provider.connect.title": "Connecter {{provider}}", - "provider.connect.title.anthropicProMax": "Connexion avec Claude Pro/Max", - "provider.connect.selectMethod": "Sélectionnez la méthode de connexion pour {{provider}}.", - "provider.connect.method.apiKey": "Clé API", - "provider.connect.status.inProgress": "Autorisation en cours...", - "provider.connect.status.waiting": "En attente d'autorisation...", - "provider.connect.status.failed": "Échec de l'autorisation : {{error}}", - "provider.connect.apiKey.description": - "Entrez votre clé API {{provider}} pour connecter votre compte et utiliser les modèles {{provider}} dans Kilo.", - "provider.connect.apiKey.label": "Clé API {{provider}}", - "provider.connect.apiKey.placeholder": "Clé API", - "provider.connect.apiKey.required": "La clé API est requise", - "provider.connect.opencodeZen.line1": - "OpenCode Zen vous donne accès à un ensemble sélectionné de modèles fiables et optimisés pour les agents de codage.", - "provider.connect.opencodeZen.line2": - "Avec une seule clé API, vous aurez accès à des modèles tels que Claude, GPT, Gemini, GLM et plus encore.", - "provider.connect.opencodeZen.visit.prefix": "Visitez ", - "provider.connect.opencodeZen.visit.link": "opencode.ai/zen", - "provider.connect.opencodeZen.visit.suffix": " pour récupérer votre clé API.", - "provider.connect.oauth.code.visit.prefix": "Visitez ", - "provider.connect.oauth.code.visit.link": "ce lien", - "provider.connect.oauth.code.visit.suffix": - " pour récupérer votre code d'autorisation afin de connecter votre compte et utiliser les modèles {{provider}} dans Kilo.", - "provider.connect.oauth.code.label": "Code d'autorisation {{method}}", - "provider.connect.oauth.code.placeholder": "Code d'autorisation", - "provider.connect.oauth.code.required": "Le code d'autorisation est requis", - "provider.connect.oauth.code.invalid": "Code d'autorisation invalide", - "provider.connect.oauth.auto.visit.prefix": "Visitez ", - "provider.connect.oauth.auto.visit.link": "ce lien", - "provider.connect.oauth.auto.visit.suffix": - " et entrez le code ci-dessous pour connecter votre compte et utiliser les modèles {{provider}} dans Kilo.", - "provider.connect.oauth.auto.confirmationCode": "Code de confirmation", - "provider.connect.toast.connected.title": "{{provider}} connecté", - "provider.connect.toast.connected.description": "Les modèles {{provider}} sont maintenant disponibles.", - "provider.custom.title": "Fournisseur personnalisé", - "provider.custom.description.prefix": "Configurer un fournisseur compatible OpenAI. Voir la ", - "provider.custom.description.link": "doc de config fournisseur", - "provider.custom.description.suffix": ".", - "provider.custom.field.providerID.label": "ID du fournisseur", - "provider.custom.field.providerID.placeholder": "monfournisseur", - "provider.custom.field.providerID.description": "Lettres minuscules, chiffres, traits d'union ou tirets bas", - "provider.custom.field.name.label": "Nom d'affichage", - "provider.custom.field.name.placeholder": "Mon fournisseur IA", - "provider.custom.field.baseURL.label": "URL de base", - "provider.custom.field.baseURL.placeholder": "https://api.monfournisseur.com/v1", - "provider.custom.field.apiKey.label": "Clé API", - "provider.custom.field.apiKey.placeholder": "Clé API", - "provider.custom.field.apiKey.description": "Optionnel. Laisser vide si vous gérez l'auth via les en-têtes.", - "provider.custom.models.label": "Modèles", - "provider.custom.models.id.label": "ID", - "provider.custom.models.id.placeholder": "id-modele", - "provider.custom.models.name.label": "Nom", - "provider.custom.models.name.placeholder": "Nom d'affichage", - "provider.custom.models.remove": "Supprimer le modèle", - "provider.custom.models.add": "Ajouter un modèle", - "provider.custom.headers.label": "En-têtes (optionnel)", - "provider.custom.headers.key.label": "En-tête", - "provider.custom.headers.key.placeholder": "Nom-En-Tête", - "provider.custom.headers.value.label": "Valeur", - "provider.custom.headers.value.placeholder": "valeur", - "provider.custom.headers.remove": "Supprimer l'en-tête", - "provider.custom.headers.add": "Ajouter un en-tête", - "provider.custom.error.providerID.required": "L'ID du fournisseur est requis", - "provider.custom.error.providerID.format": "Utilisez des lettres minuscules, chiffres, traits d'union ou tirets bas", - "provider.custom.error.providerID.exists": "Cet ID de fournisseur existe déjà", - "provider.custom.error.name.required": "Le nom d'affichage est requis", - "provider.custom.error.baseURL.required": "L'URL de base est requise", - "provider.custom.error.baseURL.format": "Doit commencer par http:// ou https://", - "provider.custom.error.required": "Requis", - "provider.custom.error.duplicate": "Doublon", - "provider.disconnect.toast.disconnected.title": "{{provider}} déconnecté", - "provider.disconnect.toast.disconnected.description": "Les modèles {{provider}} ne sont plus disponibles.", - "model.tag.free": "Gratuit", - "model.tag.latest": "Dernier", - "model.provider.anthropic": "Anthropic", - "model.provider.openai": "OpenAI", - "model.provider.google": "Google", - "model.provider.xai": "xAI", - "model.provider.meta": "Meta", - "model.input.text": "texte", - "model.input.image": "image", - "model.input.audio": "audio", - "model.input.video": "vidéo", - "model.input.pdf": "pdf", - "model.tooltip.allows": "Autorise : {{inputs}}", - "model.tooltip.reasoning.allowed": "Autorise le raisonnement", - "model.tooltip.reasoning.none": "Sans raisonnement", - "model.tooltip.context": "Limite de contexte {{limit}}", - "common.search.placeholder": "Rechercher", - "common.goBack": "Retour", - "common.goForward": "Avancer", - "common.loading": "Chargement", - "common.loading.ellipsis": "...", - "common.cancel": "Annuler", - "common.connect": "Connecter", - "common.disconnect": "Déconnecter", - "common.continue": "Soumettre", - "common.submit": "Soumettre", - "common.save": "Enregistrer", - "common.saving": "Enregistrement...", - "common.default": "Défaut", - "common.attachment": "pièce jointe", - "prompt.placeholder.shell": "Entrez une commande shell... {{example}}", - "prompt.placeholder.normal": 'Demandez n\'importe quoi... "{{example}}"', - "prompt.placeholder.simple": "Demandez n'importe quoi...", - "prompt.placeholder.summarizeComments": "Résumer les commentaires…", - "prompt.placeholder.summarizeComment": "Résumer le commentaire…", - "prompt.mode.shell": "Shell", - "prompt.mode.normal": "Prompt", - "prompt.mode.shell.exit": "esc pour quitter", - "prompt.example.1": "Corriger un TODO dans la base de code", - "prompt.example.2": "Quelle est la pile technique de ce projet ?", - "prompt.example.3": "Réparer les tests échoués", - "prompt.example.4": "Expliquer comment fonctionne l'authentification", - "prompt.example.5": "Trouver et corriger les vulnérabilités de sécurité", - "prompt.example.6": "Ajouter des tests unitaires pour le service utilisateur", - "prompt.example.7": "Refactoriser cette fonction pour être plus lisible", - "prompt.example.8": "Que signifie cette erreur ?", - "prompt.example.9": "Aidez-moi à déboguer ce problème", - "prompt.example.10": "Générer la documentation de l'API", - "prompt.example.11": "Optimiser les requêtes de base de données", - "prompt.example.12": "Ajouter une validation d'entrée", - "prompt.example.13": "Créer un nouveau composant pour...", - "prompt.example.14": "Comment déployer ce projet ?", - "prompt.example.15": "Vérifier mon code pour les meilleures pratiques", - "prompt.example.16": "Ajouter la gestion des erreurs à cette fonction", - "prompt.example.17": "Expliquer ce modèle regex", - "prompt.example.18": "Convertir ceci en TypeScript", - "prompt.example.19": "Ajouter des logs dans toute la base de code", - "prompt.example.20": "Quelles dépendances sont obsolètes ?", - "prompt.example.21": "Aidez-moi à écrire un script de migration", - "prompt.example.22": "Implémenter la mise en cache pour ce point de terminaison", - "prompt.example.23": "Ajouter la pagination à cette liste", - "prompt.example.24": "Créer une commande CLI pour...", - "prompt.example.25": "Comment fonctionnent les variables d'environnement ici ?", - "prompt.popover.emptyResults": "Aucun résultat correspondant", - "prompt.popover.emptyCommands": "Aucune commande correspondante", - "prompt.dropzone.label": "Déposez des images, des PDF ou des fichiers texte ici", - "prompt.dropzone.file.label": "Déposez pour @mentionner le fichier", - "prompt.slash.badge.custom": "personnalisé", - "prompt.slash.badge.skill": "skill", - "prompt.slash.badge.mcp": "mcp", - "prompt.context.active": "actif", - "prompt.context.includeActiveFile": "Inclure le fichier actif", - "prompt.context.removeActiveFile": "Retirer le fichier actif du contexte", - "prompt.context.removeFile": "Retirer le fichier du contexte", - "prompt.action.attachFile": "Joindre un fichier", - "prompt.attachment.remove": "Supprimer la pièce jointe", - "prompt.action.send": "Envoyer", - "prompt.action.stop": "Arrêter", - "prompt.toast.pasteUnsupported.title": "Pièce jointe non prise en charge", - "prompt.toast.pasteUnsupported.description": - "Seules les images, les PDF ou les fichiers texte peuvent être joints ici.", - "prompt.toast.modelAgentRequired.title": "Sélectionnez un agent et un modèle", - "prompt.toast.modelAgentRequired.description": "Choisissez un agent et un modèle avant d'envoyer un message.", - "prompt.toast.worktreeCreateFailed.title": "Échec de la création de l'arbre de travail", - "prompt.toast.sessionCreateFailed.title": "Échec de la création de la session", - "prompt.toast.shellSendFailed.title": "Échec de l'envoi de la commande shell", - "prompt.toast.commandSendFailed.title": "Échec de l'envoi de la commande", - "prompt.toast.promptSendFailed.title": "Échec de l'envoi du message", - "prompt.toast.promptSendFailed.description": "Impossible de récupérer la session", - "dialog.mcp.title": "MCPs", - "dialog.mcp.description": "{{enabled}} sur {{total}} activés", - "dialog.mcp.empty": "Aucun MCP configuré", - "dialog.lsp.empty": "LSPs détectés automatiquement par type de fichier", - "dialog.plugins.empty": "Plugins configurés dans opencode.json", - "mcp.status.connected": "connecté", - "mcp.status.failed": "échoué", - "mcp.status.needs_auth": "nécessite auth", - "mcp.status.disabled": "désactivé", - "dialog.fork.empty": "Aucun message à partir duquel bifurquer", - "dialog.directory.search.placeholder": "Rechercher des dossiers", - "dialog.directory.empty": "Aucun dossier trouvé", - "dialog.server.title": "Serveurs", - "dialog.server.description": "Changez le serveur Kilo auquel cette application se connecte.", - "dialog.server.search.placeholder": "Rechercher des serveurs", - "dialog.server.empty": "Aucun serveur pour l'instant", - "dialog.server.add.title": "Ajouter un serveur", - "dialog.server.add.url": "URL du serveur", - "dialog.server.add.placeholder": "http://localhost:4096", - "dialog.server.add.error": "Impossible de se connecter au serveur", - "dialog.server.add.checking": "Vérification...", - "dialog.server.add.button": "Ajouter un serveur", - "dialog.server.add.name": "Nom du serveur (optionnel)", - "dialog.server.add.namePlaceholder": "Localhost", - "dialog.server.add.username": "Nom d'utilisateur (optionnel)", - "dialog.server.add.password": "Mot de passe (optionnel)", - "dialog.server.edit.title": "Modifier le serveur", - "dialog.server.default.title": "Serveur par défaut", - "dialog.server.default.description": - "Se connecter à ce serveur au lancement de l'application au lieu de démarrer un serveur local. Nécessite un redémarrage.", - "dialog.server.default.none": "Aucun serveur sélectionné", - "dialog.server.default.set": "Définir le serveur actuel comme défaut", - "dialog.server.default.clear": "Effacer", - "dialog.server.action.remove": "Supprimer le serveur", - "dialog.server.menu.edit": "Modifier", - "dialog.server.menu.default": "Définir par défaut", - "dialog.server.menu.defaultRemove": "Supprimer par défaut", - "dialog.server.menu.delete": "Supprimer", - "dialog.server.current": "Serveur actuel", - "dialog.server.status.default": "Défaut", - "dialog.project.edit.title": "Modifier le projet", - "dialog.project.edit.name": "Nom", - "dialog.project.edit.icon": "Icône", - "dialog.project.edit.icon.alt": "Icône du projet", - "dialog.project.edit.icon.hint": "Cliquez ou faites glisser une image", - "dialog.project.edit.icon.recommended": "Recommandé : 128x128px", - "dialog.project.edit.color": "Couleur", - "dialog.project.edit.color.select": "Sélectionner la couleur {{color}}", - "dialog.project.edit.worktree.startup": "Script de démarrage de l'espace de travail", - "dialog.project.edit.worktree.startup.description": - "S'exécute après la création d'un nouvel espace de travail (arbre de travail).", - "dialog.project.edit.worktree.startup.placeholder": "p. ex. bun install", - "context.breakdown.title": "Répartition du contexte", - "context.breakdown.note": - "Répartition approximative des jetons d'entrée. \"Autre\" inclut les définitions d'outils et les frais généraux.", - "context.breakdown.system": "Système", - "context.breakdown.user": "Utilisateur", - "context.breakdown.assistant": "Assistant", - "context.breakdown.tool": "Appels d'outils", - "context.breakdown.other": "Autre", - "context.systemPrompt.title": "Prompt système", - "context.rawMessages.title": "Messages bruts", - "context.stats.session": "Session", - "context.stats.messages": "Messages", - "context.stats.provider": "Fournisseur", - "context.stats.model": "Modèle", - "context.stats.limit": "Limite de contexte", - "context.stats.totalTokens": "Total des jetons", - "context.stats.usage": "Utilisation", - "context.stats.inputTokens": "Jetons d'entrée", - "context.stats.outputTokens": "Jetons de sortie", - "context.stats.reasoningTokens": "Jetons de raisonnement", - "context.stats.cacheTokens": "Jetons de cache (lecture/écriture)", - "context.stats.userMessages": "Messages utilisateur", - "context.stats.assistantMessages": "Messages assistant", - "context.stats.totalCost": "Coût total", - "context.stats.sessionCreated": "Session créée", - "context.stats.lastActivity": "Dernière activité", - "context.usage.tokens": "Jetons", - "context.usage.usage": "Utilisation", - "context.usage.cost": "Coût", - "context.usage.clickToView": "Cliquez pour voir le contexte", - "context.usage.view": "Voir l'utilisation du contexte", - "language.en": "English", - "language.zh": "简体中文", - "language.zht": "繁體中文", - "language.ko": "한국어", - "language.de": "Deutsch", - "language.es": "Español", - "language.fr": "Français", - "language.da": "Dansk", - "language.ja": "日本語", - "language.pl": "Polski", - "language.ru": "Русский", - "language.ar": "العربية", - "language.no": "Norsk", - "language.br": "Português (Brasil)", - "language.bs": "Bosanski", - "language.th": "ไทย", - "language.tr": "Türkçe", - "toast.language.title": "Langue", - "toast.language.description": "Passé à {{language}}", - "toast.theme.title": "Thème changé", - "toast.scheme.title": "Schéma de couleurs", - "toast.workspace.enabled.title": "Espaces de travail activés", - "toast.workspace.enabled.description": "Plusieurs worktrees sont désormais affichés dans la barre latérale", - "toast.workspace.disabled.title": "Espaces de travail désactivés", - "toast.workspace.disabled.description": "Seul le worktree principal est affiché dans la barre latérale", - "toast.permissions.autoaccept.on.title": "Acceptation automatique des permissions", - "toast.permissions.autoaccept.on.description": "Les demandes de permission seront approuvées automatiquement", - "toast.permissions.autoaccept.off.title": "Acceptation automatique des permissions arrêtée", - "toast.permissions.autoaccept.off.description": "Les demandes de permission nécessiteront une approbation", - "toast.model.none.title": "Aucun modèle sélectionné", - "toast.model.none.description": "Connectez un fournisseur pour résumer cette session", - "toast.file.loadFailed.title": "Échec du chargement du fichier", - "toast.file.listFailed.title": "Échec de la liste des fichiers", - "toast.context.noLineSelection.title": "Aucune sélection de lignes", - "toast.context.noLineSelection.description": "Sélectionnez d'abord une plage de lignes dans un onglet de fichier.", - "toast.session.share.copyFailed.title": "Échec de la copie de l'URL dans le presse-papiers", - "toast.session.share.success.title": "Session partagée", - "toast.session.share.success.description": "URL de partage copiée dans le presse-papiers !", - "toast.session.share.failed.title": "Échec du partage de la session", - "toast.session.share.failed.description": "Une erreur s'est produite lors du partage de la session", - "toast.session.unshare.success.title": "Session non partagée", - "toast.session.unshare.success.description": "Session non partagée avec succès !", - "toast.session.unshare.failed.title": "Échec de l'annulation du partage", - "toast.session.unshare.failed.description": "Une erreur s'est produite lors de l'annulation du partage de la session", - "toast.session.listFailed.title": "Échec du chargement des sessions pour {{project}}", - "toast.update.title": "Mise à jour disponible", - "toast.update.description": "Une nouvelle version d'Kilo ({{version}}) est maintenant disponible pour installation.", - "toast.update.action.installRestart": "Installer et redémarrer", - "toast.update.action.notYet": "Pas encore", - "error.page.title": "Quelque chose s'est mal passé", - "error.page.description": "Une erreur s'est produite lors du chargement de l'application.", - "error.page.details.label": "Détails de l'erreur", - "error.page.action.restart": "Redémarrer", - "error.page.action.checking": "Vérification...", - "error.page.action.checkUpdates": "Vérifier les mises à jour", - "error.page.action.updateTo": "Mettre à jour vers {{version}}", - "error.page.report.prefix": "Veuillez signaler cette erreur à l'équipe Kilo", - "error.page.report.discord": "sur Discord", - "error.page.version": "Version : {{version}}", - "error.dev.rootNotFound": - "Élément racine introuvable. Avez-vous oublié de l'ajouter à votre index.html ? Ou peut-être que l'attribut id est mal orthographié ?", - "error.globalSync.connectFailed": - "Impossible de se connecter au serveur. Y a-t-il un serveur en cours d'exécution à `{{url}}` ?", - "directory.error.invalidUrl": "Répertoire invalide dans l'URL.", - "error.chain.unknown": "Erreur inconnue", - "error.chain.causedBy": "Causé par :", - "error.chain.apiError": "Erreur API", - "error.chain.status": "Statut : {{status}}", - "error.chain.retryable": "Réessayable : {{retryable}}", - "error.chain.responseBody": "Corps de la réponse :\n{{body}}", - "error.chain.didYouMean": "Vouliez-vous dire : {{suggestions}}", - "error.chain.modelNotFound": "Modèle introuvable : {{provider}}/{{model}}", - "error.chain.checkConfig": "Vérifiez votre configuration (opencode.json) pour les noms de fournisseur/modèle", - "error.chain.mcpFailed": - "Le serveur MCP \"{{name}}\" a échoué. Notez qu'Kilo ne supporte pas encore l'authentification MCP.", - "error.chain.providerAuthFailed": "Échec de l'authentification du fournisseur ({{provider}}) : {{message}}", - "error.chain.providerInitFailed": - 'Échec de l\'initialisation du fournisseur "{{provider}}". Vérifiez les identifiants et la configuration.', - "error.chain.configJsonInvalid": "Le fichier de configuration à {{path}} n'est pas un JSON(C) valide", - "error.chain.configJsonInvalidWithMessage": - "Le fichier de configuration à {{path}} n'est pas un JSON(C) valide : {{message}}", - "error.chain.configDirectoryTypo": - 'Le répertoire "{{dir}}" dans {{path}} n\'est pas valide. Renommez le répertoire en "{{suggestion}}" ou supprimez-le. C\'est une faute de frappe courante.', - "error.chain.configFrontmatterError": "Échec de l'analyse du frontmatter dans {{path}} :\n{{message}}", - "error.chain.configInvalid": "Le fichier de configuration à {{path}} est invalide", - "error.chain.configInvalidWithMessage": "Le fichier de configuration à {{path}} est invalide : {{message}}", - "notification.permission.title": "Permission requise", - "notification.permission.description": "{{sessionTitle}} dans {{projectName}} a besoin d'une permission", - "notification.question.title": "Question", - "notification.question.description": "{{sessionTitle}} dans {{projectName}} a une question", - "notification.action.goToSession": "Aller à la session", - "notification.session.responseReady.title": "Réponse prête", - "notification.session.error.title": "Erreur de session", - "notification.session.error.fallbackDescription": "Une erreur s'est produite", - "home.recentProjects": "Projets récents", - "home.empty.title": "Aucun projet récent", - "home.empty.description": "Commencez par ouvrir un projet local", - "session.tab.session": "Session", - "session.tab.review": "Revue", - "session.tab.context": "Contexte", - "session.panel.reviewAndFiles": "Revue et fichiers", - "session.review.filesChanged": "{{count}} fichiers modifiés", - "session.review.change.one": "Modification", - "session.review.change.other": "Modifications", - "session.review.loadingChanges": "Chargement des modifications...", - "session.review.empty": "Aucune modification dans cette session pour l'instant", - "session.review.noChanges": "Aucune modification", - "session.review.noVcs": "Aucun système de contrôle de version Git détecté, modifications non affichées", - "session.review.noSnapshot": - "Le suivi des instantanés est désactivé dans la configuration, les modifications de session sont donc indisponibles", - "session.files.selectToOpen": "Sélectionnez un fichier à ouvrir", - "session.files.all": "Tous les fichiers", - "session.files.empty": "Aucun fichier", - "session.files.binaryContent": "Fichier binaire (le contenu ne peut pas être affiché)", - "session.messages.renderEarlier": "Afficher les messages précédents", - "session.messages.loadingEarlier": "Chargement des messages précédents...", - "session.messages.loadEarlier": "Charger les messages précédents", - "session.messages.loading": "Chargement des messages...", - "session.messages.jumpToLatest": "Aller au dernier", - "session.context.addToContext": "Ajouter {{selection}} au contexte", - "session.todo.title": "Tâches", - "session.todo.collapse": "Réduire", - "session.todo.expand": "Développer", - "session.followupDock.summary.one": "{{count}} message en file d'attente", - "session.followupDock.summary.other": "{{count}} messages en file d'attente", - "session.followupDock.sendNow": "Envoyer maintenant", - "session.followupDock.edit": "Modifier", - "session.followupDock.collapse": "Réduire les messages en file d'attente", - "session.followupDock.expand": "Développer les messages en file d'attente", - "session.revertDock.summary.one": "{{count}} message annulé", - "session.revertDock.summary.other": "{{count}} messages annulés", - "session.revertDock.collapse": "Réduire les messages annulés", - "session.revertDock.expand": "Développer les messages annulés", - "session.revertDock.restore": "Restaurer le message", - "session.new.title": "Créez ce que vous voulez", - "session.new.worktree.main": "Branche principale", - "session.new.worktree.mainWithBranch": "Branche principale ({{branch}})", - "session.new.worktree.create": "Créer un nouvel arbre de travail", - "session.new.lastModified": "Dernière modification", - "session.header.search.placeholder": "Rechercher {{project}}", - "session.header.searchFiles": "Rechercher des fichiers", - "session.header.openIn": "Ouvrir dans", - "session.header.open.action": "Ouvrir {{app}}", - "session.header.open.ariaLabel": "Ouvrir dans {{app}}", - "session.header.open.menu": "Options d'ouverture", - "session.header.open.copyPath": "Copier le chemin", - "status.popover.trigger": "Statut", - "status.popover.ariaLabel": "Configurations des serveurs", - "status.popover.tab.servers": "Serveurs", - "status.popover.tab.mcp": "MCP", - "status.popover.tab.lsp": "LSP", - "status.popover.tab.plugins": "Plugins", - "status.popover.action.manageServers": "Gérer les serveurs", - "session.share.popover.title": "Publier sur le web", - "session.share.popover.description.shared": - "Cette session est publique sur le web. Elle est accessible à toute personne disposant du lien.", - "session.share.popover.description.unshared": - "Partager la session publiquement sur le web. Elle sera accessible à toute personne disposant du lien.", - "session.share.action.share": "Partager", - "session.share.action.publish": "Publier", - "session.share.action.publishing": "Publication...", - "session.share.action.unpublish": "Dépublier", - "session.share.action.unpublishing": "Dépublication...", - "session.share.action.view": "Voir", - "session.share.copy.copied": "Copié", - "session.share.copy.copyLink": "Copier le lien", - "lsp.tooltip.none": "Aucun serveur LSP", - "lsp.label.connected": "{{count}} LSP", - "prompt.loading": "Chargement du prompt...", - "terminal.loading": "Chargement du terminal...", - "terminal.title": "Terminal", - "terminal.title.numbered": "Terminal {{number}}", - "terminal.close": "Fermer le terminal", - "terminal.connectionLost.title": "Connexion perdue", - "terminal.connectionLost.description": - "La connexion au terminal a été interrompue. Cela peut arriver lorsque le serveur redémarre.", - "common.closeTab": "Fermer l'onglet", - "common.dismiss": "Ignorer", - "common.requestFailed": "La demande a échoué", - "common.moreOptions": "Plus d'options", - "common.learnMore": "En savoir plus", - "common.rename": "Renommer", - "common.reset": "Réinitialiser", - "common.archive": "Archiver", - "common.delete": "Supprimer", - "common.close": "Fermer", - "common.edit": "Modifier", - "common.loadMore": "Charger plus", - "common.key.esc": "ESC", - "sidebar.menu.toggle": "Basculer le menu", - "sidebar.nav.projectsAndSessions": "Projets et sessions", - "sidebar.settings": "Paramètres", - "sidebar.help": "Aide", - "sidebar.workspaces.enable": "Activer les espaces de travail", - "sidebar.workspaces.disable": "Désactiver les espaces de travail", - "sidebar.gettingStarted.title": "Commencer", - "sidebar.gettingStarted.line1": "Kilo inclut des modèles gratuits pour que vous puissiez commencer immédiatement.", - "sidebar.gettingStarted.line2": - "Connectez n'importe quel fournisseur pour utiliser des modèles, y compris Claude, GPT, Gemini etc.", - "sidebar.project.recentSessions": "Sessions récentes", - "sidebar.project.viewAllSessions": "Voir toutes les sessions", - "sidebar.project.clearNotifications": "Effacer les notifications", - "app.name.desktop": "Kilo Desktop", - "settings.section.desktop": "Bureau", - "settings.section.server": "Serveur", - "settings.tab.general": "Général", - "settings.tab.shortcuts": "Raccourcis", - "settings.desktop.section.wsl": "WSL", - "settings.desktop.wsl.title": "Intégration WSL", - "settings.desktop.wsl.description": "Exécuter le serveur Kilo dans WSL sur Windows.", - "settings.general.section.appearance": "Apparence", - "settings.general.section.notifications": "Notifications système", - "settings.general.section.updates": "Mises à jour", - "settings.general.section.sounds": "Effets sonores", - "settings.general.section.feed": "Flux", - "settings.general.section.display": "Affichage", - "settings.general.row.language.title": "Langue", - "settings.general.row.language.description": "Changer la langue d'affichage pour Kilo", - "settings.general.row.appearance.title": "Apparence", - "settings.general.row.appearance.description": "Personnaliser l'apparence d'Kilo sur votre appareil", - "settings.general.row.colorScheme.title": "Schéma de couleurs", - "settings.general.row.colorScheme.description": "Choisissez si Kilo suit le thème système, clair ou sombre", - "settings.general.row.theme.title": "Thème", - "settings.general.row.theme.description": "Personnaliser le thème d'Kilo.", - "settings.general.row.font.title": "Police de code", - "settings.general.row.font.description": "Personnaliser la police utilisée dans les blocs de code", - "settings.general.row.terminalFont.title": "Terminal Font", - "settings.general.row.terminalFont.description": "Customise the font used in the terminal", - "settings.general.row.uiFont.title": "Police de l'interface", - "settings.general.row.uiFont.description": "Personnaliser la police utilisée dans toute l'interface", - "settings.general.row.followup.title": "Comportement de suivi", - "settings.general.row.followup.description": - "Choisissez si les messages de suivi dirigent immédiatement ou attendent dans une file d'attente", - "settings.general.row.followup.option.queue": "File d'attente", - "settings.general.row.followup.option.steer": "Diriger", - "settings.general.row.reasoningSummaries.title": "Afficher les résumés de raisonnement", - "settings.general.row.reasoningSummaries.description": - "Afficher les résumés de raisonnement du modèle dans la chronologie", - "settings.general.row.shellToolPartsExpanded.title": "Développer les parties de l'outil shell", - "settings.general.row.shellToolPartsExpanded.description": - "Afficher les parties de l'outil shell développées par défaut dans la chronologie", - "settings.general.row.editToolPartsExpanded.title": "Développer les parties de l'outil edit", - "settings.general.row.editToolPartsExpanded.description": - "Afficher les parties des outils edit, write et patch développées par défaut dans la chronologie", - "settings.general.row.showSessionProgressBar.title": "Afficher la barre de progression de la session", - "settings.general.row.showSessionProgressBar.description": - "Afficher la barre de progression animée en haut de la session lorsque l'agent travaille", - "settings.general.row.wayland.title": "Utiliser Wayland natif", - "settings.general.row.wayland.description": "Désactiver le repli X11 sur Wayland. Nécessite un redémarrage.", - "settings.general.row.wayland.tooltip": - "Sur Linux avec des moniteurs à taux de rafraîchissement mixte, Wayland natif peut être plus stable.", - "settings.general.row.releaseNotes.title": "Notes de version", - "settings.general.row.releaseNotes.description": 'Afficher des pop-ups "Quoi de neuf" après les mises à jour', - "settings.updates.row.startup.title": "Vérifier les mises à jour au démarrage", - "settings.updates.row.startup.description": "Vérifier automatiquement les mises à jour au lancement d'Kilo", - "settings.updates.row.check.title": "Vérifier les mises à jour", - "settings.updates.row.check.description": "Vérifier manuellement les mises à jour et installer si disponible", - "settings.updates.action.checkNow": "Vérifier maintenant", - "settings.updates.action.checking": "Vérification...", - "settings.updates.toast.latest.title": "Vous êtes à jour", - "settings.updates.toast.latest.description": "Vous utilisez la dernière version d'Kilo.", - "sound.option.none": "Aucun", - "sound.option.alert01": "Alerte 01", - "sound.option.alert02": "Alerte 02", - "sound.option.alert03": "Alerte 03", - "sound.option.alert04": "Alerte 04", - "sound.option.alert05": "Alerte 05", - "sound.option.alert06": "Alerte 06", - "sound.option.alert07": "Alerte 07", - "sound.option.alert08": "Alerte 08", - "sound.option.alert09": "Alerte 09", - "sound.option.alert10": "Alerte 10", - "sound.option.bipbop01": "Bip-bop 01", - "sound.option.bipbop02": "Bip-bop 02", - "sound.option.bipbop03": "Bip-bop 03", - "sound.option.bipbop04": "Bip-bop 04", - "sound.option.bipbop05": "Bip-bop 05", - "sound.option.bipbop06": "Bip-bop 06", - "sound.option.bipbop07": "Bip-bop 07", - "sound.option.bipbop08": "Bip-bop 08", - "sound.option.bipbop09": "Bip-bop 09", - "sound.option.bipbop10": "Bip-bop 10", - "sound.option.staplebops01": "Staplebops 01", - "sound.option.staplebops02": "Staplebops 02", - "sound.option.staplebops03": "Staplebops 03", - "sound.option.staplebops04": "Staplebops 04", - "sound.option.staplebops05": "Staplebops 05", - "sound.option.staplebops06": "Staplebops 06", - "sound.option.staplebops07": "Staplebops 07", - "sound.option.nope01": "Non 01", - "sound.option.nope02": "Non 02", - "sound.option.nope03": "Non 03", - "sound.option.nope04": "Non 04", - "sound.option.nope05": "Non 05", - "sound.option.nope06": "Non 06", - "sound.option.nope07": "Non 07", - "sound.option.nope08": "Non 08", - "sound.option.nope09": "Non 09", - "sound.option.nope10": "Non 10", - "sound.option.nope11": "Non 11", - "sound.option.nope12": "Non 12", - "sound.option.yup01": "Oui 01", - "sound.option.yup02": "Oui 02", - "sound.option.yup03": "Oui 03", - "sound.option.yup04": "Oui 04", - "sound.option.yup05": "Oui 05", - "sound.option.yup06": "Oui 06", - "settings.general.notifications.agent.title": "Agent", - "settings.general.notifications.agent.description": - "Afficher une notification système lorsque l'agent a terminé ou nécessite une attention", - "settings.general.notifications.permissions.title": "Permissions", - "settings.general.notifications.permissions.description": - "Afficher une notification système lorsqu'une permission est requise", - "settings.general.notifications.errors.title": "Erreurs", - "settings.general.notifications.errors.description": "Afficher une notification système lorsqu'une erreur se produit", - "settings.general.sounds.agent.title": "Agent", - "settings.general.sounds.agent.description": "Jouer un son lorsque l'agent a terminé ou nécessite une attention", - "settings.general.sounds.permissions.title": "Permissions", - "settings.general.sounds.permissions.description": "Jouer un son lorsqu'une permission est requise", - "settings.general.sounds.errors.title": "Erreurs", - "settings.general.sounds.errors.description": "Jouer un son lorsqu'une erreur se produit", - "settings.shortcuts.title": "Raccourcis clavier", - "settings.shortcuts.reset.button": "Rétablir les défauts", - "settings.shortcuts.reset.toast.title": "Raccourcis réinitialisés", - "settings.shortcuts.reset.toast.description": "Les raccourcis clavier ont été réinitialisés aux valeurs par défaut.", - "settings.shortcuts.conflict.title": "Raccourci déjà utilisé", - "settings.shortcuts.conflict.description": "{{keybind}} est déjà assigné à {{titles}}.", - "settings.shortcuts.unassigned": "Non assigné", - "settings.shortcuts.pressKeys": "Appuyez sur les touches", - "settings.shortcuts.search.placeholder": "Rechercher des raccourcis", - "settings.shortcuts.search.empty": "Aucun raccourci trouvé", - "settings.shortcuts.group.general": "Général", - "settings.shortcuts.group.session": "Session", - "settings.shortcuts.group.navigation": "Navigation", - "settings.shortcuts.group.modelAndAgent": "Modèle et agent", - "settings.shortcuts.group.terminal": "Terminal", - "settings.shortcuts.group.prompt": "Prompt", - "settings.providers.title": "Fournisseurs", - "settings.providers.description": "Les paramètres des fournisseurs seront configurables ici.", - "settings.providers.section.connected": "Fournisseurs connectés", - "settings.providers.connected.empty": "Aucun fournisseur connecté", - "settings.providers.section.popular": "Fournisseurs populaires", - "settings.providers.tag.environment": "Environnement", - "settings.providers.tag.config": "Configuration", - "settings.providers.tag.custom": "Personnalisé", - "settings.providers.tag.other": "Autre", - "settings.models.title": "Modèles", - "settings.models.description": "Les paramètres des modèles seront configurables ici.", - "settings.agents.title": "Agents", - "settings.agents.description": "Les paramètres des agents seront configurables ici.", - "settings.commands.title": "Commandes", - "settings.commands.description": "Les paramètres des commandes seront configurables ici.", - "settings.mcp.title": "MCP", - "settings.mcp.description": "Les paramètres MCP seront configurables ici.", - "settings.permissions.title": "Permissions", - "settings.permissions.description": "Contrôlez les outils que le serveur peut utiliser par défaut.", - "settings.permissions.section.tools": "Outils", - "settings.permissions.toast.updateFailed.title": "Échec de la mise à jour des permissions", - "settings.permissions.action.allow": "Autoriser", - "settings.permissions.action.ask": "Demander", - "settings.permissions.action.deny": "Refuser", - "settings.permissions.tool.read.title": "Lire", - "settings.permissions.tool.read.description": "Lecture d'un fichier (correspond au chemin du fichier)", - "settings.permissions.tool.edit.title": "Modifier", - "settings.permissions.tool.edit.description": - "Modifier des fichiers, y compris les modifications, écritures, patchs et multi-modifications", - "settings.permissions.tool.glob.title": "Glob", - "settings.permissions.tool.glob.description": "Correspondre aux fichiers utilisant des modèles glob", - "settings.permissions.tool.grep.title": "Grep", - "settings.permissions.tool.grep.description": - "Rechercher dans le contenu des fichiers à l'aide d'expressions régulières", - "settings.permissions.tool.list.title": "Lister", - "settings.permissions.tool.list.description": "Lister les fichiers dans un répertoire", - "settings.permissions.tool.bash.title": "Bash", - "settings.permissions.tool.bash.description": "Exécuter des commandes shell", - "settings.permissions.tool.task.title": "Tâche", - "settings.permissions.tool.task.description": "Lancer des sous-agents", - "settings.permissions.tool.skill.title": "Compétence", - "settings.permissions.tool.skill.description": "Charger une compétence par son nom", - "settings.permissions.tool.lsp.title": "LSP", - "settings.permissions.tool.lsp.description": "Exécuter des requêtes de serveur de langage", - "settings.permissions.tool.todowrite.title": "Écrire Todo", - "settings.permissions.tool.todowrite.description": "Mettre à jour la liste de tâches", - "settings.permissions.tool.webfetch.title": "Récupération Web", - "settings.permissions.tool.webfetch.description": "Récupérer le contenu d'une URL", - "settings.permissions.tool.websearch.title": "Recherche Web", - "settings.permissions.tool.websearch.description": "Rechercher sur le web", - "settings.permissions.tool.codesearch.title": "Recherche de code", - "settings.permissions.tool.codesearch.description": "Rechercher du code sur le web", - "settings.permissions.tool.external_directory.title": "Répertoire externe", - "settings.permissions.tool.external_directory.description": "Accéder aux fichiers en dehors du répertoire du projet", - "settings.permissions.tool.doom_loop.title": "Boucle infernale", - "settings.permissions.tool.doom_loop.description": "Détecter les appels d'outils répétés avec une entrée identique", - "session.delete.failed.title": "Échec de la suppression de la session", - "session.delete.title": "Supprimer la session", - "session.delete.confirm": 'Supprimer la session "{{name}}" ?', - "session.delete.button": "Supprimer la session", - "workspace.new": "Nouvel espace de travail", - "workspace.type.local": "local", - "workspace.type.sandbox": "bac à sable", - "workspace.create.failed.title": "Échec de la création de l'espace de travail", - "workspace.delete.failed.title": "Échec de la suppression de l'espace de travail", - "workspace.resetting.title": "Réinitialisation de l'espace de travail", - "workspace.resetting.description": "Cela peut prendre une minute.", - "workspace.reset.failed.title": "Échec de la réinitialisation de l'espace de travail", - "workspace.reset.success.title": "Espace de travail réinitialisé", - "workspace.reset.success.description": "L'espace de travail correspond maintenant à la branche par défaut.", - "workspace.error.stillPreparing": "L'espace de travail est encore en cours de préparation", - "workspace.status.checking": "Vérification des modifications non fusionnées...", - "workspace.status.error": "Impossible de vérifier le statut git.", - "workspace.status.clean": "Aucune modification non fusionnée détectée.", - "workspace.status.dirty": "Modifications non fusionnées détectées dans cet espace de travail.", - "workspace.delete.title": "Supprimer l'espace de travail", - "workspace.delete.confirm": 'Supprimer l\'espace de travail "{{name}}" ?', - "workspace.delete.button": "Supprimer l'espace de travail", - "workspace.reset.title": "Réinitialiser l'espace de travail", - "workspace.reset.confirm": 'Réinitialiser l\'espace de travail "{{name}}" ?', - "workspace.reset.button": "Réinitialiser l'espace de travail", - "workspace.reset.archived.none": "Aucune session active ne sera archivée.", - "workspace.reset.archived.one": "1 session sera archivée.", - "workspace.reset.archived.many": "{{count}} sessions seront archivées.", - "workspace.reset.note": "Cela réinitialisera l'espace de travail pour correspondre à la branche par défaut.", - "common.open": "Ouvrir", - "dialog.releaseNotes.action.getStarted": "Commencer", - "dialog.releaseNotes.action.next": "Suivant", - "dialog.releaseNotes.action.hideFuture": "Ne plus afficher à l'avenir", - "dialog.releaseNotes.media.alt": "Aperçu de la version", - "toast.project.reloadFailed.title": "Échec du rechargement de {{project}}", - "error.server.invalidConfiguration": "Configuration invalide", - "common.moreCountSuffix": " (+{{count}} de plus)", - "common.time.justNow": "À l'instant", - "common.time.minutesAgo.short": "il y a {{count}}m", - "common.time.hoursAgo.short": "il y a {{count}}h", - "common.time.daysAgo.short": "il y a {{count}}j", - "settings.providers.connected.environmentDescription": "Connecté à partir de vos variables d'environnement", - "settings.providers.custom.description": "Ajouter un fournisseur compatible avec OpenAI via l'URL de base.", - - "app.server.unreachable": "Impossible de joindre {{server}}", - "app.server.retrying": "Nouvelle tentative automatique...", - "app.server.otherServers": "Autres serveurs", - "dialog.server.add.usernamePlaceholder": "nom d'utilisateur", - "dialog.server.add.passwordPlaceholder": "mot de passe", - "server.row.noUsername": "aucun nom d'utilisateur", - "session.review.noVcs.createGit.title": "Créer un dépôt Git", - "session.review.noVcs.createGit.description": "Suivre, examiner et annuler les modifications dans ce projet", - "session.review.noVcs.createGit.actionLoading": "Création du dépôt Git...", - "session.review.noVcs.createGit.action": "Créer un dépôt Git", - "session.todo.progress": "{{done}} tâches sur {{total}} terminées", - "session.question.progress": "{{current}} questions sur {{total}}", - "session.header.open.finder": "Finder", - "session.header.open.fileExplorer": "Explorateur de fichiers", - "session.header.open.fileManager": "Gestionnaire de fichiers", - "session.header.open.app.vscode": "VS Code", - "session.header.open.app.cursor": "Cursor", - "session.header.open.app.zed": "Zed", - "session.header.open.app.textmate": "TextMate", - "session.header.open.app.antigravity": "Antigravity", - "session.header.open.app.terminal": "Terminal", - "session.header.open.app.iterm2": "iTerm2", - "session.header.open.app.ghostty": "Ghostty", - "session.header.open.app.warp": "Warp", - "session.header.open.app.xcode": "Xcode", - "session.header.open.app.androidStudio": "Android Studio", - "session.header.open.app.powershell": "PowerShell", - "session.header.open.app.sublimeText": "Sublime Text", - "debugBar.ariaLabel": "Diagnostics de performance de développement", - "debugBar.na": "n/a", - "debugBar.nav.label": "NAV", - "debugBar.nav.tip": - "Dernière transition de route terminée touchant une page de session, mesurée du début du routeur jusqu'au premier affichage après stabilisation.", - "debugBar.fps.label": "FPS", - "debugBar.fps.tip": "Images par seconde glissantes sur les 5 dernières secondes.", - "debugBar.frame.label": "FRAME", - "debugBar.frame.tip": "Pire temps d'image sur les 5 dernières secondes.", - "debugBar.jank.label": "JANK", - "debugBar.jank.tip": "Images de plus de 32ms au cours des 5 dernières secondes.", - "debugBar.long.label": "LONG", - "debugBar.long.tip": - "Temps bloqué et nombre de tâches longues au cours des 5 dernières secondes. Tâche max : {{max}}.", - "debugBar.delay.label": "DELAY", - "debugBar.delay.tip": "Pire délai d'entrée observé au cours des 5 dernières secondes.", - "debugBar.inp.label": "INP", - "debugBar.inp.tip": - "Durée approximative d'interaction au cours des 5 dernières secondes. Ceci est similaire à INP, pas le INP officiel des Web Vitals.", - "debugBar.cls.label": "CLS", - "debugBar.cls.tip": "Décalage cumulatif de la mise en page pour la durée de vie actuelle de l'application.", - "debugBar.mem.label": "MEM", - "debugBar.mem.tipUnavailable": "Tas JS utilisé vs limite de tas. Chromium uniquement.", - "debugBar.mem.tip": "Tas JS utilisé vs limite de tas. {{used}} sur {{limit}}.", - "common.key.ctrl": "Ctrl", - "common.key.alt": "Alt", - "common.key.shift": "Maj", - "common.key.meta": "Méta", - "common.key.space": "Espace", - "common.key.backspace": "Retour arrière", - "common.key.enter": "Entrée", - "common.key.tab": "Tab", - "common.key.delete": "Suppr", - "common.key.home": "Début", - "common.key.end": "Fin", - "common.key.pageUp": "Page précédente", - "common.key.pageDown": "Page suivante", - "common.key.insert": "Inser", - "common.unknown": "inconnu", - "error.page.circular": "[Circulaire]", - "error.globalSDK.noServerAvailable": "Aucun serveur disponible", - "error.globalSDK.serverNotAvailable": "Serveur non disponible", - "error.childStore.persistedCacheCreateFailed": "Échec de la création du cache persistant", - "error.childStore.persistedProjectMetadataCreateFailed": - "Échec de la création des métadonnées de projet persistantes", - "error.childStore.persistedProjectIconCreateFailed": "Échec de la création de l'icône de projet persistante", - "error.childStore.storeCreateFailed": "Échec de la création du stockage", - "terminal.connectionLost.abnormalClose": "WebSocket fermé anormalement : {{code}}", -} diff --git a/packages/app/src/i18n/ja.ts b/packages/app/src/i18n/ja.ts deleted file mode 100644 index 49b8d652d7..0000000000 --- a/packages/app/src/i18n/ja.ts +++ /dev/null @@ -1,853 +0,0 @@ -export const dict = { - "command.category.suggested": "おすすめ", - "command.category.view": "表示", - "command.category.project": "プロジェクト", - "command.category.provider": "プロバイダー", - "command.category.server": "サーバー", - "command.category.session": "セッション", - "command.category.theme": "テーマ", - "command.category.language": "言語", - "command.category.file": "ファイル", - "command.category.context": "コンテキスト", - "command.category.terminal": "ターミナル", - "command.category.model": "モデル", - "command.category.mcp": "MCP", - "command.category.agent": "エージェント", - "command.category.permissions": "権限", - "command.category.workspace": "ワークスペース", - "command.category.settings": "設定", - "theme.scheme.system": "システム", - "theme.scheme.light": "ライト", - "theme.scheme.dark": "ダーク", - "command.sidebar.toggle": "サイドバーの切り替え", - "command.project.open": "プロジェクトを開く", - "command.provider.connect": "プロバイダーに接続", - "command.server.switch": "サーバーの切り替え", - "command.settings.open": "設定を開く", - "command.session.previous": "前のセッション", - "command.session.next": "次のセッション", - "command.session.previous.unseen": "前の未読セッション", - "command.session.next.unseen": "次の未読セッション", - "command.session.archive": "セッションをアーカイブ", - "command.palette": "コマンドパレット", - "command.theme.cycle": "テーマの切り替え", - "command.theme.set": "テーマを使用: {{theme}}", - "command.theme.scheme.cycle": "配色の切り替え", - "command.theme.scheme.set": "配色を使用: {{scheme}}", - "command.language.cycle": "言語の切り替え", - "command.language.set": "言語を使用: {{language}}", - "command.session.new": "新しいセッション", - "command.file.open": "ファイルを開く", - "command.tab.close": "タブを閉じる", - "command.context.addSelection": "選択範囲をコンテキストに追加", - "command.context.addSelection.description": "現在のファイルから選択した行を追加", - "command.input.focus": "入力欄にフォーカス", - "command.terminal.toggle": "ターミナルの切り替え", - "command.fileTree.toggle": "ファイルツリーを切り替え", - "command.review.toggle": "レビューの切り替え", - "command.terminal.new": "新しいターミナル", - "command.terminal.new.description": "新しいターミナルタブを作成", - "command.steps.toggle": "ステップの切り替え", - "command.steps.toggle.description": "現在のメッセージのステップを表示または非表示", - "command.message.previous": "前のメッセージ", - "command.message.previous.description": "前のユーザーメッセージに移動", - "command.message.next": "次のメッセージ", - "command.message.next.description": "次のユーザーメッセージに移動", - "command.model.choose": "モデルを選択", - "command.model.choose.description": "別のモデルを選択", - "command.mcp.toggle": "MCPの切り替え", - "command.mcp.toggle.description": "MCPを切り替える", - "command.agent.cycle": "エージェントの切り替え", - "command.agent.cycle.description": "次のエージェントに切り替え", - "command.agent.cycle.reverse": "エージェントを逆順に切り替え", - "command.agent.cycle.reverse.description": "前のエージェントに切り替え", - "command.model.variant.cycle": "思考レベルの切り替え", - "command.model.variant.cycle.description": "次の思考レベルに切り替え", - "command.prompt.mode.shell": "シェル", - "command.prompt.mode.normal": "プロンプト", - "command.permissions.autoaccept.enable": "権限を自動承認する", - "command.permissions.autoaccept.disable": "権限の自動承認を停止する", - "command.workspace.toggle": "ワークスペースを切り替え", - "command.workspace.toggle.description": "サイドバーでの複数のワークスペースの有効化・無効化", - "command.session.undo": "元に戻す", - "command.session.undo.description": "最後のメッセージを元に戻す", - "command.session.redo": "やり直す", - "command.session.redo.description": "元に戻したメッセージをやり直す", - "command.session.compact": "セッションを圧縮", - "command.session.compact.description": "セッションを要約してコンテキストサイズを削減", - "command.session.fork": "メッセージからフォーク", - "command.session.fork.description": "以前のメッセージから新しいセッションを作成", - "command.session.share": "セッションを共有", - "command.session.share.description": "このセッションを共有しURLをクリップボードにコピー", - "command.session.unshare": "セッションの共有を停止", - "command.session.unshare.description": "このセッションの共有を停止", - "palette.search.placeholder": "ファイル、コマンド、セッションを検索", - "palette.empty": "結果が見つかりません", - "palette.group.commands": "コマンド", - "palette.group.files": "ファイル", - "dialog.provider.search.placeholder": "プロバイダーを検索", - "dialog.provider.empty": "プロバイダーが見つかりません", - "dialog.provider.group.popular": "人気", - "dialog.provider.group.other": "その他", - "dialog.provider.tag.recommended": "推奨", - "dialog.provider.opencode.note": "Claude, GPT, Geminiなどを含む厳選されたモデル", - "dialog.provider.opencode.tagline": "信頼性の高い最適化モデル", - "dialog.provider.opencodeGo.tagline": "すべての人に低価格のサブスクリプション", - "dialog.provider.anthropic.note": "Claude Pro/MaxまたはAPIキーで接続", - "dialog.provider.copilot.note": "CopilotまたはAPIキーで接続", - "dialog.provider.openai.note": "ChatGPT Pro/PlusまたはAPIキーで接続", - "dialog.provider.google.note": "高速で構造化された応答のためのGeminiモデル", - "dialog.provider.openrouter.note": "1つのプロバイダーからすべてのサポートされているモデルにアクセス", - "dialog.provider.vercel.note": "スマートルーターによるAIモデルへの統合アクセス", - "dialog.model.select.title": "モデルを選択", - "dialog.model.search.placeholder": "モデルを検索", - "dialog.model.empty": "モデルが見つかりません", - "dialog.model.manage": "モデルを管理", - "dialog.model.manage.description": "モデルセレクターに表示するモデルをカスタマイズします。", - "dialog.model.manage.provider.toggle": "すべての{{provider}}モデルを切り替え", - "dialog.model.unpaid.freeModels.title": "Kiloが提供する無料モデル", - "dialog.model.unpaid.addMore.title": "人気のプロバイダーからモデルを追加", - "dialog.provider.viewAll": "さらにプロバイダーを表示", - "provider.connect.title": "{{provider}}を接続", - "provider.connect.title.anthropicProMax": "Claude Pro/Maxでログイン", - "provider.connect.selectMethod": "{{provider}}のログイン方法を選択してください。", - "provider.connect.method.apiKey": "APIキー", - "provider.connect.status.inProgress": "認証中...", - "provider.connect.status.waiting": "認証を待機中...", - "provider.connect.status.failed": "認証に失敗しました: {{error}}", - "provider.connect.apiKey.description": - "{{provider}}のAPIキーを入力してアカウントを接続し、Kiloで{{provider}}モデルを使用します。", - "provider.connect.apiKey.label": "{{provider}} APIキー", - "provider.connect.apiKey.placeholder": "APIキー", - "provider.connect.apiKey.required": "APIキーが必要です", - "provider.connect.opencodeZen.line1": - "OpenCode Zenは、コーディングエージェント向けに最適化された信頼性の高いモデルへのアクセスを提供します。", - "provider.connect.opencodeZen.line2": "1つのAPIキーで、Claude、GPT、Gemini、GLMなどのモデルにアクセスできます。", - "provider.connect.opencodeZen.visit.prefix": " ", - "provider.connect.opencodeZen.visit.link": "opencode.ai/zen", - "provider.connect.opencodeZen.visit.suffix": " にアクセスしてAPIキーを取得してください。", - "provider.connect.oauth.code.visit.prefix": " ", - "provider.connect.oauth.code.visit.link": "このリンク", - "provider.connect.oauth.code.visit.suffix": - " にアクセスして認証コードを取得し、アカウントを接続してKiloで{{provider}}モデルを使用してください。", - "provider.connect.oauth.code.label": "{{method}} 認証コード", - "provider.connect.oauth.code.placeholder": "認証コード", - "provider.connect.oauth.code.required": "認証コードが必要です", - "provider.connect.oauth.code.invalid": "無効な認証コード", - "provider.connect.oauth.auto.visit.prefix": " ", - "provider.connect.oauth.auto.visit.link": "このリンク", - "provider.connect.oauth.auto.visit.suffix": - " にアクセスし、以下のコードを入力してアカウントを接続し、Kiloで{{provider}}モデルを使用してください。", - "provider.connect.oauth.auto.confirmationCode": "確認コード", - "provider.connect.toast.connected.title": "{{provider}}が接続されました", - "provider.connect.toast.connected.description": "{{provider}}モデルが使用可能になりました。", - "provider.custom.title": "カスタムプロバイダー", - "provider.custom.description.prefix": "OpenAI互換のプロバイダーを設定します。詳細は", - "provider.custom.description.link": "プロバイダー設定ドキュメント", - "provider.custom.description.suffix": "をご覧ください。", - "provider.custom.field.providerID.label": "プロバイダーID", - "provider.custom.field.providerID.placeholder": "myprovider", - "provider.custom.field.providerID.description": "小文字、数字、ハイフン、アンダースコア", - "provider.custom.field.name.label": "表示名", - "provider.custom.field.name.placeholder": "My AI Provider", - "provider.custom.field.baseURL.label": "ベースURL", - "provider.custom.field.baseURL.placeholder": "https://api.myprovider.com/v1", - "provider.custom.field.apiKey.label": "APIキー", - "provider.custom.field.apiKey.placeholder": "APIキー", - "provider.custom.field.apiKey.description": "オプション。ヘッダーで認証を管理する場合は空のままにしてください。", - "provider.custom.models.label": "モデル", - "provider.custom.models.id.label": "ID", - "provider.custom.models.id.placeholder": "model-id", - "provider.custom.models.name.label": "名前", - "provider.custom.models.name.placeholder": "表示名", - "provider.custom.models.remove": "モデルを削除", - "provider.custom.models.add": "モデルを追加", - "provider.custom.headers.label": "ヘッダー (オプション)", - "provider.custom.headers.key.label": "ヘッダー", - "provider.custom.headers.key.placeholder": "Header-Name", - "provider.custom.headers.value.label": "値", - "provider.custom.headers.value.placeholder": "value", - "provider.custom.headers.remove": "ヘッダーを削除", - "provider.custom.headers.add": "ヘッダーを追加", - "provider.custom.error.providerID.required": "プロバイダーIDが必要です", - "provider.custom.error.providerID.format": "小文字、数字、ハイフン、アンダースコアを使用してください", - "provider.custom.error.providerID.exists": "そのプロバイダーIDは既に存在します", - "provider.custom.error.name.required": "表示名が必要です", - "provider.custom.error.baseURL.required": "ベースURLが必要です", - "provider.custom.error.baseURL.format": "http:// または https:// で始まる必要があります", - "provider.custom.error.required": "必須", - "provider.custom.error.duplicate": "重複", - "provider.disconnect.toast.disconnected.title": "{{provider}}が切断されました", - "provider.disconnect.toast.disconnected.description": "{{provider}}のモデルは利用できなくなりました。", - "model.tag.free": "無料", - "model.tag.latest": "最新", - "model.provider.anthropic": "Anthropic", - "model.provider.openai": "OpenAI", - "model.provider.google": "Google", - "model.provider.xai": "xAI", - "model.provider.meta": "Meta", - "model.input.text": "テキスト", - "model.input.image": "画像", - "model.input.audio": "音声", - "model.input.video": "動画", - "model.input.pdf": "pdf", - "model.tooltip.allows": "対応: {{inputs}}", - "model.tooltip.reasoning.allowed": "推論を許可", - "model.tooltip.reasoning.none": "推論なし", - "model.tooltip.context": "コンテキスト上限 {{limit}}", - "common.search.placeholder": "検索", - "common.goBack": "戻る", - "common.goForward": "進む", - "common.loading": "読み込み中", - "common.loading.ellipsis": "...", - "common.cancel": "キャンセル", - "common.connect": "接続", - "common.disconnect": "切断", - "common.continue": "送信", - "common.submit": "送信", - "common.save": "保存", - "common.saving": "保存中...", - "common.default": "デフォルト", - "common.attachment": "添付ファイル", - "prompt.placeholder.shell": "シェルコマンドを入力... {{example}}", - "prompt.placeholder.normal": '何でも聞いてください... "{{example}}"', - "prompt.placeholder.simple": "何でも聞いてください...", - "prompt.placeholder.summarizeComments": "コメントを要約…", - "prompt.placeholder.summarizeComment": "コメントを要約…", - "prompt.mode.shell": "シェル", - "prompt.mode.normal": "プロンプト", - "prompt.mode.shell.exit": "escで終了", - "prompt.example.1": "コードベースのTODOを修正", - "prompt.example.2": "このプロジェクトの技術スタックは何ですか?", - "prompt.example.3": "壊れたテストを修正", - "prompt.example.4": "認証の仕組みを説明して", - "prompt.example.5": "セキュリティの脆弱性を見つけて修正", - "prompt.example.6": "ユーザーサービスのユニットテストを追加", - "prompt.example.7": "この関数を読みやすくリファクタリング", - "prompt.example.8": "このエラーはどういう意味ですか?", - "prompt.example.9": "この問題のデバッグを手伝って", - "prompt.example.10": "APIドキュメントを生成", - "prompt.example.11": "データベースクエリを最適化", - "prompt.example.12": "入力バリデーションを追加", - "prompt.example.13": "〜の新しいコンポーネントを作成", - "prompt.example.14": "このプロジェクトをデプロイするには?", - "prompt.example.15": "ベストプラクティスの観点でコードをレビュー", - "prompt.example.16": "この関数にエラーハンドリングを追加", - "prompt.example.17": "この正規表現パターンを説明して", - "prompt.example.18": "これをTypeScriptに変換", - "prompt.example.19": "コードベース全体にログを追加", - "prompt.example.20": "古い依存関係はどれですか?", - "prompt.example.21": "マイグレーションスクリプトの作成を手伝って", - "prompt.example.22": "このエンドポイントにキャッシュを実装", - "prompt.example.23": "このリストにページネーションを追加", - "prompt.example.24": "〜のCLIコマンドを作成", - "prompt.example.25": "ここでは環境変数はどう機能しますか?", - "prompt.popover.emptyResults": "一致する結果がありません", - "prompt.popover.emptyCommands": "一致するコマンドがありません", - "prompt.dropzone.label": "画像、PDF、またはテキストファイルをここにドロップしてください", - "prompt.dropzone.file.label": "ドロップして@メンションファイルを追加", - "prompt.slash.badge.custom": "カスタム", - "prompt.slash.badge.skill": "スキル", - "prompt.slash.badge.mcp": "mcp", - "prompt.context.active": "アクティブ", - "prompt.context.includeActiveFile": "アクティブなファイルを含める", - "prompt.context.removeActiveFile": "コンテキストからアクティブなファイルを削除", - "prompt.context.removeFile": "コンテキストからファイルを削除", - "prompt.action.attachFile": "ファイルを添付", - "prompt.attachment.remove": "添付ファイルを削除", - "prompt.action.send": "送信", - "prompt.action.stop": "停止", - "prompt.toast.pasteUnsupported.title": "サポートされていない添付ファイル", - "prompt.toast.pasteUnsupported.description": "画像、PDF、またはテキストファイルのみ添付できます。", - "prompt.toast.modelAgentRequired.title": "エージェントとモデルを選択", - "prompt.toast.modelAgentRequired.description": "プロンプトを送信する前にエージェントとモデルを選択してください。", - "prompt.toast.worktreeCreateFailed.title": "ワークツリーの作成に失敗しました", - "prompt.toast.sessionCreateFailed.title": "セッションの作成に失敗しました", - "prompt.toast.shellSendFailed.title": "シェルコマンドの送信に失敗しました", - "prompt.toast.commandSendFailed.title": "コマンドの送信に失敗しました", - "prompt.toast.promptSendFailed.title": "プロンプトの送信に失敗しました", - "prompt.toast.promptSendFailed.description": "セッションを取得できませんでした", - "dialog.mcp.title": "MCP", - "dialog.mcp.description": "{{total}}個中{{enabled}}個が有効", - "dialog.mcp.empty": "MCPが設定されていません", - "dialog.lsp.empty": "ファイルタイプから自動検出されたLSP", - "dialog.plugins.empty": "opencode.jsonで設定されたプラグイン", - "mcp.status.connected": "接続済み", - "mcp.status.failed": "失敗", - "mcp.status.needs_auth": "認証が必要", - "mcp.status.disabled": "無効", - "dialog.fork.empty": "フォーク元のメッセージがありません", - "dialog.directory.search.placeholder": "フォルダを検索", - "dialog.directory.empty": "フォルダが見つかりません", - "dialog.server.title": "サーバー", - "dialog.server.description": "このアプリが接続するKiloサーバーを切り替えます。", - "dialog.server.search.placeholder": "サーバーを検索", - "dialog.server.empty": "サーバーはまだありません", - "dialog.server.add.title": "サーバーを追加", - "dialog.server.add.url": "サーバーURL", - "dialog.server.add.placeholder": "http://localhost:4096", - "dialog.server.add.error": "サーバーに接続できませんでした", - "dialog.server.add.checking": "確認中...", - "dialog.server.add.button": "サーバーを追加", - "dialog.server.add.name": "サーバー名 (オプション)", - "dialog.server.add.namePlaceholder": "Localhost", - "dialog.server.add.username": "ユーザー名 (オプション)", - "dialog.server.add.password": "パスワード (オプション)", - "dialog.server.edit.title": "サーバーを編集", - "dialog.server.default.title": "デフォルトサーバー", - "dialog.server.default.description": - "ローカルサーバーを起動する代わりに、アプリ起動時にこのサーバーに接続します。再起動が必要です。", - "dialog.server.default.none": "サーバーが選択されていません", - "dialog.server.default.set": "現在のサーバーをデフォルトに設定", - "dialog.server.default.clear": "クリア", - "dialog.server.action.remove": "サーバーを削除", - "dialog.server.menu.edit": "編集", - "dialog.server.menu.default": "デフォルトに設定", - "dialog.server.menu.defaultRemove": "デフォルト設定を解除", - "dialog.server.menu.delete": "削除", - "dialog.server.current": "現在のサーバー", - "dialog.server.status.default": "デフォルト", - "dialog.project.edit.title": "プロジェクトを編集", - "dialog.project.edit.name": "名前", - "dialog.project.edit.icon": "アイコン", - "dialog.project.edit.icon.alt": "プロジェクトアイコン", - "dialog.project.edit.icon.hint": "クリックまたは画像をドラッグ", - "dialog.project.edit.icon.recommended": "推奨: 128x128px", - "dialog.project.edit.color": "色", - "dialog.project.edit.color.select": "{{color}}の色を選択", - "dialog.project.edit.worktree.startup": "ワークスペース起動スクリプト", - "dialog.project.edit.worktree.startup.description": - "新しいワークスペース (ワークツリー) を作成した後に実行されます。", - "dialog.project.edit.worktree.startup.placeholder": "例: bun install", - "context.breakdown.title": "コンテキストの内訳", - "context.breakdown.note": '入力トークンのおおよその内訳です。"その他"にはツールの定義やオーバーヘッドが含まれます。', - "context.breakdown.system": "システム", - "context.breakdown.user": "ユーザー", - "context.breakdown.assistant": "アシスタント", - "context.breakdown.tool": "ツール呼び出し", - "context.breakdown.other": "その他", - "context.systemPrompt.title": "システムプロンプト", - "context.rawMessages.title": "生のメッセージ", - "context.stats.session": "セッション", - "context.stats.messages": "メッセージ", - "context.stats.provider": "プロバイダー", - "context.stats.model": "モデル", - "context.stats.limit": "コンテキスト制限", - "context.stats.totalTokens": "総トークン数", - "context.stats.usage": "使用量", - "context.stats.inputTokens": "入力トークン", - "context.stats.outputTokens": "出力トークン", - "context.stats.reasoningTokens": "推論トークン", - "context.stats.cacheTokens": "キャッシュトークン (読込/書込)", - "context.stats.userMessages": "ユーザーメッセージ", - "context.stats.assistantMessages": "アシスタントメッセージ", - "context.stats.totalCost": "総コスト", - "context.stats.sessionCreated": "セッション作成日時", - "context.stats.lastActivity": "最終アクティビティ", - "context.usage.tokens": "トークン", - "context.usage.usage": "使用量", - "context.usage.cost": "コスト", - "context.usage.clickToView": "クリックしてコンテキストを表示", - "context.usage.view": "コンテキスト使用量を表示", - "language.en": "English", - "language.zh": "简体中文", - "language.zht": "繁體中文", - "language.ko": "한국어", - "language.de": "Deutsch", - "language.es": "Español", - "language.fr": "Français", - "language.da": "Dansk", - "language.ja": "日本語", - "language.pl": "Polski", - "language.ru": "Русский", - "language.ar": "العربية", - "language.no": "Norsk", - "language.br": "Português (Brasil)", - "language.bs": "Bosanski", - "language.th": "ไทย", - "language.tr": "Türkçe", - "toast.language.title": "言語", - "toast.language.description": "{{language}}に切り替えました", - "toast.theme.title": "テーマが切り替わりました", - "toast.scheme.title": "配色", - "toast.workspace.enabled.title": "ワークスペースが有効になりました", - "toast.workspace.enabled.description": "サイドバーに複数のワークツリーが表示されます", - "toast.workspace.disabled.title": "ワークスペースが無効になりました", - "toast.workspace.disabled.description": "サイドバーにはメインのワークツリーのみが表示されます", - "toast.permissions.autoaccept.on.title": "権限を自動承認しています", - "toast.permissions.autoaccept.on.description": "権限の要求は自動的に承認されます", - "toast.permissions.autoaccept.off.title": "権限の自動承認を停止しました", - "toast.permissions.autoaccept.off.description": "権限の要求には承認が必要になります", - "toast.model.none.title": "モデルが選択されていません", - "toast.model.none.description": "このセッションを要約するにはプロバイダーを接続してください", - "toast.file.loadFailed.title": "ファイルの読み込みに失敗しました", - "toast.file.listFailed.title": "ファイル一覧の取得に失敗しました", - "toast.context.noLineSelection.title": "行が選択されていません", - "toast.context.noLineSelection.description": "まずファイルタブで行範囲を選択してください。", - "toast.session.share.copyFailed.title": "URLのコピーに失敗しました", - "toast.session.share.success.title": "セッションを共有しました", - "toast.session.share.success.description": "共有URLをクリップボードにコピーしました!", - "toast.session.share.failed.title": "セッションの共有に失敗しました", - "toast.session.share.failed.description": "セッションの共有中にエラーが発生しました", - "toast.session.unshare.success.title": "セッションの共有を解除しました", - "toast.session.unshare.success.description": "セッションの共有解除に成功しました!", - "toast.session.unshare.failed.title": "セッションの共有解除に失敗しました", - "toast.session.unshare.failed.description": "セッションの共有解除中にエラーが発生しました", - "toast.session.listFailed.title": "{{project}}のセッション読み込みに失敗しました", - "toast.update.title": "アップデートが利用可能です", - "toast.update.description": "Kiloの新しいバージョン ({{version}}) がインストール可能です。", - "toast.update.action.installRestart": "インストールして再起動", - "toast.update.action.notYet": "今はしない", - "error.page.title": "問題が発生しました", - "error.page.description": "アプリケーションの読み込み中にエラーが発生しました。", - "error.page.details.label": "エラー詳細", - "error.page.action.restart": "再起動", - "error.page.action.checking": "確認中...", - "error.page.action.checkUpdates": "アップデートを確認", - "error.page.action.updateTo": "{{version}}にアップデート", - "error.page.report.prefix": "このエラーをKiloチームに報告してください: ", - "error.page.report.discord": "Discord", - "error.page.version": "バージョン: {{version}}", - "error.dev.rootNotFound": - "ルート要素が見つかりません。index.htmlに追加するのを忘れていませんか?またはid属性のスペルが間違っていませんか?", - "error.globalSync.connectFailed": "サーバーに接続できませんでした。`{{url}}`でサーバーが実行されていますか?", - "directory.error.invalidUrl": "URL内のディレクトリが無効です。", - "error.chain.unknown": "不明なエラー", - "error.chain.causedBy": "原因:", - "error.chain.apiError": "APIエラー", - "error.chain.status": "ステータス: {{status}}", - "error.chain.retryable": "再試行可能: {{retryable}}", - "error.chain.responseBody": "レスポンス本文:\n{{body}}", - "error.chain.didYouMean": "もしかして: {{suggestions}}", - "error.chain.modelNotFound": "モデルが見つかりません: {{provider}}/{{model}}", - "error.chain.checkConfig": "config (opencode.json) のプロバイダー/モデル名を確認してください", - "error.chain.mcpFailed": 'MCPサーバー "{{name}}" が失敗しました。注意: KiloはまだMCP認証をサポートしていません。', - "error.chain.providerAuthFailed": "プロバイダー認証に失敗しました ({{provider}}): {{message}}", - "error.chain.providerInitFailed": - 'プロバイダー "{{provider}}" の初期化に失敗しました。認証情報と設定を確認してください。', - "error.chain.configJsonInvalid": "{{path}} の設定ファイルは有効なJSON(C)ではありません", - "error.chain.configJsonInvalidWithMessage": "{{path}} の設定ファイルは有効なJSON(C)ではありません: {{message}}", - "error.chain.configDirectoryTypo": - '{{path}} 内のディレクトリ "{{dir}}" は無効です。"{{suggestion}}" に名前を変更するか削除してください。これはよくあるタイプミスです。', - "error.chain.configFrontmatterError": "{{path}} のフロントマターの解析に失敗しました:\n{{message}}", - "error.chain.configInvalid": "{{path}} の設定ファイルが無効です", - "error.chain.configInvalidWithMessage": "{{path}} の設定ファイルが無効です: {{message}}", - "notification.permission.title": "権限が必要です", - "notification.permission.description": "{{projectName}} の {{sessionTitle}} が権限を必要としています", - "notification.question.title": "質問", - "notification.question.description": "{{projectName}} の {{sessionTitle}} から質問があります", - "notification.action.goToSession": "セッションへ移動", - "notification.session.responseReady.title": "応答の準備ができました", - "notification.session.error.title": "セッションエラー", - "notification.session.error.fallbackDescription": "エラーが発生しました", - "home.recentProjects": "最近のプロジェクト", - "home.empty.title": "最近のプロジェクトはありません", - "home.empty.description": "ローカルプロジェクトを開いて始めましょう", - "session.tab.session": "セッション", - "session.tab.review": "レビュー", - "session.tab.context": "コンテキスト", - "session.panel.reviewAndFiles": "レビューとファイル", - "session.review.filesChanged": "{{count}} ファイル変更", - "session.review.change.one": "変更", - "session.review.change.other": "変更", - "session.review.loadingChanges": "変更を読み込み中...", - "session.review.empty": "このセッションでの変更はまだありません", - "session.review.noVcs": "Gitバージョン管理システムが検出されないため、変更は表示されません", - "session.review.noSnapshot": "設定でスナップショット追跡が無効になっているため、セッションの変更は利用できません", - "session.review.noChanges": "変更なし", - "session.files.selectToOpen": "開くファイルを選択", - "session.files.all": "すべてのファイル", - "session.files.empty": "ファイルなし", - "session.files.binaryContent": "バイナリファイル(内容を表示できません)", - "session.messages.renderEarlier": "以前のメッセージを表示", - "session.messages.loadingEarlier": "以前のメッセージを読み込み中...", - "session.messages.loadEarlier": "以前のメッセージを読み込む", - "session.messages.loading": "メッセージを読み込み中...", - "session.messages.jumpToLatest": "最新へジャンプ", - "session.context.addToContext": "{{selection}}をコンテキストに追加", - "session.todo.title": "ToDo", - "session.todo.collapse": "折りたたむ", - "session.todo.expand": "展開", - "session.followupDock.summary.one": "{{count}} 件のメッセージが待機中", - "session.followupDock.summary.other": "{{count}} 件のメッセージが待機中", - "session.followupDock.sendNow": "今すぐ送信", - "session.followupDock.edit": "編集", - "session.followupDock.collapse": "待機中のメッセージを折りたたむ", - "session.followupDock.expand": "待機中のメッセージを展開", - "session.revertDock.summary.one": "{{count}} 件のロールバックされたメッセージ", - "session.revertDock.summary.other": "{{count}} 件のロールバックされたメッセージ", - "session.revertDock.collapse": "ロールバックされたメッセージを折りたたむ", - "session.revertDock.expand": "ロールバックされたメッセージを展開", - "session.revertDock.restore": "メッセージを復元", - "session.new.title": "何でも作る", - "session.new.worktree.main": "メインブランチ", - "session.new.worktree.mainWithBranch": "メインブランチ ({{branch}})", - "session.new.worktree.create": "新しいワークツリーを作成", - "session.new.lastModified": "最終更新", - "session.header.search.placeholder": "{{project}}を検索", - "session.header.searchFiles": "ファイルを検索", - "session.header.openIn": "で開く", - "session.header.open.action": "{{app}}を開く", - "session.header.open.ariaLabel": "{{app}}で開く", - "session.header.open.menu": "開くオプション", - "session.header.open.copyPath": "パスをコピー", - "status.popover.trigger": "ステータス", - "status.popover.ariaLabel": "サーバー設定", - "status.popover.tab.servers": "サーバー", - "status.popover.tab.mcp": "MCP", - "status.popover.tab.lsp": "LSP", - "status.popover.tab.plugins": "プラグイン", - "status.popover.action.manageServers": "サーバーを管理", - "session.share.popover.title": "ウェブで公開", - "session.share.popover.description.shared": - "このセッションはウェブで公開されています。リンクを知っている人なら誰でもアクセスできます。", - "session.share.popover.description.unshared": - "セッションをウェブで公開します。リンクを知っている人なら誰でもアクセスできるようになります。", - "session.share.action.share": "共有", - "session.share.action.publish": "公開", - "session.share.action.publishing": "公開中...", - "session.share.action.unpublish": "非公開にする", - "session.share.action.unpublishing": "非公開にしています...", - "session.share.action.view": "表示", - "session.share.copy.copied": "コピーしました", - "session.share.copy.copyLink": "リンクをコピー", - "lsp.tooltip.none": "LSPサーバーなし", - "lsp.label.connected": "{{count}} LSP", - "prompt.loading": "プロンプトを読み込み中...", - "terminal.loading": "ターミナルを読み込み中...", - "terminal.title": "ターミナル", - "terminal.title.numbered": "ターミナル {{number}}", - "terminal.close": "ターミナルを閉じる", - "terminal.connectionLost.title": "接続が失われました", - "terminal.connectionLost.description": - "ターミナルの接続が中断されました。これはサーバーが再起動したときに発生することがあります。", - "common.closeTab": "タブを閉じる", - "common.dismiss": "閉じる", - "common.requestFailed": "リクエスト失敗", - "common.moreOptions": "その他のオプション", - "common.learnMore": "詳細", - "common.rename": "名前変更", - "common.reset": "リセット", - "common.archive": "アーカイブ", - "common.delete": "削除", - "common.close": "閉じる", - "common.edit": "編集", - "common.loadMore": "さらに読み込む", - "common.key.esc": "ESC", - "sidebar.menu.toggle": "メニューを切り替え", - "sidebar.nav.projectsAndSessions": "プロジェクトとセッション", - "sidebar.settings": "設定", - "sidebar.help": "ヘルプ", - "sidebar.workspaces.enable": "ワークスペースを有効化", - "sidebar.workspaces.disable": "ワークスペースを無効化", - "sidebar.gettingStarted.title": "はじめに", - "sidebar.gettingStarted.line1": "Kiloには無料モデルが含まれているため、すぐに開始できます。", - "sidebar.gettingStarted.line2": "プロバイダーを接続して、Claude、GPT、Geminiなどのモデルを使用できます。", - "sidebar.project.recentSessions": "最近のセッション", - "sidebar.project.viewAllSessions": "すべてのセッションを表示", - "sidebar.project.clearNotifications": "通知をクリア", - "app.name.desktop": "Kilo Desktop", - "settings.section.desktop": "デスクトップ", - "settings.section.server": "サーバー", - "settings.tab.general": "一般", - "settings.tab.shortcuts": "ショートカット", - "settings.desktop.section.wsl": "WSL", - "settings.desktop.wsl.title": "WSL連携", - "settings.desktop.wsl.description": "WindowsのWSL環境でKiloサーバーを実行します。", - "settings.general.section.appearance": "外観", - "settings.general.section.notifications": "システム通知", - "settings.general.section.updates": "アップデート", - "settings.general.section.sounds": "効果音", - "settings.general.section.feed": "フィード", - "settings.general.section.display": "ディスプレイ", - "settings.general.row.language.title": "言語", - "settings.general.row.language.description": "Kiloの表示言語を変更します", - "settings.general.row.appearance.title": "外観", - "settings.general.row.appearance.description": "デバイスでのKiloの表示をカスタマイズします", - "settings.general.row.colorScheme.title": "配色", - "settings.general.row.colorScheme.description": "Kiloがシステム、ライト、またはダークテーマに従うかを選択します", - "settings.general.row.theme.title": "テーマ", - "settings.general.row.theme.description": "Kiloのテーマをカスタマイズします。", - "settings.general.row.font.title": "コードフォント", - "settings.general.row.font.description": "コードブロックで使用するフォントをカスタマイズします", - "settings.general.row.terminalFont.title": "Terminal Font", - "settings.general.row.terminalFont.description": "Customise the font used in the terminal", - "settings.general.row.uiFont.title": "UIフォント", - "settings.general.row.uiFont.description": "インターフェース全体で使用するフォントをカスタマイズします", - "settings.general.row.followup.title": "フォローアップの動作", - "settings.general.row.followup.description": - "フォローアッププロンプトを即座に実行するか、キューで待機させるかを選択します", - "settings.general.row.followup.option.queue": "キューに追加", - "settings.general.row.followup.option.steer": "即座に実行 (Steer)", - "settings.general.row.reasoningSummaries.title": "推論の要約を表示", - "settings.general.row.reasoningSummaries.description": "タイムラインにモデルの推論の要約を表示します", - "settings.general.row.shellToolPartsExpanded.title": "shell ツールパーツを展開", - "settings.general.row.shellToolPartsExpanded.description": - "タイムラインで shell ツールパーツをデフォルトで展開して表示します", - "settings.general.row.editToolPartsExpanded.title": "edit ツールパーツを展開", - "settings.general.row.editToolPartsExpanded.description": - "タイムラインで edit、write、patch ツールパーツをデフォルトで展開して表示します", - "settings.general.row.showSessionProgressBar.title": "セッション進行状況バーを表示", - "settings.general.row.showSessionProgressBar.description": - "エージェントの作業中に、セッション上部にアニメーション付きの進行状況バーを表示します", - "settings.general.row.wayland.title": "ネイティブWaylandを使用", - "settings.general.row.wayland.description": "WaylandでのX11フォールバックを無効にします。再起動が必要です。", - "settings.general.row.wayland.tooltip": - "リフレッシュレートが混在するモニターを使用しているLinuxでは、ネイティブWaylandの方が安定する場合があります。", - "settings.general.row.releaseNotes.title": "リリースノート", - "settings.general.row.releaseNotes.description": "アップデート後に「新機能」ポップアップを表示", - "settings.updates.row.startup.title": "起動時にアップデートを確認", - "settings.updates.row.startup.description": "Kilo の起動時に自動でアップデートを確認します", - "settings.updates.row.check.title": "アップデートを確認", - "settings.updates.row.check.description": "手動でアップデートを確認し、利用可能ならインストールします", - "settings.updates.action.checkNow": "今すぐ確認", - "settings.updates.action.checking": "確認中...", - "settings.updates.toast.latest.title": "最新です", - "settings.updates.toast.latest.description": "Kilo は最新バージョンです。", - "sound.option.none": "なし", - "sound.option.alert01": "アラート 01", - "sound.option.alert02": "アラート 02", - "sound.option.alert03": "アラート 03", - "sound.option.alert04": "アラート 04", - "sound.option.alert05": "アラート 05", - "sound.option.alert06": "アラート 06", - "sound.option.alert07": "アラート 07", - "sound.option.alert08": "アラート 08", - "sound.option.alert09": "アラート 09", - "sound.option.alert10": "アラート 10", - "sound.option.bipbop01": "ビップボップ 01", - "sound.option.bipbop02": "ビップボップ 02", - "sound.option.bipbop03": "ビップボップ 03", - "sound.option.bipbop04": "ビップボップ 04", - "sound.option.bipbop05": "ビップボップ 05", - "sound.option.bipbop06": "ビップボップ 06", - "sound.option.bipbop07": "ビップボップ 07", - "sound.option.bipbop08": "ビップボップ 08", - "sound.option.bipbop09": "ビップボップ 09", - "sound.option.bipbop10": "ビップボップ 10", - "sound.option.staplebops01": "ステープルボップス 01", - "sound.option.staplebops02": "ステープルボップス 02", - "sound.option.staplebops03": "ステープルボップス 03", - "sound.option.staplebops04": "ステープルボップス 04", - "sound.option.staplebops05": "ステープルボップス 05", - "sound.option.staplebops06": "ステープルボップス 06", - "sound.option.staplebops07": "ステープルボップス 07", - "sound.option.nope01": "いいえ 01", - "sound.option.nope02": "いいえ 02", - "sound.option.nope03": "いいえ 03", - "sound.option.nope04": "いいえ 04", - "sound.option.nope05": "いいえ 05", - "sound.option.nope06": "いいえ 06", - "sound.option.nope07": "いいえ 07", - "sound.option.nope08": "いいえ 08", - "sound.option.nope09": "いいえ 09", - "sound.option.nope10": "いいえ 10", - "sound.option.nope11": "いいえ 11", - "sound.option.nope12": "いいえ 12", - "sound.option.yup01": "はい 01", - "sound.option.yup02": "はい 02", - "sound.option.yup03": "はい 03", - "sound.option.yup04": "はい 04", - "sound.option.yup05": "はい 05", - "sound.option.yup06": "はい 06", - "settings.general.notifications.agent.title": "エージェント", - "settings.general.notifications.agent.description": - "エージェントが完了したか、注意が必要な場合にシステム通知を表示します", - "settings.general.notifications.permissions.title": "権限", - "settings.general.notifications.permissions.description": "権限が必要な場合にシステム通知を表示します", - "settings.general.notifications.errors.title": "エラー", - "settings.general.notifications.errors.description": "エラーが発生した場合にシステム通知を表示します", - "settings.general.sounds.agent.title": "エージェント", - "settings.general.sounds.agent.description": "エージェントが完了したか、注意が必要な場合に音を再生します", - "settings.general.sounds.permissions.title": "権限", - "settings.general.sounds.permissions.description": "権限が必要な場合に音を再生します", - "settings.general.sounds.errors.title": "エラー", - "settings.general.sounds.errors.description": "エラーが発生した場合に音を再生します", - "settings.shortcuts.title": "キーボードショートカット", - "settings.shortcuts.reset.button": "デフォルトにリセット", - "settings.shortcuts.reset.toast.title": "ショートカットをリセットしました", - "settings.shortcuts.reset.toast.description": "キーボードショートカットがデフォルトにリセットされました。", - "settings.shortcuts.conflict.title": "ショートカットは既に使用されています", - "settings.shortcuts.conflict.description": "{{keybind}} は既に {{titles}} に割り当てられています。", - "settings.shortcuts.unassigned": "未割り当て", - "settings.shortcuts.pressKeys": "キーを押してください", - "settings.shortcuts.search.placeholder": "ショートカットを検索", - "settings.shortcuts.search.empty": "ショートカットが見つかりません", - "settings.shortcuts.group.general": "一般", - "settings.shortcuts.group.session": "セッション", - "settings.shortcuts.group.navigation": "ナビゲーション", - "settings.shortcuts.group.modelAndAgent": "モデルとエージェント", - "settings.shortcuts.group.terminal": "ターミナル", - "settings.shortcuts.group.prompt": "プロンプト", - "settings.providers.title": "プロバイダー", - "settings.providers.description": "プロバイダー設定はここで構成できます。", - "settings.providers.section.connected": "接続済みプロバイダー", - "settings.providers.connected.empty": "接続済みプロバイダーはありません", - "settings.providers.section.popular": "人気のプロバイダー", - "settings.providers.tag.environment": "環境", - "settings.providers.tag.config": "設定", - "settings.providers.tag.custom": "カスタム", - "settings.providers.tag.other": "その他", - "settings.models.title": "モデル", - "settings.models.description": "モデル設定はここで構成できます。", - "settings.agents.title": "エージェント", - "settings.agents.description": "エージェント設定はここで構成できます。", - "settings.commands.title": "コマンド", - "settings.commands.description": "コマンド設定はここで構成できます。", - "settings.mcp.title": "MCP", - "settings.mcp.description": "MCP設定はここで構成できます。", - "settings.permissions.title": "権限", - "settings.permissions.description": "サーバーがデフォルトで使用できるツールを制御します。", - "settings.permissions.section.tools": "ツール", - "settings.permissions.toast.updateFailed.title": "権限の更新に失敗しました", - "settings.permissions.action.allow": "許可", - "settings.permissions.action.ask": "確認", - "settings.permissions.action.deny": "拒否", - "settings.permissions.tool.read.title": "読み込み", - "settings.permissions.tool.read.description": "ファイルの読み込み (ファイルパスに一致)", - "settings.permissions.tool.edit.title": "編集", - "settings.permissions.tool.edit.description": "ファイルの変更(編集、書き込み、パッチ、複数編集を含む)", - "settings.permissions.tool.glob.title": "Glob", - "settings.permissions.tool.glob.description": "Globパターンを使用したファイルの一致", - "settings.permissions.tool.grep.title": "Grep", - "settings.permissions.tool.grep.description": "正規表現を使用したファイル内容の検索", - "settings.permissions.tool.list.title": "リスト", - "settings.permissions.tool.list.description": "ディレクトリ内のファイル一覧表示", - "settings.permissions.tool.bash.title": "Bash", - "settings.permissions.tool.bash.description": "シェルコマンドの実行", - "settings.permissions.tool.task.title": "タスク", - "settings.permissions.tool.task.description": "サブエージェントの起動", - "settings.permissions.tool.skill.title": "スキル", - "settings.permissions.tool.skill.description": "名前によるスキルの読み込み", - "settings.permissions.tool.lsp.title": "LSP", - "settings.permissions.tool.lsp.description": "言語サーバークエリの実行", - "settings.permissions.tool.todowrite.title": "Todo書き込み", - "settings.permissions.tool.todowrite.description": "Todoリストの更新", - "settings.permissions.tool.webfetch.title": "Web取得", - "settings.permissions.tool.webfetch.description": "URLからコンテンツを取得", - "settings.permissions.tool.websearch.title": "Web検索", - "settings.permissions.tool.websearch.description": "ウェブを検索", - "settings.permissions.tool.codesearch.title": "コード検索", - "settings.permissions.tool.codesearch.description": "ウェブ上のコードを検索", - "settings.permissions.tool.external_directory.title": "外部ディレクトリ", - "settings.permissions.tool.external_directory.description": "プロジェクトディレクトリ外のファイルへのアクセス", - "settings.permissions.tool.doom_loop.title": "無限ループ", - "settings.permissions.tool.doom_loop.description": "同一入力による繰り返しのツール呼び出しを検出", - "session.delete.failed.title": "セッションの削除に失敗しました", - "session.delete.title": "セッションの削除", - "session.delete.confirm": 'セッション "{{name}}" を削除しますか?', - "session.delete.button": "セッションを削除", - "workspace.new": "新しいワークスペース", - "workspace.type.local": "ローカル", - "workspace.type.sandbox": "サンドボックス", - "workspace.create.failed.title": "ワークスペースの作成に失敗しました", - "workspace.delete.failed.title": "ワークスペースの削除に失敗しました", - "workspace.resetting.title": "ワークスペースをリセット中", - "workspace.resetting.description": "これには少し時間がかかる場合があります。", - "workspace.reset.failed.title": "ワークスペースのリセットに失敗しました", - "workspace.reset.success.title": "ワークスペースをリセットしました", - "workspace.reset.success.description": "ワークスペースはデフォルトブランチと一致しています。", - "workspace.error.stillPreparing": "ワークスペースはまだ準備中です", - "workspace.status.checking": "未マージの変更を確認中...", - "workspace.status.error": "gitステータスを確認できません。", - "workspace.status.clean": "未マージの変更は検出されませんでした。", - "workspace.status.dirty": "このワークスペースで未マージの変更が検出されました。", - "workspace.delete.title": "ワークスペースの削除", - "workspace.delete.confirm": 'ワークスペース "{{name}}" を削除しますか?', - "workspace.delete.button": "ワークスペースを削除", - "workspace.reset.title": "ワークスペースのリセット", - "workspace.reset.confirm": 'ワークスペース "{{name}}" をリセットしますか?', - "workspace.reset.button": "ワークスペースをリセット", - "workspace.reset.archived.none": "アクティブなセッションはアーカイブされません。", - "workspace.reset.archived.one": "1つのセッションがアーカイブされます。", - "workspace.reset.archived.many": "{{count}}個のセッションがアーカイブされます。", - "workspace.reset.note": "これにより、ワークスペースはデフォルトブランチと一致するようにリセットされます。", - "common.open": "開く", - "dialog.releaseNotes.action.getStarted": "始める", - "dialog.releaseNotes.action.next": "次へ", - "dialog.releaseNotes.action.hideFuture": "今後表示しない", - "dialog.releaseNotes.media.alt": "リリースのプレビュー", - "toast.project.reloadFailed.title": "{{project}} の再読み込みに失敗しました", - "error.server.invalidConfiguration": "無効な設定", - "common.moreCountSuffix": " (他 {{count}} 件)", - "common.time.justNow": "たった今", - "common.time.minutesAgo.short": "{{count}} 分前", - "common.time.hoursAgo.short": "{{count}} 時間前", - "common.time.daysAgo.short": "{{count}} 日前", - "settings.providers.connected.environmentDescription": "環境変数から接続されました", - "settings.providers.custom.description": "ベース URL を指定して OpenAI 互換のプロバイダーを追加します。", - - "app.server.unreachable": "{{server}} に到達できませんでした", - "app.server.retrying": "自動的に再試行中...", - "app.server.otherServers": "その他のサーバー", - "dialog.server.add.usernamePlaceholder": "ユーザー名", - "dialog.server.add.passwordPlaceholder": "パスワード", - "server.row.noUsername": "ユーザー名なし", - "session.review.noVcs.createGit.title": "Git リポジトリを作成", - "session.review.noVcs.createGit.description": "このプロジェクトの変更を追跡、レビュー、元に戻す", - "session.review.noVcs.createGit.actionLoading": "Git リポジトリを作成中...", - "session.review.noVcs.createGit.action": "Git リポジトリを作成", - "session.todo.progress": "{{done}} 個中 {{total}} 個の Todo が完了", - "session.question.progress": "{{total}} 問中 {{current}} 問", - "session.header.open.finder": "Finder", - "session.header.open.fileExplorer": "エクスプローラー", - "session.header.open.fileManager": "ファイルマネージャー", - "session.header.open.app.vscode": "VS Code", - "session.header.open.app.cursor": "Cursor", - "session.header.open.app.zed": "Zed", - "session.header.open.app.textmate": "TextMate", - "session.header.open.app.antigravity": "Antigravity", - "session.header.open.app.terminal": "ターミナル", - "session.header.open.app.iterm2": "iTerm2", - "session.header.open.app.ghostty": "Ghostty", - "session.header.open.app.warp": "Warp", - "session.header.open.app.xcode": "Xcode", - "session.header.open.app.androidStudio": "Android Studio", - "session.header.open.app.powershell": "PowerShell", - "session.header.open.app.sublimeText": "Sublime Text", - "debugBar.ariaLabel": "開発パフォーマンス診断", - "debugBar.na": "n/a", - "debugBar.nav.label": "NAV", - "debugBar.nav.tip": "セッションページに触れる最後に完了したルート遷移。ルーター開始から安定後の最初の描画まで測定。", - "debugBar.fps.label": "FPS", - "debugBar.fps.tip": "過去5秒間のローリングフレーム/秒。", - "debugBar.frame.label": "FRAME", - "debugBar.frame.tip": "過去5秒間の最悪フレーム時間。", - "debugBar.jank.label": "JANK", - "debugBar.jank.tip": "過去5秒間で32msを超えたフレーム。", - "debugBar.long.label": "LONG", - "debugBar.long.tip": "過去5秒間のブロック時間と長時間タスク数。最大タスク: {{max}}。", - "debugBar.delay.label": "DELAY", - "debugBar.delay.tip": "過去5秒間で観測された最悪の入力遅延。", - "debugBar.inp.label": "INP", - "debugBar.inp.tip": - "過去5秒間の概算インタラクション時間。これは INP に似ていますが、公式の Web Vitals INP ではありません。", - "debugBar.cls.label": "CLS", - "debugBar.cls.tip": "現在のアプリ寿命の累積レイアウトシフト。", - "debugBar.mem.label": "MEM", - "debugBar.mem.tipUnavailable": "使用中の JS ヒープ対ヒープ制限。Chromium のみ。", - "debugBar.mem.tip": "使用中の JS ヒープ対ヒープ制限。{{limit}} 中 {{used}}。", - "common.key.ctrl": "Ctrl", - "common.key.alt": "Alt", - "common.key.shift": "Shift", - "common.key.meta": "Meta", - "common.key.space": "Space", - "common.key.backspace": "Backspace", - "common.key.enter": "Enter", - "common.key.tab": "Tab", - "common.key.delete": "Delete", - "common.key.home": "Home", - "common.key.end": "End", - "common.key.pageUp": "Page Up", - "common.key.pageDown": "Page Down", - "common.key.insert": "Insert", - "common.unknown": "不明", - "error.page.circular": "[循環]", - "error.globalSDK.noServerAvailable": "利用可能なサーバーがありません", - "error.globalSDK.serverNotAvailable": "サーバーが利用できません", - "error.childStore.persistedCacheCreateFailed": "永続キャッシュの作成に失敗しました", - "error.childStore.persistedProjectMetadataCreateFailed": "永続プロジェクトメタデータの作成に失敗しました", - "error.childStore.persistedProjectIconCreateFailed": "永続プロジェクトアイコンの作成に失敗しました", - "error.childStore.storeCreateFailed": "ストアの作成に失敗しました", - "terminal.connectionLost.abnormalClose": "WebSocket が異常終了しました: {{code}}", -} diff --git a/packages/app/src/i18n/ko.ts b/packages/app/src/i18n/ko.ts deleted file mode 100644 index c1c6f397cc..0000000000 --- a/packages/app/src/i18n/ko.ts +++ /dev/null @@ -1,848 +0,0 @@ -export const dict = { - "command.category.suggested": "추천", - "command.category.view": "보기", - "command.category.project": "프로젝트", - "command.category.provider": "공급자", - "command.category.server": "서버", - "command.category.session": "세션", - "command.category.theme": "테마", - "command.category.language": "언어", - "command.category.file": "파일", - "command.category.context": "컨텍스트", - "command.category.terminal": "터미널", - "command.category.model": "모델", - "command.category.mcp": "MCP", - "command.category.agent": "에이전트", - "command.category.permissions": "권한", - "command.category.workspace": "작업 공간", - "command.category.settings": "설정", - "theme.scheme.system": "시스템", - "theme.scheme.light": "라이트", - "theme.scheme.dark": "다크", - "command.sidebar.toggle": "사이드바 토글", - "command.project.open": "프로젝트 열기", - "command.provider.connect": "공급자 연결", - "command.server.switch": "서버 전환", - "command.settings.open": "설정 열기", - "command.session.previous": "이전 세션", - "command.session.next": "다음 세션", - "command.session.previous.unseen": "이전 읽지 않은 세션", - "command.session.next.unseen": "다음 읽지 않은 세션", - "command.session.archive": "세션 보관", - "command.palette": "명령 팔레트", - "command.theme.cycle": "테마 순환", - "command.theme.set": "테마 사용: {{theme}}", - "command.theme.scheme.cycle": "색상 테마 순환", - "command.theme.scheme.set": "색상 테마 사용: {{scheme}}", - "command.language.cycle": "언어 순환", - "command.language.set": "언어 사용: {{language}}", - "command.session.new": "새 세션", - "command.file.open": "파일 열기", - "command.tab.close": "탭 닫기", - "command.context.addSelection": "선택 영역을 컨텍스트에 추가", - "command.context.addSelection.description": "현재 파일에서 선택한 줄을 추가", - "command.input.focus": "입력창 포커스", - "command.terminal.toggle": "터미널 토글", - "command.fileTree.toggle": "파일 트리 토글", - "command.review.toggle": "검토 토글", - "command.terminal.new": "새 터미널", - "command.terminal.new.description": "새 터미널 탭 생성", - "command.steps.toggle": "단계 토글", - "command.steps.toggle.description": "현재 메시지의 단계 표시/숨기기", - "command.message.previous": "이전 메시지", - "command.message.previous.description": "이전 사용자 메시지로 이동", - "command.message.next": "다음 메시지", - "command.message.next.description": "다음 사용자 메시지로 이동", - "command.model.choose": "모델 선택", - "command.model.choose.description": "다른 모델 선택", - "command.mcp.toggle": "MCP 토글", - "command.mcp.toggle.description": "MCP 토글", - "command.agent.cycle": "에이전트 순환", - "command.agent.cycle.description": "다음 에이전트로 전환", - "command.agent.cycle.reverse": "에이전트 역순환", - "command.agent.cycle.reverse.description": "이전 에이전트로 전환", - "command.model.variant.cycle": "생각 수준 순환", - "command.model.variant.cycle.description": "다음 생각 수준으로 전환", - "command.prompt.mode.shell": "셸", - "command.prompt.mode.normal": "프롬프트", - "command.permissions.autoaccept.enable": "권한 자동 수락", - "command.permissions.autoaccept.disable": "권한 자동 수락 중지", - "command.workspace.toggle": "작업 공간 전환", - "command.workspace.toggle.description": "사이드바에서 다중 작업 공간 활성화 또는 비활성화", - "command.session.undo": "실행 취소", - "command.session.undo.description": "마지막 메시지 실행 취소", - "command.session.redo": "다시 실행", - "command.session.redo.description": "마지막 실행 취소된 메시지 다시 실행", - "command.session.compact": "세션 압축", - "command.session.compact.description": "컨텍스트 크기를 줄이기 위해 세션 요약", - "command.session.fork": "메시지에서 분기", - "command.session.fork.description": "이전 메시지에서 새 세션 생성", - "command.session.share": "세션 공유", - "command.session.share.description": "이 세션을 공유하고 URL을 클립보드에 복사", - "command.session.unshare": "세션 공유 중지", - "command.session.unshare.description": "이 세션 공유 중지", - "palette.search.placeholder": "파일, 명령어 및 세션 검색", - "palette.empty": "결과 없음", - "palette.group.commands": "명령어", - "palette.group.files": "파일", - "dialog.provider.search.placeholder": "공급자 검색", - "dialog.provider.empty": "공급자 없음", - "dialog.provider.group.popular": "인기", - "dialog.provider.group.other": "기타", - "dialog.provider.tag.recommended": "추천", - "dialog.provider.opencode.note": "Claude, GPT, Gemini 등을 포함한 엄선된 모델", - "dialog.provider.opencode.tagline": "신뢰할 수 있는 최적화 모델", - "dialog.provider.opencodeGo.tagline": "모두를 위한 저렴한 구독", - "dialog.provider.anthropic.note": "Claude Pro/Max 또는 API 키로 연결", - "dialog.provider.copilot.note": "Copilot 또는 API 키로 연결", - "dialog.provider.openai.note": "ChatGPT Pro/Plus 또는 API 키로 연결", - "dialog.provider.google.note": "빠르고 구조화된 응답을 위한 Gemini 모델", - "dialog.provider.openrouter.note": "모든 지원 모델을 단일 공급자에서 액세스", - "dialog.provider.vercel.note": "스마트 라우팅을 통한 AI 모델 통합 액세스", - "dialog.model.select.title": "모델 선택", - "dialog.model.search.placeholder": "모델 검색", - "dialog.model.empty": "모델 결과 없음", - "dialog.model.manage": "모델 관리", - "dialog.model.manage.description": "모델 선택기에 표시할 모델 사용자 지정", - "dialog.model.manage.provider.toggle": "모든 {{provider}} 모델 토글", - "dialog.model.unpaid.freeModels.title": "Kilo에서 제공하는 무료 모델", - "dialog.model.unpaid.addMore.title": "인기 공급자의 모델 추가", - "dialog.provider.viewAll": "더 많은 공급자 보기", - "provider.connect.title": "{{provider}} 연결", - "provider.connect.title.anthropicProMax": "Claude Pro/Max로 로그인", - "provider.connect.selectMethod": "{{provider}} 로그인 방법 선택", - "provider.connect.method.apiKey": "API 키", - "provider.connect.status.inProgress": "인증 진행 중...", - "provider.connect.status.waiting": "인증 대기 중...", - "provider.connect.status.failed": "인증 실패: {{error}}", - "provider.connect.apiKey.description": - "{{provider}} API 키를 입력하여 계정을 연결하고 Kilo에서 {{provider}} 모델을 사용하세요.", - "provider.connect.apiKey.label": "{{provider}} API 키", - "provider.connect.apiKey.placeholder": "API 키", - "provider.connect.apiKey.required": "API 키가 필요합니다", - "provider.connect.opencodeZen.line1": - "OpenCode Zen은 코딩 에이전트를 위해 최적화된 신뢰할 수 있는 엄선된 모델에 대한 액세스를 제공합니다.", - "provider.connect.opencodeZen.line2": "단일 API 키로 Claude, GPT, Gemini, GLM 등 다양한 모델에 액세스할 수 있습니다.", - "provider.connect.opencodeZen.visit.prefix": "다음 ", - "provider.connect.opencodeZen.visit.link": "opencode.ai/zen", - "provider.connect.opencodeZen.visit.suffix": "을 방문하여 API 키를 받으세요.", - "provider.connect.oauth.code.visit.prefix": "다음 ", - "provider.connect.oauth.code.visit.link": "이 링크", - "provider.connect.oauth.code.visit.suffix": - "를 방문하여 인증 코드를 받아 계정을 연결하고 Kilo에서 {{provider}} 모델을 사용하세요.", - "provider.connect.oauth.code.label": "{{method}} 인증 코드", - "provider.connect.oauth.code.placeholder": "인증 코드", - "provider.connect.oauth.code.required": "인증 코드가 필요합니다", - "provider.connect.oauth.code.invalid": "유효하지 않은 인증 코드", - "provider.connect.oauth.auto.visit.prefix": "다음 ", - "provider.connect.oauth.auto.visit.link": "이 링크", - "provider.connect.oauth.auto.visit.suffix": - "를 방문하고 아래 코드를 입력하여 계정을 연결하고 Kilo에서 {{provider}} 모델을 사용하세요.", - "provider.connect.oauth.auto.confirmationCode": "확인 코드", - "provider.connect.toast.connected.title": "{{provider}} 연결됨", - "provider.connect.toast.connected.description": "이제 {{provider}} 모델을 사용할 수 있습니다.", - "provider.custom.title": "사용자 지정 공급자", - "provider.custom.description.prefix": "OpenAI 호환 공급자를 구성합니다. ", - "provider.custom.description.link": "공급자 구성 문서", - "provider.custom.description.suffix": "를 참조하세요.", - "provider.custom.field.providerID.label": "공급자 ID", - "provider.custom.field.providerID.placeholder": "myprovider", - "provider.custom.field.providerID.description": "소문자, 숫자, 하이픈 또는 밑줄", - "provider.custom.field.name.label": "표시 이름", - "provider.custom.field.name.placeholder": "내 AI 공급자", - "provider.custom.field.baseURL.label": "기본 URL", - "provider.custom.field.baseURL.placeholder": "https://api.myprovider.com/v1", - "provider.custom.field.apiKey.label": "API 키", - "provider.custom.field.apiKey.placeholder": "API 키", - "provider.custom.field.apiKey.description": "선택 사항입니다. 헤더를 통해 인증을 관리하는 경우 비워 두세요.", - "provider.custom.models.label": "모델", - "provider.custom.models.id.label": "ID", - "provider.custom.models.id.placeholder": "model-id", - "provider.custom.models.name.label": "이름", - "provider.custom.models.name.placeholder": "표시 이름", - "provider.custom.models.remove": "모델 제거", - "provider.custom.models.add": "모델 추가", - "provider.custom.headers.label": "헤더 (선택 사항)", - "provider.custom.headers.key.label": "헤더", - "provider.custom.headers.key.placeholder": "헤더 이름", - "provider.custom.headers.value.label": "값", - "provider.custom.headers.value.placeholder": "값", - "provider.custom.headers.remove": "헤더 제거", - "provider.custom.headers.add": "헤더 추가", - "provider.custom.error.providerID.required": "공급자 ID가 필요합니다", - "provider.custom.error.providerID.format": "소문자, 숫자, 하이픈 또는 밑줄을 사용하세요", - "provider.custom.error.providerID.exists": "해당 공급자 ID가 이미 존재합니다", - "provider.custom.error.name.required": "표시 이름이 필요합니다", - "provider.custom.error.baseURL.required": "기본 URL이 필요합니다", - "provider.custom.error.baseURL.format": "http:// 또는 https://로 시작해야 합니다", - "provider.custom.error.required": "필수", - "provider.custom.error.duplicate": "중복", - "provider.disconnect.toast.disconnected.title": "{{provider}} 연결 해제됨", - "provider.disconnect.toast.disconnected.description": "{{provider}} 모델을 더 이상 사용할 수 없습니다.", - "model.tag.free": "무료", - "model.tag.latest": "최신", - "model.provider.anthropic": "Anthropic", - "model.provider.openai": "OpenAI", - "model.provider.google": "Google", - "model.provider.xai": "xAI", - "model.provider.meta": "Meta", - "model.input.text": "텍스트", - "model.input.image": "이미지", - "model.input.audio": "오디오", - "model.input.video": "비디오", - "model.input.pdf": "pdf", - "model.tooltip.allows": "지원: {{inputs}}", - "model.tooltip.reasoning.allowed": "추론 허용", - "model.tooltip.reasoning.none": "추론 없음", - "model.tooltip.context": "컨텍스트 제한 {{limit}}", - "common.search.placeholder": "검색", - "common.goBack": "뒤로 가기", - "common.goForward": "앞으로 가기", - "common.loading": "로딩 중", - "common.loading.ellipsis": "...", - "common.cancel": "취소", - "common.connect": "연결", - "common.disconnect": "연결 해제", - "common.continue": "제출", - "common.submit": "제출", - "common.save": "저장", - "common.saving": "저장 중...", - "common.default": "기본값", - "common.attachment": "첨부 파일", - "prompt.placeholder.shell": "셸 명령어 입력... {{example}}", - "prompt.placeholder.normal": '무엇이든 물어보세요... "{{example}}"', - "prompt.placeholder.simple": "무엇이든 물어보세요...", - "prompt.placeholder.summarizeComments": "댓글 요약…", - "prompt.placeholder.summarizeComment": "댓글 요약…", - "prompt.mode.shell": "셸", - "prompt.mode.normal": "프롬프트", - "prompt.mode.shell.exit": "종료하려면 esc", - "prompt.example.1": "코드베이스의 TODO 수정", - "prompt.example.2": "이 프로젝트의 기술 스택이 무엇인가요?", - "prompt.example.3": "고장 난 테스트 수정", - "prompt.example.4": "인증 작동 방식 설명", - "prompt.example.5": "보안 취약점 찾기 및 수정", - "prompt.example.6": "사용자 서비스에 단위 테스트 추가", - "prompt.example.7": "이 함수를 더 읽기 쉽게 리팩터링", - "prompt.example.8": "이 오류는 무엇을 의미하나요?", - "prompt.example.9": "이 문제 디버깅 도와줘", - "prompt.example.10": "API 문서 생성", - "prompt.example.11": "데이터베이스 쿼리 최적화", - "prompt.example.12": "입력 유효성 검사 추가", - "prompt.example.13": "...를 위한 새 컴포넌트 생성", - "prompt.example.14": "이 프로젝트를 어떻게 배포하나요?", - "prompt.example.15": "모범 사례를 기준으로 내 코드 검토", - "prompt.example.16": "이 함수에 오류 처리 추가", - "prompt.example.17": "이 정규식 패턴 설명", - "prompt.example.18": "이것을 TypeScript로 변환", - "prompt.example.19": "코드베이스 전체에 로깅 추가", - "prompt.example.20": "오래된 종속성은 무엇인가요?", - "prompt.example.21": "마이그레이션 스크립트 작성 도와줘", - "prompt.example.22": "이 엔드포인트에 캐싱 구현", - "prompt.example.23": "이 목록에 페이지네이션 추가", - "prompt.example.24": "...를 위한 CLI 명령어 생성", - "prompt.example.25": "여기서 환경 변수는 어떻게 작동하나요?", - "prompt.popover.emptyResults": "일치하는 결과 없음", - "prompt.popover.emptyCommands": "일치하는 명령어 없음", - "prompt.dropzone.label": "이미지, PDF 또는 텍스트 파일을 이곳에 드롭하세요", - "prompt.dropzone.file.label": "드롭하여 파일 @멘션 추가", - "prompt.slash.badge.custom": "사용자 지정", - "prompt.slash.badge.skill": "스킬", - "prompt.slash.badge.mcp": "mcp", - "prompt.context.active": "활성", - "prompt.context.includeActiveFile": "활성 파일 포함", - "prompt.context.removeActiveFile": "컨텍스트에서 활성 파일 제거", - "prompt.context.removeFile": "컨텍스트에서 파일 제거", - "prompt.action.attachFile": "파일 첨부", - "prompt.attachment.remove": "첨부 파일 제거", - "prompt.action.send": "전송", - "prompt.action.stop": "중지", - "prompt.toast.pasteUnsupported.title": "지원되지 않는 첨부 파일", - "prompt.toast.pasteUnsupported.description": "이미지, PDF 또는 텍스트 파일만 첨부할 수 있습니다.", - "prompt.toast.modelAgentRequired.title": "에이전트 및 모델 선택", - "prompt.toast.modelAgentRequired.description": "프롬프트를 보내기 전에 에이전트와 모델을 선택하세요.", - "prompt.toast.worktreeCreateFailed.title": "작업 트리 생성 실패", - "prompt.toast.sessionCreateFailed.title": "세션 생성 실패", - "prompt.toast.shellSendFailed.title": "셸 명령 전송 실패", - "prompt.toast.commandSendFailed.title": "명령 전송 실패", - "prompt.toast.promptSendFailed.title": "프롬프트 전송 실패", - "prompt.toast.promptSendFailed.description": "세션을 가져올 수 없습니다", - "dialog.mcp.title": "MCP", - "dialog.mcp.description": "{{total}}개 중 {{enabled}}개 활성화됨", - "dialog.mcp.empty": "구성된 MCP 없음", - "dialog.lsp.empty": "파일 유형에서 자동 감지된 LSP", - "dialog.plugins.empty": "opencode.json에 구성된 플러그인", - "mcp.status.connected": "연결됨", - "mcp.status.failed": "실패", - "mcp.status.needs_auth": "인증 필요", - "mcp.status.disabled": "비활성화됨", - "dialog.fork.empty": "분기할 메시지 없음", - "dialog.directory.search.placeholder": "폴더 검색", - "dialog.directory.empty": "폴더 없음", - "dialog.server.title": "서버", - "dialog.server.description": "이 앱이 연결할 Kilo 서버를 전환합니다.", - "dialog.server.search.placeholder": "서버 검색", - "dialog.server.empty": "서버 없음", - "dialog.server.add.title": "서버 추가", - "dialog.server.add.url": "서버 URL", - "dialog.server.add.placeholder": "http://localhost:4096", - "dialog.server.add.error": "서버에 연결할 수 없습니다", - "dialog.server.add.checking": "확인 중...", - "dialog.server.add.button": "서버 추가", - "dialog.server.add.name": "서버 이름 (선택 사항)", - "dialog.server.add.namePlaceholder": "Localhost", - "dialog.server.add.username": "사용자 이름 (선택 사항)", - "dialog.server.add.password": "비밀번호 (선택 사항)", - "dialog.server.edit.title": "서버 편집", - "dialog.server.default.title": "기본 서버", - "dialog.server.default.description": - "로컬 서버를 시작하는 대신 앱 실행 시 이 서버에 연결합니다. 다시 시작해야 합니다.", - "dialog.server.default.none": "선택된 서버 없음", - "dialog.server.default.set": "현재 서버를 기본값으로 설정", - "dialog.server.default.clear": "지우기", - "dialog.server.action.remove": "서버 제거", - "dialog.server.menu.edit": "편집", - "dialog.server.menu.default": "기본값으로 설정", - "dialog.server.menu.defaultRemove": "기본값 제거", - "dialog.server.menu.delete": "삭제", - "dialog.server.current": "현재 서버", - "dialog.server.status.default": "기본값", - "dialog.project.edit.title": "프로젝트 편집", - "dialog.project.edit.name": "이름", - "dialog.project.edit.icon": "아이콘", - "dialog.project.edit.icon.alt": "프로젝트 아이콘", - "dialog.project.edit.icon.hint": "이미지를 클릭하거나 드래그하세요", - "dialog.project.edit.icon.recommended": "권장: 128x128px", - "dialog.project.edit.color": "색상", - "dialog.project.edit.color.select": "{{color}} 색상 선택", - "dialog.project.edit.worktree.startup": "작업 공간 시작 스크립트", - "dialog.project.edit.worktree.startup.description": "새 작업 공간(작업 트리)을 만든 뒤 실행됩니다.", - "dialog.project.edit.worktree.startup.placeholder": "예: bun install", - "context.breakdown.title": "컨텍스트 분석", - "context.breakdown.note": '입력 토큰의 대략적인 분석입니다. "기타"에는 도구 정의 및 오버헤드가 포함됩니다.', - "context.breakdown.system": "시스템", - "context.breakdown.user": "사용자", - "context.breakdown.assistant": "어시스턴트", - "context.breakdown.tool": "도구 호출", - "context.breakdown.other": "기타", - "context.systemPrompt.title": "시스템 프롬프트", - "context.rawMessages.title": "원시 메시지", - "context.stats.session": "세션", - "context.stats.messages": "메시지", - "context.stats.provider": "공급자", - "context.stats.model": "모델", - "context.stats.limit": "컨텍스트 제한", - "context.stats.totalTokens": "총 토큰", - "context.stats.usage": "사용량", - "context.stats.inputTokens": "입력 토큰", - "context.stats.outputTokens": "출력 토큰", - "context.stats.reasoningTokens": "추론 토큰", - "context.stats.cacheTokens": "캐시 토큰 (읽기/쓰기)", - "context.stats.userMessages": "사용자 메시지", - "context.stats.assistantMessages": "어시스턴트 메시지", - "context.stats.totalCost": "총 비용", - "context.stats.sessionCreated": "세션 생성됨", - "context.stats.lastActivity": "최근 활동", - "context.usage.tokens": "토큰", - "context.usage.usage": "사용량", - "context.usage.cost": "비용", - "context.usage.clickToView": "컨텍스트를 보려면 클릭", - "context.usage.view": "컨텍스트 사용량 보기", - "language.en": "English", - "language.zh": "简体中文", - "language.zht": "繁體中文", - "language.ko": "한국어", - "language.de": "Deutsch", - "language.es": "Español", - "language.fr": "Français", - "language.da": "Dansk", - "language.ja": "日本語", - "language.pl": "Polski", - "language.ru": "Русский", - "language.ar": "العربية", - "language.no": "Norsk", - "language.br": "Português (Brasil)", - "language.bs": "Bosanski", - "language.th": "ไทย", - "language.tr": "Türkçe", - "toast.language.title": "언어", - "toast.language.description": "{{language}}(으)로 전환됨", - "toast.theme.title": "테마 전환됨", - "toast.scheme.title": "색상 테마", - "toast.workspace.enabled.title": "작업 공간 활성화됨", - "toast.workspace.enabled.description": "이제 사이드바에 여러 작업 트리가 표시됩니다", - "toast.workspace.disabled.title": "작업 공간 비활성화됨", - "toast.workspace.disabled.description": "사이드바에 메인 작업 트리만 표시됩니다", - "toast.permissions.autoaccept.on.title": "권한 자동 수락 중", - "toast.permissions.autoaccept.on.description": "권한 요청이 자동으로 승인됩니다", - "toast.permissions.autoaccept.off.title": "권한 자동 수락 중지됨", - "toast.permissions.autoaccept.off.description": "권한 요청에 승인이 필요합니다", - "toast.model.none.title": "선택된 모델 없음", - "toast.model.none.description": "이 세션을 요약하려면 공급자를 연결하세요", - "toast.file.loadFailed.title": "파일 로드 실패", - "toast.file.listFailed.title": "파일 목록을 불러오지 못했습니다", - "toast.context.noLineSelection.title": "줄 선택 없음", - "toast.context.noLineSelection.description": "먼저 파일 탭에서 줄 범위를 선택하세요.", - "toast.session.share.copyFailed.title": "URL 클립보드 복사 실패", - "toast.session.share.success.title": "세션 공유됨", - "toast.session.share.success.description": "공유 URL이 클립보드에 복사되었습니다!", - "toast.session.share.failed.title": "세션 공유 실패", - "toast.session.share.failed.description": "세션을 공유하는 동안 오류가 발생했습니다", - "toast.session.unshare.success.title": "세션 공유 해제됨", - "toast.session.unshare.success.description": "세션 공유가 성공적으로 해제되었습니다!", - "toast.session.unshare.failed.title": "세션 공유 해제 실패", - "toast.session.unshare.failed.description": "세션 공유를 해제하는 동안 오류가 발생했습니다", - "toast.session.listFailed.title": "{{project}}에 대한 세션을 로드하지 못했습니다", - "toast.update.title": "업데이트 가능", - "toast.update.description": "Kilo의 새 버전({{version}})을 설치할 수 있습니다.", - "toast.update.action.installRestart": "설치 및 다시 시작", - "toast.update.action.notYet": "나중에", - "error.page.title": "문제가 발생했습니다", - "error.page.description": "애플리케이션을 로드하는 동안 오류가 발생했습니다.", - "error.page.details.label": "오류 세부 정보", - "error.page.action.restart": "다시 시작", - "error.page.action.checking": "확인 중...", - "error.page.action.checkUpdates": "업데이트 확인", - "error.page.action.updateTo": "{{version}} 버전으로 업데이트", - "error.page.report.prefix": "이 오류를 Kilo 팀에 제보해 주세요: ", - "error.page.report.discord": "Discord", - "error.page.version": "버전: {{version}}", - "error.dev.rootNotFound": - "루트 요소를 찾을 수 없습니다. index.html에 추가하는 것을 잊으셨나요? 또는 id 속성의 철자가 틀렸을 수 있습니다.", - "error.globalSync.connectFailed": "서버에 연결할 수 없습니다. `{{url}}`에서 서버가 실행 중인가요?", - "directory.error.invalidUrl": "URL에 유효하지 않은 디렉터리가 있습니다.", - "error.chain.unknown": "알 수 없는 오류", - "error.chain.causedBy": "원인:", - "error.chain.apiError": "API 오류", - "error.chain.status": "상태: {{status}}", - "error.chain.retryable": "재시도 가능: {{retryable}}", - "error.chain.responseBody": "응답 본문:\n{{body}}", - "error.chain.didYouMean": "혹시 {{suggestions}}을(를) 의미하셨나요?", - "error.chain.modelNotFound": "모델을 찾을 수 없음: {{provider}}/{{model}}", - "error.chain.checkConfig": "구성(opencode.json)의 공급자/모델 이름을 확인하세요", - "error.chain.mcpFailed": 'MCP 서버 "{{name}}" 실패. 참고: Kilo는 아직 MCP 인증을 지원하지 않습니다.', - "error.chain.providerAuthFailed": "공급자 인증 실패 ({{provider}}): {{message}}", - "error.chain.providerInitFailed": '공급자 "{{provider}}" 초기화 실패. 자격 증명과 구성을 확인하세요.', - "error.chain.configJsonInvalid": "{{path}}의 구성 파일이 유효한 JSON(C)가 아닙니다", - "error.chain.configJsonInvalidWithMessage": "{{path}}의 구성 파일이 유효한 JSON(C)가 아닙니다: {{message}}", - "error.chain.configDirectoryTypo": - '{{path}}의 "{{dir}}" 디렉터리가 유효하지 않습니다. 디렉터리 이름을 "{{suggestion}}"으로 변경하거나 제거하세요. 이는 흔한 오타입니다.', - "error.chain.configFrontmatterError": "{{path}}의 frontmatter 파싱 실패:\n{{message}}", - "error.chain.configInvalid": "{{path}}의 구성 파일이 유효하지 않습니다", - "error.chain.configInvalidWithMessage": "{{path}}의 구성 파일이 유효하지 않습니다: {{message}}", - "notification.permission.title": "권한 필요", - "notification.permission.description": "{{projectName}}의 {{sessionTitle}}에서 권한이 필요합니다", - "notification.question.title": "질문", - "notification.question.description": "{{projectName}}의 {{sessionTitle}}에서 질문이 있습니다", - "notification.action.goToSession": "세션으로 이동", - "notification.session.responseReady.title": "응답 준비됨", - "notification.session.error.title": "세션 오류", - "notification.session.error.fallbackDescription": "오류가 발생했습니다", - "home.recentProjects": "최근 프로젝트", - "home.empty.title": "최근 프로젝트 없음", - "home.empty.description": "로컬 프로젝트를 열어 시작하세요", - "session.tab.session": "세션", - "session.tab.review": "검토", - "session.tab.context": "컨텍스트", - "session.panel.reviewAndFiles": "검토 및 파일", - "session.review.filesChanged": "{{count}}개 파일 변경됨", - "session.review.change.one": "변경", - "session.review.change.other": "변경", - "session.review.loadingChanges": "변경 사항 로드 중...", - "session.review.empty": "이 세션에 변경 사항이 아직 없습니다", - "session.review.noVcs": "Git 버전 관리 시스템이 감지되지 않아 변경 사항이 표시되지 않습니다", - "session.review.noSnapshot": "구성에서 스냅샷 추적이 비활성화되어 있어 세션 변경 사항을 사용할 수 없습니다", - "session.review.noChanges": "변경 없음", - "session.files.selectToOpen": "열 파일을 선택하세요", - "session.files.all": "모든 파일", - "session.files.empty": "파일 없음", - "session.files.binaryContent": "바이너리 파일 (내용을 표시할 수 없음)", - "session.messages.renderEarlier": "이전 메시지 렌더링", - "session.messages.loadingEarlier": "이전 메시지 로드 중...", - "session.messages.loadEarlier": "이전 메시지 로드", - "session.messages.loading": "메시지 로드 중...", - "session.messages.jumpToLatest": "최신으로 이동", - "session.context.addToContext": "컨텍스트에 {{selection}} 추가", - "session.todo.title": "할 일", - "session.todo.collapse": "접기", - "session.todo.expand": "펼치기", - "session.followupDock.summary.one": "{{count}}개의 대기 중인 메시지", - "session.followupDock.summary.other": "{{count}}개의 대기 중인 메시지", - "session.followupDock.sendNow": "지금 전송", - "session.followupDock.edit": "편집", - "session.followupDock.collapse": "대기 중인 메시지 접기", - "session.followupDock.expand": "대기 중인 메시지 펼치기", - "session.revertDock.summary.one": "{{count}}개의 롤백된 메시지", - "session.revertDock.summary.other": "{{count}}개의 롤백된 메시지", - "session.revertDock.collapse": "롤백된 메시지 접기", - "session.revertDock.expand": "롤백된 메시지 펼치기", - "session.revertDock.restore": "메시지 복원", - "session.new.title": "무엇이든 만들기", - "session.new.worktree.main": "메인 브랜치", - "session.new.worktree.mainWithBranch": "메인 브랜치 ({{branch}})", - "session.new.worktree.create": "새 작업 트리 생성", - "session.new.lastModified": "최근 수정", - "session.header.search.placeholder": "{{project}} 검색", - "session.header.searchFiles": "파일 검색", - "session.header.openIn": "다음에서 열기", - "session.header.open.action": "{{app}} 열기", - "session.header.open.ariaLabel": "{{app}}에서 열기", - "session.header.open.menu": "열기 옵션", - "session.header.open.copyPath": "경로 복사", - "status.popover.trigger": "상태", - "status.popover.ariaLabel": "서버 구성", - "status.popover.tab.servers": "서버", - "status.popover.tab.mcp": "MCP", - "status.popover.tab.lsp": "LSP", - "status.popover.tab.plugins": "플러그인", - "status.popover.action.manageServers": "서버 관리", - "session.share.popover.title": "웹에 게시", - "session.share.popover.description.shared": "이 세션은 웹에 공개되었습니다. 링크가 있는 누구나 액세스할 수 있습니다.", - "session.share.popover.description.unshared": - "세션을 웹에 공개적으로 공유합니다. 링크가 있는 누구나 액세스할 수 있습니다.", - "session.share.action.share": "공유", - "session.share.action.publish": "게시", - "session.share.action.publishing": "게시 중...", - "session.share.action.unpublish": "게시 취소", - "session.share.action.unpublishing": "게시 취소 중...", - "session.share.action.view": "보기", - "session.share.copy.copied": "복사됨", - "session.share.copy.copyLink": "링크 복사", - "lsp.tooltip.none": "LSP 서버 없음", - "lsp.label.connected": "{{count}} LSP", - "prompt.loading": "프롬프트 로드 중...", - "terminal.loading": "터미널 로드 중...", - "terminal.title": "터미널", - "terminal.title.numbered": "터미널 {{number}}", - "terminal.close": "터미널 닫기", - "terminal.connectionLost.title": "연결 끊김", - "terminal.connectionLost.description": - "터미널 연결이 중단되었습니다. 서버가 재시작하면 이런 일이 발생할 수 있습니다.", - "common.closeTab": "탭 닫기", - "common.dismiss": "닫기", - "common.requestFailed": "요청 실패", - "common.moreOptions": "더 많은 옵션", - "common.learnMore": "더 알아보기", - "common.rename": "이름 바꾸기", - "common.reset": "초기화", - "common.archive": "보관", - "common.delete": "삭제", - "common.close": "닫기", - "common.edit": "편집", - "common.loadMore": "더 불러오기", - "common.key.esc": "ESC", - "sidebar.menu.toggle": "메뉴 토글", - "sidebar.nav.projectsAndSessions": "프로젝트 및 세션", - "sidebar.settings": "설정", - "sidebar.help": "도움말", - "sidebar.workspaces.enable": "작업 공간 활성화", - "sidebar.workspaces.disable": "작업 공간 비활성화", - "sidebar.gettingStarted.title": "시작하기", - "sidebar.gettingStarted.line1": "Kilo에는 무료 모델이 포함되어 있어 즉시 시작할 수 있습니다.", - "sidebar.gettingStarted.line2": "Claude, GPT, Gemini 등을 포함한 모델을 사용하려면 공급자를 연결하세요.", - "sidebar.project.recentSessions": "최근 세션", - "sidebar.project.viewAllSessions": "모든 세션 보기", - "sidebar.project.clearNotifications": "알림 지우기", - "app.name.desktop": "Kilo Desktop", - "settings.section.desktop": "데스크톱", - "settings.section.server": "서버", - "settings.tab.general": "일반", - "settings.tab.shortcuts": "단축키", - "settings.desktop.section.wsl": "WSL", - "settings.desktop.wsl.title": "WSL 통합", - "settings.desktop.wsl.description": "Windows의 WSL 내부에서 Kilo 서버를 실행합니다.", - "settings.general.section.appearance": "모양", - "settings.general.section.notifications": "시스템 알림", - "settings.general.section.updates": "업데이트", - "settings.general.section.sounds": "효과음", - "settings.general.section.feed": "피드", - "settings.general.section.display": "디스플레이", - "settings.general.row.language.title": "언어", - "settings.general.row.language.description": "Kilo 표시 언어 변경", - "settings.general.row.appearance.title": "모양", - "settings.general.row.appearance.description": "기기에서 Kilo가 보이는 방식 사용자 지정", - "settings.general.row.colorScheme.title": "색상 테마", - "settings.general.row.colorScheme.description": "Kilo가 시스템, 라이트 또는 다크 테마를 따를지 선택하세요", - "settings.general.row.theme.title": "테마", - "settings.general.row.theme.description": "Kilo 테마 사용자 지정", - "settings.general.row.font.title": "코드 글꼴", - "settings.general.row.font.description": "코드 블록에 사용되는 글꼴을 사용자 지정", - "settings.general.row.terminalFont.title": "Terminal Font", - "settings.general.row.terminalFont.description": "Customise the font used in the terminal", - "settings.general.row.uiFont.title": "UI 글꼴", - "settings.general.row.uiFont.description": "인터페이스 전반에 사용되는 글꼴을 사용자 지정", - "settings.general.row.followup.title": "후속 조치 동작", - "settings.general.row.followup.description": "후속 프롬프트를 즉시 실행할지 대기열에 넣을지 선택하세요", - "settings.general.row.followup.option.queue": "대기열", - "settings.general.row.followup.option.steer": "조종", - "settings.general.row.reasoningSummaries.title": "추론 요약 표시", - "settings.general.row.reasoningSummaries.description": "타임라인에 모델 추론 요약 표시", - "settings.general.row.shellToolPartsExpanded.title": "shell 도구 파트 펼치기", - "settings.general.row.shellToolPartsExpanded.description": - "타임라인에서 기본적으로 shell 도구 파트를 펼친 상태로 표시합니다", - "settings.general.row.editToolPartsExpanded.title": "edit 도구 파트 펼치기", - "settings.general.row.editToolPartsExpanded.description": - "타임라인에서 기본적으로 edit, write, patch 도구 파트를 펼친 상태로 표시합니다", - "settings.general.row.showSessionProgressBar.title": "세션 진행 표시줄 표시", - "settings.general.row.showSessionProgressBar.description": - "에이전트가 작업 중일 때 세션 상단에 애니메이션 진행 표시줄을 표시합니다", - "settings.general.row.wayland.title": "네이티브 Wayland 사용", - "settings.general.row.wayland.description": "Wayland에서 X11 폴백을 비활성화합니다. 다시 시작해야 합니다.", - "settings.general.row.wayland.tooltip": - "혼합 주사율 모니터가 있는 Linux에서는 네이티브 Wayland가 더 안정적일 수 있습니다.", - "settings.general.row.releaseNotes.title": "릴리스 노트", - "settings.general.row.releaseNotes.description": "업데이트 후 '새 소식' 팝업 표시", - "settings.updates.row.startup.title": "시작 시 업데이트 확인", - "settings.updates.row.startup.description": "Kilo를 실행할 때 업데이트를 자동으로 확인합니다", - "settings.updates.row.check.title": "업데이트 확인", - "settings.updates.row.check.description": "업데이트를 수동으로 확인하고, 사용 가능하면 설치합니다", - "settings.updates.action.checkNow": "지금 확인", - "settings.updates.action.checking": "확인 중...", - "settings.updates.toast.latest.title": "최신 상태입니다", - "settings.updates.toast.latest.description": "현재 최신 버전의 Kilo를 사용 중입니다.", - "sound.option.none": "없음", - "sound.option.alert01": "알림 01", - "sound.option.alert02": "알림 02", - "sound.option.alert03": "알림 03", - "sound.option.alert04": "알림 04", - "sound.option.alert05": "알림 05", - "sound.option.alert06": "알림 06", - "sound.option.alert07": "알림 07", - "sound.option.alert08": "알림 08", - "sound.option.alert09": "알림 09", - "sound.option.alert10": "알림 10", - "sound.option.bipbop01": "빕-밥 01", - "sound.option.bipbop02": "빕-밥 02", - "sound.option.bipbop03": "빕-밥 03", - "sound.option.bipbop04": "빕-밥 04", - "sound.option.bipbop05": "빕-밥 05", - "sound.option.bipbop06": "빕-밥 06", - "sound.option.bipbop07": "빕-밥 07", - "sound.option.bipbop08": "빕-밥 08", - "sound.option.bipbop09": "빕-밥 09", - "sound.option.bipbop10": "빕-밥 10", - "sound.option.staplebops01": "스테이플밥스 01", - "sound.option.staplebops02": "스테이플밥스 02", - "sound.option.staplebops03": "스테이플밥스 03", - "sound.option.staplebops04": "스테이플밥스 04", - "sound.option.staplebops05": "스테이플밥스 05", - "sound.option.staplebops06": "스테이플밥스 06", - "sound.option.staplebops07": "스테이플밥스 07", - "sound.option.nope01": "아니오 01", - "sound.option.nope02": "아니오 02", - "sound.option.nope03": "아니오 03", - "sound.option.nope04": "아니오 04", - "sound.option.nope05": "아니오 05", - "sound.option.nope06": "아니오 06", - "sound.option.nope07": "아니오 07", - "sound.option.nope08": "아니오 08", - "sound.option.nope09": "아니오 09", - "sound.option.nope10": "아니오 10", - "sound.option.nope11": "아니오 11", - "sound.option.nope12": "아니오 12", - "sound.option.yup01": "네 01", - "sound.option.yup02": "네 02", - "sound.option.yup03": "네 03", - "sound.option.yup04": "네 04", - "sound.option.yup05": "네 05", - "sound.option.yup06": "네 06", - "settings.general.notifications.agent.title": "에이전트", - "settings.general.notifications.agent.description": "에이전트가 완료되거나 주의가 필요할 때 시스템 알림 표시", - "settings.general.notifications.permissions.title": "권한", - "settings.general.notifications.permissions.description": "권한이 필요할 때 시스템 알림 표시", - "settings.general.notifications.errors.title": "오류", - "settings.general.notifications.errors.description": "오류가 발생했을 때 시스템 알림 표시", - "settings.general.sounds.agent.title": "에이전트", - "settings.general.sounds.agent.description": "에이전트가 완료되거나 주의가 필요할 때 소리 재생", - "settings.general.sounds.permissions.title": "권한", - "settings.general.sounds.permissions.description": "권한이 필요할 때 소리 재생", - "settings.general.sounds.errors.title": "오류", - "settings.general.sounds.errors.description": "오류가 발생했을 때 소리 재생", - "settings.shortcuts.title": "키보드 단축키", - "settings.shortcuts.reset.button": "기본값으로 초기화", - "settings.shortcuts.reset.toast.title": "단축키 초기화됨", - "settings.shortcuts.reset.toast.description": "키보드 단축키가 기본값으로 초기화되었습니다.", - "settings.shortcuts.conflict.title": "단축키가 이미 사용 중임", - "settings.shortcuts.conflict.description": "{{keybind}}은(는) 이미 {{titles}}에 할당되어 있습니다.", - "settings.shortcuts.unassigned": "할당되지 않음", - "settings.shortcuts.pressKeys": "키 누르기", - "settings.shortcuts.search.placeholder": "단축키 검색", - "settings.shortcuts.search.empty": "단축키를 찾을 수 없습니다", - "settings.shortcuts.group.general": "일반", - "settings.shortcuts.group.session": "세션", - "settings.shortcuts.group.navigation": "탐색", - "settings.shortcuts.group.modelAndAgent": "모델 및 에이전트", - "settings.shortcuts.group.terminal": "터미널", - "settings.shortcuts.group.prompt": "프롬프트", - "settings.providers.title": "공급자", - "settings.providers.description": "공급자 설정은 여기서 구성할 수 있습니다.", - "settings.providers.section.connected": "연결된 공급자", - "settings.providers.connected.empty": "연결된 공급자 없음", - "settings.providers.section.popular": "인기 공급자", - "settings.providers.tag.environment": "환경", - "settings.providers.tag.config": "구성", - "settings.providers.tag.custom": "사용자 지정", - "settings.providers.tag.other": "기타", - "settings.models.title": "모델", - "settings.models.description": "모델 설정은 여기서 구성할 수 있습니다.", - "settings.agents.title": "에이전트", - "settings.agents.description": "에이전트 설정은 여기서 구성할 수 있습니다.", - "settings.commands.title": "명령어", - "settings.commands.description": "명령어 설정은 여기서 구성할 수 있습니다.", - "settings.mcp.title": "MCP", - "settings.mcp.description": "MCP 설정은 여기서 구성할 수 있습니다.", - "settings.permissions.title": "권한", - "settings.permissions.description": "서버가 기본적으로 사용할 수 있는 도구를 제어합니다.", - "settings.permissions.section.tools": "도구", - "settings.permissions.toast.updateFailed.title": "권한 업데이트 실패", - "settings.permissions.action.allow": "허용", - "settings.permissions.action.ask": "묻기", - "settings.permissions.action.deny": "거부", - "settings.permissions.tool.read.title": "읽기", - "settings.permissions.tool.read.description": "파일 읽기 (파일 경로와 일치)", - "settings.permissions.tool.edit.title": "편집", - "settings.permissions.tool.edit.description": "파일 수정 (편집, 쓰기, 패치 및 다중 편집 포함)", - "settings.permissions.tool.glob.title": "Glob", - "settings.permissions.tool.glob.description": "glob 패턴을 사용하여 파일 일치", - "settings.permissions.tool.grep.title": "Grep", - "settings.permissions.tool.grep.description": "정규식을 사용하여 파일 내용 검색", - "settings.permissions.tool.list.title": "목록", - "settings.permissions.tool.list.description": "디렉터리 내 파일 나열", - "settings.permissions.tool.bash.title": "Bash", - "settings.permissions.tool.bash.description": "셸 명령어 실행", - "settings.permissions.tool.task.title": "작업", - "settings.permissions.tool.task.description": "하위 에이전트 실행", - "settings.permissions.tool.skill.title": "기술", - "settings.permissions.tool.skill.description": "이름으로 기술 로드", - "settings.permissions.tool.lsp.title": "LSP", - "settings.permissions.tool.lsp.description": "언어 서버 쿼리 실행", - "settings.permissions.tool.todowrite.title": "할 일 쓰기", - "settings.permissions.tool.todowrite.description": "할 일 목록 업데이트", - "settings.permissions.tool.webfetch.title": "웹 가져오기", - "settings.permissions.tool.webfetch.description": "URL에서 콘텐츠 가져오기", - "settings.permissions.tool.websearch.title": "웹 검색", - "settings.permissions.tool.websearch.description": "웹 검색", - "settings.permissions.tool.codesearch.title": "코드 검색", - "settings.permissions.tool.codesearch.description": "웹에서 코드 검색", - "settings.permissions.tool.external_directory.title": "외부 디렉터리", - "settings.permissions.tool.external_directory.description": "프로젝트 디렉터리 외부의 파일에 액세스", - "settings.permissions.tool.doom_loop.title": "무한 반복", - "settings.permissions.tool.doom_loop.description": "동일한 입력으로 반복되는 도구 호출 감지", - "session.delete.failed.title": "세션 삭제 실패", - "session.delete.title": "세션 삭제", - "session.delete.confirm": '"{{name}}" 세션을 삭제하시겠습니까?', - "session.delete.button": "세션 삭제", - "workspace.new": "새 작업 공간", - "workspace.type.local": "로컬", - "workspace.type.sandbox": "샌드박스", - "workspace.create.failed.title": "작업 공간 생성 실패", - "workspace.delete.failed.title": "작업 공간 삭제 실패", - "workspace.resetting.title": "작업 공간 재설정 중", - "workspace.resetting.description": "잠시 시간이 걸릴 수 있습니다.", - "workspace.reset.failed.title": "작업 공간 재설정 실패", - "workspace.reset.success.title": "작업 공간 재설정됨", - "workspace.reset.success.description": "작업 공간이 이제 기본 브랜치와 일치합니다.", - "workspace.error.stillPreparing": "작업 공간이 아직 준비 중입니다", - "workspace.status.checking": "병합되지 않은 변경 사항 확인 중...", - "workspace.status.error": "Git 상태를 확인할 수 없습니다.", - "workspace.status.clean": "병합되지 않은 변경 사항이 감지되지 않았습니다.", - "workspace.status.dirty": "이 작업 공간에서 병합되지 않은 변경 사항이 감지되었습니다.", - "workspace.delete.title": "작업 공간 삭제", - "workspace.delete.confirm": '"{{name}}" 작업 공간을 삭제하시겠습니까?', - "workspace.delete.button": "작업 공간 삭제", - "workspace.reset.title": "작업 공간 재설정", - "workspace.reset.confirm": '"{{name}}" 작업 공간을 재설정하시겠습니까?', - "workspace.reset.button": "작업 공간 재설정", - "workspace.reset.archived.none": "활성 세션이 보관되지 않습니다.", - "workspace.reset.archived.one": "1개의 세션이 보관됩니다.", - "workspace.reset.archived.many": "{{count}}개의 세션이 보관됩니다.", - "workspace.reset.note": "이 작업은 작업 공간을 기본 브랜치와 일치하도록 재설정합니다.", - "common.open": "열기", - "dialog.releaseNotes.action.getStarted": "시작하기", - "dialog.releaseNotes.action.next": "다음", - "dialog.releaseNotes.action.hideFuture": "다시 보지 않기", - "dialog.releaseNotes.media.alt": "릴리스 미리보기", - "toast.project.reloadFailed.title": "{{project}} 다시 불러오기 실패", - "error.server.invalidConfiguration": "잘못된 구성", - "common.moreCountSuffix": " (외 {{count}}개)", - "common.time.justNow": "방금 전", - "common.time.minutesAgo.short": "{{count}}분 전", - "common.time.hoursAgo.short": "{{count}}시간 전", - "common.time.daysAgo.short": "{{count}}일 전", - "settings.providers.connected.environmentDescription": "환경 변수에서 연결됨", - "settings.providers.custom.description": "기본 URL로 OpenAI 호환 공급자를 추가합니다.", - - "app.server.unreachable": "{{server}}에 연결할 수 없습니다", - "app.server.retrying": "자동으로 재시도 중...", - "app.server.otherServers": "다른 서버", - "dialog.server.add.usernamePlaceholder": "사용자 이름", - "dialog.server.add.passwordPlaceholder": "비밀번호", - "server.row.noUsername": "사용자 이름 없음", - "session.review.noVcs.createGit.title": "Git 저장소 생성", - "session.review.noVcs.createGit.description": "이 프로젝트의 변경 사항을 추적, 검토 및 실행 취소", - "session.review.noVcs.createGit.actionLoading": "Git 저장소 생성 중...", - "session.review.noVcs.createGit.action": "Git 저장소 생성", - "session.todo.progress": "{{total}}개의 할 일 중 {{done}}개 완료", - "session.question.progress": "{{total}}개의 질문 중 {{current}}개", - "session.header.open.finder": "Finder", - "session.header.open.fileExplorer": "파일 탐색기", - "session.header.open.fileManager": "파일 관리자", - "session.header.open.app.vscode": "VS Code", - "session.header.open.app.cursor": "Cursor", - "session.header.open.app.zed": "Zed", - "session.header.open.app.textmate": "TextMate", - "session.header.open.app.antigravity": "Antigravity", - "session.header.open.app.terminal": "터미널", - "session.header.open.app.iterm2": "iTerm2", - "session.header.open.app.ghostty": "Ghostty", - "session.header.open.app.warp": "Warp", - "session.header.open.app.xcode": "Xcode", - "session.header.open.app.androidStudio": "Android Studio", - "session.header.open.app.powershell": "PowerShell", - "session.header.open.app.sublimeText": "Sublime Text", - "debugBar.ariaLabel": "개발 성능 진단", - "debugBar.na": "해당 없음", - "debugBar.nav.label": "NAV", - "debugBar.nav.tip": - "세션 페이지에 닿은 마지막 완료된 라우트 전환. 라우터 시작부터 정착 후 첫 번째 페인트까지 측정됨.", - "debugBar.fps.label": "FPS", - "debugBar.fps.tip": "지난 5초간의 초당 프레임 수.", - "debugBar.frame.label": "FRAME", - "debugBar.frame.tip": "지난 5초간의 최악의 프레임 시간.", - "debugBar.jank.label": "JANK", - "debugBar.jank.tip": "지난 5초간 32ms를 초과한 프레임.", - "debugBar.long.label": "LONG", - "debugBar.long.tip": "지난 5초간의 차단된 시간 및 긴 작업 수. 최대 작업: {{max}}.", - "debugBar.delay.label": "DELAY", - "debugBar.delay.tip": "지난 5초간 관찰된 최악의 입력 지연.", - "debugBar.inp.label": "INP", - "debugBar.inp.tip": "지난 5초간의 대략적인 상호작용 지속 시간. 이것은 공식 Web Vitals INP가 아닌 INP와 유사합니다.", - "debugBar.cls.label": "CLS", - "debugBar.cls.tip": "현재 앱 수명 동안의 누적 레이아웃 이동.", - "debugBar.mem.label": "MEM", - "debugBar.mem.tipUnavailable": "사용된 JS 힙 대 힙 제한. Chromium 전용.", - "debugBar.mem.tip": "사용된 JS 힙 대 힙 제한. {{limit}} 중 {{used}}.", - "common.key.ctrl": "Ctrl", - "common.key.alt": "Alt", - "common.key.shift": "Shift", - "common.key.meta": "Meta", - "common.key.space": "Space", - "common.key.backspace": "Backspace", - "common.key.enter": "Enter", - "common.key.tab": "Tab", - "common.key.delete": "Delete", - "common.key.home": "Home", - "common.key.end": "End", - "common.key.pageUp": "Page Up", - "common.key.pageDown": "Page Down", - "common.key.insert": "Insert", - "common.unknown": "알 수 없음", - "error.page.circular": "[순환]", - "error.globalSDK.noServerAvailable": "사용 가능한 서버 없음", - "error.globalSDK.serverNotAvailable": "서버를 사용할 수 없음", - "error.childStore.persistedCacheCreateFailed": "영구 캐시 생성 실패", - "error.childStore.persistedProjectMetadataCreateFailed": "영구 프로젝트 메타데이터 생성 실패", - "error.childStore.persistedProjectIconCreateFailed": "영구 프로젝트 아이콘 생성 실패", - "error.childStore.storeCreateFailed": "저장소 생성 실패", - "terminal.connectionLost.abnormalClose": "WebSocket이 비정상적으로 닫힘: {{code}}", -} diff --git a/packages/app/src/i18n/nl.ts b/packages/app/src/i18n/nl.ts deleted file mode 100644 index 376aaa42bb..0000000000 --- a/packages/app/src/i18n/nl.ts +++ /dev/null @@ -1,855 +0,0 @@ -export const dict = { - "command.category.suggested": "Voorgesteld", - "command.category.view": "Weergave", - "command.category.project": "Project", - "command.category.provider": "Provider", - "command.category.server": "Server", - "command.category.session": "Sessie", - "command.category.theme": "Thema", - "command.category.language": "Taal", - "command.category.file": "Bestand", - "command.category.context": "Context", - "command.category.terminal": "Terminal", - "command.category.model": "Model", - "command.category.mcp": "MCP", - "command.category.agent": "Agent", - "command.category.permissions": "Machtigingen", - "command.category.workspace": "Werkruimte", - "command.category.settings": "Instellingen", - - "theme.scheme.system": "Systeem", - "theme.scheme.light": "Licht", - "theme.scheme.dark": "Donker", - - "command.sidebar.toggle": "Zijbalk in-/uitschakelen", - "command.project.open": "Project openen", - "command.provider.connect": "Provider verbinden", - "command.server.switch": "Server wisselen", - "command.settings.open": "Instellingen openen", - "command.session.previous": "Vorige sessie", - "command.session.next": "Volgende sessie", - "command.session.previous.unseen": "Vorige ongelezen sessie", - "command.session.next.unseen": "Volgende ongelezen sessie", - "command.session.archive": "Sessie archiveren", - - "command.palette": "Opdrachtenpalet", - - "command.theme.cycle": "Volgend thema", - "command.theme.set": "Gebruik thema: {{theme}}", - "command.theme.scheme.cycle": "Volgend kleurenschema", - "command.theme.scheme.set": "Gebruik kleurenschema: {{scheme}}", - - "command.language.cycle": "Volgende taal", - "command.language.set": "Gebruik taal: {{language}}", - - "command.session.new": "Nieuwe sessie", - "command.file.open": "Bestand openen", - "command.tab.close": "Tabblad sluiten", - "command.context.addSelection": "Selectie toevoegen aan context", - "command.context.addSelection.description": "Geselecteerde regels van het huidige bestand toevoegen", - "command.input.focus": "Invoer focussen", - "command.terminal.toggle": "Terminal in-/uitschakelen", - "command.fileTree.toggle": "Bestandsboom in-/uitschakelen", - "command.review.toggle": "Review in-/uitschakelen", - "command.terminal.new": "Nieuwe terminal", - "command.terminal.new.description": "Een nieuw terminal-tabblad maken", - "command.steps.toggle": "Stappen in-/uitschakelen", - "command.steps.toggle.description": "Stappen voor het huidige bericht tonen of verbergen", - "command.message.previous": "Vorig bericht", - "command.message.previous.description": "Ga naar het vorige gebruikersbericht", - "command.message.next": "Volgend bericht", - "command.message.next.description": "Ga naar het volgende gebruikersbericht", - "command.model.choose": "Model kiezen", - "command.model.choose.description": "Selecteer een ander model", - "command.mcp.toggle": "MCPs in-/uitschakelen", - "command.mcp.toggle.description": "MCPs in-/uitschakelen", - "command.agent.cycle": "Volgende agent", - "command.agent.cycle.description": "Overschakelen naar de volgende agent", - "command.agent.cycle.reverse": "Vorige agent", - "command.agent.cycle.reverse.description": "Overschakelen naar de vorige agent", - "command.model.variant.cycle": "Volgende denkinspanning", - "command.model.variant.cycle.description": "Overschakelen naar het volgende inspanningsniveau", - "command.prompt.mode.shell": "Shell", - "command.prompt.mode.normal": "Prompt", - "command.permissions.autoaccept.enable": "Machtigingen automatisch accepteren", - "command.permissions.autoaccept.disable": "Machtigingen niet langer automatisch accepteren", - "command.workspace.toggle": "Werkruimtes in-/uitschakelen", - "command.workspace.toggle.description": "Meerdere werkruimtes in de zijbalk in- of uitschakelen", - "command.session.undo": "Ongedaan maken", - "command.session.undo.description": "Laatste bericht ongedaan maken", - "command.session.redo": "Opnieuw uitvoeren", - "command.session.redo.description": "Het laatst ongedaan gemaakte bericht opnieuw uitvoeren", - "command.session.compact": "Sessie comprimeren", - "command.session.compact.description": "Sessie samenvatten om contextgrootte te verkleinen", - "command.session.fork": "Afsplitsen van bericht", - "command.session.fork.description": "Nieuwe sessie maken van een eerder bericht", - "command.session.share": "Sessie delen", - "command.session.share.description": "Deze sessie delen en de URL naar klembord kopiëren", - "command.session.unshare": "Sessie niet meer delen", - "command.session.unshare.description": "Stoppen met het delen van deze sessie", - - "palette.search.placeholder": "Zoek bestanden, opdrachten en sessies", - "palette.empty": "Geen resultaten gevonden", - "palette.group.commands": "Opdrachten", - "palette.group.files": "Bestanden", - - "dialog.provider.search.placeholder": "Zoek providers", - "dialog.provider.empty": "Geen providers gevonden", - "dialog.provider.group.popular": "Populair", - "dialog.provider.group.other": "Overig", - "dialog.provider.tag.recommended": "Aanbevolen", - "dialog.provider.opencode.note": "Geselecteerde modellen inclusief Claude, GPT, Gemini en meer", - "dialog.provider.opencode.tagline": "Betrouwbare geoptimaliseerde modellen", - "dialog.provider.opencodeGo.tagline": "Goedkoop abonnement voor iedereen", - "dialog.provider.anthropic.note": "Directe toegang tot Claude modellen, inclusief Pro en Max", - "dialog.provider.copilot.note": "AI-modellen voor codeerassistentie via GitHub Copilot", - "dialog.provider.openai.note": "GPT modellen voor snelle, capabele algemene AI-taken", - "dialog.provider.google.note": "Gemini modellen voor snelle, gestructureerde antwoorden", - "dialog.provider.openrouter.note": "Toegang tot alle ondersteunde modellen via één provider", - "dialog.provider.vercel.note": "Geünificeerde toegang tot AI-modellen met slimme routering", - - "dialog.model.select.title": "Selecteer model", - "dialog.model.search.placeholder": "Zoek modellen", - "dialog.model.empty": "Geen modelresultaten", - "dialog.model.manage": "Modellen beheren", - "dialog.model.manage.description": "Pas aan welke modellen in de modelkiezer verschijnen.", - "dialog.model.manage.provider.toggle": "Alle {{provider}} modellen in-/uitschakelen", - - "dialog.model.unpaid.freeModels.title": "Gratis modellen aangeboden door Kilo", - "dialog.model.unpaid.addMore.title": "Meer modellen toevoegen van populaire providers", - - "dialog.provider.viewAll": "Toon meer providers", - - "provider.connect.title": "{{provider}} verbinden", - "provider.connect.title.anthropicProMax": "Inloggen met Claude Pro/Max", - "provider.connect.selectMethod": "Selecteer inlogmethode voor {{provider}}.", - "provider.connect.method.apiKey": "API key", - "provider.connect.status.inProgress": "Autorisatie in uitvoering...", - "provider.connect.status.waiting": "Wachten op autorisatie...", - "provider.connect.status.failed": "Autorisatie mislukt: {{error}}", - "provider.connect.apiKey.description": - "Voer uw {{provider}} API key in om uw account te verbinden en {{provider}} modellen te gebruiken in Kilo.", - "provider.connect.apiKey.label": "{{provider}} API key", - "provider.connect.apiKey.placeholder": "API key", - "provider.connect.apiKey.required": "API key is vereist", - "provider.connect.opencodeZen.line1": - "OpenCode Zen geeft u toegang tot een geselecteerde set betrouwbare en geoptimaliseerde modellen voor coderingsagenten.", - "provider.connect.opencodeZen.line2": - "Met een enkele API key krijgt u toegang tot modellen zoals Claude, GPT, Gemini, GLM en meer.", - "provider.connect.opencodeZen.visit.prefix": "Bezoek ", - "provider.connect.opencodeZen.visit.link": "opencode.ai/zen", - "provider.connect.opencodeZen.visit.suffix": " om uw API key te verkrijgen.", - "provider.connect.oauth.code.visit.prefix": "Bezoek ", - "provider.connect.oauth.code.visit.link": "deze link", - "provider.connect.oauth.code.visit.suffix": - " om uw autorisatiecode te verkrijgen, uw account te verbinden en {{provider}} modellen te gebruiken in Kilo.", - "provider.connect.oauth.code.label": "{{method}} autorisatiecode", - "provider.connect.oauth.code.placeholder": "Autorisatiecode", - "provider.connect.oauth.code.required": "Autorisatiecode is vereist", - "provider.connect.oauth.code.invalid": "Ongeldige autorisatiecode", - "provider.connect.oauth.auto.visit.prefix": "Bezoek ", - "provider.connect.oauth.auto.visit.link": "deze link", - "provider.connect.oauth.auto.visit.suffix": - " en voer de onderstaande code in om uw account te verbinden en {{provider}} modellen te gebruiken in Kilo.", - "provider.connect.oauth.auto.confirmationCode": "Bevestigingscode", - "provider.connect.toast.connected.title": "{{provider}} verbonden", - "provider.connect.toast.connected.description": "{{provider}} modellen zijn nu beschikbaar om te gebruiken.", - - "provider.custom.title": "Aangepaste provider", - "provider.custom.description.prefix": "Configureer een OpenAI-compatibele provider. Zie de ", - "provider.custom.description.link": "provider configuratiedocumentatie", - "provider.custom.description.suffix": ".", - "provider.custom.field.providerID.label": "Provider ID", - "provider.custom.field.providerID.placeholder": "myprovider", - "provider.custom.field.providerID.description": "Kleine letters, cijfers, koppeltekens of underscores", - "provider.custom.field.name.label": "Weergavenaam", - "provider.custom.field.name.placeholder": "Mijn AI Provider", - "provider.custom.field.baseURL.label": "Basis-URL", - "provider.custom.field.baseURL.placeholder": "https://api.myprovider.com/v1", - "provider.custom.field.apiKey.label": "API key", - "provider.custom.field.apiKey.placeholder": "API key", - "provider.custom.field.apiKey.description": "Optioneel. Laat leeg als u auth via headers beheert.", - "provider.custom.models.label": "Modellen", - "provider.custom.models.id.label": "ID", - "provider.custom.models.id.placeholder": "model-id", - "provider.custom.models.name.label": "Naam", - "provider.custom.models.name.placeholder": "Weergavenaam", - "provider.custom.models.remove": "Model verwijderen", - "provider.custom.models.add": "Model toevoegen", - "provider.custom.headers.label": "Headers (optioneel)", - "provider.custom.headers.key.label": "Header", - "provider.custom.headers.key.placeholder": "Header-Naam", - "provider.custom.headers.value.label": "Waarde", - "provider.custom.headers.value.placeholder": "waarde", - "provider.custom.headers.remove": "Header verwijderen", - "provider.custom.headers.add": "Header toevoegen", - "provider.custom.error.providerID.required": "Provider ID is vereist", - "provider.custom.error.providerID.format": "Gebruik kleine letters, cijfers, koppeltekens of underscores", - "provider.custom.error.providerID.exists": "Deze provider ID bestaat al", - "provider.custom.error.name.required": "Weergavenaam is vereist", - "provider.custom.error.baseURL.required": "Basis-URL is vereist", - "provider.custom.error.baseURL.format": "Moet beginnen met http:// of https://", - "provider.custom.error.required": "Vereist", - "provider.custom.error.duplicate": "Duplicaat", - - "provider.disconnect.toast.disconnected.title": "{{provider}} losgekoppeld", - "provider.disconnect.toast.disconnected.description": "{{provider}} modellen zijn niet langer beschikbaar.", - - "model.tag.free": "Gratis", - "model.tag.latest": "Nieuwste", - "model.provider.anthropic": "Anthropic", - "model.provider.openai": "OpenAI", - "model.provider.google": "Google", - "model.provider.xai": "xAI", - "model.provider.meta": "Meta", - "model.input.text": "tekst", - "model.input.image": "afbeelding", - "model.input.audio": "audio", - "model.input.video": "video", - "model.input.pdf": "pdf", - "model.tooltip.allows": "Staat toe: {{inputs}}", - "model.tooltip.reasoning.allowed": "Staat redenering toe", - "model.tooltip.reasoning.none": "Geen redenering", - "model.tooltip.context": "Contextlimiet {{limit}}", - - "common.search.placeholder": "Zoeken", - "common.goBack": "Ga terug", - "common.goForward": "Ga vooruit", - "common.loading": "Laden", - "common.loading.ellipsis": "...", - "common.cancel": "Annuleren", - "common.open": "Openen", - "common.connect": "Verbinden", - "common.disconnect": "Loskoppelen", - "common.submit": "Verzenden", - "common.save": "Opslaan", - "common.saving": "Opslaan...", - "common.default": "Standaard", - "common.attachment": "bijlage", - - "prompt.placeholder.shell": "Voer shell-opdracht in...", - "prompt.placeholder.normal": 'Vraag alles... "{{example}}"', - "prompt.placeholder.simple": "Vraag alles...", - "prompt.placeholder.summarizeComments": "Reacties samenvatten...", - "prompt.placeholder.summarizeComment": "Reactie samenvatten...", - "prompt.mode.shell": "Shell", - "prompt.mode.normal": "Prompt", - "prompt.mode.shell.exit": "esc om te verlaten", - - "prompt.example.1": "Fix een TODO in de codebase", - "prompt.example.2": "Wat is de tech stack van dit project?", - "prompt.example.3": "Repareer kapotte tests", - "prompt.example.4": "Leg uit hoe authenticatie werkt", - "prompt.example.5": "Vind en herstel beveiligingskwetsbaarheden", - "prompt.example.6": "Voeg unittests toe voor de gebruikersservice", - "prompt.example.7": "Herschrijf deze functie om deze beter leesbaar te maken", - "prompt.example.8": "Wat betekent deze fout?", - "prompt.example.9": "Help me met het debuggen van dit probleem", - "prompt.example.10": "Genereer API-documentatie", - "prompt.example.11": "Optimaliseer databasequery's", - "prompt.example.12": "Voeg invoervalidatie toe", - "prompt.example.13": "Maak een nieuwe component voor...", - "prompt.example.14": "Hoe implementeer ik dit project?", - "prompt.example.15": "Bekijk mijn code voor best practices", - "prompt.example.16": "Voeg foutafhandeling toe aan deze functie", - "prompt.example.17": "Leg dit regex-patroon uit", - "prompt.example.18": "Zet dit om naar TypeScript", - "prompt.example.19": "Voeg logging toe in de hele codebase", - "prompt.example.20": "Welke afhankelijkheden zijn verouderd?", - "prompt.example.21": "Help me een migratiescript te schrijven", - "prompt.example.22": "Implementeer caching voor dit eindpunt", - "prompt.example.23": "Voeg paginering toe aan deze lijst", - "prompt.example.24": "Maak een CLI-opdracht voor...", - "prompt.example.25": "Hoe werken omgevingsvariabelen hier?", - - "prompt.popover.emptyResults": "Geen overeenkomende resultaten", - "prompt.popover.emptyCommands": "Geen overeenkomende opdrachten", - "prompt.dropzone.label": "Sleep afbeeldingen of PDF's hierheen", - "prompt.dropzone.file.label": "Sleep om bestand te @vermelden", - "prompt.slash.badge.custom": "aangepast", - "prompt.slash.badge.skill": "skill", - "prompt.slash.badge.mcp": "mcp", - "prompt.context.active": "actief", - "prompt.context.includeActiveFile": "Neem actief bestand op", - "prompt.context.removeActiveFile": "Verwijder actief bestand uit context", - "prompt.context.removeFile": "Verwijder bestand uit context", - "prompt.action.attachFile": "Bestand toevoegen", - "prompt.attachment.remove": "Bijlage verwijderen", - "prompt.action.send": "Verzenden", - "prompt.action.stop": "Stop", - - "prompt.toast.pasteUnsupported.title": "Niet-ondersteunde plakbewerking", - "prompt.toast.pasteUnsupported.description": "Hier kunnen alleen afbeeldingen of PDF's worden geplakt.", - "prompt.toast.modelAgentRequired.title": "Selecteer een agent en model", - "prompt.toast.modelAgentRequired.description": "Kies een agent en model voordat u een prompt verzendt.", - "prompt.toast.worktreeCreateFailed.title": "Maken van worktree mislukt", - "prompt.toast.sessionCreateFailed.title": "Maken van sessie mislukt", - "prompt.toast.shellSendFailed.title": "Verzenden van shell-opdracht mislukt", - "prompt.toast.commandSendFailed.title": "Verzenden van opdracht mislukt", - "prompt.toast.promptSendFailed.title": "Verzenden van prompt mislukt", - "prompt.toast.promptSendFailed.description": "Kan sessie niet ophalen", - - "dialog.mcp.title": "MCPs", - "dialog.mcp.description": "{{enabled}} van {{total}} ingeschakeld", - "dialog.mcp.empty": "Geen MCP's geconfigureerd", - - "dialog.lsp.empty": "LSP's automatisch gedetecteerd aan de hand van bestandstypen", - "dialog.plugins.empty": "Plug-ins geconfigureerd in opencode.json", - - "mcp.status.connected": "verbonden", - "mcp.status.failed": "mislukt", - "mcp.status.needs_auth": "authenticatie vereist", - "mcp.status.disabled": "uitgeschakeld", - - "dialog.fork.empty": "Geen berichten om van af te splitsen", - - "dialog.directory.search.placeholder": "Zoek mappen", - "dialog.directory.empty": "Geen mappen gevonden", - - "dialog.server.title": "Servers", - "dialog.server.description": "Wissel met welke Kilo server deze app verbindt.", - "dialog.server.search.placeholder": "Zoek servers", - "dialog.server.empty": "Nog geen servers", - "dialog.server.add.title": "Server toevoegen", - "dialog.server.add.url": "Serveradres", - "dialog.server.add.placeholder": "http://localhost:4096", - "dialog.server.add.error": "Kan geen verbinding maken met server", - "dialog.server.add.checking": "Controleren...", - "dialog.server.add.button": "Server toevoegen", - "dialog.server.add.name": "Servernaam (optioneel)", - "dialog.server.add.namePlaceholder": "Localhost", - "dialog.server.add.username": "Gebruikersnaam (optioneel)", - "dialog.server.add.password": "Wachtwoord (optioneel)", - "dialog.server.edit.title": "Server bewerken", - "dialog.server.default.title": "Standaardserver", - "dialog.server.default.description": - "Maak verbinding met deze server bij de lancering van de app in plaats van een lokale server te starten. Vereist een herstart.", - "dialog.server.default.none": "Geen server geselecteerd", - "dialog.server.default.set": "Huidige server als standaard instellen", - "dialog.server.default.clear": "Wissen", - "dialog.server.action.remove": "Server verwijderen", - - "dialog.server.menu.edit": "Bewerken", - "dialog.server.menu.default": "Als standaard instellen", - "dialog.server.menu.defaultRemove": "Standaard verwijderen", - "dialog.server.menu.delete": "Verwijderen", - "dialog.server.current": "Huidige Server", - "dialog.server.status.default": "Standaard", - - "dialog.project.edit.title": "Project bewerken", - "dialog.project.edit.name": "Naam", - "dialog.project.edit.icon": "Icoon", - "dialog.project.edit.icon.alt": "Projecticoon", - "dialog.project.edit.icon.hint": "Klik of sleep een afbeelding", - "dialog.project.edit.icon.recommended": "Aanbevolen: 128x128px", - "dialog.project.edit.color": "Kleur", - "dialog.project.edit.color.select": "Selecteer {{color}} kleur", - "dialog.project.edit.worktree.startup": "Werkruimte opstartscript", - "dialog.project.edit.worktree.startup.description": - "Wordt uitgevoerd na het maken van een nieuwe werkruimte (worktree).", - "dialog.project.edit.worktree.startup.placeholder": "bijv. bun install", - - "dialog.releaseNotes.action.getStarted": "Aan de slag", - "dialog.releaseNotes.action.next": "Volgende", - "dialog.releaseNotes.action.hideFuture": "Toon deze niet in de toekomst", - "dialog.releaseNotes.media.alt": "Releasevoorbeeld", - - "context.breakdown.title": "Contextverdeling", - "context.breakdown.note": 'Geschatte verdeling van invoertokens. "Overig" bevat tooldefinities en overhead.', - "context.breakdown.system": "Systeem", - "context.breakdown.user": "Gebruiker", - "context.breakdown.assistant": "Assistent", - "context.breakdown.tool": "Tool-aanroepen", - "context.breakdown.other": "Overig", - - "context.systemPrompt.title": "Systeemprompt", - "context.rawMessages.title": "Ruwe berichten", - - "context.stats.session": "Sessie", - "context.stats.messages": "Berichten", - "context.stats.provider": "Provider", - "context.stats.model": "Model", - "context.stats.limit": "Contextlimiet", - "context.stats.totalTokens": "Totaal Tokens", - "context.stats.usage": "Gebruik", - "context.stats.inputTokens": "Invoertokens", - "context.stats.outputTokens": "Uitvoertokens", - "context.stats.reasoningTokens": "Redeneertokens", - "context.stats.cacheTokens": "Cachetokens (lezen/schrijven)", - "context.stats.userMessages": "Gebruikersberichten", - "context.stats.assistantMessages": "Assistentberichten", - "context.stats.totalCost": "Totale Kosten", - "context.stats.sessionCreated": "Sessie Aangemaakt", - "context.stats.lastActivity": "Laatste Activiteit", - - "context.usage.tokens": "Tokens", - "context.usage.usage": "Gebruik", - "context.usage.cost": "Kosten", - "context.usage.clickToView": "Klik om context te bekijken", - "context.usage.view": "Contextgebruik bekijken", - - "language.en": "English", - "language.zh": "简体中文", - "language.zht": "繁體中文", - "language.ko": "한국어", - "language.de": "Deutsch", - "language.es": "Español", - "language.fr": "Français", - "language.da": "Dansk", - "language.ja": "日本語", - "language.pl": "Polski", - "language.ru": "Русский", - "language.ar": "العربية", - "language.no": "Norsk", - "language.br": "Português (Brasil)", - "language.bs": "Bosanski", - "language.nl": "Nederlands", - "language.th": "ไทย", - "language.tr": "Türkçe", - - "toast.language.title": "Taal", - "toast.language.description": "Omgeschakeld naar {{language}}", - - "toast.theme.title": "Thema gewisseld", - "toast.scheme.title": "Kleurenschema", - - "toast.workspace.enabled.title": "Werkruimtes ingeschakeld", - "toast.workspace.enabled.description": "Meerdere worktrees worden nu weergegeven in de zijbalk", - "toast.workspace.disabled.title": "Werkruimtes uitgeschakeld", - "toast.workspace.disabled.description": "Alleen de hoofdworktree wordt weergegeven in de zijbalk", - - "toast.permissions.autoaccept.on.title": "Machtigingen automatisch accepteren", - "toast.permissions.autoaccept.on.description": "Toestemmingsverzoeken worden automatisch goedgekeurd", - "toast.permissions.autoaccept.off.title": "Gestopt met automatisch accepteren van machtigingen", - "toast.permissions.autoaccept.off.description": "Toestemmingsverzoeken vereisen goedkeuring", - - "toast.model.none.title": "Geen model geselecteerd", - "toast.model.none.description": "Verbind een provider om deze sessie samen te vatten", - - "toast.file.loadFailed.title": "Laden van bestand mislukt", - "toast.file.listFailed.title": "Lijst met bestanden ophalen mislukt", - - "toast.context.noLineSelection.title": "Geen regelselectie", - "toast.context.noLineSelection.description": "Selecteer eerst een regelbereik in een bestandstabblad.", - - "toast.session.share.copyFailed.title": "Kopiëren van URL naar klembord mislukt", - "toast.session.share.success.title": "Sessie gedeeld", - "toast.session.share.success.description": "Deel-URL naar klembord gekopieerd!", - "toast.session.share.failed.title": "Delen van sessie mislukt", - "toast.session.share.failed.description": "Er is een fout opgetreden bij het delen van de sessie", - - "toast.session.unshare.success.title": "Sessie niet meer gedeeld", - "toast.session.unshare.success.description": "Sessie succesvol niet meer gedeeld!", - "toast.session.unshare.failed.title": "Sessie niet meer delen mislukt", - "toast.session.unshare.failed.description": - "Er is een fout opgetreden bij het ongedaan maken van delen van de sessie", - - "toast.session.listFailed.title": "Laden van sessies voor {{project}} mislukt", - "toast.project.reloadFailed.title": "Herladen van {{project}} mislukt", - - "toast.update.title": "Update beschikbaar", - "toast.update.description": "Een nieuwe versie van Kilo ({{version}}) is nu beschikbaar om te installeren.", - "toast.update.action.installRestart": "Installeren en herstarten", - "toast.update.action.notYet": "Nog niet", - - "error.page.title": "Er is iets misgegaan", - "error.page.description": "Er is een fout opgetreden tijdens het laden van de applicatie.", - "error.page.details.label": "Foutdetails", - "error.page.action.restart": "Herstarten", - "error.page.action.checking": "Controleren...", - "error.page.action.checkUpdates": "Controleren op updates", - "error.page.action.updateTo": "Updaten naar {{version}}", - "error.page.report.prefix": "Meld deze fout alstublieft aan het Kilo team", - "error.page.report.discord": "op Discord", - "error.page.version": "Versie: {{version}}", - - "error.dev.rootNotFound": - "Root-element niet gevonden. Bent u vergeten dit toe te voegen aan uw index.html? Of misschien is het id-attribuut verkeerd gespeld?", - - "error.globalSync.connectFailed": "Kon geen verbinding maken met server. Draait er een server op `{{url}}`?", - "directory.error.invalidUrl": "Ongeldige map in URL.", - - "error.chain.unknown": "Onbekende fout", - "error.server.invalidConfiguration": "Ongeldige configuratie", - "error.chain.causedBy": "Veroorzaakt door:", - "error.chain.apiError": "API-fout", - "error.chain.status": "Status: {{status}}", - "error.chain.retryable": "Opnieuw te proberen: {{retryable}}", - "error.chain.responseBody": "Reactie-body:\n{{body}}", - "error.chain.didYouMean": "Bedoelde u: {{suggestions}}", - "error.chain.modelNotFound": "Model niet gevonden: {{provider}}/{{model}}", - "error.chain.checkConfig": "Controleer uw config (opencode.json) provider/modelnamen", - "error.chain.mcpFailed": 'MCP-server "{{name}}" is mislukt. Let op, Kilo ondersteunt nog geen MCP-authenticatie.', - "error.chain.providerAuthFailed": "Providerauthenticatie mislukt ({{provider}}): {{message}}", - "error.chain.providerInitFailed": - 'Initialiseren van provider "{{provider}}" mislukt. Controleer referenties en configuratie.', - "error.chain.configJsonInvalid": "Configuratiebestand op {{path}} is geen geldige JSON(C)", - "error.chain.configJsonInvalidWithMessage": "Configuratiebestand op {{path}} is geen geldige JSON(C): {{message}}", - "error.chain.configDirectoryTypo": - 'Map "{{dir}}" in {{path}} is niet geldig. Hernoem de map naar "{{suggestion}}" of verwijder deze. Dit is een veelvoorkomende typfout.', - "error.chain.configFrontmatterError": "Parsen van frontmatter mislukt in {{path}}:\n{{message}}", - "error.chain.configInvalid": "Configuratiebestand op {{path}} is ongeldig", - "error.chain.configInvalidWithMessage": "Configuratiebestand op {{path}} is ongeldig: {{message}}", - - "notification.permission.title": "Machtiging vereist", - "notification.permission.description": "{{sessionTitle}} in {{projectName}} heeft machtiging nodig", - "notification.question.title": "Vraag", - "notification.question.description": "{{sessionTitle}} in {{projectName}} heeft een vraag", - "notification.action.goToSession": "Ga naar sessie", - - "notification.session.responseReady.title": "Antwoord gereed", - "notification.session.error.title": "Sessiefout", - "notification.session.error.fallbackDescription": "Er is een fout opgetreden", - - "home.recentProjects": "Recente projecten", - "home.empty.title": "Geen recente projecten", - "home.empty.description": "Ga aan de slag door een lokaal project te openen", - - "session.tab.session": "Sessie", - "session.tab.review": "Review", - "session.tab.context": "Context", - "session.panel.reviewAndFiles": "Review en bestanden", - "session.review.filesChanged": "{{count}} Bestanden Gewijzigd", - "session.review.change.one": "Wijziging", - "session.review.change.other": "Wijzigingen", - "session.review.loadingChanges": "Wijzigingen laden...", - "session.review.empty": "Nog geen wijzigingen in deze sessie", - "session.review.noVcs": "Geen Git Version Control System gedetecteerd, wijzigingen niet weergegeven", - "session.review.noSnapshot": - "Snapshot-tracking is uitgeschakeld in configuratie, dus sessiewijzigingen zijn niet beschikbaar", - "session.review.noChanges": "Geen wijzigingen", - - "session.files.selectToOpen": "Selecteer een bestand om te openen", - "session.files.all": "Alle bestanden", - "session.files.empty": "Geen bestanden", - "session.files.binaryContent": "Binair bestand (inhoud kan niet worden weergegeven)", - - "session.messages.renderEarlier": "Eerdere berichten renderen", - "session.messages.loadingEarlier": "Eerdere berichten laden...", - "session.messages.loadEarlier": "Eerdere berichten laden", - "session.messages.loading": "Berichten laden...", - "session.messages.jumpToLatest": "Spring naar nieuwste", - - "session.context.addToContext": "Voeg {{selection}} toe aan context", - "session.todo.title": "To-do's", - "session.todo.collapse": "Inklappen", - "session.todo.expand": "Uitklappen", - - "session.new.title": "Bouw wat u wilt", - "session.new.worktree.main": "Hoofdbranch", - "session.new.worktree.mainWithBranch": "Hoofdbranch ({{branch}})", - "session.new.worktree.create": "Nieuwe worktree maken", - "session.new.lastModified": "Laatst gewijzigd", - - "session.header.search.placeholder": "Zoek {{project}}", - "session.header.searchFiles": "Zoek bestanden", - "session.header.openIn": "Openen in", - "session.header.open.action": "Open {{app}}", - "session.header.open.ariaLabel": "Openen in {{app}}", - "session.header.open.menu": "Open-opties", - "session.header.open.copyPath": "Pad kopiëren", - - "status.popover.trigger": "Status", - "status.popover.ariaLabel": "Serverconfiguraties", - "status.popover.tab.servers": "Servers", - "status.popover.tab.mcp": "MCP", - "status.popover.tab.lsp": "LSP", - "status.popover.tab.plugins": "Plug-ins", - "status.popover.action.manageServers": "Servers beheren", - - "session.share.popover.title": "Op web publiceren", - "session.share.popover.description.shared": - "Deze sessie is openbaar op het web. Het is toegankelijk voor iedereen met de link.", - "session.share.popover.description.unshared": - "Sessie openbaar delen op het web. Het zal toegankelijk zijn voor iedereen met de link.", - "session.share.action.share": "Delen", - "session.share.action.publish": "Publiceren", - "session.share.action.publishing": "Publiceren...", - "session.share.action.unpublish": "Publicatie ongedaan maken", - "session.share.action.unpublishing": "Publicatie ongedaan maken...", - "session.share.action.view": "Bekijken", - "session.share.copy.copied": "Gekopieerd", - "session.share.copy.copyLink": "Link kopiëren", - - "lsp.tooltip.none": "Geen LSP-servers", - "lsp.label.connected": "{{count}} LSP", - - "prompt.loading": "Prompt laden...", - "terminal.loading": "Terminal laden...", - "terminal.title": "Terminal", - "terminal.title.numbered": "Terminal {{number}}", - "terminal.close": "Terminal sluiten", - "terminal.connectionLost.title": "Verbinding verbroken", - "terminal.connectionLost.description": - "De terminalverbinding is onderbroken. Dit kan gebeuren wanneer de server herstart.", - - "common.closeTab": "Tabblad sluiten", - "common.dismiss": "Negeren", - "common.moreCountSuffix": " (nog +{{count}})", - "common.requestFailed": "Verzoek mislukt", - "common.moreOptions": "Meer opties", - "common.learnMore": "Meer informatie", - "common.rename": "Hernoemen", - "common.reset": "Reset", - "common.archive": "Archiveren", - "common.delete": "Verwijderen", - "common.close": "Sluiten", - "common.edit": "Bewerken", - "common.loadMore": "Meer laden", - "common.key.esc": "ESC", - - "common.time.justNow": "Zojuist", - "common.time.minutesAgo.short": "{{count}}m geleden", - "common.time.hoursAgo.short": "{{count}}u geleden", - "common.time.daysAgo.short": "{{count}}d geleden", - - "sidebar.menu.toggle": "Menu in-/uitschakelen", - "sidebar.nav.projectsAndSessions": "Projecten en sessies", - "sidebar.settings": "Instellingen", - "sidebar.help": "Help", - "sidebar.workspaces.enable": "Werkruimtes inschakelen", - "sidebar.workspaces.disable": "Werkruimtes uitschakelen", - "sidebar.gettingStarted.title": "Aan de slag", - "sidebar.gettingStarted.line1": "Kilo bevat gratis modellen zodat u direct kunt beginnen.", - "sidebar.gettingStarted.line2": "Verbind een provider om modellen te gebruiken, incl. Claude, GPT, Gemini etc.", - "sidebar.project.recentSessions": "Recente sessies", - "sidebar.project.viewAllSessions": "Bekijk alle sessies", - "sidebar.project.clearNotifications": "Meldingen wissen", - - "app.name.desktop": "Kilo Desktop", - - "settings.section.desktop": "Desktop", - "settings.section.server": "Server", - "settings.tab.general": "Algemeen", - "settings.tab.shortcuts": "Sneltoetsen", - "settings.desktop.section.wsl": "WSL", - "settings.desktop.wsl.title": "WSL-integratie", - "settings.desktop.wsl.description": "Draai de Kilo server binnen WSL op Windows.", - - "settings.general.section.appearance": "Uiterlijk", - "settings.general.section.notifications": "Systeemmeldingen", - "settings.general.section.updates": "Updates", - "settings.general.section.sounds": "Geluidseffecten", - "settings.general.section.feed": "Feed", - "settings.general.section.display": "Weergave", - - "settings.general.row.language.title": "Taal", - "settings.general.row.language.description": "Wijzig de weergavetaal voor Kilo", - "settings.general.row.appearance.title": "Uiterlijk", - "settings.general.row.appearance.description": "Pas aan hoe Kilo eruitziet op uw apparaat", - "settings.general.row.theme.title": "Thema", - "settings.general.row.theme.description": "Pas het thema van Kilo aan.", - "settings.general.row.font.title": "Lettertype", - "settings.general.row.font.description": "Pas het monospace-lettertype in codeblokken aan", - "settings.general.row.reasoningSummaries.title": "Redeneersamenvattingen weergeven", - "settings.general.row.reasoningSummaries.description": "Toon model-redeneersamenvattingen in de tijdlijn", - "settings.general.row.shellToolPartsExpanded.title": "Shell-toolonderdelen uitklappen", - "settings.general.row.shellToolPartsExpanded.description": - "Toon shell-toolonderdelen standaard uitgeklapt in de tijdlijn", - "settings.general.row.editToolPartsExpanded.title": "Bewerk-toolonderdelen uitklappen", - "settings.general.row.editToolPartsExpanded.description": - "Toon bewerk-, schrijf- en patch-toolonderdelen standaard uitgeklapt in de tijdlijn", - - "settings.general.row.wayland.title": "Gebruik native Wayland", - "settings.general.row.wayland.description": "Schakel X11-terugval uit op Wayland. Vereist een herstart.", - "settings.general.row.wayland.tooltip": - "Op Linux met monitoren met gemengde verversingssnelheden kan native Wayland stabieler zijn.", - - "settings.general.row.releaseNotes.title": "Release notes", - "settings.general.row.releaseNotes.description": "Toon Wat is nieuw pop-ups na updates", - - "settings.updates.row.startup.title": "Controleren op updates bij opstarten", - "settings.updates.row.startup.description": "Automatisch op updates controleren wanneer Kilo start", - "settings.updates.row.check.title": "Controleren op updates", - "settings.updates.row.check.description": "Handmatig op updates controleren en installeren indien beschikbaar", - "settings.updates.action.checkNow": "Nu controleren", - "settings.updates.action.checking": "Controleren...", - "settings.updates.toast.latest.title": "U bent up-to-date", - "settings.updates.toast.latest.description": "U draait de nieuwste versie van Kilo.", - "font.option.ibmPlexMono": "IBM Plex Mono", - "font.option.cascadiaCode": "Cascadia Code", - "font.option.firaCode": "Fira Code", - "font.option.hack": "Hack", - "font.option.inconsolata": "Inconsolata", - "font.option.intelOneMono": "Intel One Mono", - "font.option.iosevka": "Iosevka", - "font.option.jetbrainsMono": "JetBrains Mono", - "font.option.mesloLgs": "Meslo LGS", - "font.option.robotoMono": "Roboto Mono", - "font.option.sourceCodePro": "Source Code Pro", - "font.option.ubuntuMono": "Ubuntu Mono", - "font.option.geistMono": "Geist Mono", - "sound.option.none": "Geen", - "sound.option.alert01": "Melding 01", - "sound.option.alert02": "Melding 02", - "sound.option.alert03": "Melding 03", - "sound.option.alert04": "Melding 04", - "sound.option.alert05": "Melding 05", - "sound.option.alert06": "Melding 06", - "sound.option.alert07": "Melding 07", - "sound.option.alert08": "Melding 08", - "sound.option.alert09": "Melding 09", - "sound.option.alert10": "Melding 10", - "sound.option.bipbop01": "Bip-bop 01", - "sound.option.bipbop02": "Bip-bop 02", - "sound.option.bipbop03": "Bip-bop 03", - "sound.option.bipbop04": "Bip-bop 04", - "sound.option.bipbop05": "Bip-bop 05", - "sound.option.bipbop06": "Bip-bop 06", - "sound.option.bipbop07": "Bip-bop 07", - "sound.option.bipbop08": "Bip-bop 08", - "sound.option.bipbop09": "Bip-bop 09", - "sound.option.bipbop10": "Bip-bop 10", - "sound.option.staplebops01": "Staplebops 01", - "sound.option.staplebops02": "Staplebops 02", - "sound.option.staplebops03": "Staplebops 03", - "sound.option.staplebops04": "Staplebops 04", - "sound.option.staplebops05": "Staplebops 05", - "sound.option.staplebops06": "Staplebops 06", - "sound.option.staplebops07": "Staplebops 07", - "sound.option.nope01": "Nee 01", - "sound.option.nope02": "Nee 02", - "sound.option.nope03": "Nee 03", - "sound.option.nope04": "Nee 04", - "sound.option.nope05": "Nee 05", - "sound.option.nope06": "Nee 06", - "sound.option.nope07": "Nee 07", - "sound.option.nope08": "Nee 08", - "sound.option.nope09": "Nee 09", - "sound.option.nope10": "Nee 10", - "sound.option.nope11": "Nee 11", - "sound.option.nope12": "Nee 12", - "sound.option.yup01": "Ja 01", - "sound.option.yup02": "Ja 02", - "sound.option.yup03": "Ja 03", - "sound.option.yup04": "Ja 04", - "sound.option.yup05": "Ja 05", - "sound.option.yup06": "Ja 06", - - "settings.general.notifications.agent.title": "Agent", - "settings.general.notifications.agent.description": - "Toon systeemmelding wanneer de agent klaar is of aandacht nodig heeft", - "settings.general.notifications.permissions.title": "Machtigingen", - "settings.general.notifications.permissions.description": "Toon systeemmelding wanneer een machtiging vereist is", - "settings.general.notifications.errors.title": "Fouten", - "settings.general.notifications.errors.description": "Toon systeemmelding wanneer er een fout optreedt", - - "settings.general.sounds.agent.title": "Agent", - "settings.general.sounds.agent.description": "Speel geluid af wanneer de agent klaar is of aandacht nodig heeft", - "settings.general.sounds.permissions.title": "Machtigingen", - "settings.general.sounds.permissions.description": "Speel geluid af wanneer een machtiging vereist is", - "settings.general.sounds.errors.title": "Fouten", - "settings.general.sounds.errors.description": "Speel geluid af wanneer er een fout optreedt", - - "settings.shortcuts.title": "Sneltoetsen", - "settings.shortcuts.reset.button": "Terugzetten naar standaard", - "settings.shortcuts.reset.toast.title": "Sneltoetsen gereset", - "settings.shortcuts.reset.toast.description": "Sneltoetsen zijn teruggezet naar de standaardwaarden.", - "settings.shortcuts.conflict.title": "Sneltoets al in gebruik", - "settings.shortcuts.conflict.description": "{{keybind}} is al toegewezen aan {{titles}}.", - "settings.shortcuts.unassigned": "Niet toegewezen", - "settings.shortcuts.pressKeys": "Druk op toetsen", - "settings.shortcuts.search.placeholder": "Zoek sneltoetsen", - "settings.shortcuts.search.empty": "Geen sneltoetsen gevonden", - - "settings.shortcuts.group.general": "Algemeen", - "settings.shortcuts.group.session": "Sessie", - "settings.shortcuts.group.navigation": "Navigatie", - "settings.shortcuts.group.modelAndAgent": "Model en agent", - "settings.shortcuts.group.terminal": "Terminal", - "settings.shortcuts.group.prompt": "Prompt", - - "settings.providers.title": "Providers", - "settings.providers.description": "Provider-instellingen kunnen hier worden geconfigureerd.", - "settings.providers.section.connected": "Verbonden providers", - "settings.providers.connected.empty": "Geen verbonden providers", - "settings.providers.connected.environmentDescription": "Verbonden vanuit uw omgevingsvariabelen", - "settings.providers.section.popular": "Populaire providers", - "settings.providers.custom.description": "Voeg een OpenAI-compatibele provider toe via basis-URL.", - "settings.providers.tag.environment": "Omgeving", - "settings.providers.tag.config": "Config", - "settings.providers.tag.custom": "Aangepast", - "settings.providers.tag.other": "Overig", - "settings.models.title": "Modellen", - "settings.models.description": "Model-instellingen kunnen hier worden geconfigureerd.", - "settings.agents.title": "Agenten", - "settings.agents.description": "Agent-instellingen kunnen hier worden geconfigureerd.", - "settings.commands.title": "Opdrachten", - "settings.commands.description": "Opdracht-instellingen kunnen hier worden geconfigureerd.", - "settings.mcp.title": "MCP", - "settings.mcp.description": "MCP-instellingen kunnen hier worden geconfigureerd.", - - "settings.permissions.title": "Machtigingen", - "settings.permissions.description": "Bepaal welke tools de server standaard mag gebruiken.", - "settings.permissions.section.tools": "Tools", - "settings.permissions.toast.updateFailed.title": "Bijwerken van machtigingen mislukt", - - "settings.permissions.action.allow": "Toestaan", - "settings.permissions.action.ask": "Vragen", - "settings.permissions.action.deny": "Weigeren", - - "settings.permissions.tool.read.title": "Lezen (Read)", - "settings.permissions.tool.read.description": "Een bestand lezen (komt overeen met het bestandspad)", - "settings.permissions.tool.edit.title": "Bewerken (Edit)", - "settings.permissions.tool.edit.description": - "Bestanden wijzigen, inclusief bewerkingen, wegschrijven, patches en multi-edits", - "settings.permissions.tool.glob.title": "Glob", - "settings.permissions.tool.glob.description": "Bestanden matchen met glob-patronen", - "settings.permissions.tool.grep.title": "Grep", - "settings.permissions.tool.grep.description": "Bestandsinhoud doorzoeken met behulp van reguliere expressies", - "settings.permissions.tool.list.title": "Lijst (List)", - "settings.permissions.tool.list.description": "Bestanden in een map weergeven", - "settings.permissions.tool.bash.title": "Bash", - "settings.permissions.tool.bash.description": "Shell-opdrachten uitvoeren", - "settings.permissions.tool.task.title": "Taak (Task)", - "settings.permissions.tool.task.description": "Sub-agenten lanceren", - "settings.permissions.tool.skill.title": "Skill", - "settings.permissions.tool.skill.description": "Een skill laden op naam", - "settings.permissions.tool.lsp.title": "LSP", - "settings.permissions.tool.lsp.description": "Language server query's uitvoeren", - "settings.permissions.tool.todoread.title": "To-do lezen (Todo Read)", - "settings.permissions.tool.todoread.description": "De to-do lijst lezen", - "settings.permissions.tool.todowrite.title": "To-do schrijven (Todo Write)", - "settings.permissions.tool.todowrite.description": "De to-do lijst bijwerken", - "settings.permissions.tool.webfetch.title": "Web ophalen (Web Fetch)", - "settings.permissions.tool.webfetch.description": "Inhoud ophalen van een URL", - "settings.permissions.tool.websearch.title": "Web zoeken (Web Search)", - "settings.permissions.tool.websearch.description": "Het web doorzoeken", - "settings.permissions.tool.codesearch.title": "Code zoeken (Code Search)", - "settings.permissions.tool.codesearch.description": "Zoeken naar code op het web", - "settings.permissions.tool.external_directory.title": "Externe Map", - "settings.permissions.tool.external_directory.description": "Toegang tot bestanden buiten de projectmap", - "settings.permissions.tool.doom_loop.title": "Eindeloze Loop (Doom Loop)", - "settings.permissions.tool.doom_loop.description": "Gedetecteerde herhaalde tool-aanroepen met identieke invoer", - - "session.delete.failed.title": "Verwijderen van sessie mislukt", - "session.delete.title": "Sessie verwijderen", - "session.delete.confirm": 'Sessie "{{name}}" verwijderen?', - "session.delete.button": "Sessie verwijderen", - - "workspace.new": "Nieuwe werkruimte", - "workspace.type.local": "lokaal", - "workspace.type.sandbox": "sandbox", - "workspace.create.failed.title": "Maken van werkruimte mislukt", - "workspace.delete.failed.title": "Verwijderen van werkruimte mislukt", - "workspace.resetting.title": "Werkruimte resetten", - "workspace.resetting.description": "Dit kan een minuut duren.", - "workspace.reset.failed.title": "Resetten van werkruimte mislukt", - "workspace.reset.success.title": "Werkruimte gereset", - "workspace.reset.success.description": "Werkruimte komt nu overeen met de standaardbranch.", - "workspace.error.stillPreparing": "Werkruimte is nog aan het voorbereiden", - "workspace.status.checking": "Controleren op ongemergde wijzigingen...", - "workspace.status.error": "Kan git-status niet verifiëren.", - "workspace.status.clean": "Geen ongemergde wijzigingen gedetecteerd.", - "workspace.status.dirty": "Ongemergde wijzigingen gedetecteerd in deze werkruimte.", - "workspace.delete.title": "Werkruimte verwijderen", - "workspace.delete.confirm": 'Werkruimte "{{name}}" verwijderen?', - "workspace.delete.button": "Werkruimte verwijderen", - "workspace.reset.title": "Werkruimte resetten", - "workspace.reset.confirm": 'Werkruimte "{{name}}" resetten?', - "workspace.reset.button": "Werkruimte resetten", - "workspace.reset.archived.none": "Geen actieve sessies zullen worden gearchiveerd.", - "workspace.reset.archived.one": "1 sessie zal worden gearchiveerd.", - "workspace.reset.archived.many": "{{count}} sessies zullen worden gearchiveerd.", - "workspace.reset.note": "Dit zal de werkruimte resetten zodat deze overeenkomt met de standaardbranch.", -} diff --git a/packages/app/src/i18n/no.ts b/packages/app/src/i18n/no.ts deleted file mode 100644 index 1551f95682..0000000000 --- a/packages/app/src/i18n/no.ts +++ /dev/null @@ -1,936 +0,0 @@ -import { dict as en } from "./en" -type Keys = keyof typeof en - -export const dict = { - "command.category.suggested": "Foreslått", - "command.category.view": "Visning", - "command.category.project": "Prosjekt", - "command.category.provider": "Leverandør", - "command.category.server": "Server", - "command.category.session": "Sesjon", - "command.category.theme": "Tema", - "command.category.language": "Språk", - "command.category.file": "Fil", - "command.category.context": "Kontekst", - "command.category.terminal": "Terminal", - "command.category.model": "Modell", - "command.category.mcp": "MCP", - "command.category.agent": "Agent", - "command.category.permissions": "Tillatelser", - "command.category.workspace": "Arbeidsområde", - "command.category.settings": "Innstillinger", - - "theme.scheme.system": "System", - "theme.scheme.light": "Lys", - "theme.scheme.dark": "Mørk", - - "command.sidebar.toggle": "Veksle sidepanel", - "command.project.open": "Åpne prosjekt", - "command.provider.connect": "Koble til leverandør", - "command.server.switch": "Bytt server", - "command.settings.open": "Åpne innstillinger", - "command.session.previous": "Forrige sesjon", - "command.session.next": "Neste sesjon", - "command.session.previous.unseen": "Forrige uleste økt", - "command.session.next.unseen": "Neste uleste økt", - "command.session.archive": "Arkiver sesjon", - - "command.palette": "Kommandopalett", - - "command.theme.cycle": "Bytt tema", - "command.theme.set": "Bruk tema: {{theme}}", - "command.theme.scheme.cycle": "Bytt fargevalg", - "command.theme.scheme.set": "Bruk fargevalg: {{scheme}}", - - "command.language.cycle": "Bytt språk", - "command.language.set": "Bruk språk: {{language}}", - - "command.session.new": "Ny sesjon", - "command.file.open": "Åpne fil", - "command.tab.close": "Lukk fane", - "command.context.addSelection": "Legg til markering i kontekst", - "command.context.addSelection.description": "Legg til valgte linjer fra gjeldende fil", - "command.input.focus": "Fokuser inndata", - "command.terminal.toggle": "Veksle terminal", - "command.fileTree.toggle": "Veksle filtre", - "command.review.toggle": "Veksle gjennomgang", - "command.terminal.new": "Ny terminal", - "command.terminal.new.description": "Opprett en ny terminalfane", - "command.steps.toggle": "Veksle trinn", - "command.steps.toggle.description": "Vis eller skjul trinn for gjeldende melding", - "command.message.previous": "Forrige melding", - "command.message.previous.description": "Gå til forrige brukermelding", - "command.message.next": "Neste melding", - "command.message.next.description": "Gå til neste brukermelding", - "command.model.choose": "Velg modell", - "command.model.choose.description": "Velg en annen modell", - "command.mcp.toggle": "Veksle MCP-er", - "command.mcp.toggle.description": "Veksle MCP-er", - "command.agent.cycle": "Bytt agent", - "command.agent.cycle.description": "Bytt til neste agent", - "command.agent.cycle.reverse": "Bytt agent bakover", - "command.agent.cycle.reverse.description": "Bytt til forrige agent", - "command.model.variant.cycle": "Bytt tenkeinnsats", - "command.model.variant.cycle.description": "Bytt til neste innsatsnivå", - "command.prompt.mode.shell": "Shell", - "command.prompt.mode.normal": "Prompt", - "command.permissions.autoaccept.enable": "Aksepter tillatelser automatisk", - "command.permissions.autoaccept.disable": "Stopp automatisk akseptering av tillatelser", - "command.workspace.toggle": "Veksle arbeidsområder", - "command.workspace.toggle.description": "Enable or disable multiple workspaces in the sidebar", - "command.session.undo": "Angre", - "command.session.undo.description": "Angre siste melding", - "command.session.redo": "Gjør om", - "command.session.redo.description": "Gjør om siste angrede melding", - "command.session.compact": "Komprimer sesjon", - "command.session.compact.description": "Oppsummer sesjonen for å redusere kontekststørrelsen", - "command.session.fork": "Forgren fra melding", - "command.session.fork.description": "Opprett en ny sesjon fra en tidligere melding", - "command.session.share": "Del sesjon", - "command.session.share.description": "Del denne sesjonen og kopier URL-en til utklippstavlen", - "command.session.unshare": "Slutt å dele sesjon", - "command.session.unshare.description": "Slutt å dele denne sesjonen", - - "palette.search.placeholder": "Søk i filer, kommandoer og sesjoner", - "palette.empty": "Ingen resultater funnet", - "palette.group.commands": "Kommandoer", - "palette.group.files": "Filer", - - "dialog.provider.search.placeholder": "Søk etter leverandører", - "dialog.provider.empty": "Ingen leverandører funnet", - "dialog.provider.group.popular": "Populære", - "dialog.provider.group.other": "Andre", - "dialog.provider.tag.recommended": "Anbefalt", - "dialog.provider.opencode.note": "Utvalgte modeller inkludert Claude, GPT, Gemini og mer", - "dialog.provider.opencode.tagline": "Pålitelige, optimaliserte modeller", - "dialog.provider.opencodeGo.tagline": "Rimelig abonnement for alle", - "dialog.provider.anthropic.note": "Direkte tilgang til Claude-modeller, inkludert Pro og Max", - "dialog.provider.copilot.note": "AI-modeller for kodeassistanse via GitHub Copilot", - "dialog.provider.openai.note": "GPT-modeller for raske, dyktige generelle AI-oppgaver", - "dialog.provider.google.note": "Gemini-modeller for raske, strukturerte svar", - "dialog.provider.openrouter.note": "Tilgang til alle støttede modeller fra én leverandør", - "dialog.provider.vercel.note": "Enhetlig tilgang til AI-modeller med smart ruting", - - "dialog.model.select.title": "Velg modell", - "dialog.model.search.placeholder": "Søk etter modeller", - "dialog.model.empty": "Ingen modellresultater", - "dialog.model.manage": "Administrer modeller", - "dialog.model.manage.description": "Tilpass hvilke modeller som vises i modellvelgeren.", - "dialog.model.manage.provider.toggle": "Veksle alle {{provider}}-modeller", - - "dialog.model.unpaid.freeModels.title": "Gratis modeller levert av Kilo", - "dialog.model.unpaid.addMore.title": "Legg til flere modeller fra populære leverandører", - - "dialog.provider.viewAll": "Vis flere leverandører", - - "provider.connect.title": "Koble til {{provider}}", - "provider.connect.title.anthropicProMax": "Logg inn med Claude Pro/Max", - "provider.connect.selectMethod": "Velg innloggingsmetode for {{provider}}.", - "provider.connect.method.apiKey": "API-nøkkel", - "provider.connect.status.inProgress": "Autorisering pågår...", - "provider.connect.status.waiting": "Venter på autorisering...", - "provider.connect.status.failed": "Autorisering mislyktes: {{error}}", - "provider.connect.apiKey.description": - "Skriv inn din {{provider}} API-nøkkel for å koble til kontoen din og bruke {{provider}}-modeller i Kilo.", - "provider.connect.apiKey.label": "{{provider}} API-nøkkel", - "provider.connect.apiKey.placeholder": "API-nøkkel", - "provider.connect.apiKey.required": "API-nøkkel er påkrevd", - "provider.connect.opencodeZen.line1": - "OpenCode Zen gir deg tilgang til et utvalg av pålitelige optimaliserte modeller for kodeagenter.", - "provider.connect.opencodeZen.line2": - "Med én enkelt API-nøkkel får du tilgang til modeller som Claude, GPT, Gemini, GLM og flere.", - "provider.connect.opencodeZen.visit.prefix": "Besøk ", - "provider.connect.opencodeZen.visit.link": "opencode.ai/zen", - "provider.connect.opencodeZen.visit.suffix": " for å hente API-nøkkelen din.", - "provider.connect.oauth.code.visit.prefix": "Besøk ", - "provider.connect.oauth.code.visit.link": "denne lenken", - "provider.connect.oauth.code.visit.suffix": - " for å hente autorisasjonskoden din for å koble til kontoen din og bruke {{provider}}-modeller i Kilo.", - "provider.connect.oauth.code.label": "{{method}} autorisasjonskode", - "provider.connect.oauth.code.placeholder": "Autorisasjonskode", - "provider.connect.oauth.code.required": "Autorisasjonskode er påkrevd", - "provider.connect.oauth.code.invalid": "Ugyldig autorisasjonskode", - "provider.connect.oauth.auto.visit.prefix": "Besøk ", - "provider.connect.oauth.auto.visit.link": "denne lenken", - "provider.connect.oauth.auto.visit.suffix": - " og skriv inn koden nedenfor for å koble til kontoen din og bruke {{provider}}-modeller i Kilo.", - "provider.connect.oauth.auto.confirmationCode": "Bekreftelseskode", - "provider.connect.toast.connected.title": "{{provider}} tilkoblet", - "provider.connect.toast.connected.description": "{{provider}}-modeller er nå tilgjengelige.", - - "provider.custom.title": "Egendefinert leverandør", - "provider.custom.description.prefix": "Konfigurer en OpenAI-kompatibel leverandør. Se ", - "provider.custom.description.link": "dokumentasjon for leverandørkonfigurasjon", - "provider.custom.description.suffix": ".", - "provider.custom.field.providerID.label": "Leverandør-ID", - "provider.custom.field.providerID.placeholder": "minleverandør", - "provider.custom.field.providerID.description": "Små bokstaver, tall, bindestreker eller understreker", - "provider.custom.field.name.label": "Visningsnavn", - "provider.custom.field.name.placeholder": "Min AI-leverandør", - "provider.custom.field.baseURL.label": "Base-URL", - "provider.custom.field.baseURL.placeholder": "https://api.myprovider.com/v1", - "provider.custom.field.apiKey.label": "API-nøkkel", - "provider.custom.field.apiKey.placeholder": "API-nøkkel", - "provider.custom.field.apiKey.description": "Valgfritt. La stå tomt hvis du administrerer autentisering via headers.", - "provider.custom.models.label": "Modeller", - "provider.custom.models.id.label": "ID", - "provider.custom.models.id.placeholder": "modell-id", - "provider.custom.models.name.label": "Navn", - "provider.custom.models.name.placeholder": "Visningsnavn", - "provider.custom.models.remove": "Fjern modell", - "provider.custom.models.add": "Legg til modell", - "provider.custom.headers.label": "Headers (valgfritt)", - "provider.custom.headers.key.label": "Header", - "provider.custom.headers.key.placeholder": "Header-Navn", - "provider.custom.headers.value.label": "Verdi", - "provider.custom.headers.value.placeholder": "verdi", - "provider.custom.headers.remove": "Fjern header", - "provider.custom.headers.add": "Legg til header", - "provider.custom.error.providerID.required": "Leverandør-ID er påkrevd", - "provider.custom.error.providerID.format": "Bruk små bokstaver, tall, bindestreker eller understreker", - "provider.custom.error.providerID.exists": "Den leverandør-IDen finnes allerede", - "provider.custom.error.name.required": "Visningsnavn er påkrevd", - "provider.custom.error.baseURL.required": "Base-URL er påkrevd", - "provider.custom.error.baseURL.format": "Må starte med http:// eller https://", - "provider.custom.error.required": "Påkrevd", - "provider.custom.error.duplicate": "Duplikat", - - "provider.disconnect.toast.disconnected.title": "{{provider}} frakoblet", - "provider.disconnect.toast.disconnected.description": "Modeller fra {{provider}} er ikke lenger tilgjengelige.", - - "model.tag.free": "Gratis", - "model.tag.latest": "Nyeste", - "model.provider.anthropic": "Anthropic", - "model.provider.openai": "OpenAI", - "model.provider.google": "Google", - "model.provider.xai": "xAI", - "model.provider.meta": "Meta", - "model.input.text": "tekst", - "model.input.image": "bilde", - "model.input.audio": "lyd", - "model.input.video": "video", - "model.input.pdf": "pdf", - "model.tooltip.allows": "Tillater: {{inputs}}", - "model.tooltip.reasoning.allowed": "Tillater resonnering", - "model.tooltip.reasoning.none": "Ingen resonnering", - "model.tooltip.context": "Kontekstgrense {{limit}}", - - "common.search.placeholder": "Søk", - "common.goBack": "Gå tilbake", - "common.goForward": "Gå frem", - "common.loading": "Laster", - "common.loading.ellipsis": "...", - "common.cancel": "Avbryt", - "common.connect": "Koble til", - "common.disconnect": "Koble fra", - "common.continue": "Send inn", - "common.submit": "Send inn", - "common.save": "Lagre", - "common.saving": "Lagrer...", - "common.default": "Standard", - "common.attachment": "vedlegg", - - "prompt.placeholder.shell": "Skriv inn shell-kommando... {{example}}", - "prompt.placeholder.normal": 'Spør om hva som helst... "{{example}}"', - "prompt.placeholder.simple": "Spør om hva som helst...", - "prompt.placeholder.summarizeComments": "Oppsummer kommentarer…", - "prompt.placeholder.summarizeComment": "Oppsummer kommentar…", - "prompt.mode.shell": "Shell", - "prompt.mode.normal": "Prompt", - "prompt.mode.shell.exit": "ESC for å avslutte", - - "prompt.example.1": "Fiks en TODO i kodebasen", - "prompt.example.2": "Hva er teknologistabelen i dette prosjektet?", - "prompt.example.3": "Fiks ødelagte tester", - "prompt.example.4": "Forklar hvordan autentisering fungerer", - "prompt.example.5": "Finn og fiks sikkerhetssårbarheter", - "prompt.example.6": "Legg til enhetstester for brukerservicen", - "prompt.example.7": "Refaktorer denne funksjonen for bedre lesbarhet", - "prompt.example.8": "Hva betyr denne feilen?", - "prompt.example.9": "Hjelp meg med å feilsøke dette problemet", - "prompt.example.10": "Generer API-dokumentasjon", - "prompt.example.11": "Optimaliser databasespørringer", - "prompt.example.12": "Legg til inputvalidering", - "prompt.example.13": "Lag en ny komponent for...", - "prompt.example.14": "Hvordan deployer jeg dette prosjektet?", - "prompt.example.15": "Gjennomgå koden min for beste praksis", - "prompt.example.16": "Legg til feilhåndtering i denne funksjonen", - "prompt.example.17": "Forklar dette regex-mønsteret", - "prompt.example.18": "Konverter dette til TypeScript", - "prompt.example.19": "Legg til logging i hele kodebasen", - "prompt.example.20": "Hvilke avhengigheter er utdaterte?", - "prompt.example.21": "Hjelp meg med å skrive et migreringsskript", - "prompt.example.22": "Implementer caching for dette endepunktet", - "prompt.example.23": "Legg til paginering i denne listen", - "prompt.example.24": "Lag en CLI-kommando for...", - "prompt.example.25": "Hvordan fungerer miljøvariabler her?", - - "prompt.popover.emptyResults": "Ingen matchende resultater", - "prompt.popover.emptyCommands": "Ingen matchende kommandoer", - "prompt.dropzone.label": "Slipp bilder, PDF-er eller tekstfiler her", - "prompt.dropzone.file.label": "Slipp for å @nevne fil", - "prompt.slash.badge.custom": "egendefinert", - "prompt.slash.badge.skill": "skill", - "prompt.slash.badge.mcp": "mcp", - "prompt.context.active": "aktiv", - "prompt.context.includeActiveFile": "Inkluder aktiv fil", - "prompt.context.removeActiveFile": "Fjern aktiv fil fra kontekst", - "prompt.context.removeFile": "Fjern fil fra kontekst", - "prompt.action.attachFile": "Legg ved fil", - "prompt.attachment.remove": "Fjern vedlegg", - "prompt.action.send": "Send", - "prompt.action.stop": "Stopp", - - "prompt.toast.pasteUnsupported.title": "Ikke støttet vedlegg", - "prompt.toast.pasteUnsupported.description": "Kun bilder, PDF-er eller tekstfiler kan legges ved her.", - "prompt.toast.modelAgentRequired.title": "Velg en agent og modell", - "prompt.toast.modelAgentRequired.description": "Velg en agent og modell før du sender en forespørsel.", - "prompt.toast.worktreeCreateFailed.title": "Kunne ikke opprette worktree", - "prompt.toast.sessionCreateFailed.title": "Kunne ikke opprette sesjon", - "prompt.toast.shellSendFailed.title": "Kunne ikke sende shell-kommando", - "prompt.toast.commandSendFailed.title": "Kunne ikke sende kommando", - "prompt.toast.promptSendFailed.title": "Kunne ikke sende forespørsel", - "prompt.toast.promptSendFailed.description": "Kunne ikke hente økt", - - "dialog.mcp.title": "MCP-er", - "dialog.mcp.description": "{{enabled}} av {{total}} aktivert", - "dialog.mcp.empty": "Ingen MCP-er konfigurert", - - "dialog.lsp.empty": "LSP-er automatisk oppdaget fra filtyper", - "dialog.plugins.empty": "Plugins konfigurert i opencode.json", - - "mcp.status.connected": "tilkoblet", - "mcp.status.failed": "mislyktes", - "mcp.status.needs_auth": "trenger autentisering", - "mcp.status.disabled": "deaktivert", - - "dialog.fork.empty": "Ingen meldinger å forgrene fra", - - "dialog.directory.search.placeholder": "Søk etter mapper", - "dialog.directory.empty": "Ingen mapper funnet", - - "dialog.server.title": "Servere", - "dialog.server.description": "Bytt hvilken Kilo-server denne appen kobler til.", - "dialog.server.search.placeholder": "Søk etter servere", - "dialog.server.empty": "Ingen servere ennå", - "dialog.server.add.title": "Legg til en server", - "dialog.server.add.url": "Server-URL", - "dialog.server.add.placeholder": "http://localhost:4096", - "dialog.server.add.error": "Kunne ikke koble til server", - "dialog.server.add.checking": "Sjekker...", - "dialog.server.add.button": "Legg til server", - "dialog.server.add.name": "Servernavn (valgfritt)", - "dialog.server.add.namePlaceholder": "Localhost", - "dialog.server.add.username": "Brukernavn (valgfritt)", - "dialog.server.add.password": "Passord (valgfritt)", - "dialog.server.edit.title": "Rediger server", - "dialog.server.default.title": "Standardserver", - "dialog.server.default.description": - "Koble til denne serveren ved oppstart i stedet for å starte en lokal server. Krever omstart.", - "dialog.server.default.none": "Ingen server valgt", - "dialog.server.default.set": "Sett gjeldende server som standard", - "dialog.server.default.clear": "Tøm", - "dialog.server.action.remove": "Fjern server", - - "dialog.server.menu.edit": "Rediger", - "dialog.server.menu.default": "Sett som standard", - "dialog.server.menu.defaultRemove": "Fjern standard", - "dialog.server.menu.delete": "Slett", - "dialog.server.current": "Gjeldende server", - "dialog.server.status.default": "Standard", - - "dialog.project.edit.title": "Rediger prosjekt", - "dialog.project.edit.name": "Navn", - "dialog.project.edit.icon": "Ikon", - "dialog.project.edit.icon.alt": "Prosjektikon", - "dialog.project.edit.icon.hint": "Klikk eller dra et bilde", - "dialog.project.edit.icon.recommended": "Anbefalt: 128x128px", - "dialog.project.edit.color": "Farge", - "dialog.project.edit.color.select": "Velg fargen {{color}}", - "dialog.project.edit.worktree.startup": "Oppstartsskript for arbeidsområde", - "dialog.project.edit.worktree.startup.description": "Kjører etter at et nytt arbeidsområde (worktree) er opprettet.", - "dialog.project.edit.worktree.startup.placeholder": "f.eks. bun install", - - "context.breakdown.title": "Kontekstfordeling", - "context.breakdown.note": 'Omtrentlig fordeling av input-tokens. "Annet" inkluderer verktøydefinisjoner og overhead.', - "context.breakdown.system": "System", - "context.breakdown.user": "Bruker", - "context.breakdown.assistant": "Assistent", - "context.breakdown.tool": "Verktøykall", - "context.breakdown.other": "Annet", - - "context.systemPrompt.title": "Systemprompt", - "context.rawMessages.title": "Rå meldinger", - - "context.stats.session": "Sesjon", - "context.stats.messages": "Meldinger", - "context.stats.provider": "Leverandør", - "context.stats.model": "Modell", - "context.stats.limit": "Kontekstgrense", - "context.stats.totalTokens": "Totalt antall tokens", - "context.stats.usage": "Forbruk", - "context.stats.inputTokens": "Input-tokens", - "context.stats.outputTokens": "Output-tokens", - "context.stats.reasoningTokens": "Resonnerings-tokens", - "context.stats.cacheTokens": "Cache-tokens (les/skriv)", - "context.stats.userMessages": "Brukermeldinger", - "context.stats.assistantMessages": "Assistentmeldinger", - "context.stats.totalCost": "Total kostnad", - "context.stats.sessionCreated": "Sesjon opprettet", - "context.stats.lastActivity": "Siste aktivitet", - - "context.usage.tokens": "Tokens", - "context.usage.usage": "Forbruk", - "context.usage.cost": "Kostnad", - "context.usage.clickToView": "Klikk for å se kontekst", - "context.usage.view": "Se kontekstforbruk", - - "language.en": "English", - "language.zh": "简体中文", - "language.zht": "繁體中文", - "language.ko": "한국어", - "language.de": "Deutsch", - "language.es": "Español", - "language.fr": "Français", - "language.da": "Dansk", - "language.ja": "日本語", - "language.pl": "Polski", - "language.ru": "Русский", - "language.ar": "العربية", - "language.no": "Norsk", - "language.br": "Português (Brasil)", - "language.bs": "Bosanski", - "language.th": "ไทย", - "language.tr": "Türkçe", - - "toast.language.title": "Språk", - "toast.language.description": "Byttet til {{language}}", - - "toast.theme.title": "Tema byttet", - "toast.scheme.title": "Fargevalg", - - "toast.workspace.enabled.title": "Arbeidsområder aktivert", - "toast.workspace.enabled.description": "Flere worktrees vises nå i sidefeltet", - "toast.workspace.disabled.title": "Arbeidsområder deaktivert", - "toast.workspace.disabled.description": "Kun hoved-worktree vises i sidefeltet", - - "toast.permissions.autoaccept.on.title": "Aksepterer tillatelser automatisk", - "toast.permissions.autoaccept.on.description": "Forespørsler om tillatelse vil bli godkjent automatisk", - "toast.permissions.autoaccept.off.title": "Stoppet automatisk akseptering av tillatelser", - "toast.permissions.autoaccept.off.description": "Forespørsler om tillatelse vil kreve godkjenning", - - "toast.model.none.title": "Ingen modell valgt", - "toast.model.none.description": "Koble til en leverandør for å oppsummere denne sesjonen", - - "toast.file.loadFailed.title": "Kunne ikke laste fil", - "toast.file.listFailed.title": "Kunne ikke liste filer", - - "toast.context.noLineSelection.title": "Ingen linjevalg", - "toast.context.noLineSelection.description": "Velg først et linjeområde i en filfane.", - - "toast.session.share.copyFailed.title": "Kunne ikke kopiere URL til utklippstavlen", - "toast.session.share.success.title": "Sesjon delt", - "toast.session.share.success.description": "Delings-URL kopiert til utklippstavlen!", - "toast.session.share.failed.title": "Kunne ikke dele sesjon", - "toast.session.share.failed.description": "Det oppstod en feil under deling av sesjonen", - - "toast.session.unshare.success.title": "Deling av sesjon stoppet", - "toast.session.unshare.success.description": "Sesjonen deles ikke lenger!", - "toast.session.unshare.failed.title": "Kunne ikke stoppe deling av sesjon", - "toast.session.unshare.failed.description": "Det oppstod en feil da delingen av sesjonen skulle stoppes", - - "toast.session.listFailed.title": "Kunne ikke laste sesjoner for {{project}}", - - "toast.update.title": "Oppdatering tilgjengelig", - "toast.update.description": "En ny versjon av Kilo ({{version}}) er nå tilgjengelig for installasjon.", - "toast.update.action.installRestart": "Installer og start på nytt", - "toast.update.action.notYet": "Ikke nå", - - "error.page.title": "Noe gikk galt", - "error.page.description": "Det oppstod en feil under lasting av applikasjonen.", - "error.page.details.label": "Feildetaljer", - "error.page.action.restart": "Start på nytt", - "error.page.action.checking": "Sjekker...", - "error.page.action.checkUpdates": "Se etter oppdateringer", - "error.page.action.updateTo": "Oppdater til {{version}}", - "error.page.report.prefix": "Vennligst rapporter denne feilen til Kilo-teamet", - "error.page.report.discord": "på Discord", - "error.page.version": "Versjon: {{version}}", - - "error.dev.rootNotFound": - "Rotelement ikke funnet. Glemte du å legge det til i index.html? Eller kanskje id-attributten er feilstavet?", - - "error.globalSync.connectFailed": "Kunne ikke koble til server. Kjører det en server på `{{url}}`?", - "directory.error.invalidUrl": "Invalid directory in URL.", - - "error.chain.unknown": "Ukjent feil", - "error.chain.causedBy": "Forårsaket av:", - "error.chain.apiError": "API-feil", - "error.chain.status": "Status: {{status}}", - "error.chain.retryable": "Kan prøves på nytt: {{retryable}}", - "error.chain.responseBody": "Responsinnhold:\n{{body}}", - "error.chain.didYouMean": "Mente du: {{suggestions}}", - "error.chain.modelNotFound": "Modell ikke funnet: {{provider}}/{{model}}", - "error.chain.checkConfig": "Sjekk leverandør-/modellnavnene i konfigurasjonen din (opencode.json)", - "error.chain.mcpFailed": 'MCP-server "{{name}}" mislyktes. Merk at Kilo ikke støtter MCP-autentisering ennå.', - "error.chain.providerAuthFailed": "Leverandørautentisering mislyktes ({{provider}}): {{message}}", - "error.chain.providerInitFailed": - 'Kunne ikke initialisere leverandør "{{provider}}". Sjekk legitimasjon og konfigurasjon.', - "error.chain.configJsonInvalid": "Konfigurasjonsfilen på {{path}} er ikke gyldig JSON(C)", - "error.chain.configJsonInvalidWithMessage": "Konfigurasjonsfilen på {{path}} er ikke gyldig JSON(C): {{message}}", - "error.chain.configDirectoryTypo": - 'Mappen "{{dir}}" i {{path}} er ikke gyldig. Gi mappen nytt navn til "{{suggestion}}" eller fjern den. Dette er en vanlig skrivefeil.', - "error.chain.configFrontmatterError": "Kunne ikke analysere frontmatter i {{path}}:\n{{message}}", - "error.chain.configInvalid": "Konfigurasjonsfilen på {{path}} er ugyldig", - "error.chain.configInvalidWithMessage": "Konfigurasjonsfilen på {{path}} er ugyldig: {{message}}", - - "notification.permission.title": "Tillatelse påkrevd", - "notification.permission.description": "{{sessionTitle}} i {{projectName}} trenger tillatelse", - "notification.question.title": "Spørsmål", - "notification.question.description": "{{sessionTitle}} i {{projectName}} har et spørsmål", - "notification.action.goToSession": "Gå til sesjon", - - "notification.session.responseReady.title": "Svar klart", - "notification.session.error.title": "Sesjonsfeil", - "notification.session.error.fallbackDescription": "Det oppstod en feil", - - "home.recentProjects": "Nylige prosjekter", - "home.empty.title": "Ingen nylige prosjekter", - "home.empty.description": "Kom i gang ved å åpne et lokalt prosjekt", - - "session.tab.session": "Sesjon", - "session.tab.review": "Gjennomgang", - "session.tab.context": "Kontekst", - "session.panel.reviewAndFiles": "Gjennomgang og filer", - "session.review.filesChanged": "{{count}} filer endret", - "session.review.change.one": "Endring", - "session.review.change.other": "Endringer", - "session.review.loadingChanges": "Laster endringer...", - "session.review.empty": "Ingen endringer i denne sesjonen ennå", - "session.review.noVcs": "Ingen Git-versjonskontrollsystem oppdaget, endringer vises ikke", - "session.review.noSnapshot": - "Snapshot-sporing er deaktivert i konfigurasjonen, så sesjonsendringer er ikke tilgjengelige", - "session.review.noChanges": "Ingen endringer", - - "session.files.selectToOpen": "Velg en fil å åpne", - "session.files.all": "Alle filer", - "session.files.empty": "Ingen filer", - "session.files.binaryContent": "Binær fil (innhold kan ikke vises)", - - "session.messages.renderEarlier": "Vis tidligere meldinger", - "session.messages.loadingEarlier": "Laster inn tidligere meldinger...", - "session.messages.loadEarlier": "Last inn tidligere meldinger", - "session.messages.loading": "Laster meldinger...", - "session.messages.jumpToLatest": "Hopp til nyeste", - - "session.context.addToContext": "Legg til {{selection}} i kontekst", - "session.todo.title": "Oppgaver", - "session.todo.collapse": "Skjul", - "session.todo.expand": "Utvid", - "session.followupDock.summary.one": "{{count}} melding i kø", - "session.followupDock.summary.other": "{{count}} meldinger i kø", - "session.followupDock.sendNow": "Send nå", - "session.followupDock.edit": "Rediger", - "session.followupDock.collapse": "Skjul meldinger i kø", - "session.followupDock.expand": "Utvid meldinger i kø", - "session.revertDock.summary.one": "{{count}} tilbakestilt melding", - "session.revertDock.summary.other": "{{count}} tilbakestilte meldinger", - "session.revertDock.collapse": "Skjul tilbakestilte meldinger", - "session.revertDock.expand": "Utvid tilbakestilte meldinger", - "session.revertDock.restore": "Gjenopprett melding", - - "session.new.title": "Bygg hva som helst", - "session.new.worktree.main": "Hovedgren", - "session.new.worktree.mainWithBranch": "Hovedgren ({{branch}})", - "session.new.worktree.create": "Opprett nytt worktree", - "session.new.lastModified": "Sist endret", - - "session.header.search.placeholder": "Søk i {{project}}", - "session.header.searchFiles": "Søk etter filer", - "session.header.openIn": "Åpne i", - "session.header.open.action": "Åpne {{app}}", - "session.header.open.ariaLabel": "Åpne i {{app}}", - "session.header.open.menu": "Åpne alternativer", - "session.header.open.copyPath": "Kopier bane", - - "status.popover.trigger": "Status", - "status.popover.ariaLabel": "Serverkonfigurasjoner", - "status.popover.tab.servers": "Servere", - "status.popover.tab.mcp": "MCP", - "status.popover.tab.lsp": "LSP", - "status.popover.tab.plugins": "Plugins", - "status.popover.action.manageServers": "Administrer servere", - - "session.share.popover.title": "Publiser på nett", - "session.share.popover.description.shared": - "Denne sesjonen er offentlig på nettet. Den er tilgjengelig for alle med lenken.", - "session.share.popover.description.unshared": - "Del sesjonen offentlig på nettet. Den vil være tilgjengelig for alle med lenken.", - "session.share.action.share": "Del", - "session.share.action.publish": "Publiser", - "session.share.action.publishing": "Publiserer...", - "session.share.action.unpublish": "Avpubliser", - "session.share.action.unpublishing": "Avpubliserer...", - "session.share.action.view": "Vis", - "session.share.copy.copied": "Kopiert", - "session.share.copy.copyLink": "Kopier lenke", - - "lsp.tooltip.none": "Ingen LSP-servere", - "lsp.label.connected": "{{count}} LSP", - - "prompt.loading": "Laster prompt...", - "terminal.loading": "Laster terminal...", - "terminal.title": "Terminal", - "terminal.title.numbered": "Terminal {{number}}", - "terminal.close": "Lukk terminal", - "terminal.connectionLost.title": "Tilkobling mistet", - "terminal.connectionLost.description": - "Terminalforbindelsen ble avbrutt. Dette kan skje når serveren starter på nytt.", - - "common.closeTab": "Lukk fane", - "common.dismiss": "Avvis", - "common.requestFailed": "Forespørsel mislyktes", - "common.moreOptions": "Flere alternativer", - "common.learnMore": "Lær mer", - "common.rename": "Gi nytt navn", - "common.reset": "Tilbakestill", - "common.archive": "Arkiver", - "common.delete": "Slett", - "common.close": "Lukk", - "common.edit": "Rediger", - "common.loadMore": "Last flere", - "common.key.esc": "ESC", - - "sidebar.menu.toggle": "Veksle meny", - "sidebar.nav.projectsAndSessions": "Prosjekter og sesjoner", - "sidebar.settings": "Innstillinger", - "sidebar.help": "Hjelp", - "sidebar.workspaces.enable": "Aktiver arbeidsområder", - "sidebar.workspaces.disable": "Deaktiver arbeidsområder", - "sidebar.gettingStarted.title": "Kom i gang", - "sidebar.gettingStarted.line1": "Kilo inkluderer gratis modeller så du kan starte umiddelbart.", - "sidebar.gettingStarted.line2": "Koble til en leverandør for å bruke modeller, inkl. Claude, GPT, Gemini osv.", - "sidebar.project.recentSessions": "Nylige sesjoner", - "sidebar.project.viewAllSessions": "Vis alle sesjoner", - "sidebar.project.clearNotifications": "Fjern varsler", - - "app.name.desktop": "Kilo Desktop", - - "settings.section.desktop": "Skrivebord", - "settings.section.server": "Server", - "settings.tab.general": "Generelt", - "settings.tab.shortcuts": "Snarveier", - "settings.desktop.section.wsl": "WSL", - "settings.desktop.wsl.title": "WSL-integrasjon", - "settings.desktop.wsl.description": "Kjør Kilo-serveren i WSL på Windows.", - - "settings.general.section.appearance": "Utseende", - "settings.general.section.notifications": "Systemvarsler", - "settings.general.section.updates": "Oppdateringer", - "settings.general.section.sounds": "Lydeffekter", - "settings.general.section.feed": "Feed", - "settings.general.section.display": "Skjerm", - - "settings.general.row.language.title": "Språk", - "settings.general.row.language.description": "Endre visningsspråket for Kilo", - "settings.general.row.appearance.title": "Utseende", - "settings.general.row.appearance.description": "Tilpass hvordan Kilo ser ut på enheten din", - "settings.general.row.colorScheme.title": "Fargevalg", - "settings.general.row.colorScheme.description": "Velg om Kilo skal følge systemets, lyst eller mørkt tema", - "settings.general.row.theme.title": "Tema", - "settings.general.row.theme.description": "Tilpass hvordan Kilo er tematisert.", - "settings.general.row.font.title": "Kodefont", - "settings.general.row.font.description": "Tilpass skrifttypen som brukes i kodeblokker", - "settings.general.row.terminalFont.title": "Terminal Font", - "settings.general.row.terminalFont.description": "Customise the font used in the terminal", - "settings.general.row.uiFont.title": "UI-skrift", - "settings.general.row.uiFont.description": "Tilpass skrifttypen som brukes i hele grensesnittet", - "settings.general.row.followup.title": "Oppfølgingsadferd", - "settings.general.row.followup.description": "Velg om oppfølgingsspørsmål skal kjøres umiddelbart eller vente i kø", - "settings.general.row.followup.option.queue": "Kø", - "settings.general.row.followup.option.steer": "Styr", - "settings.general.row.reasoningSummaries.title": "Vis resonneringssammendrag", - "settings.general.row.reasoningSummaries.description": "Vis sammendrag av modellresonnering i tidslinjen", - "settings.general.row.shellToolPartsExpanded.title": "Utvid shell-verktøydeler", - "settings.general.row.shellToolPartsExpanded.description": "Vis shell-verktøydeler utvidet som standard i tidslinjen", - "settings.general.row.editToolPartsExpanded.title": "Utvid edit-verktøydeler", - "settings.general.row.editToolPartsExpanded.description": - "Vis edit-, write- og patch-verktøydeler utvidet som standard i tidslinjen", - "settings.general.row.showSessionProgressBar.title": "Vis fremdriftslinje for sesjonen", - "settings.general.row.showSessionProgressBar.description": - "Vis den animerte fremdriftslinjen øverst i sesjonen når agenten jobber", - "settings.general.row.wayland.title": "Bruk innebygd Wayland", - "settings.general.row.wayland.description": "Deaktiver X11-fallback på Wayland. Krever omstart.", - "settings.general.row.wayland.tooltip": - "På Linux med skjermer med blandet oppdateringsfrekvens kan innebygd Wayland være mer stabilt.", - - "settings.general.row.releaseNotes.title": "Utgivelsesnotater", - "settings.general.row.releaseNotes.description": 'Vis "Hva er nytt"-vinduer etter oppdateringer', - - "settings.updates.row.startup.title": "Se etter oppdateringer ved oppstart", - "settings.updates.row.startup.description": "Se automatisk etter oppdateringer når Kilo starter", - "settings.updates.row.check.title": "Se etter oppdateringer", - "settings.updates.row.check.description": "Se etter oppdateringer manuelt og installer hvis tilgjengelig", - "settings.updates.action.checkNow": "Sjekk nå", - "settings.updates.action.checking": "Sjekker...", - "settings.updates.toast.latest.title": "Du er oppdatert", - "settings.updates.toast.latest.description": "Du bruker den nyeste versjonen av Kilo.", - "sound.option.none": "Ingen", - "sound.option.alert01": "Varsel 01", - "sound.option.alert02": "Varsel 02", - "sound.option.alert03": "Varsel 03", - "sound.option.alert04": "Varsel 04", - "sound.option.alert05": "Varsel 05", - "sound.option.alert06": "Varsel 06", - "sound.option.alert07": "Varsel 07", - "sound.option.alert08": "Varsel 08", - "sound.option.alert09": "Varsel 09", - "sound.option.alert10": "Varsel 10", - "sound.option.bipbop01": "Bip-bop 01", - "sound.option.bipbop02": "Bip-bop 02", - "sound.option.bipbop03": "Bip-bop 03", - "sound.option.bipbop04": "Bip-bop 04", - "sound.option.bipbop05": "Bip-bop 05", - "sound.option.bipbop06": "Bip-bop 06", - "sound.option.bipbop07": "Bip-bop 07", - "sound.option.bipbop08": "Bip-bop 08", - "sound.option.bipbop09": "Bip-bop 09", - "sound.option.bipbop10": "Bip-bop 10", - "sound.option.staplebops01": "Staplebops 01", - "sound.option.staplebops02": "Staplebops 02", - "sound.option.staplebops03": "Staplebops 03", - "sound.option.staplebops04": "Staplebops 04", - "sound.option.staplebops05": "Staplebops 05", - "sound.option.staplebops06": "Staplebops 06", - "sound.option.staplebops07": "Staplebops 07", - "sound.option.nope01": "Nei 01", - "sound.option.nope02": "Nei 02", - "sound.option.nope03": "Nei 03", - "sound.option.nope04": "Nei 04", - "sound.option.nope05": "Nei 05", - "sound.option.nope06": "Nei 06", - "sound.option.nope07": "Nei 07", - "sound.option.nope08": "Nei 08", - "sound.option.nope09": "Nei 09", - "sound.option.nope10": "Nei 10", - "sound.option.nope11": "Nei 11", - "sound.option.nope12": "Nei 12", - "sound.option.yup01": "Ja 01", - "sound.option.yup02": "Ja 02", - "sound.option.yup03": "Ja 03", - "sound.option.yup04": "Ja 04", - "sound.option.yup05": "Ja 05", - "sound.option.yup06": "Ja 06", - - "settings.general.notifications.agent.title": "Agent", - "settings.general.notifications.agent.description": - "Vis systemvarsel når agenten er ferdig eller trenger oppmerksomhet", - "settings.general.notifications.permissions.title": "Tillatelser", - "settings.general.notifications.permissions.description": "Vis systemvarsel når en tillatelse er påkrevd", - "settings.general.notifications.errors.title": "Feil", - "settings.general.notifications.errors.description": "Vis systemvarsel når det oppstår en feil", - - "settings.general.sounds.agent.title": "Agent", - "settings.general.sounds.agent.description": "Spill av lyd når agenten er ferdig eller trenger oppmerksomhet", - "settings.general.sounds.permissions.title": "Tillatelser", - "settings.general.sounds.permissions.description": "Spill av lyd når en tillatelse er påkrevd", - "settings.general.sounds.errors.title": "Feil", - "settings.general.sounds.errors.description": "Spill av lyd når det oppstår en feil", - - "settings.shortcuts.title": "Tastatursnarveier", - "settings.shortcuts.reset.button": "Tilbakestill til standard", - "settings.shortcuts.reset.toast.title": "Snarveier tilbakestilt", - "settings.shortcuts.reset.toast.description": "Tastatursnarveier er tilbakestilt til standard.", - "settings.shortcuts.conflict.title": "Snarvei allerede i bruk", - "settings.shortcuts.conflict.description": "{{keybind}} er allerede tilordnet til {{titles}}.", - "settings.shortcuts.unassigned": "Ikke tilordnet", - "settings.shortcuts.pressKeys": "Trykk taster", - "settings.shortcuts.search.placeholder": "Søk etter snarveier", - "settings.shortcuts.search.empty": "Ingen snarveier funnet", - - "settings.shortcuts.group.general": "Generelt", - "settings.shortcuts.group.session": "Sesjon", - "settings.shortcuts.group.navigation": "Navigasjon", - "settings.shortcuts.group.modelAndAgent": "Modell og agent", - "settings.shortcuts.group.terminal": "Terminal", - "settings.shortcuts.group.prompt": "Prompt", - - "settings.providers.title": "Leverandører", - "settings.providers.description": "Leverandørinnstillinger vil kunne konfigureres her.", - "settings.providers.section.connected": "Tilkoblede leverandører", - "settings.providers.connected.empty": "Ingen tilkoblede leverandører", - "settings.providers.section.popular": "Populære leverandører", - "settings.providers.tag.environment": "Miljø", - "settings.providers.tag.config": "Konfigurasjon", - "settings.providers.tag.custom": "Tilpasset", - "settings.providers.tag.other": "Annet", - "settings.models.title": "Modeller", - "settings.models.description": "Modellinnstillinger vil kunne konfigureres her.", - "settings.agents.title": "Agenter", - "settings.agents.description": "Agentinnstillinger vil kunne konfigureres her.", - "settings.commands.title": "Kommandoer", - "settings.commands.description": "Kommandoinnstillinger vil kunne konfigureres her.", - "settings.mcp.title": "MCP", - "settings.mcp.description": "MCP-innstillinger vil kunne konfigureres her.", - - "settings.permissions.title": "Tillatelser", - "settings.permissions.description": "Kontroller hvilke verktøy serveren kan bruke som standard.", - "settings.permissions.section.tools": "Verktøy", - "settings.permissions.toast.updateFailed.title": "Kunne ikke oppdatere tillatelser", - - "settings.permissions.action.allow": "Tillat", - "settings.permissions.action.ask": "Spør", - "settings.permissions.action.deny": "Avslå", - - "settings.permissions.tool.read.title": "Les", - "settings.permissions.tool.read.description": "Lesing av en fil (matcher filbanen)", - "settings.permissions.tool.edit.title": "Rediger", - "settings.permissions.tool.edit.description": - "Endre filer, inkludert redigeringer, skriving, patcher og multi-redigeringer", - "settings.permissions.tool.glob.title": "Glob", - "settings.permissions.tool.glob.description": "Match filer ved hjelp av glob-mønstre", - "settings.permissions.tool.grep.title": "Grep", - "settings.permissions.tool.grep.description": "Søk i filinnhold ved hjelp av regulære uttrykk", - "settings.permissions.tool.list.title": "Liste", - "settings.permissions.tool.list.description": "List filer i en mappe", - "settings.permissions.tool.bash.title": "Bash", - "settings.permissions.tool.bash.description": "Kjør shell-kommandoer", - "settings.permissions.tool.task.title": "Oppgave", - "settings.permissions.tool.task.description": "Start underagenter", - "settings.permissions.tool.skill.title": "Ferdighet", - "settings.permissions.tool.skill.description": "Last en ferdighet etter navn", - "settings.permissions.tool.lsp.title": "LSP", - "settings.permissions.tool.lsp.description": "Kjør språkserverforespørsler", - "settings.permissions.tool.todowrite.title": "Skriv gjøremål", - "settings.permissions.tool.todowrite.description": "Oppdater gjøremålslisten", - "settings.permissions.tool.webfetch.title": "Webhenting", - "settings.permissions.tool.webfetch.description": "Hent innhold fra en URL", - "settings.permissions.tool.websearch.title": "Websøk", - "settings.permissions.tool.websearch.description": "Søk på nettet", - "settings.permissions.tool.codesearch.title": "Kodesøk", - "settings.permissions.tool.codesearch.description": "Søk etter kode på nettet", - "settings.permissions.tool.external_directory.title": "Ekstern mappe", - "settings.permissions.tool.external_directory.description": "Få tilgang til filer utenfor prosjektmappen", - "settings.permissions.tool.doom_loop.title": "Doom Loop", - "settings.permissions.tool.doom_loop.description": "Oppdager gjentatte verktøykall med identisk input", - - "session.delete.failed.title": "Kunne ikke slette sesjon", - "session.delete.title": "Slett sesjon", - "session.delete.confirm": 'Slette sesjonen "{{name}}"?', - "session.delete.button": "Slett sesjon", - - "workspace.new": "Nytt arbeidsområde", - "workspace.type.local": "lokal", - "workspace.type.sandbox": "sandkasse", - "workspace.create.failed.title": "Kunne ikke opprette arbeidsområde", - "workspace.delete.failed.title": "Kunne ikke slette arbeidsområde", - "workspace.resetting.title": "Tilbakestiller arbeidsområde", - "workspace.resetting.description": "Dette kan ta et minutt.", - "workspace.reset.failed.title": "Kunne ikke tilbakestille arbeidsområde", - "workspace.reset.success.title": "Arbeidsområde tilbakestilt", - "workspace.reset.success.description": "Arbeidsområdet samsvarer nå med standardgrenen.", - "workspace.error.stillPreparing": "Arbeidsområdet klargjøres fortsatt", - "workspace.status.checking": "Sjekker for ikke-sammenslåtte endringer...", - "workspace.status.error": "Kunne ikke bekrefte git-status.", - "workspace.status.clean": "Ingen ikke-sammenslåtte endringer oppdaget.", - "workspace.status.dirty": "Ikke-sammenslåtte endringer oppdaget i dette arbeidsområdet.", - "workspace.delete.title": "Slett arbeidsområde", - "workspace.delete.confirm": 'Slette arbeidsområdet "{{name}}"?', - "workspace.delete.button": "Slett arbeidsområde", - "workspace.reset.title": "Tilbakestill arbeidsområde", - "workspace.reset.confirm": 'Tilbakestille arbeidsområdet "{{name}}"?', - "workspace.reset.button": "Tilbakestill arbeidsområde", - "workspace.reset.archived.none": "Ingen aktive sesjoner vil bli arkivert.", - "workspace.reset.archived.one": "1 sesjon vil bli arkivert.", - "workspace.reset.archived.many": "{{count}} sesjoner vil bli arkivert.", - "workspace.reset.note": "Dette vil tilbakestille arbeidsområdet til å samsvare med standardgrenen.", - "common.open": "Åpne", - "dialog.releaseNotes.action.getStarted": "Kom i gang", - "dialog.releaseNotes.action.next": "Neste", - "dialog.releaseNotes.action.hideFuture": "Ikke vis disse igjen", - "dialog.releaseNotes.media.alt": "Forhåndsvisning av utgivelse", - "toast.project.reloadFailed.title": "Kunne ikke laste inn {{project}} på nytt", - "error.server.invalidConfiguration": "Ugyldig konfigurasjon", - "common.moreCountSuffix": " (+{{count}} mer)", - "common.time.justNow": "Akkurat nå", - "common.time.minutesAgo.short": "{{count}} m siden", - "common.time.hoursAgo.short": "{{count}} t siden", - "common.time.daysAgo.short": "{{count}} d siden", - "settings.providers.connected.environmentDescription": "Koblet til fra miljøvariablene dine", - "settings.providers.custom.description": "Legg til en OpenAI-kompatibel leverandør via basis-URL.", - - "app.server.unreachable": "Kunne ikke nå {{server}}", - "app.server.retrying": "Prøver på nytt automatisk...", - "app.server.otherServers": "Andre servere", - "dialog.server.add.usernamePlaceholder": "brukernavn", - "dialog.server.add.passwordPlaceholder": "passord", - "server.row.noUsername": "inget brukernavn", - "session.review.noVcs.createGit.title": "Opprett et Git-depot", - "session.review.noVcs.createGit.description": "Spor, gjennomgå og angre endringer i dette prosjektet", - "session.review.noVcs.createGit.actionLoading": "Oppretter Git-depot...", - "session.review.noVcs.createGit.action": "Opprett Git-depot", - "session.todo.progress": "{{done}} av {{total}} oppgaver fullført", - "session.question.progress": "{{current}} av {{total}} spørsmål", - "session.header.open.finder": "Finder", - "session.header.open.fileExplorer": "Filutforsker", - "session.header.open.fileManager": "Filbehandler", - "session.header.open.app.vscode": "VS Code", - "session.header.open.app.cursor": "Cursor", - "session.header.open.app.zed": "Zed", - "session.header.open.app.textmate": "TextMate", - "session.header.open.app.antigravity": "Antigravity", - "session.header.open.app.terminal": "Terminal", - "session.header.open.app.iterm2": "iTerm2", - "session.header.open.app.ghostty": "Ghostty", - "session.header.open.app.warp": "Warp", - "session.header.open.app.xcode": "Xcode", - "session.header.open.app.androidStudio": "Android Studio", - "session.header.open.app.powershell": "PowerShell", - "session.header.open.app.sublimeText": "Sublime Text", - "debugBar.ariaLabel": "Utviklingsytelsesdiagnostikk", - "debugBar.na": "i/t", - "debugBar.nav.label": "NAV", - "debugBar.nav.tip": - "Siste fullførte ruteovergang som berører en sesjonsside, målt fra ruterstart til første opptegning etter at den har roet seg.", - "debugBar.fps.label": "FPS", - "debugBar.fps.tip": "Rullende bilder per sekund over de siste 5 sekundene.", - "debugBar.frame.label": "FRAME", - "debugBar.frame.tip": "Verste bildetid over de siste 5 sekundene.", - "debugBar.jank.label": "JANK", - "debugBar.jank.tip": "Bilder over 32ms i de siste 5 sekundene.", - "debugBar.long.label": "LONG", - "debugBar.long.tip": "Blokkert tid og antall lange oppgaver i de siste 5 sekundene. Maks oppgave: {{max}}.", - "debugBar.delay.label": "DELAY", - "debugBar.delay.tip": "Verste observerte inndataforsinkelse i de siste 5 sekundene.", - "debugBar.inp.label": "INP", - "debugBar.inp.tip": - "Omtrentlig interaksjonsvarighet over de siste 5 sekundene. Dette er INP-lignende, ikke den offisielle Web Vitals INP.", - "debugBar.cls.label": "CLS", - "debugBar.cls.tip": "Kumulativ layoutforskyvning for gjeldende app-levetid.", - "debugBar.mem.label": "MEM", - "debugBar.mem.tipUnavailable": "Brukt JS-heap vs heap-grense. Kun Chromium.", - "debugBar.mem.tip": "Brukt JS-heap vs heap-grense. {{used}} av {{limit}}.", - "common.key.ctrl": "Ctrl", - "common.key.alt": "Alt", - "common.key.shift": "Shift", - "common.key.meta": "Meta", - "common.key.space": "Mellomrom", - "common.key.backspace": "Backspace", - "common.key.enter": "Enter", - "common.key.tab": "Tab", - "common.key.delete": "Delete", - "common.key.home": "Home", - "common.key.end": "End", - "common.key.pageUp": "Page Up", - "common.key.pageDown": "Page Down", - "common.key.insert": "Insert", - "common.unknown": "ukjent", - "error.page.circular": "[Sirkulær]", - "error.globalSDK.noServerAvailable": "Ingen server tilgjengelig", - "error.globalSDK.serverNotAvailable": "Server ikke tilgjengelig", - "error.childStore.persistedCacheCreateFailed": "Kunne ikke opprette vedvarende hurtigbuffer", - "error.childStore.persistedProjectMetadataCreateFailed": "Kunne ikke opprette vedvarende prosjektmetadata", - "error.childStore.persistedProjectIconCreateFailed": "Kunne ikke opprette vedvarende prosjektikon", - "error.childStore.storeCreateFailed": "Kunne ikke opprette lager", - "terminal.connectionLost.abnormalClose": "WebSocket lukket unormalt: {{code}}", -} satisfies Partial> diff --git a/packages/app/src/i18n/parity.test.ts b/packages/app/src/i18n/parity.test.ts deleted file mode 100644 index c06a55ab17..0000000000 --- a/packages/app/src/i18n/parity.test.ts +++ /dev/null @@ -1,32 +0,0 @@ -import { describe, expect, test } from "bun:test" -import { dict as en } from "./en" -import { dict as ar } from "./ar" -import { dict as br } from "./br" -import { dict as bs } from "./bs" -import { dict as da } from "./da" -import { dict as de } from "./de" -import { dict as es } from "./es" -import { dict as fr } from "./fr" -import { dict as ja } from "./ja" -import { dict as ko } from "./ko" -import { dict as no } from "./no" -import { dict as pl } from "./pl" -import { dict as ru } from "./ru" -import { dict as th } from "./th" -import { dict as zh } from "./zh" -import { dict as zht } from "./zht" -import { dict as tr } from "./tr" - -const locales = [ar, br, bs, da, de, es, fr, ja, ko, no, pl, ru, th, tr, zh, zht] -const keys = ["command.session.previous.unseen", "command.session.next.unseen"] as const - -describe("i18n parity", () => { - test("non-English locales translate targeted unseen session keys", () => { - for (const locale of locales) { - for (const key of keys) { - expect(locale[key]).toBeDefined() - expect(locale[key]).not.toBe(en[key]) - } - } - }) -}) diff --git a/packages/app/src/i18n/pl.ts b/packages/app/src/i18n/pl.ts deleted file mode 100644 index 49e7e12d99..0000000000 --- a/packages/app/src/i18n/pl.ts +++ /dev/null @@ -1,857 +0,0 @@ -export const dict = { - "command.category.suggested": "Sugerowane", - "command.category.view": "Widok", - "command.category.project": "Projekt", - "command.category.provider": "Dostawca", - "command.category.server": "Serwer", - "command.category.session": "Sesja", - "command.category.theme": "Motyw", - "command.category.language": "Język", - "command.category.file": "Plik", - "command.category.context": "Kontekst", - "command.category.terminal": "Terminal", - "command.category.model": "Model", - "command.category.mcp": "MCP", - "command.category.agent": "Agent", - "command.category.permissions": "Uprawnienia", - "command.category.workspace": "Przestrzeń robocza", - "command.category.settings": "Ustawienia", - "theme.scheme.system": "Systemowy", - "theme.scheme.light": "Jasny", - "theme.scheme.dark": "Ciemny", - "command.sidebar.toggle": "Przełącz pasek boczny", - "command.project.open": "Otwórz projekt", - "command.provider.connect": "Połącz dostawcę", - "command.server.switch": "Przełącz serwer", - "command.settings.open": "Otwórz ustawienia", - "command.session.previous": "Poprzednia sesja", - "command.session.next": "Następna sesja", - "command.session.previous.unseen": "Poprzednia nieprzeczytana sesja", - "command.session.next.unseen": "Następna nieprzeczytana sesja", - "command.session.archive": "Zarchiwizuj sesję", - "command.palette": "Paleta poleceń", - "command.theme.cycle": "Przełącz motyw", - "command.theme.set": "Użyj motywu: {{theme}}", - "command.theme.scheme.cycle": "Przełącz schemat kolorów", - "command.theme.scheme.set": "Użyj schematu kolorów: {{scheme}}", - "command.language.cycle": "Przełącz język", - "command.language.set": "Użyj języka: {{language}}", - "command.session.new": "Nowa sesja", - "command.file.open": "Otwórz plik", - "command.tab.close": "Zamknij kartę", - "command.context.addSelection": "Dodaj zaznaczenie do kontekstu", - "command.context.addSelection.description": "Dodaj zaznaczone linie z bieżącego pliku", - "command.input.focus": "Fokus na pole wejściowe", - "command.terminal.toggle": "Przełącz terminal", - "command.fileTree.toggle": "Przełącz drzewo plików", - "command.review.toggle": "Przełącz przegląd", - "command.terminal.new": "Nowy terminal", - "command.terminal.new.description": "Utwórz nową kartę terminala", - "command.steps.toggle": "Przełącz kroki", - "command.steps.toggle.description": "Pokaż lub ukryj kroki dla bieżącej wiadomości", - "command.message.previous": "Poprzednia wiadomość", - "command.message.previous.description": "Przejdź do poprzedniej wiadomości użytkownika", - "command.message.next": "Następna wiadomość", - "command.message.next.description": "Przejdź do następnej wiadomości użytkownika", - "command.model.choose": "Wybierz model", - "command.model.choose.description": "Wybierz inny model", - "command.mcp.toggle": "Przełącz MCP", - "command.mcp.toggle.description": "Przełącz MCP", - "command.agent.cycle": "Przełącz agenta", - "command.agent.cycle.description": "Przełącz na następnego agenta", - "command.agent.cycle.reverse": "Przełącz agenta wstecz", - "command.agent.cycle.reverse.description": "Przełącz na poprzedniego agenta", - "command.model.variant.cycle": "Przełącz wysiłek myślowy", - "command.model.variant.cycle.description": "Przełącz na następny poziom wysiłku", - "command.prompt.mode.shell": "Terminal", - "command.prompt.mode.normal": "Prompt", - "command.permissions.autoaccept.enable": "Automatycznie akceptuj uprawnienia", - "command.permissions.autoaccept.disable": "Zatrzymaj automatyczne akceptowanie uprawnień", - "command.workspace.toggle": "Przełącz przestrzenie robocze", - "command.workspace.toggle.description": "Włącz lub wyłącz wiele przestrzeni roboczych na pasku bocznym", - "command.session.undo": "Cofnij", - "command.session.undo.description": "Cofnij ostatnią wiadomość", - "command.session.redo": "Ponów", - "command.session.redo.description": "Ponów ostatnią cofniętą wiadomość", - "command.session.compact": "Kompaktuj sesję", - "command.session.compact.description": "Podsumuj sesję, aby zmniejszyć rozmiar kontekstu", - "command.session.fork": "Rozwidlij od wiadomości", - "command.session.fork.description": "Utwórz nową sesję od poprzedniej wiadomości", - "command.session.share": "Udostępnij sesję", - "command.session.share.description": "Udostępnij tę sesję i skopiuj URL do schowka", - "command.session.unshare": "Przestań udostępniać sesję", - "command.session.unshare.description": "Zatrzymaj udostępnianie tej sesji", - "palette.search.placeholder": "Szukaj plików, poleceń i sesji", - "palette.empty": "Brak wyników", - "palette.group.commands": "Polecenia", - "palette.group.files": "Pliki", - "dialog.provider.search.placeholder": "Szukaj dostawców", - "dialog.provider.empty": "Nie znaleziono dostawców", - "dialog.provider.group.popular": "Popularne", - "dialog.provider.group.other": "Inne", - "dialog.provider.tag.recommended": "Zalecane", - "dialog.provider.opencode.note": "Wyselekcjonowane modele, w tym Claude, GPT, Gemini i inne", - "dialog.provider.opencode.tagline": "Niezawodne, zoptymalizowane modele", - "dialog.provider.opencodeGo.tagline": "Tania subskrypcja dla każdego", - "dialog.provider.anthropic.note": "Bezpośredni dostęp do modeli Claude, w tym Pro i Max", - "dialog.provider.copilot.note": "Modele AI do pomocy w kodowaniu przez GitHub Copilot", - "dialog.provider.openai.note": "Modele GPT do szybkich i wszechstronnych zadań AI", - "dialog.provider.google.note": "Modele Gemini do szybkich i ustrukturyzowanych odpowiedzi", - "dialog.provider.openrouter.note": "Dostęp do wszystkich obsługiwanych modeli od jednego dostawcy", - "dialog.provider.vercel.note": "Ujednolicony dostęp do modeli AI z inteligentnym routingiem", - "dialog.model.select.title": "Wybierz model", - "dialog.model.search.placeholder": "Szukaj modeli", - "dialog.model.empty": "Brak wyników modelu", - "dialog.model.manage": "Zarządzaj modelami", - "dialog.model.manage.description": "Dostosuj, które modele pojawiają się w wyborze modelu.", - "dialog.model.manage.provider.toggle": "Przełącz wszystkie modele {{provider}}", - "dialog.model.unpaid.freeModels.title": "Darmowe modele dostarczane przez Kilo", - "dialog.model.unpaid.addMore.title": "Dodaj więcej modeli od popularnych dostawców", - "dialog.provider.viewAll": "Zobacz więcej dostawców", - "provider.connect.title": "Połącz {{provider}}", - "provider.connect.title.anthropicProMax": "Zaloguj się z Claude Pro/Max", - "provider.connect.selectMethod": "Wybierz metodę logowania dla {{provider}}.", - "provider.connect.method.apiKey": "Klucz API", - "provider.connect.status.inProgress": "Autoryzacja w toku...", - "provider.connect.status.waiting": "Oczekiwanie na autoryzację...", - "provider.connect.status.failed": "Autoryzacja nie powiodła się: {{error}}", - "provider.connect.apiKey.description": - "Wprowadź swój klucz API {{provider}}, aby połączyć konto i używać modeli {{provider}} w Kilo.", - "provider.connect.apiKey.label": "Klucz API {{provider}}", - "provider.connect.apiKey.placeholder": "Klucz API", - "provider.connect.apiKey.required": "Klucz API jest wymagany", - "provider.connect.opencodeZen.line1": - "OpenCode Zen daje dostęp do wybranego zestawu niezawodnych, zoptymalizowanych modeli dla agentów kodujących.", - "provider.connect.opencodeZen.line2": - "Z jednym kluczem API uzyskasz dostęp do modeli takich jak Claude, GPT, Gemini, GLM i więcej.", - "provider.connect.opencodeZen.visit.prefix": "Odwiedź ", - "provider.connect.opencodeZen.visit.link": "opencode.ai/zen", - "provider.connect.opencodeZen.visit.suffix": ", aby odebrać swój klucz API.", - "provider.connect.oauth.code.visit.prefix": "Odwiedź ", - "provider.connect.oauth.code.visit.link": "ten link", - "provider.connect.oauth.code.visit.suffix": - ", aby odebrać kod autoryzacyjny, połączyć konto i używać modeli {{provider}} w Kilo.", - "provider.connect.oauth.code.label": "Kod autoryzacyjny {{method}}", - "provider.connect.oauth.code.placeholder": "Kod autoryzacyjny", - "provider.connect.oauth.code.required": "Kod autoryzacyjny jest wymagany", - "provider.connect.oauth.code.invalid": "Nieprawidłowy kod autoryzacyjny", - "provider.connect.oauth.auto.visit.prefix": "Odwiedź ", - "provider.connect.oauth.auto.visit.link": "ten link", - "provider.connect.oauth.auto.visit.suffix": - " i wprowadź poniższy kod, aby połączyć konto i używać modeli {{provider}} w Kilo.", - "provider.connect.oauth.auto.confirmationCode": "Kod potwierdzający", - "provider.connect.toast.connected.title": "Połączono {{provider}}", - "provider.connect.toast.connected.description": "Modele {{provider}} są teraz dostępne do użycia.", - "provider.custom.title": "Dostawca niestandardowy", - "provider.custom.description.prefix": "Skonfiguruj dostawcę zgodnego z OpenAI. Zobacz ", - "provider.custom.description.link": "dokumentację konfiguracji dostawcy", - "provider.custom.description.suffix": ".", - "provider.custom.field.providerID.label": "ID dostawcy", - "provider.custom.field.providerID.placeholder": "mojdostawca", - "provider.custom.field.providerID.description": "Małe litery, cyfry, łączniki lub podkreślenia", - "provider.custom.field.name.label": "Nazwa wyświetlana", - "provider.custom.field.name.placeholder": "Mój Dostawca AI", - "provider.custom.field.baseURL.label": "Bazowy URL", - "provider.custom.field.baseURL.placeholder": "https://api.mojdostawca.com/v1", - "provider.custom.field.apiKey.label": "Klucz API", - "provider.custom.field.apiKey.placeholder": "Klucz API", - "provider.custom.field.apiKey.description": - "Opcjonalne. Pozostaw puste, jeśli zarządzasz autoryzacją przez nagłówki.", - "provider.custom.models.label": "Modele", - "provider.custom.models.id.label": "ID", - "provider.custom.models.id.placeholder": "model-id", - "provider.custom.models.name.label": "Nazwa", - "provider.custom.models.name.placeholder": "Nazwa wyświetlana", - "provider.custom.models.remove": "Usuń model", - "provider.custom.models.add": "Dodaj model", - "provider.custom.headers.label": "Nagłówki (opcjonalne)", - "provider.custom.headers.key.label": "Nagłówek", - "provider.custom.headers.key.placeholder": "Nazwa-Naglowka", - "provider.custom.headers.value.label": "Wartość", - "provider.custom.headers.value.placeholder": "wartość", - "provider.custom.headers.remove": "Usuń nagłówek", - "provider.custom.headers.add": "Dodaj nagłówek", - "provider.custom.error.providerID.required": "ID dostawcy jest wymagane", - "provider.custom.error.providerID.format": "Używaj małych liter, cyfr, łączników lub podkreśleń", - "provider.custom.error.providerID.exists": "To ID dostawcy już istnieje", - "provider.custom.error.name.required": "Nazwa wyświetlana jest wymagana", - "provider.custom.error.baseURL.required": "Bazowy URL jest wymagany", - "provider.custom.error.baseURL.format": "Musi zaczynać się od http:// lub https://", - "provider.custom.error.required": "Wymagane", - "provider.custom.error.duplicate": "Duplikat", - "provider.disconnect.toast.disconnected.title": "Rozłączono {{provider}}", - "provider.disconnect.toast.disconnected.description": "Modele {{provider}} nie są już dostępne.", - "model.tag.free": "Darmowy", - "model.tag.latest": "Najnowszy", - "model.provider.anthropic": "Anthropic", - "model.provider.openai": "OpenAI", - "model.provider.google": "Google", - "model.provider.xai": "xAI", - "model.provider.meta": "Meta", - "model.input.text": "tekst", - "model.input.image": "obraz", - "model.input.audio": "audio", - "model.input.video": "wideo", - "model.input.pdf": "pdf", - "model.tooltip.allows": "Obsługuje: {{inputs}}", - "model.tooltip.reasoning.allowed": "Obsługuje wnioskowanie", - "model.tooltip.reasoning.none": "Brak wnioskowania", - "model.tooltip.context": "Limit kontekstu {{limit}}", - "common.search.placeholder": "Szukaj", - "common.goBack": "Wstecz", - "common.goForward": "Dalej", - "common.loading": "Ładowanie", - "common.loading.ellipsis": "...", - "common.cancel": "Anuluj", - "common.connect": "Połącz", - "common.disconnect": "Rozłącz", - "common.continue": "Prześlij", - "common.submit": "Prześlij", - "common.save": "Zapisz", - "common.saving": "Zapisywanie...", - "common.default": "Domyślny", - "common.attachment": "załącznik", - "prompt.placeholder.shell": "Wpisz polecenie terminala... {{example}}", - "prompt.placeholder.normal": 'Zapytaj o cokolwiek... "{{example}}"', - "prompt.placeholder.simple": "Zapytaj o cokolwiek...", - "prompt.placeholder.summarizeComments": "Podsumuj komentarze…", - "prompt.placeholder.summarizeComment": "Podsumuj komentarz…", - "prompt.mode.shell": "Terminal", - "prompt.mode.normal": "Prompt", - "prompt.mode.shell.exit": "esc aby wyjść", - "prompt.example.1": "Napraw TODO w bazie kodu", - "prompt.example.2": "Jaki jest stos technologiczny tego projektu?", - "prompt.example.3": "Napraw zepsute testy", - "prompt.example.4": "Wyjaśnij jak działa uwierzytelnianie", - "prompt.example.5": "Znajdź i napraw luki w zabezpieczeniach", - "prompt.example.6": "Dodaj testy jednostkowe dla serwisu użytkownika", - "prompt.example.7": "Zrefaktoryzuj tę funkcję, aby była bardziej czytelna", - "prompt.example.8": "Co oznacza ten błąd?", - "prompt.example.9": "Pomóż mi zdebugować ten problem", - "prompt.example.10": "Wygeneruj dokumentację API", - "prompt.example.11": "Zoptymalizuj zapytania do bazy danych", - "prompt.example.12": "Dodaj walidację danych wejściowych", - "prompt.example.13": "Utwórz nowy komponent dla...", - "prompt.example.14": "Jak wdrożyć ten projekt?", - "prompt.example.15": "Sprawdź mój kod pod kątem najlepszych praktyk", - "prompt.example.16": "Dodaj obsługę błędów do tej funkcję", - "prompt.example.17": "Wyjaśnij ten wzorzec regex", - "prompt.example.18": "Przekonwertuj to na TypeScript", - "prompt.example.19": "Dodaj logowanie w całej bazie kodu", - "prompt.example.20": "Które zależności są przestarzałe?", - "prompt.example.21": "Pomóż mi napisać skrypt migracyjny", - "prompt.example.22": "Zaimplementuj cachowanie dla tego punktu końcowego", - "prompt.example.23": "Dodaj stronicowanie do tej listy", - "prompt.example.24": "Utwórz polecenie CLI dla...", - "prompt.example.25": "Jak działają tutaj zmienne środowiskowe?", - "prompt.popover.emptyResults": "Brak pasujących wyników", - "prompt.popover.emptyCommands": "Brak pasujących poleceń", - "prompt.dropzone.label": "Upuść tutaj obrazy, pliki PDF lub pliki tekstowe", - "prompt.dropzone.file.label": "Upuść, aby @wspomnieć plik", - "prompt.slash.badge.custom": "własne", - "prompt.slash.badge.skill": "skill", - "prompt.slash.badge.mcp": "mcp", - "prompt.context.active": "aktywny", - "prompt.context.includeActiveFile": "Dołącz aktywny plik", - "prompt.context.removeActiveFile": "Usuń aktywny plik z kontekstu", - "prompt.context.removeFile": "Usuń plik z kontekstu", - "prompt.action.attachFile": "Załącz plik", - "prompt.attachment.remove": "Usuń załącznik", - "prompt.action.send": "Wyślij", - "prompt.action.stop": "Zatrzymaj", - "prompt.toast.pasteUnsupported.title": "Nieobsługiwany załącznik", - "prompt.toast.pasteUnsupported.description": "Można tutaj załączać tylko obrazy, pliki PDF lub pliki tekstowe.", - "prompt.toast.modelAgentRequired.title": "Wybierz agenta i model", - "prompt.toast.modelAgentRequired.description": "Wybierz agenta i model przed wysłaniem zapytania.", - "prompt.toast.worktreeCreateFailed.title": "Nie udało się utworzyć drzewa roboczego", - "prompt.toast.sessionCreateFailed.title": "Nie udało się utworzyć sesji", - "prompt.toast.shellSendFailed.title": "Nie udało się wysłać polecenia powłoki", - "prompt.toast.commandSendFailed.title": "Nie udało się wysłać polecenia", - "prompt.toast.promptSendFailed.title": "Nie udało się wysłać zapytania", - "prompt.toast.promptSendFailed.description": "Nie udało się pobrać sesji", - "dialog.mcp.title": "MCP", - "dialog.mcp.description": "{{enabled}} z {{total}} włączone", - "dialog.mcp.empty": "Brak skonfigurowanych MCP", - "dialog.lsp.empty": "LSP wykryte automatycznie na podstawie typów plików", - "dialog.plugins.empty": "Wtyczki skonfigurowane w opencode.json", - "mcp.status.connected": "połączono", - "mcp.status.failed": "niepowodzenie", - "mcp.status.needs_auth": "wymaga autoryzacji", - "mcp.status.disabled": "wyłączone", - "dialog.fork.empty": "Brak wiadomości do rozwidlenia", - "dialog.directory.search.placeholder": "Szukaj folderów", - "dialog.directory.empty": "Nie znaleziono folderów", - "dialog.server.title": "Serwery", - "dialog.server.description": "Przełącz serwer Kilo, z którym łączy się ta aplikacja.", - "dialog.server.search.placeholder": "Szukaj serwerów", - "dialog.server.empty": "Brak serwerów", - "dialog.server.add.title": "Dodaj serwer", - "dialog.server.add.url": "URL serwera", - "dialog.server.add.placeholder": "http://localhost:4096", - "dialog.server.add.error": "Nie można połączyć się z serwerem", - "dialog.server.add.checking": "Sprawdzanie...", - "dialog.server.add.button": "Dodaj serwer", - "dialog.server.add.name": "Nazwa serwera (opcjonalnie)", - "dialog.server.add.namePlaceholder": "Localhost", - "dialog.server.add.username": "Nazwa użytkownika (opcjonalnie)", - "dialog.server.add.password": "Hasło (opcjonalnie)", - "dialog.server.edit.title": "Edytuj serwer", - "dialog.server.default.title": "Domyślny serwer", - "dialog.server.default.description": - "Połącz z tym serwerem przy uruchomieniu aplikacji zamiast uruchamiać lokalny serwer. Wymaga restartu.", - "dialog.server.default.none": "Nie wybrano serwera", - "dialog.server.default.set": "Ustaw bieżący serwer jako domyślny", - "dialog.server.default.clear": "Wyczyść", - "dialog.server.action.remove": "Usuń serwer", - "dialog.server.menu.edit": "Edytuj", - "dialog.server.menu.default": "Ustaw jako domyślny", - "dialog.server.menu.defaultRemove": "Usuń domyślny", - "dialog.server.menu.delete": "Usuń", - "dialog.server.current": "Obecny serwer", - "dialog.server.status.default": "Domyślny", - "dialog.project.edit.title": "Edytuj projekt", - "dialog.project.edit.name": "Nazwa", - "dialog.project.edit.icon": "Ikona", - "dialog.project.edit.icon.alt": "Ikona projektu", - "dialog.project.edit.icon.hint": "Kliknij lub przeciągnij obraz", - "dialog.project.edit.icon.recommended": "Zalecane: 128x128px", - "dialog.project.edit.color": "Kolor", - "dialog.project.edit.color.select": "Wybierz kolor {{color}}", - "dialog.project.edit.worktree.startup": "Skrypt uruchamiania przestrzeni roboczej", - "dialog.project.edit.worktree.startup.description": "Runs after creating a new workspace (worktree).", - "dialog.project.edit.worktree.startup.placeholder": "np. bun install", - "context.breakdown.title": "Podział kontekstu", - "context.breakdown.note": 'Przybliżony podział tokenów wejściowych. "Inne" obejmuje definicje narzędzi i narzut.', - "context.breakdown.system": "System", - "context.breakdown.user": "Użytkownik", - "context.breakdown.assistant": "Asystent", - "context.breakdown.tool": "Wywołania narzędzi", - "context.breakdown.other": "Inne", - "context.systemPrompt.title": "Prompt systemowy", - "context.rawMessages.title": "Surowe wiadomości", - "context.stats.session": "Sesja", - "context.stats.messages": "Wiadomości", - "context.stats.provider": "Dostawca", - "context.stats.model": "Model", - "context.stats.limit": "Limit kontekstu", - "context.stats.totalTokens": "Całkowita liczba tokenów", - "context.stats.usage": "Użycie", - "context.stats.inputTokens": "Tokeny wejściowe", - "context.stats.outputTokens": "Tokeny wyjściowe", - "context.stats.reasoningTokens": "Tokeny wnioskowania", - "context.stats.cacheTokens": "Tokeny pamięci podręcznej (odczyt/zapis)", - "context.stats.userMessages": "Wiadomości użytkownika", - "context.stats.assistantMessages": "Wiadomości asystenta", - "context.stats.totalCost": "Całkowity koszt", - "context.stats.sessionCreated": "Utworzono sesję", - "context.stats.lastActivity": "Ostatnia aktywność", - "context.usage.tokens": "Tokeny", - "context.usage.usage": "Użycie", - "context.usage.cost": "Koszt", - "context.usage.clickToView": "Kliknij, aby zobaczyć kontekst", - "context.usage.view": "Pokaż użycie kontekstu", - "language.en": "English", - "language.zh": "简体中文", - "language.zht": "繁體中文", - "language.ko": "한국어", - "language.de": "Deutsch", - "language.es": "Español", - "language.fr": "Français", - "language.da": "Dansk", - "language.ja": "日本語", - "language.pl": "Polski", - "language.ru": "Русский", - "language.ar": "العربية", - "language.no": "Norsk", - "language.br": "Português (Brasil)", - "language.bs": "Bosanski", - "language.th": "ไทย", - "language.tr": "Türkçe", - "toast.language.title": "Język", - "toast.language.description": "Przełączono na {{language}}", - "toast.theme.title": "Przełączono motyw", - "toast.scheme.title": "Schemat kolorów", - "toast.workspace.enabled.title": "Przestrzenie robocze włączone", - "toast.workspace.enabled.description": "Kilka worktree jest teraz wyświetlanych na pasku bocznym", - "toast.workspace.disabled.title": "Przestrzenie robocze wyłączone", - "toast.workspace.disabled.description": "Tylko główny worktree jest wyświetlany na pasku bocznym", - "toast.permissions.autoaccept.on.title": "Automatyczne akceptowanie uprawnień", - "toast.permissions.autoaccept.on.description": "Żądania uprawnień będą automatycznie zatwierdzane", - "toast.permissions.autoaccept.off.title": "Zatrzymano automatyczne akceptowanie uprawnień", - "toast.permissions.autoaccept.off.description": "Żądania uprawnień będą wymagały zatwierdzenia", - "toast.model.none.title": "Nie wybrano modelu", - "toast.model.none.description": "Połącz dostawcę, aby podsumować tę sesję", - "toast.file.loadFailed.title": "Nie udało się załadować pliku", - "toast.file.listFailed.title": "Nie udało się wyświetlić listy plików", - "toast.context.noLineSelection.title": "Brak zaznaczenia linii", - "toast.context.noLineSelection.description": "Najpierw wybierz zakres linii w zakładce pliku.", - "toast.session.share.copyFailed.title": "Nie udało się skopiować URL do schowka", - "toast.session.share.success.title": "Sesja udostępniona", - "toast.session.share.success.description": "Link udostępniania skopiowany do schowka!", - "toast.session.share.failed.title": "Nie udało się udostępnić sesji", - "toast.session.share.failed.description": "Wystąpił błąd podczas udostępniania sesji", - "toast.session.unshare.success.title": "Zatrzymano udostępnianie sesji", - "toast.session.unshare.success.description": "Udostępnianie sesji zostało pomyślnie zatrzymane!", - "toast.session.unshare.failed.title": "Nie udało się zatrzymać udostępniania sesji", - "toast.session.unshare.failed.description": "Wystąpił błąd podczas zatrzymywania udostępniania sesji", - "toast.session.listFailed.title": "Nie udało się załadować sesji dla {{project}}", - "toast.update.title": "Dostępna aktualizacja", - "toast.update.description": "Nowa wersja Kilo ({{version}}) jest teraz dostępna do instalacji.", - "toast.update.action.installRestart": "Zainstaluj i zrestartuj", - "toast.update.action.notYet": "Jeszcze nie", - "error.page.title": "Coś poszło nie tak", - "error.page.description": "Wystąpił błąd podczas ładowania aplikacji.", - "error.page.details.label": "Szczegóły błędu", - "error.page.action.restart": "Restartuj", - "error.page.action.checking": "Sprawdzanie...", - "error.page.action.checkUpdates": "Sprawdź aktualizacje", - "error.page.action.updateTo": "Zaktualizuj do {{version}}", - "error.page.report.prefix": "Proszę zgłosić ten błąd do zespołu Kilo", - "error.page.report.discord": "na Discordzie", - "error.page.version": "Wersja: {{version}}", - "error.dev.rootNotFound": - "Nie znaleziono elementu głównego. Czy zapomniałeś dodać go do swojego index.html? A może atrybut id został błędnie wpisany?", - "error.globalSync.connectFailed": "Nie można połączyć się z serwerem. Czy serwer działa pod adresem `{{url}}`?", - "directory.error.invalidUrl": "Nieprawidłowy katalog w URL.", - "error.chain.unknown": "Nieznany błąd", - "error.chain.causedBy": "Spowodowany przez:", - "error.chain.apiError": "Błąd API", - "error.chain.status": "Status: {{status}}", - "error.chain.retryable": "Można ponowić: {{retryable}}", - "error.chain.responseBody": "Treść odpowiedzi:\n{{body}}", - "error.chain.didYouMean": "Czy miałeś na myśli: {{suggestions}}", - "error.chain.modelNotFound": "Model nie znaleziony: {{provider}}/{{model}}", - "error.chain.checkConfig": "Sprawdź swoją konfigurację (opencode.json) nazwy dostawców/modeli", - "error.chain.mcpFailed": 'MCP server "{{name}}" failed. Note, Kilo does not support MCP authentication yet.', - "error.chain.providerAuthFailed": "Uwierzytelnianie dostawcy nie powiodło się ({{provider}}): {{message}}", - "error.chain.providerInitFailed": - 'Nie udało się zainicjować dostawcy "{{provider}}". Sprawdź poświadczenia i konfigurację.', - "error.chain.configJsonInvalid": "Plik konfiguracyjny w {{path}} nie jest poprawnym JSON(C)", - "error.chain.configJsonInvalidWithMessage": "Plik konfiguracyjny w {{path}} nie jest poprawnym JSON(C): {{message}}", - "error.chain.configDirectoryTypo": - 'Katalog "{{dir}}" w {{path}} jest nieprawidłowy. Zmień nazwę katalogu na "{{suggestion}}" lub usuń go. To częsta literówka.', - "error.chain.configFrontmatterError": "Nie udało się przetworzyć frontmatter w {{path}}:\n{{message}}", - "error.chain.configInvalid": "Plik konfiguracyjny w {{path}} jest nieprawidłowy", - "error.chain.configInvalidWithMessage": "Plik konfiguracyjny w {{path}} jest nieprawidłowy: {{message}}", - "notification.permission.title": "Wymagane uprawnienie", - "notification.permission.description": "{{sessionTitle}} w {{projectName}} potrzebuje uprawnienia", - "notification.question.title": "Pytanie", - "notification.question.description": "{{sessionTitle}} w {{projectName}} ma pytanie", - "notification.action.goToSession": "Przejdź do sesji", - "notification.session.responseReady.title": "Odpowiedź gotowa", - "notification.session.error.title": "Błąd sesji", - "notification.session.error.fallbackDescription": "Wystąpił błąd", - "home.recentProjects": "Ostatnie projekty", - "home.empty.title": "Brak ostatnich projektów", - "home.empty.description": "Zacznij od otwarcia lokalnego projektu", - "session.tab.session": "Sesja", - "session.tab.review": "Przegląd", - "session.tab.context": "Kontekst", - "session.panel.reviewAndFiles": "Przegląd i pliki", - "session.review.filesChanged": "Zmieniono {{count}} plików", - "session.review.change.one": "Zmiana", - "session.review.change.other": "Zmiany", - "session.review.loadingChanges": "Ładowanie zmian...", - "session.review.empty": "Brak zmian w tej sesji", - "session.review.noVcs": "Nie wykryto systemu kontroli wersji Git, zmiany nie są wyświetlane", - "session.review.noSnapshot": "Śledzenie migawek jest wyłączone w konfiguracji, więc zmiany w sesji są niedostępne", - "session.review.noChanges": "Brak zmian", - "session.files.selectToOpen": "Wybierz plik do otwarcia", - "session.files.all": "Wszystkie pliki", - "session.files.empty": "Brak plików", - "session.files.binaryContent": "Plik binarny (zawartość nie może być wyświetlona)", - "session.messages.renderEarlier": "Renderuj wcześniejsze wiadomości", - "session.messages.loadingEarlier": "Ładowanie wcześniejszych wiadomości...", - "session.messages.loadEarlier": "Załaduj wcześniejsze wiadomości", - "session.messages.loading": "Ładowanie wiadomości...", - "session.messages.jumpToLatest": "Przejdź do najnowszych", - "session.context.addToContext": "Dodaj {{selection}} do kontekstu", - "session.todo.title": "Zadania", - "session.todo.collapse": "Zwiń", - "session.todo.expand": "Rozwiń", - "session.followupDock.summary.one": "{{count}} wiadomość w kolejce", - "session.followupDock.summary.other": "{{count}} wiadomości w kolejce", - "session.followupDock.sendNow": "Wyślij teraz", - "session.followupDock.edit": "Edytuj", - "session.followupDock.collapse": "Zwiń wiadomości w kolejce", - "session.followupDock.expand": "Rozwiń wiadomości w kolejce", - "session.revertDock.summary.one": "{{count}} cofnięta wiadomość", - "session.revertDock.summary.other": "{{count}} cofnięte wiadomości", - "session.revertDock.collapse": "Zwiń cofnięte wiadomości", - "session.revertDock.expand": "Rozwiń cofnięte wiadomości", - "session.revertDock.restore": "Przywróć wiadomość", - "session.new.title": "Zbuduj cokolwiek", - "session.new.worktree.main": "Główna gałąź", - "session.new.worktree.mainWithBranch": "Główna gałąź ({{branch}})", - "session.new.worktree.create": "Utwórz nowe drzewo robocze", - "session.new.lastModified": "Ostatnio zmodyfikowano", - "session.header.search.placeholder": "Szukaj {{project}}", - "session.header.searchFiles": "Szukaj plików", - "session.header.openIn": "Otwórz w", - "session.header.open.action": "Otwórz {{app}}", - "session.header.open.ariaLabel": "Otwórz w {{app}}", - "session.header.open.menu": "Opcje otwierania", - "session.header.open.copyPath": "Kopiuj ścieżkę", - "status.popover.trigger": "Status", - "status.popover.ariaLabel": "Konfiguracje serwerów", - "status.popover.tab.servers": "Serwery", - "status.popover.tab.mcp": "MCP", - "status.popover.tab.lsp": "LSP", - "status.popover.tab.plugins": "Wtyczki", - "status.popover.action.manageServers": "Zarządzaj serwerami", - "session.share.popover.title": "Opublikuj w sieci", - "session.share.popover.description.shared": - "Ta sesja jest publiczna w sieci. Jest dostępna dla każdego, kto posiada link.", - "session.share.popover.description.unshared": - "Udostępnij sesję publicznie w sieci. Będzie dostępna dla każdego, kto posiada link.", - "session.share.action.share": "Udostępnij", - "session.share.action.publish": "Opublikuj", - "session.share.action.publishing": "Publikowanie...", - "session.share.action.unpublish": "Cofnij publikację", - "session.share.action.unpublishing": "Cofanie publikacji...", - "session.share.action.view": "Widok", - "session.share.copy.copied": "Skopiowano", - "session.share.copy.copyLink": "Kopiuj link", - "lsp.tooltip.none": "Brak serwerów LSP", - "lsp.label.connected": "{{count}} LSP", - "prompt.loading": "Ładowanie promptu...", - "terminal.loading": "Ładowanie terminala...", - "terminal.title": "Terminal", - "terminal.title.numbered": "Terminal {{number}}", - "terminal.close": "Zamknij terminal", - "terminal.connectionLost.title": "Utracono połączenie", - "terminal.connectionLost.description": - "Połączenie z terminalem zostało przerwane. Może się to zdarzyć przy restarcie serwera.", - "common.closeTab": "Zamknij kartę", - "common.dismiss": "Odrzuć", - "common.requestFailed": "Żądanie nie powiodło się", - "common.moreOptions": "Więcej opcji", - "common.learnMore": "Dowiedz się więcej", - "common.rename": "Zmień nazwę", - "common.reset": "Resetuj", - "common.archive": "Archiwizuj", - "common.delete": "Usuń", - "common.close": "Zamknij", - "common.edit": "Edytuj", - "common.loadMore": "Załaduj więcej", - "common.key.esc": "ESC", - "sidebar.menu.toggle": "Przełącz menu", - "sidebar.nav.projectsAndSessions": "Projekty i sesje", - "sidebar.settings": "Ustawienia", - "sidebar.help": "Pomoc", - "sidebar.workspaces.enable": "Włącz przestrzenie robocze", - "sidebar.workspaces.disable": "Wyłącz przestrzenie robocze", - "sidebar.gettingStarted.title": "Pierwsze kroki", - "sidebar.gettingStarted.line1": "Kilo zawiera darmowe modele, więc możesz zacząć od razu.", - "sidebar.gettingStarted.line2": "Połącz dowolnego dostawcę, aby używać modeli, w tym Claude, GPT, Gemini itp.", - "sidebar.project.recentSessions": "Ostatnie sesje", - "sidebar.project.viewAllSessions": "Zobacz wszystkie sesje", - "sidebar.project.clearNotifications": "Wyczyść powiadomienia", - "app.name.desktop": "Kilo Desktop", - "settings.section.desktop": "Pulpit", - "settings.section.server": "Serwer", - "settings.tab.general": "Ogólne", - "settings.tab.shortcuts": "Skróty", - "settings.desktop.section.wsl": "WSL", - "settings.desktop.wsl.title": "WSL integration", - "settings.desktop.wsl.description": "Run the Kilo server inside WSL on Windows.", - "settings.general.section.appearance": "Wygląd", - "settings.general.section.notifications": "Powiadomienia systemowe", - "settings.general.section.updates": "Aktualizacje", - "settings.general.section.sounds": "Efekty dźwiękowe", - "settings.general.section.feed": "Kanał", - "settings.general.section.display": "Ekran", - "settings.general.row.language.title": "Język", - "settings.general.row.language.description": "Zmień język wyświetlania dla Kilo", - "settings.general.row.appearance.title": "Wygląd", - "settings.general.row.appearance.description": "Dostosuj wygląd Kilo na swoim urządzeniu", - "settings.general.row.colorScheme.title": "Schemat kolorów", - "settings.general.row.colorScheme.description": - "Wybierz, czy Kilo ma używać motywu systemowego, jasnego czy ciemnego", - "settings.general.row.theme.title": "Motyw", - "settings.general.row.theme.description": "Dostosuj motyw Kilo.", - "settings.general.row.font.title": "Czcionka kodu", - "settings.general.row.font.description": "Dostosuj czcionkę używaną w blokach kodu", - "settings.general.row.terminalFont.title": "Terminal Font", - "settings.general.row.terminalFont.description": "Customise the font used in the terminal", - "settings.general.row.uiFont.title": "Czcionka interfejsu", - "settings.general.row.uiFont.description": "Dostosuj czcionkę używaną w całym interfejsie", - "settings.general.row.followup.title": "Zachowanie kontynuacji", - "settings.general.row.followup.description": "Wybierz, czy kontynuacja ma być natychmiastowa, czy czekać w kolejce", - "settings.general.row.followup.option.queue": "Kolejka", - "settings.general.row.followup.option.steer": "Sterowanie", - "settings.general.row.reasoningSummaries.title": "Pokaż podsumowania wnioskowania", - "settings.general.row.reasoningSummaries.description": "Wyświetlaj podsumowania wnioskowania modelu na osi czasu", - "settings.general.row.shellToolPartsExpanded.title": "Rozwijaj elementy narzędzia shell", - "settings.general.row.shellToolPartsExpanded.description": - "Domyślnie pokazuj rozwinięte elementy narzędzia shell na osi czasu", - "settings.general.row.editToolPartsExpanded.title": "Rozwijaj elementy narzędzia edit", - "settings.general.row.editToolPartsExpanded.description": - "Domyślnie pokazuj rozwinięte elementy narzędzi edit, write i patch na osi czasu", - "settings.general.row.showSessionProgressBar.title": "Pokazuj pasek postępu sesji", - "settings.general.row.showSessionProgressBar.description": - "Wyświetlaj animowany pasek postępu u góry sesji, gdy agent pracuje", - "settings.general.row.wayland.title": "Użyj natywnego Wayland", - "settings.general.row.wayland.description": "Wyłącz fallback X11 na Wayland. Wymaga restartu.", - "settings.general.row.wayland.tooltip": - "Na Linuxie z monitorami o różnym odświeżaniu, natywny Wayland może być bardziej stabilny.", - "settings.general.row.releaseNotes.title": "Informacje o wydaniu", - "settings.general.row.releaseNotes.description": 'Pokazuj wyskakujące okna "Co nowego" po aktualizacjach', - "settings.updates.row.startup.title": "Sprawdzaj aktualizacje przy uruchomieniu", - "settings.updates.row.startup.description": "Automatycznie sprawdzaj aktualizacje podczas uruchamiania Kilo", - "settings.updates.row.check.title": "Sprawdź aktualizacje", - "settings.updates.row.check.description": "Ręcznie sprawdź aktualizacje i zainstaluj, jeśli są dostępne", - "settings.updates.action.checkNow": "Sprawdź teraz", - "settings.updates.action.checking": "Sprawdzanie...", - "settings.updates.toast.latest.title": "Masz najnowszą wersję", - "settings.updates.toast.latest.description": "Korzystasz z najnowszej wersji Kilo.", - "sound.option.none": "Brak", - "sound.option.alert01": "Alert 01", - "sound.option.alert02": "Alert 02", - "sound.option.alert03": "Alert 03", - "sound.option.alert04": "Alert 04", - "sound.option.alert05": "Alert 05", - "sound.option.alert06": "Alert 06", - "sound.option.alert07": "Alert 07", - "sound.option.alert08": "Alert 08", - "sound.option.alert09": "Alert 09", - "sound.option.alert10": "Alert 10", - "sound.option.bipbop01": "Bip-bop 01", - "sound.option.bipbop02": "Bip-bop 02", - "sound.option.bipbop03": "Bip-bop 03", - "sound.option.bipbop04": "Bip-bop 04", - "sound.option.bipbop05": "Bip-bop 05", - "sound.option.bipbop06": "Bip-bop 06", - "sound.option.bipbop07": "Bip-bop 07", - "sound.option.bipbop08": "Bip-bop 08", - "sound.option.bipbop09": "Bip-bop 09", - "sound.option.bipbop10": "Bip-bop 10", - "sound.option.staplebops01": "Staplebops 01", - "sound.option.staplebops02": "Staplebops 02", - "sound.option.staplebops03": "Staplebops 03", - "sound.option.staplebops04": "Staplebops 04", - "sound.option.staplebops05": "Staplebops 05", - "sound.option.staplebops06": "Staplebops 06", - "sound.option.staplebops07": "Staplebops 07", - "sound.option.nope01": "Nope 01", - "sound.option.nope02": "Nope 02", - "sound.option.nope03": "Nope 03", - "sound.option.nope04": "Nope 04", - "sound.option.nope05": "Nope 05", - "sound.option.nope06": "Nope 06", - "sound.option.nope07": "Nope 07", - "sound.option.nope08": "Nope 08", - "sound.option.nope09": "Nope 09", - "sound.option.nope10": "Nope 10", - "sound.option.nope11": "Nope 11", - "sound.option.nope12": "Nope 12", - "sound.option.yup01": "Yup 01", - "sound.option.yup02": "Yup 02", - "sound.option.yup03": "Yup 03", - "sound.option.yup04": "Yup 04", - "sound.option.yup05": "Yup 05", - "sound.option.yup06": "Yup 06", - "settings.general.notifications.agent.title": "Agent", - "settings.general.notifications.agent.description": - "Pokaż powiadomienie systemowe, gdy agent zakończy pracę lub wymaga uwagi", - "settings.general.notifications.permissions.title": "Uprawnienia", - "settings.general.notifications.permissions.description": - "Pokaż powiadomienie systemowe, gdy wymagane jest uprawnienie", - "settings.general.notifications.errors.title": "Błędy", - "settings.general.notifications.errors.description": "Pokaż powiadomienie systemowe, gdy wystąpi błąd", - "settings.general.sounds.agent.title": "Agent", - "settings.general.sounds.agent.description": "Odtwórz dźwięk, gdy agent zakończy pracę lub wymaga uwagi", - "settings.general.sounds.permissions.title": "Uprawnienia", - "settings.general.sounds.permissions.description": "Odtwórz dźwięk, gdy wymagane jest uprawnienie", - "settings.general.sounds.errors.title": "Błędy", - "settings.general.sounds.errors.description": "Odtwórz dźwięk, gdy wystąpi błąd", - "settings.shortcuts.title": "Skróty klawiszowe", - "settings.shortcuts.reset.button": "Przywróć domyślne", - "settings.shortcuts.reset.toast.title": "Zresetowano skróty", - "settings.shortcuts.reset.toast.description": "Skróty klawiszowe zostały przywrócone do ustawień domyślnych.", - "settings.shortcuts.conflict.title": "Skrót już w użyciu", - "settings.shortcuts.conflict.description": "{{keybind}} jest już przypisany do {{titles}}.", - "settings.shortcuts.unassigned": "Nieprzypisany", - "settings.shortcuts.pressKeys": "Naciśnij klawisze", - "settings.shortcuts.search.placeholder": "Szukaj skrótów", - "settings.shortcuts.search.empty": "Nie znaleziono skrótów", - "settings.shortcuts.group.general": "Ogólne", - "settings.shortcuts.group.session": "Sesja", - "settings.shortcuts.group.navigation": "Nawigacja", - "settings.shortcuts.group.modelAndAgent": "Model i agent", - "settings.shortcuts.group.terminal": "Terminal", - "settings.shortcuts.group.prompt": "Prompt", - "settings.providers.title": "Dostawcy", - "settings.providers.description": "Ustawienia dostawców będą tutaj konfigurowalne.", - "settings.providers.section.connected": "Połączeni dostawcy", - "settings.providers.connected.empty": "Brak połączonych dostawców", - "settings.providers.section.popular": "Popularni dostawcy", - "settings.providers.tag.environment": "Środowisko", - "settings.providers.tag.config": "Konfiguracja", - "settings.providers.tag.custom": "Niestandardowe", - "settings.providers.tag.other": "Inne", - "settings.models.title": "Modele", - "settings.models.description": "Ustawienia modeli będą tutaj konfigurowalne.", - "settings.agents.title": "Agenci", - "settings.agents.description": "Ustawienia agentów będą tutaj konfigurowalne.", - "settings.commands.title": "Polecenia", - "settings.commands.description": "Ustawienia poleceń będą tutaj konfigurowalne.", - "settings.mcp.title": "MCP", - "settings.mcp.description": "Ustawienia MCP będą tutaj konfigurowalne.", - "settings.permissions.title": "Uprawnienia", - "settings.permissions.description": "Kontroluj, jakich narzędzi serwer może używać domyślnie.", - "settings.permissions.section.tools": "Narzędzia", - "settings.permissions.toast.updateFailed.title": "Nie udało się zaktualizować uprawnień", - "settings.permissions.action.allow": "Zezwól", - "settings.permissions.action.ask": "Pytaj", - "settings.permissions.action.deny": "Odmów", - "settings.permissions.tool.read.title": "Odczyt", - "settings.permissions.tool.read.description": "Odczyt pliku (pasuje do ścieżki pliku)", - "settings.permissions.tool.edit.title": "Edycja", - "settings.permissions.tool.edit.description": "Modyfikacja plików, w tym edycje, zapisy, łatki i multi-edycje", - "settings.permissions.tool.glob.title": "Glob", - "settings.permissions.tool.glob.description": "Dopasowywanie plików za pomocą wzorców glob", - "settings.permissions.tool.grep.title": "Grep", - "settings.permissions.tool.grep.description": "Przeszukiwanie zawartości plików za pomocą wyrażeń regularnych", - "settings.permissions.tool.list.title": "Lista", - "settings.permissions.tool.list.description": "Wyświetlanie listy plików w katalogu", - "settings.permissions.tool.bash.title": "Bash", - "settings.permissions.tool.bash.description": "Uruchamianie poleceń powłoki", - "settings.permissions.tool.task.title": "Zadanie", - "settings.permissions.tool.task.description": "Uruchamianie pod-agentów", - "settings.permissions.tool.skill.title": "Umiejętność", - "settings.permissions.tool.skill.description": "Ładowanie umiejętności według nazwy", - "settings.permissions.tool.lsp.title": "LSP", - "settings.permissions.tool.lsp.description": "Uruchamianie zapytań serwera językowego", - "settings.permissions.tool.todowrite.title": "Zapis Todo", - "settings.permissions.tool.todowrite.description": "Aktualizacja listy zadań", - "settings.permissions.tool.webfetch.title": "Pobieranie z sieci", - "settings.permissions.tool.webfetch.description": "Pobieranie zawartości z adresu URL", - "settings.permissions.tool.websearch.title": "Wyszukiwanie w sieci", - "settings.permissions.tool.websearch.description": "Przeszukiwanie sieci", - "settings.permissions.tool.codesearch.title": "Wyszukiwanie kodu", - "settings.permissions.tool.codesearch.description": "Przeszukiwanie kodu w sieci", - "settings.permissions.tool.external_directory.title": "Katalog zewnętrzny", - "settings.permissions.tool.external_directory.description": "Dostęp do plików poza katalogiem projektu", - "settings.permissions.tool.doom_loop.title": "Zapętlenie", - "settings.permissions.tool.doom_loop.description": "Wykrywanie powtarzających się wywołań narzędzi (doom loop)", - "session.delete.failed.title": "Nie udało się usunąć sesji", - "session.delete.title": "Usuń sesję", - "session.delete.confirm": 'Usunąć sesję "{{name}}"?', - "session.delete.button": "Usuń sesję", - "workspace.new": "Nowa przestrzeń robocza", - "workspace.type.local": "lokalna", - "workspace.type.sandbox": "piaskownica", - "workspace.create.failed.title": "Nie udało się utworzyć przestrzeni roboczej", - "workspace.delete.failed.title": "Nie udało się usunąć przestrzeni roboczej", - "workspace.resetting.title": "Resetowanie przestrzeni roboczej", - "workspace.resetting.description": "To może potrwać minutę.", - "workspace.reset.failed.title": "Nie udało się zresetować przestrzeni roboczej", - "workspace.reset.success.title": "Przestrzeń robocza zresetowana", - "workspace.reset.success.description": "Przestrzeń robocza odpowiada teraz domyślnej gałęzi.", - "workspace.error.stillPreparing": "Przestrzeń robocza jest wciąż przygotowywana", - "workspace.status.checking": "Sprawdzanie niezscalonych zmian...", - "workspace.status.error": "Nie można zweryfikować statusu git.", - "workspace.status.clean": "Nie wykryto niezscalonych zmian.", - "workspace.status.dirty": "Wykryto niezscalone zmiany w tej przestrzeni roboczej.", - "workspace.delete.title": "Usuń przestrzeń roboczą", - "workspace.delete.confirm": 'Usunąć przestrzeń roboczą "{{name}}"?', - "workspace.delete.button": "Usuń przestrzeń roboczą", - "workspace.reset.title": "Resetuj przestrzeń roboczą", - "workspace.reset.confirm": 'Zresetować przestrzeń roboczą "{{name}}"?', - "workspace.reset.button": "Resetuj przestrzeń roboczą", - "workspace.reset.archived.none": "Żadne aktywne sesje nie zostaną zarchiwizowane.", - "workspace.reset.archived.one": "1 sesja zostanie zarchiwizowana.", - "workspace.reset.archived.many": "{{count}} sesji zostanie zarchiwizowanych.", - "workspace.reset.note": "To zresetuje przestrzeń roboczą, aby odpowiadała domyślnej gałęzi.", - "common.open": "Otwórz", - "dialog.releaseNotes.action.getStarted": "Rozpocznij", - "dialog.releaseNotes.action.next": "Dalej", - "dialog.releaseNotes.action.hideFuture": "Nie pokazuj tego w przyszłości", - "dialog.releaseNotes.media.alt": "Podgląd wydania", - "toast.project.reloadFailed.title": "Nie udało się ponownie wczytać {{project}}", - "error.server.invalidConfiguration": "Nieprawidłowa konfiguracja", - "common.moreCountSuffix": " (jeszcze {{count}})", - "common.time.justNow": "Przed chwilą", - "common.time.minutesAgo.short": "{{count}} min temu", - "common.time.hoursAgo.short": "{{count}} godz. temu", - "common.time.daysAgo.short": "{{count}} dni temu", - "settings.providers.connected.environmentDescription": "Połączono ze zmiennymi środowiskowymi", - "settings.providers.custom.description": "Dodaj dostawcę zgodnego z OpenAI poprzez podstawowy URL.", - - "app.server.unreachable": "Nie można połączyć z {{server}}", - "app.server.retrying": "Ponawianie automatycznie...", - "app.server.otherServers": "Inne serwery", - "dialog.server.add.usernamePlaceholder": "nazwa użytkownika", - "dialog.server.add.passwordPlaceholder": "hasło", - "server.row.noUsername": "brak nazwy użytkownika", - "session.review.noVcs.createGit.title": "Utwórz repozytorium Git", - "session.review.noVcs.createGit.description": "Śledź, przeglądaj i cofaj zmiany w tym projekcie", - "session.review.noVcs.createGit.actionLoading": "Tworzenie repozytorium Git...", - "session.review.noVcs.createGit.action": "Utwórz repozytorium Git", - "session.todo.progress": "Ukończono {{done}} z {{total}} zadań", - "session.question.progress": "{{current}} z {{total}} pytań", - "session.header.open.finder": "Finder", - "session.header.open.fileExplorer": "Eksplorator plików", - "session.header.open.fileManager": "Menedżer plików", - "session.header.open.app.vscode": "VS Code", - "session.header.open.app.cursor": "Cursor", - "session.header.open.app.zed": "Zed", - "session.header.open.app.textmate": "TextMate", - "session.header.open.app.antigravity": "Antigravity", - "session.header.open.app.terminal": "Terminal", - "session.header.open.app.iterm2": "iTerm2", - "session.header.open.app.ghostty": "Ghostty", - "session.header.open.app.warp": "Warp", - "session.header.open.app.xcode": "Xcode", - "session.header.open.app.androidStudio": "Android Studio", - "session.header.open.app.powershell": "PowerShell", - "session.header.open.app.sublimeText": "Sublime Text", - "debugBar.ariaLabel": "Diagnostyka wydajności deweloperskiej", - "debugBar.na": "n.d.", - "debugBar.nav.label": "NAV", - "debugBar.nav.tip": - "Ostatnie zakończone przejście trasy dotykające strony sesji, mierzone od startu routera do pierwszego odrysowania po ustaleniu.", - "debugBar.fps.label": "FPS", - "debugBar.fps.tip": "Średnia liczba klatek na sekundę w ciągu ostatnich 5 sekund.", - "debugBar.frame.label": "FRAME", - "debugBar.frame.tip": "Najgorszy czas klatki w ciągu ostatnich 5 sekund.", - "debugBar.jank.label": "JANK", - "debugBar.jank.tip": "Klatki powyżej 32ms w ciągu ostatnich 5 sekund.", - "debugBar.long.label": "LONG", - "debugBar.long.tip": - "Zablokowany czas i liczba długich zadań w ciągu ostatnich 5 sekund. Maksymalne zadanie: {{max}}.", - "debugBar.delay.label": "DELAY", - "debugBar.delay.tip": "Najgorsze zaobserwowane opóźnienie wejścia w ciągu ostatnich 5 sekund.", - "debugBar.inp.label": "INP", - "debugBar.inp.tip": - "Przybliżony czas trwania interakcji w ciągu ostatnich 5 sekund. Jest to podobne do INP, a nie oficjalne Web Vitals INP.", - "debugBar.cls.label": "CLS", - "debugBar.cls.tip": "Skumulowane przesunięcie układu dla bieżącego czasu życia aplikacji.", - "debugBar.mem.label": "MEM", - "debugBar.mem.tipUnavailable": "Użyta sterta JS vs limit sterty. Tylko Chromium.", - "debugBar.mem.tip": "Użyta sterta JS vs limit sterty. {{used}} z {{limit}}.", - "common.key.ctrl": "Ctrl", - "common.key.alt": "Alt", - "common.key.shift": "Shift", - "common.key.meta": "Meta", - "common.key.space": "Spacja", - "common.key.backspace": "Backspace", - "common.key.enter": "Enter", - "common.key.tab": "Tab", - "common.key.delete": "Delete", - "common.key.home": "Home", - "common.key.end": "End", - "common.key.pageUp": "Page Up", - "common.key.pageDown": "Page Down", - "common.key.insert": "Insert", - "common.unknown": "nieznany", - "error.page.circular": "[Cykliczne]", - "error.globalSDK.noServerAvailable": "Brak dostępnego serwera", - "error.globalSDK.serverNotAvailable": "Serwer niedostępny", - "error.childStore.persistedCacheCreateFailed": "Nie udało się utworzyć trwałej pamięci podręcznej", - "error.childStore.persistedProjectMetadataCreateFailed": "Nie udało się utworzyć trwałych metadanych projektu", - "error.childStore.persistedProjectIconCreateFailed": "Nie udało się utworzyć trwałej ikony projektu", - "error.childStore.storeCreateFailed": "Nie udało się utworzyć magazynu", - "terminal.connectionLost.abnormalClose": "WebSocket zamknięty nieprawidłowo: {{code}}", -} diff --git a/packages/app/src/i18n/ru.ts b/packages/app/src/i18n/ru.ts deleted file mode 100644 index 6436f62c1f..0000000000 --- a/packages/app/src/i18n/ru.ts +++ /dev/null @@ -1,938 +0,0 @@ -export const dict = { - "command.category.suggested": "Предложено", - "command.category.view": "Просмотр", - "command.category.project": "Проект", - "command.category.provider": "Провайдер", - "command.category.server": "Сервер", - "command.category.session": "Сессия", - "command.category.theme": "Тема", - "command.category.language": "Язык", - "command.category.file": "Файл", - "command.category.context": "Контекст", - "command.category.terminal": "Терминал", - "command.category.model": "Модель", - "command.category.mcp": "MCP", - "command.category.agent": "Агент", - "command.category.permissions": "Разрешения", - "command.category.workspace": "Рабочее пространство", - "command.category.settings": "Настройки", - - "theme.scheme.system": "Системная", - "theme.scheme.light": "Светлая", - "theme.scheme.dark": "Тёмная", - - "command.sidebar.toggle": "Переключить боковую панель", - "command.project.open": "Открыть проект", - "command.provider.connect": "Подключить провайдера", - "command.server.switch": "Переключить сервер", - "command.settings.open": "Открыть настройки", - "command.session.previous": "Предыдущая сессия", - "command.session.next": "Следующая сессия", - "command.session.previous.unseen": "Предыдущая непрочитанная сессия", - "command.session.next.unseen": "Следующая непрочитанная сессия", - "command.session.archive": "Архивировать сессию", - - "command.palette": "Палитра команд", - - "command.theme.cycle": "Цикл тем", - "command.theme.set": "Использовать тему: {{theme}}", - "command.theme.scheme.cycle": "Цикл цветовой схемы", - "command.theme.scheme.set": "Использовать цветовую схему: {{scheme}}", - - "command.language.cycle": "Цикл языков", - "command.language.set": "Использовать язык: {{language}}", - - "command.session.new": "Новая сессия", - "command.file.open": "Открыть файл", - "command.tab.close": "Закрыть вкладку", - "command.context.addSelection": "Добавить выделение в контекст", - "command.context.addSelection.description": "Добавить выбранные строки из текущего файла", - "command.input.focus": "Фокус на поле ввода", - "command.terminal.toggle": "Переключить терминал", - "command.fileTree.toggle": "Переключить дерево файлов", - "command.review.toggle": "Переключить обзор", - "command.terminal.new": "Новый терминал", - "command.terminal.new.description": "Создать новую вкладку терминала", - "command.steps.toggle": "Переключить шаги", - "command.steps.toggle.description": "Показать или скрыть шаги для текущего сообщения", - "command.message.previous": "Предыдущее сообщение", - "command.message.previous.description": "Перейти к предыдущему сообщению пользователя", - "command.message.next": "Следующее сообщение", - "command.message.next.description": "Перейти к следующему сообщению пользователя", - "command.model.choose": "Выбрать модель", - "command.model.choose.description": "Выбрать другую модель", - "command.mcp.toggle": "Переключить MCP", - "command.mcp.toggle.description": "Переключить MCP", - "command.agent.cycle": "Цикл агентов", - "command.agent.cycle.description": "Переключиться к следующему агенту", - "command.agent.cycle.reverse": "Цикл агентов назад", - "command.agent.cycle.reverse.description": "Переключиться к предыдущему агенту", - "command.model.variant.cycle": "Цикл режимов мышления", - "command.model.variant.cycle.description": "Переключиться к следующему уровню усилий", - "command.prompt.mode.shell": "Оболочка", - "command.prompt.mode.normal": "Промпт", - "command.permissions.autoaccept.enable": "Автоматически принимать разрешения", - "command.permissions.autoaccept.disable": "Остановить автоматическое принятие разрешений", - "command.workspace.toggle": "Переключить рабочие пространства", - "command.workspace.toggle.description": "Включить или отключить несколько рабочих пространств в боковой панели", - "command.session.undo": "Отменить", - "command.session.undo.description": "Отменить последнее сообщение", - "command.session.redo": "Повторить", - "command.session.redo.description": "Повторить отменённое сообщение", - "command.session.compact": "Сжать сессию", - "command.session.compact.description": "Сократить сессию для уменьшения размера контекста", - "command.session.fork": "Создать ответвление", - "command.session.fork.description": "Создать новую сессию из сообщения", - "command.session.share": "Поделиться сессией", - "command.session.share.description": "Поделиться сессией и скопировать URL в буфер обмена", - "command.session.unshare": "Отменить публикацию", - "command.session.unshare.description": "Прекратить публикацию сессии", - - "palette.search.placeholder": "Поиск файлов, команд и сессий", - "palette.empty": "Ничего не найдено", - "palette.group.commands": "Команды", - "palette.group.files": "Файлы", - - "dialog.provider.search.placeholder": "Поиск провайдеров", - "dialog.provider.empty": "Провайдеры не найдены", - "dialog.provider.group.popular": "Популярные", - "dialog.provider.group.other": "Другие", - "dialog.provider.tag.recommended": "Рекомендуемые", - "dialog.provider.opencode.note": "Отобранные модели, включая Claude, GPT, Gemini и другие", - "dialog.provider.opencode.tagline": "Надежные оптимизированные модели", - "dialog.provider.opencodeGo.tagline": "Доступная подписка для всех", - "dialog.provider.anthropic.note": "Прямой доступ к моделям Claude, включая Pro и Max", - "dialog.provider.copilot.note": "ИИ-модели для помощи в кодировании через GitHub Copilot", - "dialog.provider.openai.note": "Модели GPT для быстрых и мощных задач общего ИИ", - "dialog.provider.google.note": "Модели Gemini для быстрых и структурированных ответов", - "dialog.provider.openrouter.note": "Доступ ко всем поддерживаемым моделям через одного провайдера", - "dialog.provider.vercel.note": "Единый доступ к ИИ-моделям с умной маршрутизацией", - - "dialog.model.select.title": "Выбрать модель", - "dialog.model.search.placeholder": "Поиск моделей", - "dialog.model.empty": "Модели не найдены", - "dialog.model.manage": "Управление моделями", - "dialog.model.manage.description": "Настройте какие модели появляются в выборе модели", - "dialog.model.manage.provider.toggle": "Переключить все модели {{provider}}", - - "dialog.model.unpaid.freeModels.title": "Бесплатные модели от Kilo", - "dialog.model.unpaid.addMore.title": "Добавьте больше моделей от популярных провайдеров", - - "dialog.provider.viewAll": "Показать больше провайдеров", - - "provider.connect.title": "Подключить {{provider}}", - "provider.connect.title.anthropicProMax": "Войти с помощью Claude Pro/Max", - "provider.connect.selectMethod": "Выберите способ входа для {{provider}}.", - "provider.connect.method.apiKey": "API ключ", - "provider.connect.status.inProgress": "Авторизация...", - "provider.connect.status.waiting": "Ожидание авторизации...", - "provider.connect.status.failed": "Ошибка авторизации: {{error}}", - "provider.connect.apiKey.description": - "Введите ваш API ключ {{provider}} для подключения аккаунта и использования моделей {{provider}} в Kilo.", - "provider.connect.apiKey.label": "{{provider}} API ключ", - "provider.connect.apiKey.placeholder": "API ключ", - "provider.connect.apiKey.required": "API ключ обязателен", - "provider.connect.opencodeZen.line1": - "OpenCode Zen даёт вам доступ к отобранным надёжным оптимизированным моделям для агентов программирования.", - "provider.connect.opencodeZen.line2": - "С одним API ключом вы получите доступ к таким моделям как Claude, GPT, Gemini, GLM и другим.", - "provider.connect.opencodeZen.visit.prefix": "Посетите ", - "provider.connect.opencodeZen.visit.link": "opencode.ai/zen", - "provider.connect.opencodeZen.visit.suffix": " чтобы получить ваш API ключ.", - "provider.connect.oauth.code.visit.prefix": "Посетите ", - "provider.connect.oauth.code.visit.link": "эту ссылку", - "provider.connect.oauth.code.visit.suffix": - " чтобы получить код авторизации для подключения аккаунта и использования моделей {{provider}} в Kilo.", - "provider.connect.oauth.code.label": "{{method}} код авторизации", - "provider.connect.oauth.code.placeholder": "Код авторизации", - "provider.connect.oauth.code.required": "Код авторизации обязателен", - "provider.connect.oauth.code.invalid": "Неверный код авторизации", - "provider.connect.oauth.auto.visit.prefix": "Посетите ", - "provider.connect.oauth.auto.visit.link": "эту ссылку", - "provider.connect.oauth.auto.visit.suffix": - " и введите код ниже для подключения аккаунта и использования моделей {{provider}} в Kilo.", - "provider.connect.oauth.auto.confirmationCode": "Код подтверждения", - "provider.connect.toast.connected.title": "{{provider}} подключён", - "provider.connect.toast.connected.description": "Модели {{provider}} теперь доступны.", - - "provider.custom.title": "Пользовательский провайдер", - "provider.custom.description.prefix": "Настройте OpenAI-совместимого провайдера. См. ", - "provider.custom.description.link": "документацию по настройке провайдера", - "provider.custom.description.suffix": ".", - "provider.custom.field.providerID.label": "ID провайдера", - "provider.custom.field.providerID.placeholder": "myprovider", - "provider.custom.field.providerID.description": "Строчные буквы, цифры, дефисы или подчёркивания", - "provider.custom.field.name.label": "Отображаемое имя", - "provider.custom.field.name.placeholder": "Мой AI провайдер", - "provider.custom.field.baseURL.label": "Базовый URL", - "provider.custom.field.baseURL.placeholder": "https://api.myprovider.com/v1", - "provider.custom.field.apiKey.label": "API ключ", - "provider.custom.field.apiKey.placeholder": "API ключ", - "provider.custom.field.apiKey.description": - "Необязательно. Оставьте пустым, если управляете авторизацией через заголовки.", - "provider.custom.models.label": "Модели", - "provider.custom.models.id.label": "ID", - "provider.custom.models.id.placeholder": "model-id", - "provider.custom.models.name.label": "Имя", - "provider.custom.models.name.placeholder": "Отображаемое имя", - "provider.custom.models.remove": "Удалить модель", - "provider.custom.models.add": "Добавить модель", - "provider.custom.headers.label": "Заголовки (необязательно)", - "provider.custom.headers.key.label": "Заголовок", - "provider.custom.headers.key.placeholder": "Header-Name", - "provider.custom.headers.value.label": "Значение", - "provider.custom.headers.value.placeholder": "значение", - "provider.custom.headers.remove": "Удалить заголовок", - "provider.custom.headers.add": "Добавить заголовок", - "provider.custom.error.providerID.required": "Требуется ID провайдера", - "provider.custom.error.providerID.format": "Используйте строчные буквы, цифры, дефисы или подчёркивания", - "provider.custom.error.providerID.exists": "Такой ID провайдера уже существует", - "provider.custom.error.name.required": "Требуется отображаемое имя", - "provider.custom.error.baseURL.required": "Требуется базовый URL", - "provider.custom.error.baseURL.format": "Должен начинаться с http:// или https://", - "provider.custom.error.required": "Обязательно", - "provider.custom.error.duplicate": "Дубликат", - - "provider.disconnect.toast.disconnected.title": "{{provider}} отключён", - "provider.disconnect.toast.disconnected.description": "Модели {{provider}} больше недоступны.", - "model.tag.free": "Бесплатно", - "model.tag.latest": "Последняя", - "model.provider.anthropic": "Anthropic", - "model.provider.openai": "OpenAI", - "model.provider.google": "Google", - "model.provider.xai": "xAI", - "model.provider.meta": "Meta", - "model.input.text": "текст", - "model.input.image": "изображение", - "model.input.audio": "аудио", - "model.input.video": "видео", - "model.input.pdf": "pdf", - "model.tooltip.allows": "Разрешено: {{inputs}}", - "model.tooltip.reasoning.allowed": "Разрешает рассуждение", - "model.tooltip.reasoning.none": "Без рассуждения", - "model.tooltip.context": "Лимит контекста {{limit}}", - - "common.search.placeholder": "Поиск", - "common.goBack": "Назад", - "common.goForward": "Вперёд", - "common.loading": "Загрузка", - "common.loading.ellipsis": "...", - "common.cancel": "Отмена", - "common.connect": "Подключить", - "common.disconnect": "Отключить", - "common.continue": "Отправить", - "common.submit": "Отправить", - "common.save": "Сохранить", - "common.saving": "Сохранение...", - "common.default": "По умолчанию", - "common.attachment": "вложение", - - "prompt.placeholder.shell": "Введите команду оболочки... {{example}}", - "prompt.placeholder.normal": 'Спросите что угодно... "{{example}}"', - "prompt.placeholder.simple": "Спросите что угодно...", - "prompt.placeholder.summarizeComments": "Суммировать комментарии…", - "prompt.placeholder.summarizeComment": "Суммировать комментарий…", - "prompt.mode.shell": "Оболочка", - "prompt.mode.normal": "Промпт", - "prompt.mode.shell.exit": "esc для выхода", - - "prompt.example.1": "Исправить TODO в коде", - "prompt.example.2": "Какой технологический стек этого проекта?", - "prompt.example.3": "Исправить сломанные тесты", - "prompt.example.4": "Объясни как работает аутентификация", - "prompt.example.5": "Найти и исправить уязвимости безопасности", - "prompt.example.6": "Добавить юнит-тесты для сервиса пользователя", - "prompt.example.7": "Рефакторить эту функцию для лучшей читаемости", - "prompt.example.8": "Что означает эта ошибка?", - "prompt.example.9": "Помоги мне отладить эту проблему", - "prompt.example.10": "Сгенерировать документацию API", - "prompt.example.11": "Оптимизировать запросы к базе данных", - "prompt.example.12": "Добавить валидацию ввода", - "prompt.example.13": "Создать новый компонент для...", - "prompt.example.14": "Как развернуть этот проект?", - "prompt.example.15": "Проверь мой код на лучшие практики", - "prompt.example.16": "Добавить обработку ошибок в эту функцию", - "prompt.example.17": "Объясни этот паттерн regex", - "prompt.example.18": "Конвертировать это в TypeScript", - "prompt.example.19": "Добавить логирование по всему проекту", - "prompt.example.20": "Какие зависимости устарели?", - "prompt.example.21": "Помоги написать скрипт миграции", - "prompt.example.22": "Реализовать кэширование для этой конечной точки", - "prompt.example.23": "Добавить пагинацию в этот список", - "prompt.example.24": "Создать CLI команду для...", - "prompt.example.25": "Как работают переменные окружения здесь?", - - "prompt.popover.emptyResults": "Нет совпадений", - "prompt.popover.emptyCommands": "Нет совпадающих команд", - "prompt.dropzone.label": "Перетащите сюда изображения, PDF или текстовые файлы", - "prompt.dropzone.file.label": "Отпустите для @упоминания файла", - "prompt.slash.badge.custom": "своё", - "prompt.slash.badge.skill": "навык", - "prompt.slash.badge.mcp": "mcp", - "prompt.context.active": "активно", - "prompt.context.includeActiveFile": "Включить активный файл", - "prompt.context.removeActiveFile": "Удалить активный файл из контекста", - "prompt.context.removeFile": "Удалить файл из контекста", - "prompt.action.attachFile": "Прикрепить файл", - "prompt.attachment.remove": "Удалить вложение", - "prompt.action.send": "Отправить", - "prompt.action.stop": "Остановить", - - "prompt.toast.pasteUnsupported.title": "Неподдерживаемое вложение", - "prompt.toast.pasteUnsupported.description": "Здесь можно прикрепить только изображения, PDF или текстовые файлы.", - "prompt.toast.modelAgentRequired.title": "Выберите агента и модель", - "prompt.toast.modelAgentRequired.description": "Выберите агента и модель перед отправкой запроса.", - "prompt.toast.worktreeCreateFailed.title": "Не удалось создать worktree", - "prompt.toast.sessionCreateFailed.title": "Не удалось создать сессию", - "prompt.toast.shellSendFailed.title": "Не удалось отправить команду оболочки", - "prompt.toast.commandSendFailed.title": "Не удалось отправить команду", - "prompt.toast.promptSendFailed.title": "Не удалось отправить запрос", - "prompt.toast.promptSendFailed.description": "Не удалось получить сессию", - - "dialog.mcp.title": "MCP", - "dialog.mcp.description": "{{enabled}} из {{total}} включено", - "dialog.mcp.empty": "MCP не настроены", - - "dialog.lsp.empty": "LSP автоматически обнаружены по типам файлов", - "dialog.plugins.empty": "Плагины настроены в opencode.json", - - "mcp.status.connected": "подключено", - "mcp.status.failed": "ошибка", - "mcp.status.needs_auth": "требуется авторизация", - "mcp.status.disabled": "отключено", - - "dialog.fork.empty": "Нет сообщений для ответвления", - - "dialog.directory.search.placeholder": "Поиск папок", - "dialog.directory.empty": "Папки не найдены", - - "dialog.server.title": "Серверы", - "dialog.server.description": "Переключите сервер Kilo к которому подключается приложение.", - "dialog.server.search.placeholder": "Поиск серверов", - "dialog.server.empty": "Серверов пока нет", - "dialog.server.add.title": "Добавить сервер", - "dialog.server.add.url": "URL сервера", - "dialog.server.add.placeholder": "http://localhost:4096", - "dialog.server.add.error": "Не удалось подключиться к серверу", - "dialog.server.add.checking": "Проверка...", - "dialog.server.add.button": "Добавить сервер", - "dialog.server.add.name": "Имя сервера (необязательно)", - "dialog.server.add.namePlaceholder": "Localhost", - "dialog.server.add.username": "Имя пользователя (необязательно)", - "dialog.server.add.password": "Пароль (необязательно)", - "dialog.server.edit.title": "Редактировать сервер", - "dialog.server.default.title": "Сервер по умолчанию", - "dialog.server.default.description": - "Подключаться к этому серверу при запуске приложения вместо запуска локального сервера. Требуется перезапуск.", - "dialog.server.default.none": "Сервер не выбран", - "dialog.server.default.set": "Установить текущий сервер по умолчанию", - "dialog.server.default.clear": "Очистить", - "dialog.server.action.remove": "Удалить сервер", - - "dialog.server.menu.edit": "Редактировать", - "dialog.server.menu.default": "Сделать по умолчанию", - "dialog.server.menu.defaultRemove": "Удалить по умолчанию", - "dialog.server.menu.delete": "Удалить", - "dialog.server.current": "Текущий сервер", - "dialog.server.status.default": "По умолч.", - - "dialog.project.edit.title": "Редактировать проект", - "dialog.project.edit.name": "Название", - "dialog.project.edit.icon": "Иконка", - "dialog.project.edit.icon.alt": "Иконка проекта", - "dialog.project.edit.icon.hint": "Нажмите или перетащите изображение", - "dialog.project.edit.icon.recommended": "Рекомендуется: 128x128px", - "dialog.project.edit.color": "Цвет", - "dialog.project.edit.color.select": "Выбрать цвет {{color}}", - - "dialog.project.edit.worktree.startup": "Скрипт запуска рабочего пространства", - "dialog.project.edit.worktree.startup.description": - "Запускается после создания нового рабочего пространства (worktree).", - "dialog.project.edit.worktree.startup.placeholder": "например, bun install", - "context.breakdown.title": "Разбивка контекста", - "context.breakdown.note": - 'Приблизительная разбивка входных токенов. "Другое" включает определения инструментов и накладные расходы.', - "context.breakdown.system": "Система", - "context.breakdown.user": "Пользователь", - "context.breakdown.assistant": "Ассистент", - "context.breakdown.tool": "Вызовы инструментов", - "context.breakdown.other": "Другое", - - "context.systemPrompt.title": "Системный промпт", - "context.rawMessages.title": "Исходные сообщения", - - "context.stats.session": "Сессия", - "context.stats.messages": "Сообщения", - "context.stats.provider": "Провайдер", - "context.stats.model": "Модель", - "context.stats.limit": "Лимит контекста", - "context.stats.totalTokens": "Всего токенов", - "context.stats.usage": "Использование", - "context.stats.inputTokens": "Входные токены", - "context.stats.outputTokens": "Выходные токены", - "context.stats.reasoningTokens": "Токены рассуждения", - "context.stats.cacheTokens": "Токены кэша (чтение/запись)", - "context.stats.userMessages": "Сообщения пользователя", - "context.stats.assistantMessages": "Сообщения ассистента", - "context.stats.totalCost": "Общая стоимость", - "context.stats.sessionCreated": "Сессия создана", - "context.stats.lastActivity": "Последняя активность", - - "context.usage.tokens": "Токены", - "context.usage.usage": "Использование", - "context.usage.cost": "Стоимость", - "context.usage.clickToView": "Нажмите для просмотра контекста", - "context.usage.view": "Показать использование контекста", - - "language.en": "English", - "language.zh": "简体中文", - "language.zht": "繁體中文", - "language.ko": "한국어", - "language.de": "Deutsch", - "language.es": "Español", - "language.fr": "Français", - "language.da": "Dansk", - "language.ja": "日本語", - "language.pl": "Polski", - "language.ru": "Русский", - "language.ar": "العربية", - "language.no": "Norsk", - "language.br": "Português (Brasil)", - "language.bs": "Bosanski", - "language.th": "ไทย", - "language.tr": "Türkçe", - - "toast.language.title": "Язык", - "toast.language.description": "Переключено на {{language}}", - - "toast.theme.title": "Тема переключена", - "toast.scheme.title": "Цветовая схема", - - "toast.permissions.autoaccept.on.title": "Разрешения принимаются автоматически", - "toast.permissions.autoaccept.on.description": "Запросы на разрешения будут одобряться автоматически", - "toast.permissions.autoaccept.off.title": "Автоматическое принятие разрешений остановлено", - "toast.permissions.autoaccept.off.description": "Запросы на разрешения будут требовать одобрения", - - "toast.workspace.enabled.title": "Рабочие пространства включены", - "toast.workspace.enabled.description": "В боковой панели теперь отображаются несколько рабочих деревьев", - "toast.workspace.disabled.title": "Рабочие пространства отключены", - "toast.workspace.disabled.description": "В боковой панели отображается только главное рабочее дерево", - - "toast.model.none.title": "Модель не выбрана", - "toast.model.none.description": "Подключите провайдера для суммаризации сессии", - - "toast.file.loadFailed.title": "Не удалось загрузить файл", - - "toast.file.listFailed.title": "Не удалось получить список файлов", - "toast.context.noLineSelection.title": "Нет выделения строк", - "toast.context.noLineSelection.description": "Сначала выберите диапазон строк во вкладке файла.", - "toast.session.share.copyFailed.title": "Не удалось скопировать URL в буфер обмена", - "toast.session.share.success.title": "Сессия опубликована", - "toast.session.share.success.description": "URL скопирован в буфер обмена!", - "toast.session.share.failed.title": "Не удалось опубликовать сессию", - "toast.session.share.failed.description": "Произошла ошибка при публикации сессии", - - "toast.session.unshare.success.title": "Публикация отменена", - "toast.session.unshare.success.description": "Публикация успешно отменена!", - "toast.session.unshare.failed.title": "Не удалось отменить публикацию", - "toast.session.unshare.failed.description": "Произошла ошибка при отмене публикации", - - "toast.session.listFailed.title": "Не удалось загрузить сессии для {{project}}", - - "toast.update.title": "Доступно обновление", - "toast.update.description": "Новая версия Kilo ({{version}}) доступна для установки.", - "toast.update.action.installRestart": "Установить и перезапустить", - "toast.update.action.notYet": "Пока нет", - - "error.page.title": "Что-то пошло не так", - "error.page.description": "Произошла ошибка при загрузке приложения.", - "error.page.details.label": "Детали ошибки", - "error.page.action.restart": "Перезапустить", - "error.page.action.checking": "Проверка...", - "error.page.action.checkUpdates": "Проверить обновления", - "error.page.action.updateTo": "Обновить до {{version}}", - "error.page.report.prefix": "Пожалуйста, сообщите об этой ошибке команде Kilo", - "error.page.report.discord": "в Discord", - "error.page.version": "Версия: {{version}}", - - "error.dev.rootNotFound": - "Корневой элемент не найден. Вы забыли добавить его в index.html? Или, может быть, атрибут id был написан неправильно?", - - "error.globalSync.connectFailed": "Не удалось подключиться к серверу. Запущен ли сервер по адресу `{{url}}`?", - "directory.error.invalidUrl": "Недопустимая директория в URL.", - - "error.chain.unknown": "Неизвестная ошибка", - "error.chain.causedBy": "Причина:", - "error.chain.apiError": "Ошибка API", - "error.chain.status": "Статус: {{status}}", - "error.chain.retryable": "Повторная попытка: {{retryable}}", - "error.chain.responseBody": "Тело ответа:\n{{body}}", - "error.chain.didYouMean": "Возможно, вы имели в виду: {{suggestions}}", - "error.chain.modelNotFound": "Модель не найдена: {{provider}}/{{model}}", - "error.chain.checkConfig": "Проверьте названия провайдера/модели в конфиге (opencode.json)", - "error.chain.mcpFailed": - 'MCP сервер "{{name}}" завершился с ошибкой. Обратите внимание, что Kilo пока не поддерживает MCP авторизацию.', - "error.chain.providerAuthFailed": "Ошибка аутентификации провайдера ({{provider}}): {{message}}", - "error.chain.providerInitFailed": - 'Не удалось инициализировать провайдера "{{provider}}". Проверьте учётные данные и конфигурацию.', - "error.chain.configJsonInvalid": "Конфигурационный файл по адресу {{path}} не является валидным JSON(C)", - "error.chain.configJsonInvalidWithMessage": - "Конфигурационный файл по адресу {{path}} не является валидным JSON(C): {{message}}", - "error.chain.configDirectoryTypo": - 'Папка "{{dir}}" в {{path}} невалидна. Переименуйте папку в "{{suggestion}}" или удалите её. Это распространённая опечатка.', - "error.chain.configFrontmatterError": "Не удалось разобрать frontmatter в {{path}}:\n{{message}}", - "error.chain.configInvalid": "Конфигурационный файл по адресу {{path}} невалиден", - "error.chain.configInvalidWithMessage": "Конфигурационный файл по адресу {{path}} невалиден: {{message}}", - - "notification.permission.title": "Требуется разрешение", - "notification.permission.description": "{{sessionTitle}} в {{projectName}} требуется разрешение", - "notification.question.title": "Вопрос", - "notification.question.description": "У {{sessionTitle}} в {{projectName}} есть вопрос", - "notification.action.goToSession": "Перейти к сессии", - - "notification.session.responseReady.title": "Ответ готов", - "notification.session.error.title": "Ошибка сессии", - "notification.session.error.fallbackDescription": "Произошла ошибка", - - "home.recentProjects": "Недавние проекты", - "home.empty.title": "Нет недавних проектов", - "home.empty.description": "Начните с открытия локального проекта", - - "session.tab.session": "Сессия", - "session.tab.review": "Обзор", - "session.tab.context": "Контекст", - "session.panel.reviewAndFiles": "Обзор и файлы", - "session.review.filesChanged": "{{count}} файлов изменено", - "session.review.change.one": "Изменение", - "session.review.change.other": "Изменения", - "session.review.loadingChanges": "Загрузка изменений...", - "session.review.empty": "Изменений в этой сессии пока нет", - "session.review.noVcs": "Система контроля версий Git не обнаружена, изменения не отображаются", - "session.review.noSnapshot": "Отслеживание снимков отключено в настройках, поэтому изменения сессии недоступны", - "session.review.noChanges": "Нет изменений", - "session.files.selectToOpen": "Выберите файл, чтобы открыть", - "session.files.all": "Все файлы", - "session.files.empty": "Нет файлов", - "session.files.binaryContent": "Двоичный файл (содержимое не может быть отображено)", - "session.messages.renderEarlier": "Показать предыдущие сообщения", - "session.messages.loadingEarlier": "Загрузка предыдущих сообщений...", - "session.messages.loadEarlier": "Загрузить предыдущие сообщения", - "session.messages.loading": "Загрузка сообщений...", - "session.messages.jumpToLatest": "Перейти к последнему", - - "session.context.addToContext": "Добавить {{selection}} в контекст", - "session.todo.title": "Задачи", - "session.todo.collapse": "Свернуть", - "session.todo.expand": "Развернуть", - "session.followupDock.summary.one": "{{count}} сообщение в очереди", - "session.followupDock.summary.other": "{{count}} сообщений в очереди", - "session.followupDock.sendNow": "Отправить сейчас", - "session.followupDock.edit": "Редактировать", - "session.followupDock.collapse": "Свернуть сообщения в очереди", - "session.followupDock.expand": "Развернуть сообщения в очереди", - "session.revertDock.summary.one": "{{count}} сообщение возвращено", - "session.revertDock.summary.other": "{{count}} сообщений возвращено", - "session.revertDock.collapse": "Свернуть возвращённые сообщения", - "session.revertDock.expand": "Развернуть возвращённые сообщения", - "session.revertDock.restore": "Восстановить сообщение", - - "session.new.title": "Создавайте что угодно", - "session.new.worktree.main": "Основная ветка", - "session.new.worktree.mainWithBranch": "Основная ветка ({{branch}})", - "session.new.worktree.create": "Создать новый worktree", - "session.new.lastModified": "Последнее изменение", - - "session.header.search.placeholder": "Поиск {{project}}", - "session.header.searchFiles": "Поиск файлов", - "session.header.openIn": "Открыть в", - "session.header.open.action": "Открыть {{app}}", - "session.header.open.ariaLabel": "Открыть в {{app}}", - "session.header.open.menu": "Варианты открытия", - "session.header.open.copyPath": "Копировать путь", - - "status.popover.trigger": "Статус", - "status.popover.ariaLabel": "Настройки серверов", - "status.popover.tab.servers": "Серверы", - "status.popover.tab.mcp": "MCP", - "status.popover.tab.lsp": "LSP", - "status.popover.tab.plugins": "Плагины", - "status.popover.action.manageServers": "Управлять серверами", - - "session.share.popover.title": "Опубликовать в интернете", - "session.share.popover.description.shared": - "Эта сессия общедоступна. Доступ к ней может получить любой, у кого есть ссылка.", - "session.share.popover.description.unshared": - "Опубликуйте сессию в интернете. Доступ к ней сможет получить любой, у кого есть ссылка.", - "session.share.action.share": "Поделиться", - "session.share.action.publish": "Опубликовать", - "session.share.action.publishing": "Публикация...", - "session.share.action.unpublish": "Отменить публикацию", - "session.share.action.unpublishing": "Отмена публикации...", - "session.share.action.view": "Посмотреть", - "session.share.copy.copied": "Скопировано", - "session.share.copy.copyLink": "Копировать ссылку", - - "lsp.tooltip.none": "Нет LSP серверов", - "lsp.label.connected": "{{count}} LSP", - - "prompt.loading": "Загрузка запроса...", - "terminal.loading": "Загрузка терминала...", - "terminal.title": "Терминал", - "terminal.title.numbered": "Терминал {{number}}", - "terminal.close": "Закрыть терминал", - "terminal.connectionLost.title": "Соединение потеряно", - "terminal.connectionLost.description": - "Соединение с терминалом прервано. Это может произойти при перезапуске сервера.", - - "common.closeTab": "Закрыть вкладку", - "common.dismiss": "Закрыть", - "common.requestFailed": "Запрос не выполнен", - "common.moreOptions": "Дополнительные опции", - "common.learnMore": "Подробнее", - "common.rename": "Переименовать", - "common.reset": "Сбросить", - "common.archive": "Архивировать", - "common.delete": "Удалить", - "common.close": "Закрыть", - "common.edit": "Редактировать", - "common.loadMore": "Загрузить ещё", - "common.key.esc": "ESC", - - "sidebar.menu.toggle": "Переключить меню", - "sidebar.nav.projectsAndSessions": "Проекты и сессии", - "sidebar.settings": "Настройки", - "sidebar.help": "Помощь", - "sidebar.workspaces.enable": "Включить рабочие пространства", - "sidebar.workspaces.disable": "Отключить рабочие пространства", - "sidebar.gettingStarted.title": "Начало работы", - "sidebar.gettingStarted.line1": "Kilo включает бесплатные модели, чтобы вы могли начать сразу.", - "sidebar.gettingStarted.line2": - "Подключите любого провайдера для использования моделей, включая Claude, GPT, Gemini и др.", - "sidebar.project.recentSessions": "Недавние сессии", - "sidebar.project.viewAllSessions": "Посмотреть все сессии", - "sidebar.project.clearNotifications": "Очистить уведомления", - - "app.name.desktop": "Kilo Desktop", - "settings.section.desktop": "Приложение", - "settings.section.server": "Сервер", - "settings.tab.general": "Основные", - "settings.tab.shortcuts": "Горячие клавиши", - "settings.desktop.section.wsl": "WSL", - "settings.desktop.wsl.title": "Интеграция с WSL", - "settings.desktop.wsl.description": "Запускать сервер Kilo внутри WSL на Windows.", - - "settings.general.section.appearance": "Внешний вид", - "settings.general.section.notifications": "Системные уведомления", - "settings.general.section.updates": "Обновления", - "settings.general.section.sounds": "Звуковые эффекты", - "settings.general.section.feed": "Лента", - "settings.general.section.display": "Дисплей", - - "settings.general.row.language.title": "Язык", - "settings.general.row.language.description": "Изменить язык отображения Kilo", - "settings.general.row.appearance.title": "Внешний вид", - "settings.general.row.appearance.description": "Настройте как Kilo выглядит на вашем устройстве", - "settings.general.row.colorScheme.title": "Цветовая схема", - "settings.general.row.colorScheme.description": "Выберите, следует ли Kilo системной, светлой или тёмной теме", - "settings.general.row.theme.title": "Тема", - "settings.general.row.theme.description": "Настройте оформление Kilo.", - "settings.general.row.font.title": "Шрифт кода", - "settings.general.row.font.description": "Настройте шрифт, используемый в блоках кода", - "settings.general.row.terminalFont.title": "Terminal Font", - "settings.general.row.terminalFont.description": "Customise the font used in the terminal", - "settings.general.row.uiFont.title": "Шрифт интерфейса", - "settings.general.row.uiFont.description": "Настройте шрифт, используемый во всем интерфейсе", - "settings.general.row.followup.title": "Поведение уточняющих вопросов", - "settings.general.row.followup.description": - "Выберите, отправлять ли уточняющие вопросы сразу или помещать их в очередь", - "settings.general.row.followup.option.queue": "Очередь", - "settings.general.row.followup.option.steer": "Направлять", - "settings.general.row.reasoningSummaries.title": "Показывать сводки рассуждений", - "settings.general.row.reasoningSummaries.description": "Отображать сводки рассуждений модели в ленте", - - "settings.general.row.shellToolPartsExpanded.title": "Разворачивать элементы инструмента shell", - "settings.general.row.shellToolPartsExpanded.description": - "Показывать элементы инструмента shell в ленте развернутыми по умолчанию", - "settings.general.row.editToolPartsExpanded.title": "Разворачивать элементы инструмента edit", - "settings.general.row.editToolPartsExpanded.description": - "Показывать элементы инструментов edit, write и patch в ленте развернутыми по умолчанию", - "settings.general.row.showSessionProgressBar.title": "Показывать индикатор прогресса сессии", - "settings.general.row.showSessionProgressBar.description": - "Показывать анимированный индикатор прогресса вверху сессии, когда агент работает", - "settings.general.row.wayland.title": "Использовать нативный Wayland", - "settings.general.row.wayland.description": "Отключить X11 fallback на Wayland. Требуется перезапуск.", - "settings.general.row.wayland.tooltip": - "На Linux с мониторами разной частоты обновления нативный Wayland может быть стабильнее.", - - "settings.general.row.releaseNotes.title": "Примечания к выпуску", - "settings.general.row.releaseNotes.description": 'Показывать всплывающие окна "Что нового" после обновлений', - - "settings.updates.row.startup.title": "Проверять обновления при запуске", - "settings.updates.row.startup.description": "Автоматически проверять обновления при запуске Kilo", - "settings.updates.row.check.title": "Проверить обновления", - "settings.updates.row.check.description": "Проверить обновления вручную и установить, если доступны", - "settings.updates.action.checkNow": "Проверить сейчас", - "settings.updates.action.checking": "Проверка...", - "settings.updates.toast.latest.title": "У вас последняя версия", - "settings.updates.toast.latest.description": "Вы используете последнюю версию Kilo.", - "sound.option.none": "Нет", - "sound.option.alert01": "Alert 01", - "sound.option.alert02": "Alert 02", - "sound.option.alert03": "Alert 03", - "sound.option.alert04": "Alert 04", - "sound.option.alert05": "Alert 05", - "sound.option.alert06": "Alert 06", - "sound.option.alert07": "Alert 07", - "sound.option.alert08": "Alert 08", - "sound.option.alert09": "Alert 09", - "sound.option.alert10": "Alert 10", - "sound.option.bipbop01": "Bip-bop 01", - "sound.option.bipbop02": "Bip-bop 02", - "sound.option.bipbop03": "Bip-bop 03", - "sound.option.bipbop04": "Bip-bop 04", - "sound.option.bipbop05": "Bip-bop 05", - "sound.option.bipbop06": "Bip-bop 06", - "sound.option.bipbop07": "Bip-bop 07", - "sound.option.bipbop08": "Bip-bop 08", - "sound.option.bipbop09": "Bip-bop 09", - "sound.option.bipbop10": "Bip-bop 10", - "sound.option.staplebops01": "Staplebops 01", - "sound.option.staplebops02": "Staplebops 02", - "sound.option.staplebops03": "Staplebops 03", - "sound.option.staplebops04": "Staplebops 04", - "sound.option.staplebops05": "Staplebops 05", - "sound.option.staplebops06": "Staplebops 06", - "sound.option.staplebops07": "Staplebops 07", - "sound.option.nope01": "Nope 01", - "sound.option.nope02": "Nope 02", - "sound.option.nope03": "Nope 03", - "sound.option.nope04": "Nope 04", - "sound.option.nope05": "Nope 05", - "sound.option.nope06": "Nope 06", - "sound.option.nope07": "Nope 07", - "sound.option.nope08": "Nope 08", - "sound.option.nope09": "Nope 09", - "sound.option.nope10": "Nope 10", - "sound.option.nope11": "Nope 11", - "sound.option.nope12": "Nope 12", - "sound.option.yup01": "Yup 01", - "sound.option.yup02": "Yup 02", - "sound.option.yup03": "Yup 03", - "sound.option.yup04": "Yup 04", - "sound.option.yup05": "Yup 05", - "sound.option.yup06": "Yup 06", - - "settings.general.notifications.agent.title": "Агент", - "settings.general.notifications.agent.description": - "Показывать системное уведомление когда агент завершён или требует внимания", - "settings.general.notifications.permissions.title": "Разрешения", - "settings.general.notifications.permissions.description": - "Показывать системное уведомление когда требуется разрешение", - "settings.general.notifications.errors.title": "Ошибки", - "settings.general.notifications.errors.description": "Показывать системное уведомление когда происходит ошибка", - - "settings.general.sounds.agent.title": "Агент", - "settings.general.sounds.agent.description": "Воспроизводить звук когда агент завершён или требует внимания", - "settings.general.sounds.permissions.title": "Разрешения", - "settings.general.sounds.permissions.description": "Воспроизводить звук когда требуется разрешение", - "settings.general.sounds.errors.title": "Ошибки", - "settings.general.sounds.errors.description": "Воспроизводить звук когда происходит ошибка", - - "settings.shortcuts.title": "Горячие клавиши", - "settings.shortcuts.reset.button": "Сбросить к умолчаниям", - "settings.shortcuts.reset.toast.title": "Горячие клавиши сброшены", - "settings.shortcuts.reset.toast.description": "Горячие клавиши были сброшены к значениям по умолчанию.", - "settings.shortcuts.conflict.title": "Сочетание уже используется", - "settings.shortcuts.conflict.description": "{{keybind}} уже назначено для {{titles}}.", - "settings.shortcuts.unassigned": "Не назначено", - "settings.shortcuts.pressKeys": "Нажмите клавиши", - "settings.shortcuts.search.placeholder": "Поиск горячих клавиш", - "settings.shortcuts.search.empty": "Горячие клавиши не найдены", - - "settings.shortcuts.group.general": "Основные", - "settings.shortcuts.group.session": "Сессия", - "settings.shortcuts.group.navigation": "Навигация", - "settings.shortcuts.group.modelAndAgent": "Модель и агент", - "settings.shortcuts.group.terminal": "Терминал", - "settings.shortcuts.group.prompt": "Запрос", - - "settings.providers.title": "Провайдеры", - "settings.providers.description": "Настройки провайдеров будут доступны здесь.", - "settings.providers.section.connected": "Подключённые провайдеры", - "settings.providers.connected.empty": "Нет подключённых провайдеров", - "settings.providers.section.popular": "Популярные провайдеры", - "settings.providers.tag.environment": "Среда", - "settings.providers.tag.config": "Конфигурация", - "settings.providers.tag.custom": "Пользовательский", - "settings.providers.tag.other": "Другое", - "settings.models.title": "Модели", - "settings.models.description": "Настройки моделей будут доступны здесь.", - "settings.agents.title": "Агенты", - "settings.agents.description": "Настройки агентов будут доступны здесь.", - "settings.commands.title": "Команды", - "settings.commands.description": "Настройки команд будут доступны здесь.", - "settings.mcp.title": "MCP", - "settings.mcp.description": "Настройки MCP будут доступны здесь.", - - "settings.permissions.title": "Разрешения", - "settings.permissions.description": "Контролируйте какие инструменты сервер может использовать по умолчанию.", - "settings.permissions.section.tools": "Инструменты", - "settings.permissions.toast.updateFailed.title": "Не удалось обновить разрешения", - - "settings.permissions.action.allow": "Разрешить", - "settings.permissions.action.ask": "Спрашивать", - "settings.permissions.action.deny": "Запретить", - - "settings.permissions.tool.read.title": "Чтение", - "settings.permissions.tool.read.description": "Чтение файла (по совпадению пути)", - "settings.permissions.tool.edit.title": "Редактирование", - "settings.permissions.tool.edit.description": - "Изменение файлов, включая редактирование, запись, патчи и мульти-редактирование", - "settings.permissions.tool.glob.title": "Glob", - "settings.permissions.tool.glob.description": "Сопоставление файлов по паттернам glob", - "settings.permissions.tool.grep.title": "Grep", - "settings.permissions.tool.grep.description": "Поиск по содержимому файлов с использованием регулярных выражений", - "settings.permissions.tool.list.title": "List", - "settings.permissions.tool.list.description": "Список файлов в директории", - "settings.permissions.tool.bash.title": "Bash", - "settings.permissions.tool.bash.description": "Запуск команд оболочки", - "settings.permissions.tool.task.title": "Task", - "settings.permissions.tool.task.description": "Запуск подагентов", - "settings.permissions.tool.skill.title": "Skill", - "settings.permissions.tool.skill.description": "Загрузка навыка по имени", - "settings.permissions.tool.lsp.title": "LSP", - "settings.permissions.tool.lsp.description": "Запросы к языковому серверу", - "settings.permissions.tool.todowrite.title": "Todo Write", - "settings.permissions.tool.todowrite.description": "Обновление списка задач", - "settings.permissions.tool.webfetch.title": "Web Fetch", - "settings.permissions.tool.webfetch.description": "Получение контента по URL", - "settings.permissions.tool.websearch.title": "Web Search", - "settings.permissions.tool.websearch.description": "Поиск в интернете", - "settings.permissions.tool.codesearch.title": "Code Search", - "settings.permissions.tool.codesearch.description": "Поиск кода в интернете", - "settings.permissions.tool.external_directory.title": "Внешняя директория", - "settings.permissions.tool.external_directory.description": "Доступ к файлам вне директории проекта", - "settings.permissions.tool.doom_loop.title": "Doom Loop", - "settings.permissions.tool.doom_loop.description": - "Обнаружение повторяющихся вызовов инструментов с одинаковыми входными данными", - - "session.delete.failed.title": "Не удалось удалить сессию", - "session.delete.title": "Удалить сессию", - "session.delete.confirm": 'Удалить сессию "{{name}}"?', - "session.delete.button": "Удалить сессию", - - "workspace.new": "Новое рабочее пространство", - "workspace.type.local": "локальное", - "workspace.type.sandbox": "песочница", - "workspace.create.failed.title": "Не удалось создать рабочее пространство", - "workspace.delete.failed.title": "Не удалось удалить рабочее пространство", - "workspace.resetting.title": "Сброс рабочего пространства", - "workspace.resetting.description": "Это может занять минуту.", - "workspace.reset.failed.title": "Не удалось сбросить рабочее пространство", - "workspace.reset.success.title": "Рабочее пространство сброшено", - "workspace.reset.success.description": "Рабочее пространство теперь соответствует ветке по умолчанию.", - "workspace.error.stillPreparing": "Рабочее пространство всё ещё подготавливается", - "workspace.status.checking": "Проверка незафиксированных изменений...", - "workspace.status.error": "Не удалось проверить статус git.", - "workspace.status.clean": "Незафиксированных изменений не обнаружено.", - "workspace.status.dirty": "В этом рабочем пространстве обнаружены незафиксированные изменения.", - "workspace.delete.title": "Удалить рабочее пространство", - "workspace.delete.confirm": 'Удалить рабочее пространство "{{name}}"?', - "workspace.delete.button": "Удалить рабочее пространство", - "workspace.reset.title": "Сбросить рабочее пространство", - "workspace.reset.confirm": 'Сбросить рабочее пространство "{{name}}"?', - "workspace.reset.button": "Сбросить рабочее пространство", - "workspace.reset.archived.none": "Активные сессии не будут архивированы.", - "workspace.reset.archived.one": "1 сессия будет архивирована.", - "workspace.reset.archived.many": "{{count}} сессий будет архивировано.", - "workspace.reset.note": "Это сбросит рабочее пространство до соответствия ветке по умолчанию.", - "common.open": "Открыть", - "dialog.releaseNotes.action.getStarted": "Начать", - "dialog.releaseNotes.action.next": "Далее", - "dialog.releaseNotes.action.hideFuture": "Больше не показывать", - "dialog.releaseNotes.media.alt": "Превью релиза", - "toast.project.reloadFailed.title": "Не удалось перезагрузить {{project}}", - "error.server.invalidConfiguration": "Недопустимая конфигурация", - "common.moreCountSuffix": " (ещё {{count}})", - "common.time.justNow": "Только что", - "common.time.minutesAgo.short": "{{count}} мин назад", - "common.time.hoursAgo.short": "{{count}} ч назад", - "common.time.daysAgo.short": "{{count}} д назад", - "settings.providers.connected.environmentDescription": "Подключено из ваших переменных окружения", - "settings.providers.custom.description": "Добавить провайдера, совместимого с OpenAI, по базовому URL.", - - "app.server.unreachable": "Не удалось связаться с {{server}}", - "app.server.retrying": "Автоматическая повторная попытка...", - "app.server.otherServers": "Другие серверы", - "dialog.server.add.usernamePlaceholder": "имя пользователя", - "dialog.server.add.passwordPlaceholder": "пароль", - "server.row.noUsername": "нет имени пользователя", - "session.review.noVcs.createGit.title": "Создать репозиторий Git", - "session.review.noVcs.createGit.description": "Отслеживайте, просматривайте и отменяйте изменения в этом проекте", - "session.review.noVcs.createGit.actionLoading": "Создание репозитория Git...", - "session.review.noVcs.createGit.action": "Создать репозиторий Git", - "session.todo.progress": "Выполнено {{done}} из {{total}} задач", - "session.question.progress": "{{current}} из {{total}} вопросов", - "session.header.open.finder": "Finder", - "session.header.open.fileExplorer": "Проводник", - "session.header.open.fileManager": "Файловый менеджер", - "session.header.open.app.vscode": "VS Code", - "session.header.open.app.cursor": "Cursor", - "session.header.open.app.zed": "Zed", - "session.header.open.app.textmate": "TextMate", - "session.header.open.app.antigravity": "Antigravity", - "session.header.open.app.terminal": "Терминал", - "session.header.open.app.iterm2": "iTerm2", - "session.header.open.app.ghostty": "Ghostty", - "session.header.open.app.warp": "Warp", - "session.header.open.app.xcode": "Xcode", - "session.header.open.app.androidStudio": "Android Studio", - "session.header.open.app.powershell": "PowerShell", - "session.header.open.app.sublimeText": "Sublime Text", - "debugBar.ariaLabel": "Диагностика производительности разработки", - "debugBar.na": "н/д", - "debugBar.nav.label": "NAV", - "debugBar.nav.tip": - "Последний завершенный переход маршрута, затрагивающий страницу сеанса, измеренный от запуска маршрутизатора до первой отрисовки после стабилизации.", - "debugBar.fps.label": "FPS", - "debugBar.fps.tip": "Скользящая частота кадров в секунду за последние 5 секунд.", - "debugBar.frame.label": "FRAME", - "debugBar.frame.tip": "Худшее время кадра за последние 5 секунд.", - "debugBar.jank.label": "JANK", - "debugBar.jank.tip": "Кадры более 32 мс за последние 5 секунд.", - "debugBar.long.label": "LONG", - "debugBar.long.tip": "Заблокированное время и количество длинных задач за последние 5 секунд. Макс. задача: {{max}}.", - "debugBar.delay.label": "DELAY", - "debugBar.delay.tip": "Худшая наблюдаемая задержка ввода за последние 5 секунд.", - "debugBar.inp.label": "INP", - "debugBar.inp.tip": - "Приблизительная продолжительность взаимодействия за последние 5 секунд. Это похоже на INP, а не официальный Web Vitals INP.", - "debugBar.cls.label": "CLS", - "debugBar.cls.tip": "Кумулятивный сдвиг макета за текущее время жизни приложения.", - "debugBar.mem.label": "MEM", - "debugBar.mem.tipUnavailable": "Используемая куча JS по сравнению с лимитом кучи. Только Chromium.", - "debugBar.mem.tip": "Используемая куча JS по сравнению с лимитом кучи. {{used}} из {{limit}}.", - "common.key.ctrl": "Ctrl", - "common.key.alt": "Alt", - "common.key.shift": "Shift", - "common.key.meta": "Meta", - "common.key.space": "Пробел", - "common.key.backspace": "Backspace", - "common.key.enter": "Enter", - "common.key.tab": "Tab", - "common.key.delete": "Delete", - "common.key.home": "Home", - "common.key.end": "End", - "common.key.pageUp": "Page Up", - "common.key.pageDown": "Page Down", - "common.key.insert": "Insert", - "common.unknown": "неизвестно", - "error.page.circular": "[Циклично]", - "error.globalSDK.noServerAvailable": "Нет доступного сервера", - "error.globalSDK.serverNotAvailable": "Сервер недоступен", - "error.childStore.persistedCacheCreateFailed": "Не удалось создать постоянный кэш", - "error.childStore.persistedProjectMetadataCreateFailed": "Не удалось создать постоянные метаданные проекта", - "error.childStore.persistedProjectIconCreateFailed": "Не удалось создать постоянный значок проекта", - "error.childStore.storeCreateFailed": "Не удалось создать хранилище", - "terminal.connectionLost.abnormalClose": "WebSocket закрыт аварийно: {{code}}", -} diff --git a/packages/app/src/i18n/th.ts b/packages/app/src/i18n/th.ts deleted file mode 100644 index f1bc3e9f0e..0000000000 --- a/packages/app/src/i18n/th.ts +++ /dev/null @@ -1,924 +0,0 @@ -export const dict = { - "command.category.suggested": "แนะนำ", - "command.category.view": "มุมมอง", - "command.category.project": "โปรเจกต์", - "command.category.provider": "ผู้ให้บริการ", - "command.category.server": "เซิร์ฟเวอร์", - "command.category.session": "เซสชัน", - "command.category.theme": "ธีม", - "command.category.language": "ภาษา", - "command.category.file": "ไฟล์", - "command.category.context": "บริบท", - "command.category.terminal": "เทอร์มินัล", - "command.category.model": "โมเดล", - "command.category.mcp": "MCP", - "command.category.agent": "เอเจนต์", - "command.category.permissions": "สิทธิ์", - "command.category.workspace": "พื้นที่ทำงาน", - "command.category.settings": "การตั้งค่า", - - "theme.scheme.system": "ระบบ", - "theme.scheme.light": "สว่าง", - "theme.scheme.dark": "มืด", - - "command.sidebar.toggle": "สลับแถบข้าง", - "command.project.open": "เปิดโปรเจกต์", - "command.provider.connect": "เชื่อมต่อผู้ให้บริการ", - "command.server.switch": "สลับเซิร์ฟเวอร์", - "command.settings.open": "เปิดการตั้งค่า", - "command.session.previous": "เซสชันก่อนหน้า", - "command.session.next": "เซสชันถัดไป", - "command.session.previous.unseen": "เซสชันที่ยังไม่ได้อ่านก่อนหน้า", - "command.session.next.unseen": "เซสชันที่ยังไม่ได้อ่านถัดไป", - "command.session.archive": "จัดเก็บเซสชัน", - - "command.palette": "คำสั่งค้นหา", - - "command.theme.cycle": "เปลี่ยนธีม", - "command.theme.set": "ใช้ธีม: {{theme}}", - "command.theme.scheme.cycle": "เปลี่ยนโทนสี", - "command.theme.scheme.set": "ใช้โทนสี: {{scheme}}", - - "command.language.cycle": "เปลี่ยนภาษา", - "command.language.set": "ใช้ภาษา: {{language}}", - - "command.session.new": "เซสชันใหม่", - "command.file.open": "เปิดไฟล์", - "command.tab.close": "ปิดแท็บ", - "command.context.addSelection": "เพิ่มส่วนที่เลือกไปยังบริบท", - "command.context.addSelection.description": "เพิ่มบรรทัดที่เลือกจากไฟล์ปัจจุบัน", - "command.input.focus": "โฟกัสช่องป้อนข้อมูล", - "command.terminal.toggle": "สลับเทอร์มินัล", - "command.fileTree.toggle": "สลับต้นไม้ไฟล์", - "command.review.toggle": "สลับการตรวจสอบ", - "command.terminal.new": "เทอร์มินัลใหม่", - "command.terminal.new.description": "สร้างแท็บเทอร์มินัลใหม่", - "command.steps.toggle": "สลับขั้นตอน", - "command.steps.toggle.description": "แสดงหรือซ่อนขั้นตอนสำหรับข้อความปัจจุบัน", - "command.message.previous": "ข้อความก่อนหน้า", - "command.message.previous.description": "ไปที่ข้อความผู้ใช้ก่อนหน้า", - "command.message.next": "ข้อความถัดไป", - "command.message.next.description": "ไปที่ข้อความผู้ใช้ถัดไป", - "command.model.choose": "เลือกโมเดล", - "command.model.choose.description": "เลือกโมเดลอื่น", - "command.mcp.toggle": "สลับ MCPs", - "command.mcp.toggle.description": "สลับ MCPs", - "command.agent.cycle": "เปลี่ยนเอเจนต์", - "command.agent.cycle.description": "สลับไปยังเอเจนต์ถัดไป", - "command.agent.cycle.reverse": "เปลี่ยนเอเจนต์ย้อนกลับ", - "command.agent.cycle.reverse.description": "สลับไปยังเอเจนต์ก่อนหน้า", - "command.model.variant.cycle": "เปลี่ยนความพยายามในการคิด", - "command.model.variant.cycle.description": "สลับไปยังระดับความพยายามถัดไป", - "command.prompt.mode.shell": "เชลล์", - "command.prompt.mode.normal": "พรอมต์", - "command.permissions.autoaccept.enable": "ยอมรับสิทธิ์โดยอัตโนมัติ", - "command.permissions.autoaccept.disable": "หยุดยอมรับสิทธิ์โดยอัตโนมัติ", - "command.workspace.toggle": "สลับพื้นที่ทำงาน", - "command.workspace.toggle.description": "เปิดหรือปิดใช้งานพื้นที่ทำงานหลายรายการในแถบด้านข้าง", - "command.session.undo": "ยกเลิก", - "command.session.undo.description": "ยกเลิกข้อความล่าสุด", - "command.session.redo": "ทำซ้ำ", - "command.session.redo.description": "ทำซ้ำข้อความที่ถูกยกเลิกล่าสุด", - "command.session.compact": "บีบอัดเซสชัน", - "command.session.compact.description": "สรุปเซสชันเพื่อลดขนาดบริบท", - "command.session.fork": "แตกแขนงจากข้อความ", - "command.session.fork.description": "สร้างเซสชันใหม่จากข้อความก่อนหน้า", - "command.session.share": "แชร์เซสชัน", - "command.session.share.description": "แชร์เซสชันนี้และคัดลอก URL ไปยังคลิปบอร์ด", - "command.session.unshare": "ยกเลิกการแชร์เซสชัน", - "command.session.unshare.description": "หยุดการแชร์เซสชันนี้", - - "palette.search.placeholder": "ค้นหาไฟล์ คำสั่ง และเซสชัน", - "palette.empty": "ไม่พบผลลัพธ์", - "palette.group.commands": "คำสั่ง", - "palette.group.files": "ไฟล์", - - "dialog.provider.search.placeholder": "ค้นหาผู้ให้บริการ", - "dialog.provider.empty": "ไม่พบผู้ให้บริการ", - "dialog.provider.group.popular": "ยอดนิยม", - "dialog.provider.group.other": "อื่น ๆ", - "dialog.provider.tag.recommended": "แนะนำ", - "dialog.provider.opencode.note": "โมเดลที่คัดสรร รวมถึง Claude, GPT, Gemini และอื่น ๆ", - "dialog.provider.opencode.tagline": "โมเดลที่เชื่อถือได้และปรับให้เหมาะสม", - "dialog.provider.opencodeGo.tagline": "การสมัครสมาชิกราคาประหยัดสำหรับทุกคน", - "dialog.provider.anthropic.note": "เข้าถึงโมเดล Claude โดยตรง รวมถึง Pro และ Max", - "dialog.provider.copilot.note": "โมเดล AI สำหรับการช่วยเหลือในการเขียนโค้ดผ่าน GitHub Copilot", - "dialog.provider.openai.note": "โมเดล GPT สำหรับงาน AI ทั่วไปที่รวดเร็วและมีความสามารถ", - "dialog.provider.google.note": "โมเดล Gemini สำหรับการตอบสนองที่รวดเร็วและมีโครงสร้าง", - "dialog.provider.openrouter.note": "เข้าถึงโมเดลที่รองรับทั้งหมดจากผู้ให้บริการเดียว", - "dialog.provider.vercel.note": "การเข้าถึงโมเดล AI แบบรวมด้วยการกำหนดเส้นทางอัจฉริยะ", - - "dialog.model.select.title": "เลือกโมเดล", - "dialog.model.search.placeholder": "ค้นหาโมเดล", - "dialog.model.empty": "ไม่พบผลลัพธ์โมเดล", - "dialog.model.manage": "จัดการโมเดล", - "dialog.model.manage.description": "ปรับแต่งโมเดลที่จะปรากฏในตัวเลือกโมเดล", - "dialog.model.manage.provider.toggle": "สลับโมเดลทั้งหมดของ {{provider}}", - - "dialog.model.unpaid.freeModels.title": "โมเดลฟรีที่จัดหาให้โดย Kilo", - "dialog.model.unpaid.addMore.title": "เพิ่มโมเดลเพิ่มเติมจากผู้ให้บริการยอดนิยม", - - "dialog.provider.viewAll": "แสดงผู้ให้บริการเพิ่มเติม", - - "provider.connect.title": "เชื่อมต่อ {{provider}}", - "provider.connect.title.anthropicProMax": "เข้าสู่ระบบด้วย Claude Pro/Max", - "provider.connect.selectMethod": "เลือกวิธีการเข้าสู่ระบบสำหรับ {{provider}}", - "provider.connect.method.apiKey": "คีย์ API", - "provider.connect.status.inProgress": "กำลังอนุญาต...", - "provider.connect.status.waiting": "รอการอนุญาต...", - "provider.connect.status.failed": "การอนุญาตล้มเหลว: {{error}}", - "provider.connect.apiKey.description": - "ป้อนคีย์ API ของ {{provider}} เพื่อเชื่อมต่อบัญชีและใช้โมเดล {{provider}} ใน Kilo", - "provider.connect.apiKey.label": "คีย์ API ของ {{provider}}", - "provider.connect.apiKey.placeholder": "คีย์ API", - "provider.connect.apiKey.required": "ต้องใช้คีย์ API", - "provider.connect.opencodeZen.line1": - "OpenCode Zen ให้คุณเข้าถึงชุดโมเดลที่เชื่อถือได้และปรับแต่งแล้วสำหรับเอเจนต์การเขียนโค้ด", - "provider.connect.opencodeZen.line2": - "ด้วยคีย์ API เดียวคุณจะได้รับการเข้าถึงโมเดล เช่น Claude, GPT, Gemini, GLM และอื่น ๆ", - "provider.connect.opencodeZen.visit.prefix": "เยี่ยมชม ", - "provider.connect.opencodeZen.visit.link": "opencode.ai/zen", - "provider.connect.opencodeZen.visit.suffix": " เพื่อรวบรวมคีย์ API ของคุณ", - "provider.connect.oauth.code.visit.prefix": "เยี่ยมชม ", - "provider.connect.oauth.code.visit.link": "ลิงก์นี้", - "provider.connect.oauth.code.visit.suffix": - " เพื่อรวบรวมรหัสการอนุญาตของคุณเพื่อเชื่อมต่อบัญชีและใช้โมเดล {{provider}} ใน Kilo", - "provider.connect.oauth.code.label": "รหัสการอนุญาต {{method}}", - "provider.connect.oauth.code.placeholder": "รหัสการอนุญาต", - "provider.connect.oauth.code.required": "ต้องใช้รหัสการอนุญาต", - "provider.connect.oauth.code.invalid": "รหัสการอนุญาตไม่ถูกต้อง", - "provider.connect.oauth.auto.visit.prefix": "เยี่ยมชม ", - "provider.connect.oauth.auto.visit.link": "ลิงก์นี้", - "provider.connect.oauth.auto.visit.suffix": " และป้อนรหัสด้านล่างเพื่อเชื่อมต่อบัญชีและใช้โมเดล {{provider}} ใน Kilo", - "provider.connect.oauth.auto.confirmationCode": "รหัสยืนยัน", - "provider.connect.toast.connected.title": "{{provider}} ที่เชื่อมต่อแล้ว", - "provider.connect.toast.connected.description": "โมเดล {{provider}} พร้อมใช้งานแล้ว", - - "provider.custom.title": "ผู้ให้บริการที่กำหนดเอง", - "provider.custom.description.prefix": "กำหนดค่าผู้ให้บริการที่เข้ากันได้กับ OpenAI ดู ", - "provider.custom.description.link": "เอกสารการกำหนดค่าผู้ให้บริการ", - "provider.custom.description.suffix": ".", - "provider.custom.field.providerID.label": "รหัสผู้ให้บริการ", - "provider.custom.field.providerID.placeholder": "myprovider", - "provider.custom.field.providerID.description": "ตัวอักษรพิมพ์เล็ก ตัวเลข ยัติภังค์ หรือขีดล่าง", - "provider.custom.field.name.label": "ชื่อที่แสดง", - "provider.custom.field.name.placeholder": "My AI Provider", - "provider.custom.field.baseURL.label": "URL พื้นฐาน", - "provider.custom.field.baseURL.placeholder": "https://api.myprovider.com/v1", - "provider.custom.field.apiKey.label": "คีย์ API", - "provider.custom.field.apiKey.placeholder": "คีย์ API", - "provider.custom.field.apiKey.description": "ไม่บังคับ เว้นว่างไว้หากคุณจัดการการยืนยันตัวตนผ่านส่วนหัว", - "provider.custom.models.label": "โมเดล", - "provider.custom.models.id.label": "รหัส", - "provider.custom.models.id.placeholder": "model-id", - "provider.custom.models.name.label": "ชื่อ", - "provider.custom.models.name.placeholder": "ชื่อที่แสดง", - "provider.custom.models.remove": "ลบโมเดล", - "provider.custom.models.add": "เพิ่มโมเดล", - "provider.custom.headers.label": "ส่วนหัว (ไม่บังคับ)", - "provider.custom.headers.key.label": "ส่วนหัว", - "provider.custom.headers.key.placeholder": "Header-Name", - "provider.custom.headers.value.label": "ค่า", - "provider.custom.headers.value.placeholder": "ค่า", - "provider.custom.headers.remove": "ลบส่วนหัว", - "provider.custom.headers.add": "เพิ่มส่วนหัว", - "provider.custom.error.providerID.required": "ต้องระบุรหัสผู้ให้บริการ", - "provider.custom.error.providerID.format": "ใช้ตัวอักษรพิมพ์เล็ก ตัวเลข ยัติภังค์ หรือขีดล่าง", - "provider.custom.error.providerID.exists": "รหัสผู้ให้บริการนั้นมีอยู่แล้ว", - "provider.custom.error.name.required": "ต้องระบุชื่อที่แสดง", - "provider.custom.error.baseURL.required": "ต้องระบุ URL พื้นฐาน", - "provider.custom.error.baseURL.format": "ต้องขึ้นต้นด้วย http:// หรือ https://", - "provider.custom.error.required": "จำเป็น", - "provider.custom.error.duplicate": "ซ้ำ", - - "provider.disconnect.toast.disconnected.title": "{{provider}} ที่ยกเลิกการเชื่อมต่อแล้ว", - "provider.disconnect.toast.disconnected.description": "โมเดล {{provider}} ไม่พร้อมใช้งานอีกต่อไป", - - "model.tag.free": "ฟรี", - "model.tag.latest": "ล่าสุด", - "model.provider.anthropic": "Anthropic", - "model.provider.openai": "OpenAI", - "model.provider.google": "Google", - "model.provider.xai": "xAI", - "model.provider.meta": "Meta", - "model.input.text": "ข้อความ", - "model.input.image": "รูปภาพ", - "model.input.audio": "เสียง", - "model.input.video": "วิดีโอ", - "model.input.pdf": "PDF", - "model.tooltip.allows": "อนุญาต: {{inputs}}", - "model.tooltip.reasoning.allowed": "อนุญาตการใช้เหตุผล", - "model.tooltip.reasoning.none": "ไม่มีการใช้เหตุผล", - "model.tooltip.context": "ขีดจำกัดบริบท {{limit}}", - - "common.search.placeholder": "ค้นหา", - "common.goBack": "ย้อนกลับ", - "common.goForward": "นำทางไปข้างหน้า", - "common.loading": "กำลังโหลด", - "common.loading.ellipsis": "...", - "common.cancel": "ยกเลิก", - "common.connect": "เชื่อมต่อ", - "common.disconnect": "ยกเลิกการเชื่อมต่อ", - "common.continue": "ส่ง", - "common.submit": "ส่ง", - "common.save": "บันทึก", - "common.saving": "กำลังบันทึก...", - "common.default": "ค่าเริ่มต้น", - "common.attachment": "ไฟล์แนบ", - - "prompt.placeholder.shell": "ป้อนคำสั่งเชลล์... {{example}}", - "prompt.placeholder.normal": 'ถามอะไรก็ได้... "{{example}}"', - "prompt.placeholder.simple": "ถามอะไรก็ได้...", - "prompt.placeholder.summarizeComments": "สรุปความคิดเห็น…", - "prompt.placeholder.summarizeComment": "สรุปความคิดเห็น…", - "prompt.mode.shell": "เชลล์", - "prompt.mode.normal": "พรอมต์", - "prompt.mode.shell.exit": "กด esc เพื่อออก", - - "prompt.example.1": "แก้ไข TODO ในโค้ดเบส", - "prompt.example.2": "เทคโนโลยีของโปรเจกต์นี้คืออะไร?", - "prompt.example.3": "แก้ไขการทดสอบที่เสีย", - "prompt.example.4": "อธิบายวิธีการทำงานของการตรวจสอบสิทธิ์", - "prompt.example.5": "ค้นหาและแก้ไขช่องโหว่ความปลอดภัย", - "prompt.example.6": "เพิ่มการทดสอบหน่วยสำหรับบริการผู้ใช้", - "prompt.example.7": "ปรับโครงสร้างฟังก์ชันนี้ให้อ่านง่ายขึ้น", - "prompt.example.8": "ข้อผิดพลาดนี้หมายความว่าอะไร?", - "prompt.example.9": "ช่วยฉันดีบักปัญหานี้", - "prompt.example.10": "สร้างเอกสาร API", - "prompt.example.11": "ปรับปรุงการสืบค้นฐานข้อมูล", - "prompt.example.12": "เพิ่มการตรวจสอบข้อมูลนำเข้า", - "prompt.example.13": "สร้างคอมโพเนนต์ใหม่สำหรับ...", - "prompt.example.14": "ฉันจะทำให้โปรเจกต์นี้ทำงานได้อย่างไร?", - "prompt.example.15": "ตรวจสอบโค้ดของฉันเพื่อแนวทางปฏิบัติที่ดีที่สุด", - "prompt.example.16": "เพิ่มการจัดการข้อผิดพลาดในฟังก์ชันนี้", - "prompt.example.17": "อธิบายรูปแบบ regex นี้", - "prompt.example.18": "แปลงสิ่งนี้เป็น TypeScript", - "prompt.example.19": "เพิ่มการบันทึกทั่วทั้งโค้ดเบส", - "prompt.example.20": "มีการพึ่งพาอะไรที่ล้าสมัยอยู่?", - "prompt.example.21": "ช่วยฉันเขียนสคริปต์การย้ายข้อมูล", - "prompt.example.22": "ใช้งานแคชสำหรับจุดสิ้นสุดนี้", - "prompt.example.23": "เพิ่มการแบ่งหน้าในรายการนี้", - "prompt.example.24": "สร้างคำสั่ง CLI สำหรับ...", - "prompt.example.25": "ตัวแปรสภาพแวดล้อมทำงานอย่างไรที่นี่?", - - "prompt.popover.emptyResults": "ไม่พบผลลัพธ์ที่ตรงกัน", - "prompt.popover.emptyCommands": "ไม่พบคำสั่งที่ตรงกัน", - "prompt.dropzone.label": "ลากรูปภาพ, PDF หรือไฟล์ข้อความมาวางที่นี่", - "prompt.dropzone.file.label": "วางเพื่อ @กล่าวถึงไฟล์", - "prompt.slash.badge.custom": "กำหนดเอง", - "prompt.slash.badge.skill": "ทักษะ", - "prompt.slash.badge.mcp": "MCP", - "prompt.context.active": "ใช้งานอยู่", - "prompt.context.includeActiveFile": "รวมไฟล์ที่ใช้งานอยู่", - "prompt.context.removeActiveFile": "เอาไฟล์ที่ใช้งานอยู่ออกจากบริบท", - "prompt.context.removeFile": "เอาไฟล์ออกจากบริบท", - "prompt.action.attachFile": "แนบไฟล์", - "prompt.attachment.remove": "เอาไฟล์แนบออก", - "prompt.action.send": "ส่ง", - "prompt.action.stop": "หยุด", - - "prompt.toast.pasteUnsupported.title": "ไฟล์แนบที่ไม่รองรับ", - "prompt.toast.pasteUnsupported.description": "แนบได้เฉพาะรูปภาพ, PDF หรือไฟล์ข้อความเท่านั้น", - "prompt.toast.modelAgentRequired.title": "เลือกเอเจนต์และโมเดล", - "prompt.toast.modelAgentRequired.description": "เลือกเอเจนต์และโมเดลก่อนส่งพร้อมท์", - "prompt.toast.worktreeCreateFailed.title": "ไม่สามารถสร้าง worktree", - "prompt.toast.sessionCreateFailed.title": "ไม่สามารถสร้างเซสชัน", - "prompt.toast.shellSendFailed.title": "ไม่สามารถส่งคำสั่งเชลล์", - "prompt.toast.commandSendFailed.title": "ไม่สามารถส่งคำสั่ง", - "prompt.toast.promptSendFailed.title": "ไม่สามารถส่งพร้อมท์", - "prompt.toast.promptSendFailed.description": "ไม่สามารถดึงเซสชันได้", - - "dialog.mcp.title": "MCPs", - "dialog.mcp.description": "{{enabled}} จาก {{total}} ที่เปิดใช้งาน", - "dialog.mcp.empty": "ไม่มี MCP ที่กำหนดค่า", - - "dialog.lsp.empty": "LSPs ตรวจจับอัตโนมัติจากประเภทไฟล์", - "dialog.plugins.empty": "ปลั๊กอินที่กำหนดค่าใน opencode.json", - - "mcp.status.connected": "เชื่อมต่อแล้ว", - "mcp.status.failed": "ล้มเหลว", - "mcp.status.needs_auth": "ต้องการการตรวจสอบสิทธิ์", - "mcp.status.disabled": "ปิดใช้งาน", - - "dialog.fork.empty": "ไม่มีข้อความให้แตกแขนง", - - "dialog.directory.search.placeholder": "ค้นหาโฟลเดอร์", - "dialog.directory.empty": "ไม่พบโฟลเดอร์", - - "dialog.server.title": "เซิร์ฟเวอร์", - "dialog.server.description": "สลับเซิร์ฟเวอร์ Kilo ที่แอปนี้เชื่อมต่อด้วย", - "dialog.server.search.placeholder": "ค้นหาเซิร์ฟเวอร์", - "dialog.server.empty": "ยังไม่มีเซิร์ฟเวอร์", - "dialog.server.add.title": "เพิ่มเซิร์ฟเวอร์", - "dialog.server.add.url": "URL เซิร์ฟเวอร์", - "dialog.server.add.placeholder": "http://localhost:4096", - "dialog.server.add.error": "ไม่สามารถเชื่อมต่อกับเซิร์ฟเวอร์", - "dialog.server.add.checking": "กำลังตรวจสอบ...", - "dialog.server.add.button": "เพิ่มเซิร์ฟเวอร์", - "dialog.server.add.name": "ชื่อเซิร์ฟเวอร์ (ไม่บังคับ)", - "dialog.server.add.namePlaceholder": "Localhost", - "dialog.server.add.username": "ชื่อผู้ใช้ (ไม่บังคับ)", - "dialog.server.add.password": "รหัสผ่าน (ไม่บังคับ)", - "dialog.server.edit.title": "แก้ไขเซิร์ฟเวอร์", - "dialog.server.default.title": "เซิร์ฟเวอร์เริ่มต้น", - "dialog.server.default.description": - "เชื่อมต่อกับเซิร์ฟเวอร์นี้เมื่อเปิดแอปแทนการเริ่มเซิร์ฟเวอร์ในเครื่อง ต้องรีสตาร์ท", - "dialog.server.default.none": "ไม่ได้เลือกเซิร์ฟเวอร์", - "dialog.server.default.set": "ตั้งเซิร์ฟเวอร์ปัจจุบันเป็นค่าเริ่มต้น", - "dialog.server.default.clear": "ล้าง", - "dialog.server.action.remove": "เอาเซิร์ฟเวอร์ออก", - - "dialog.server.menu.edit": "แก้ไข", - "dialog.server.menu.default": "ตั้งเป็นค่าเริ่มต้น", - "dialog.server.menu.defaultRemove": "เอาค่าเริ่มต้นออก", - "dialog.server.menu.delete": "ลบ", - "dialog.server.current": "เซิร์ฟเวอร์ปัจจุบัน", - "dialog.server.status.default": "ค่าเริ่มต้น", - - "dialog.project.edit.title": "แก้ไขโปรเจกต์", - "dialog.project.edit.name": "ชื่อ", - "dialog.project.edit.icon": "ไอคอน", - "dialog.project.edit.icon.alt": "ไอคอนโปรเจกต์", - "dialog.project.edit.icon.hint": "คลิกหรือลากรูปภาพ", - "dialog.project.edit.icon.recommended": "แนะนำ: 128x128px", - "dialog.project.edit.color": "สี", - "dialog.project.edit.color.select": "เลือกสี {{color}}", - "dialog.project.edit.worktree.startup": "สคริปต์เริ่มต้นพื้นที่ทำงาน", - "dialog.project.edit.worktree.startup.description": "ทำงานหลังจากสร้างพื้นที่ทำงานใหม่ (worktree)", - "dialog.project.edit.worktree.startup.placeholder": "เช่น bun install", - - "context.breakdown.title": "การแบ่งบริบท", - "context.breakdown.note": 'การแบ่งโดยประมาณของโทเค็นนำเข้า "อื่น ๆ" รวมถึงคำนิยามเครื่องมือและโอเวอร์เฮด', - "context.breakdown.system": "ระบบ", - "context.breakdown.user": "ผู้ใช้", - "context.breakdown.assistant": "ผู้ช่วย", - "context.breakdown.tool": "การเรียกเครื่องมือ", - "context.breakdown.other": "อื่น ๆ", - - "context.systemPrompt.title": "พร้อมท์ระบบ", - "context.rawMessages.title": "ข้อความดิบ", - - "context.stats.session": "เซสชัน", - "context.stats.messages": "ข้อความ", - "context.stats.provider": "ผู้ให้บริการ", - "context.stats.model": "โมเดล", - "context.stats.limit": "ขีดจำกัดบริบท", - "context.stats.totalTokens": "โทเค็นทั้งหมด", - "context.stats.usage": "การใช้งาน", - "context.stats.inputTokens": "โทเค็นนำเข้า", - "context.stats.outputTokens": "โทเค็นส่งออก", - "context.stats.reasoningTokens": "โทเค็นการใช้เหตุผล", - "context.stats.cacheTokens": "โทเค็นแคช (อ่าน/เขียน)", - "context.stats.userMessages": "ข้อความผู้ใช้", - "context.stats.assistantMessages": "ข้อความผู้ช่วย", - "context.stats.totalCost": "ต้นทุนทั้งหมด", - "context.stats.sessionCreated": "สร้างเซสชันเมื่อ", - "context.stats.lastActivity": "กิจกรรมล่าสุด", - - "context.usage.tokens": "โทเค็น", - "context.usage.usage": "การใช้งาน", - "context.usage.cost": "ต้นทุน", - "context.usage.clickToView": "คลิกเพื่อดูบริบท", - "context.usage.view": "ดูการใช้บริบท", - - "language.en": "English", - "language.zh": "简体中文", - "language.zht": "繁體中文", - "language.ko": "한국어", - "language.de": "Deutsch", - "language.es": "Español", - "language.fr": "Français", - "language.da": "Dansk", - "language.ja": "日本語", - "language.pl": "Polski", - "language.ru": "Русский", - "language.ar": "العربية", - "language.no": "Norsk", - "language.br": "Português (Brasil)", - "language.bs": "Bosanski", - "language.th": "ไทย", - "language.tr": "Türkçe", - - "toast.language.title": "ภาษา", - "toast.language.description": "สลับไปที่ {{language}}", - - "toast.theme.title": "สลับธีมแล้ว", - "toast.scheme.title": "โทนสี", - - "toast.workspace.enabled.title": "เปิดใช้งานพื้นที่ทำงานแล้ว", - "toast.workspace.enabled.description": "ตอนนี้จะแสดง worktree หลายรายการในแถบด้านข้าง", - "toast.workspace.disabled.title": "ปิดใช้งานพื้นที่ทำงานแล้ว", - "toast.workspace.disabled.description": "จะแสดงเฉพาะ worktree หลักในแถบด้านข้าง", - - "toast.permissions.autoaccept.on.title": "กำลังยอมรับสิทธิ์โดยอัตโนมัติ", - "toast.permissions.autoaccept.on.description": "คำขอสิทธิ์จะได้รับการอนุมัติโดยอัตโนมัติ", - "toast.permissions.autoaccept.off.title": "หยุดยอมรับสิทธิ์โดยอัตโนมัติแล้ว", - "toast.permissions.autoaccept.off.description": "คำขอสิทธิ์จะต้องได้รับการอนุมัติ", - - "toast.model.none.title": "ไม่ได้เลือกโมเดล", - "toast.model.none.description": "เชื่อมต่อผู้ให้บริการเพื่อสรุปเซสชันนี้", - - "toast.file.loadFailed.title": "ไม่สามารถโหลดไฟล์", - "toast.file.listFailed.title": "ไม่สามารถแสดงรายการไฟล์", - - "toast.context.noLineSelection.title": "ไม่มีการเลือกบรรทัด", - "toast.context.noLineSelection.description": "เลือกช่วงบรรทัดในแท็บไฟล์ก่อน", - - "toast.session.share.copyFailed.title": "ไม่สามารถคัดลอก URL ไปยังคลิปบอร์ด", - "toast.session.share.success.title": "แชร์เซสชันแล้ว", - "toast.session.share.success.description": "คัดลอก URL แชร์ไปยังคลิปบอร์ดแล้ว!", - "toast.session.share.failed.title": "ไม่สามารถแชร์เซสชัน", - "toast.session.share.failed.description": "เกิดข้อผิดพลาดระหว่างการแชร์เซสชัน", - - "toast.session.unshare.success.title": "ยกเลิกการแชร์เซสชันแล้ว", - "toast.session.unshare.success.description": "ยกเลิกการแชร์เซสชันสำเร็จ!", - "toast.session.unshare.failed.title": "ไม่สามารถยกเลิกการแชร์เซสชัน", - "toast.session.unshare.failed.description": "เกิดข้อผิดพลาดระหว่างการยกเลิกการแชร์เซสชัน", - - "toast.session.listFailed.title": "ไม่สามารถโหลดเซสชันสำหรับ {{project}}", - - "toast.update.title": "มีการอัปเดต", - "toast.update.description": "เวอร์ชันใหม่ของ Kilo ({{version}}) พร้อมใช้งานสำหรับติดตั้ง", - "toast.update.action.installRestart": "ติดตั้งและรีสตาร์ท", - "toast.update.action.notYet": "ยังไม่", - - "error.page.title": "เกิดข้อผิดพลาด", - "error.page.description": "เกิดข้อผิดพลาดระหว่างการโหลดแอปพลิเคชัน", - "error.page.details.label": "รายละเอียดข้อผิดพลาด", - "error.page.action.restart": "รีสตาร์ท", - "error.page.action.checking": "กำลังตรวจสอบ...", - "error.page.action.checkUpdates": "ตรวจสอบการอัปเดต", - "error.page.action.updateTo": "อัปเดตเป็น {{version}}", - "error.page.report.prefix": "โปรดรายงานข้อผิดพลาดนี้ให้ทีม Kilo", - "error.page.report.discord": "บน Discord", - "error.page.version": "เวอร์ชัน: {{version}}", - - "error.dev.rootNotFound": "ไม่พบองค์ประกอบรูท คุณลืมเพิ่มใน index.html หรือบางทีแอตทริบิวต์ id อาจสะกดผิด?", - - "error.globalSync.connectFailed": "ไม่สามารถเชื่อมต่อกับเซิร์ฟเวอร์ มีเซิร์ฟเวอร์ทำงานอยู่ที่ `{{url}}` หรือไม่?", - "directory.error.invalidUrl": "ไดเรกทอรีใน URL ไม่ถูกต้อง", - - "error.chain.unknown": "ข้อผิดพลาดที่ไม่รู้จัก", - "error.chain.causedBy": "สาเหตุ:", - "error.chain.apiError": "ข้อผิดพลาด API", - "error.chain.status": "สถานะ: {{status}}", - "error.chain.retryable": "สามารถลองใหม่: {{retryable}}", - "error.chain.responseBody": "เนื้อหาการตอบสนอง:\n{{body}}", - "error.chain.didYouMean": "คุณหมายถึง: {{suggestions}}", - "error.chain.modelNotFound": "ไม่พบโมเดล: {{provider}}/{{model}}", - "error.chain.checkConfig": "ตรวจสอบการกำหนดค่าของคุณ (opencode.json) ชื่อผู้ให้บริการ/โมเดล", - "error.chain.mcpFailed": 'เซิร์ฟเวอร์ MCP "{{name}}" ล้มเหลว โปรดทราบว่า Kilo ยังไม่รองรับการตรวจสอบสิทธิ์ MCP', - "error.chain.providerAuthFailed": "การตรวจสอบสิทธิ์ผู้ให้บริการล้มเหลว ({{provider}}): {{message}}", - "error.chain.providerInitFailed": 'ไม่สามารถเริ่มต้นผู้ให้บริการ "{{provider}}" ตรวจสอบข้อมูลรับรองและการกำหนดค่า', - "error.chain.configJsonInvalid": "ไฟล์กำหนดค่าที่ {{path}} ไม่ใช่ JSON(C) ที่ถูกต้อง", - "error.chain.configJsonInvalidWithMessage": "ไฟล์กำหนดค่าที่ {{path}} ไม่ใช่ JSON(C) ที่ถูกต้อง: {{message}}", - "error.chain.configDirectoryTypo": - 'ไดเรกทอรี "{{dir}}" ใน {{path}} ไม่ถูกต้อง เปลี่ยนชื่อไดเรกทอรีเป็น "{{suggestion}}" หรือเอาออก นี่เป็นการสะกดผิดทั่วไป', - "error.chain.configFrontmatterError": "ไม่สามารถแยกวิเคราะห์ frontmatter ใน {{path}}:\n{{message}}", - "error.chain.configInvalid": "ไฟล์กำหนดค่าที่ {{path}} ไม่ถูกต้อง", - "error.chain.configInvalidWithMessage": "ไฟล์กำหนดค่าที่ {{path}} ไม่ถูกต้อง: {{message}}", - - "notification.permission.title": "ต้องการสิทธิ์", - "notification.permission.description": "{{sessionTitle}} ใน {{projectName}} ต้องการสิทธิ์", - "notification.question.title": "คำถาม", - "notification.question.description": "{{sessionTitle}} ใน {{projectName}} มีคำถาม", - "notification.action.goToSession": "ไปที่เซสชัน", - - "notification.session.responseReady.title": "การตอบสนองพร้อม", - "notification.session.error.title": "ข้อผิดพลาดเซสชัน", - "notification.session.error.fallbackDescription": "เกิดข้อผิดพลาด", - - "home.recentProjects": "โปรเจกต์ล่าสุด", - "home.empty.title": "ไม่มีโปรเจกต์ล่าสุด", - "home.empty.description": "เริ่มต้นโดยเปิดโปรเจกต์ในเครื่อง", - - "session.tab.session": "เซสชัน", - "session.tab.review": "ตรวจสอบ", - "session.tab.context": "บริบท", - "session.panel.reviewAndFiles": "ตรวจสอบและไฟล์", - "session.review.filesChanged": "{{count}} ไฟล์ที่เปลี่ยนแปลง", - "session.review.change.one": "การเปลี่ยนแปลง", - "session.review.change.other": "การเปลี่ยนแปลง", - "session.review.loadingChanges": "กำลังโหลดการเปลี่ยนแปลง...", - "session.review.empty": "ยังไม่มีการเปลี่ยนแปลงในเซสชันนี้", - "session.review.noVcs": "ไม่ตรวจพบระบบควบคุมเวอร์ชัน Git การเปลี่ยนแปลงจะไม่แสดง", - "session.review.noSnapshot": "การติดตามสแนปชอตถูกปิดใช้งานในการกำหนดค่า ดังนั้นการเปลี่ยนแปลงเซสชันจึงไม่พร้อมใช้งาน", - "session.review.noChanges": "ไม่มีการเปลี่ยนแปลง", - - "session.files.selectToOpen": "เลือกไฟล์เพื่อเปิด", - "session.files.empty": "ไม่มีไฟล์", - "session.files.all": "ไฟล์ทั้งหมด", - "session.files.binaryContent": "ไฟล์ไบนารี (ไม่สามารถแสดงเนื้อหาได้)", - - "session.messages.renderEarlier": "แสดงข้อความก่อนหน้า", - "session.messages.loadingEarlier": "กำลังโหลดข้อความก่อนหน้า...", - "session.messages.loadEarlier": "โหลดข้อความก่อนหน้า", - "session.messages.loading": "กำลังโหลดข้อความ...", - "session.messages.jumpToLatest": "ไปที่ล่าสุด", - - "session.context.addToContext": "เพิ่ม {{selection}} ไปยังบริบท", - "session.todo.title": "สิ่งที่ต้องทำ", - "session.todo.collapse": "ย่อ", - "session.todo.expand": "ขยาย", - "session.followupDock.summary.one": "{{count}} ข้อความในคิว", - "session.followupDock.summary.other": "{{count}} ข้อความในคิว", - "session.followupDock.sendNow": "ส่งทันที", - "session.followupDock.edit": "แก้ไข", - "session.followupDock.collapse": "ย่อข้อความในคิว", - "session.followupDock.expand": "ขยายข้อความในคิว", - "session.revertDock.summary.one": "{{count}} ข้อความที่ถูกย้อนกลับ", - "session.revertDock.summary.other": "{{count}} ข้อความที่ถูกย้อนกลับ", - "session.revertDock.collapse": "ย่อข้อความที่ถูกย้อนกลับ", - "session.revertDock.expand": "ขยายข้อความที่ถูกย้อนกลับ", - "session.revertDock.restore": "กู้คืนข้อความ", - - "session.new.title": "สร้างอะไรก็ได้", - "session.new.worktree.main": "สาขาหลัก", - "session.new.worktree.mainWithBranch": "สาขาหลัก ({{branch}})", - "session.new.worktree.create": "สร้าง worktree ใหม่", - "session.new.lastModified": "แก้ไขล่าสุด", - - "session.header.search.placeholder": "ค้นหา {{project}}", - "session.header.searchFiles": "ค้นหาไฟล์", - "session.header.openIn": "เปิดใน", - "session.header.open.action": "เปิด {{app}}", - "session.header.open.ariaLabel": "เปิดใน {{app}}", - "session.header.open.menu": "ตัวเลือกการเปิด", - "session.header.open.copyPath": "คัดลอกเส้นทาง", - - "status.popover.trigger": "สถานะ", - "status.popover.ariaLabel": "การกำหนดค่าเซิร์ฟเวอร์", - "status.popover.tab.servers": "เซิร์ฟเวอร์", - "status.popover.tab.mcp": "MCP", - "status.popover.tab.lsp": "LSP", - "status.popover.tab.plugins": "ปลั๊กอิน", - "status.popover.action.manageServers": "จัดการเซิร์ฟเวอร์", - - "session.share.popover.title": "เผยแพร่บนเว็บ", - "session.share.popover.description.shared": "เซสชันนี้เป็นสาธารณะบนเว็บ สามารถเข้าถึงได้โดยผู้ที่มีลิงก์", - "session.share.popover.description.unshared": "แชร์เซสชันสาธารณะบนเว็บ จะเข้าถึงได้โดยผู้ที่มีลิงก์", - "session.share.action.share": "แชร์", - "session.share.action.publish": "เผยแพร่", - "session.share.action.publishing": "กำลังเผยแพร่...", - "session.share.action.unpublish": "ยกเลิกการเผยแพร่", - "session.share.action.unpublishing": "กำลังยกเลิกการเผยแพร่...", - "session.share.action.view": "ดู", - "session.share.copy.copied": "คัดลอกแล้ว", - "session.share.copy.copyLink": "คัดลอกลิงก์", - - "lsp.tooltip.none": "ไม่มีเซิร์ฟเวอร์ LSP", - "lsp.label.connected": "{{count}} LSP", - - "prompt.loading": "กำลังโหลดพร้อมท์...", - "terminal.loading": "กำลังโหลดเทอร์มินัล...", - "terminal.title": "เทอร์มินัล", - "terminal.title.numbered": "เทอร์มินัล {{number}}", - "terminal.close": "ปิดเทอร์มินัล", - "terminal.connectionLost.title": "การเชื่อมต่อขาดหาย", - "terminal.connectionLost.description": "การเชื่อมต่อเทอร์มินัลถูกขัดจังหวะ อาจเกิดขึ้นเมื่อเซิร์ฟเวอร์รีสตาร์ท", - - "common.closeTab": "ปิดแท็บ", - "common.dismiss": "ปิด", - "common.requestFailed": "คำขอล้มเหลว", - "common.moreOptions": "ตัวเลือกเพิ่มเติม", - "common.learnMore": "เรียนรู้เพิ่มเติม", - "common.rename": "เปลี่ยนชื่อ", - "common.reset": "รีเซ็ต", - "common.archive": "จัดเก็บ", - "common.delete": "ลบ", - "common.close": "ปิด", - "common.edit": "แก้ไข", - "common.loadMore": "โหลดเพิ่มเติม", - "common.key.esc": "ESC", - - "sidebar.menu.toggle": "สลับเมนู", - "sidebar.nav.projectsAndSessions": "โปรเจกต์และเซสชัน", - "sidebar.settings": "การตั้งค่า", - "sidebar.help": "ช่วยเหลือ", - "sidebar.workspaces.enable": "เปิดใช้งานพื้นที่ทำงาน", - "sidebar.workspaces.disable": "ปิดใช้งานพื้นที่ทำงาน", - "sidebar.gettingStarted.title": "เริ่มต้นใช้งาน", - "sidebar.gettingStarted.line1": "Kilo รวมถึงโมเดลฟรีเพื่อให้คุณเริ่มต้นได้ทันที", - "sidebar.gettingStarted.line2": "เชื่อมต่อผู้ให้บริการใด ๆ เพื่อใช้โมเดล รวมถึง Claude, GPT, Gemini ฯลฯ", - "sidebar.project.recentSessions": "เซสชันล่าสุด", - "sidebar.project.viewAllSessions": "ดูเซสชันทั้งหมด", - "sidebar.project.clearNotifications": "ล้างการแจ้งเตือน", - - "app.name.desktop": "Kilo Desktop", - - "settings.section.desktop": "เดสก์ท็อป", - "settings.section.server": "เซิร์ฟเวอร์", - "settings.tab.general": "ทั่วไป", - "settings.tab.shortcuts": "ทางลัด", - "settings.desktop.section.wsl": "WSL", - "settings.desktop.wsl.title": "การรวม WSL", - "settings.desktop.wsl.description": "เรียกใช้เซิร์ฟเวอร์ Kilo ภายใน WSL บน Windows", - - "settings.general.section.appearance": "รูปลักษณ์", - "settings.general.section.notifications": "การแจ้งเตือนระบบ", - "settings.general.section.updates": "การอัปเดต", - "settings.general.section.sounds": "เสียงเอฟเฟกต์", - "settings.general.section.feed": "ฟีด", - "settings.general.section.display": "การแสดงผล", - - "settings.general.row.language.title": "ภาษา", - "settings.general.row.language.description": "เปลี่ยนภาษาที่แสดงสำหรับ Kilo", - "settings.general.row.appearance.title": "รูปลักษณ์", - "settings.general.row.appearance.description": "ปรับแต่งวิธีการที่ Kilo มีลักษณะบนอุปกรณ์ของคุณ", - "settings.general.row.colorScheme.title": "โทนสี", - "settings.general.row.colorScheme.description": "เลือกว่าจะให้ Kilo ใช้ธีมตามระบบ สว่าง หรือมืด", - "settings.general.row.theme.title": "ธีม", - "settings.general.row.theme.description": "ปรับแต่งวิธีการที่ Kilo มีธีม", - "settings.general.row.font.title": "ฟอนต์โค้ด", - "settings.general.row.font.description": "ปรับแต่งฟอนต์ที่ใช้ในบล็อกโค้ด", - "settings.general.row.terminalFont.title": "Terminal Font", - "settings.general.row.terminalFont.description": "Customise the font used in the terminal", - "settings.general.row.uiFont.title": "ฟอนต์ UI", - "settings.general.row.uiFont.description": "ปรับแต่งฟอนต์ที่ใช้ทั่วทั้งอินเทอร์เฟซ", - "settings.general.row.followup.title": "พฤติกรรมการติดตามผล", - "settings.general.row.followup.description": "เลือกว่าจะให้พร้อมท์ติดตามผลทำงานทันทีหรือรอในคิว", - "settings.general.row.followup.option.queue": "คิว", - "settings.general.row.followup.option.steer": "นำทาง", - "settings.general.row.reasoningSummaries.title": "แสดงสรุปการใช้เหตุผล", - "settings.general.row.reasoningSummaries.description": "แสดงสรุปการใช้เหตุผลของโมเดลในไทม์ไลน์", - "settings.general.row.shellToolPartsExpanded.title": "ขยายส่วนเครื่องมือ shell", - "settings.general.row.shellToolPartsExpanded.description": "แสดงส่วนเครื่องมือ shell แบบขยายตามค่าเริ่มต้นในไทม์ไลน์", - "settings.general.row.editToolPartsExpanded.title": "ขยายส่วนเครื่องมือ edit", - "settings.general.row.editToolPartsExpanded.description": - "แสดงส่วนเครื่องมือ edit, write และ patch แบบขยายตามค่าเริ่มต้นในไทม์ไลน์", - "settings.general.row.showSessionProgressBar.title": "แสดงแถบความคืบหน้าของเซสชัน", - "settings.general.row.showSessionProgressBar.description": - "แสดงแถบความคืบหน้าแบบเคลื่อนไหวที่ด้านบนของเซสชันเมื่อเอเจนต์กำลังทำงาน", - "settings.general.row.wayland.title": "ใช้ Wayland แบบเนทีฟ", - "settings.general.row.wayland.description": "ปิดใช้งาน X11 fallback บน Wayland ต้องรีสตาร์ท", - "settings.general.row.wayland.tooltip": "บน Linux ที่มีจอภาพรีเฟรชเรตแบบผสม Wayland แบบเนทีฟอาจเสถียรกว่า", - - "settings.general.row.releaseNotes.title": "บันทึกการอัปเดต", - "settings.general.row.releaseNotes.description": "แสดงป๊อปอัพ What's New หลังจากอัปเดต", - - "settings.updates.row.startup.title": "ตรวจสอบการอัปเดตเมื่อเริ่มต้น", - "settings.updates.row.startup.description": "ตรวจสอบการอัปเดตโดยอัตโนมัติเมื่อ Kilo เปิดใช้งาน", - "settings.updates.row.check.title": "ตรวจสอบการอัปเดต", - "settings.updates.row.check.description": "ตรวจสอบการอัปเดตด้วยตนเองและติดตั้งหากมี", - "settings.updates.action.checkNow": "ตรวจสอบทันที", - "settings.updates.action.checking": "กำลังตรวจสอบ...", - "settings.updates.toast.latest.title": "คุณเป็นเวอร์ชันล่าสุดแล้ว", - "settings.updates.toast.latest.description": "คุณกำลังใช้งาน Kilo เวอร์ชันล่าสุด", - - "sound.option.none": "ไม่มี", - "sound.option.alert01": "เสียงเตือน 01", - "sound.option.alert02": "เสียงเตือน 02", - "sound.option.alert03": "เสียงเตือน 03", - "sound.option.alert04": "เสียงเตือน 04", - "sound.option.alert05": "เสียงเตือน 05", - "sound.option.alert06": "เสียงเตือน 06", - "sound.option.alert07": "เสียงเตือน 07", - "sound.option.alert08": "เสียงเตือน 08", - "sound.option.alert09": "เสียงเตือน 09", - "sound.option.alert10": "เสียงเตือน 10", - "sound.option.bipbop01": "Bip-bop 01", - "sound.option.bipbop02": "Bip-bop 02", - "sound.option.bipbop03": "Bip-bop 03", - "sound.option.bipbop04": "Bip-bop 04", - "sound.option.bipbop05": "Bip-bop 05", - "sound.option.bipbop06": "Bip-bop 06", - "sound.option.bipbop07": "Bip-bop 07", - "sound.option.bipbop08": "Bip-bop 08", - "sound.option.bipbop09": "Bip-bop 09", - "sound.option.bipbop10": "Bip-bop 10", - "sound.option.staplebops01": "Staplebops 01", - "sound.option.staplebops02": "Staplebops 02", - "sound.option.staplebops03": "Staplebops 03", - "sound.option.staplebops04": "Staplebops 04", - "sound.option.staplebops05": "Staplebops 05", - "sound.option.staplebops06": "Staplebops 06", - "sound.option.staplebops07": "Staplebops 07", - "sound.option.nope01": "Nope 01", - "sound.option.nope02": "Nope 02", - "sound.option.nope03": "Nope 03", - "sound.option.nope04": "Nope 04", - "sound.option.nope05": "Nope 05", - "sound.option.nope06": "Nope 06", - "sound.option.nope07": "Nope 07", - "sound.option.nope08": "Nope 08", - "sound.option.nope09": "Nope 09", - "sound.option.nope10": "Nope 10", - "sound.option.nope11": "Nope 11", - "sound.option.nope12": "Nope 12", - "sound.option.yup01": "Yup 01", - "sound.option.yup02": "Yup 02", - "sound.option.yup03": "Yup 03", - "sound.option.yup04": "Yup 04", - "sound.option.yup05": "Yup 05", - "sound.option.yup06": "Yup 06", - - "settings.general.notifications.agent.title": "เอเจนต์", - "settings.general.notifications.agent.description": "แสดงการแจ้งเตือนระบบเมื่อเอเจนต์เสร็จสิ้นหรือต้องการความสนใจ", - "settings.general.notifications.permissions.title": "สิทธิ์", - "settings.general.notifications.permissions.description": "แสดงการแจ้งเตือนระบบเมื่อต้องการสิทธิ์", - "settings.general.notifications.errors.title": "ข้อผิดพลาด", - "settings.general.notifications.errors.description": "แสดงการแจ้งเตือนระบบเมื่อเกิดข้อผิดพลาด", - - "settings.general.sounds.agent.title": "เอเจนต์", - "settings.general.sounds.agent.description": "เล่นเสียงเมื่อเอเจนต์เสร็จสิ้นหรือต้องการความสนใจ", - "settings.general.sounds.permissions.title": "สิทธิ์", - "settings.general.sounds.permissions.description": "เล่นเสียงเมื่อต้องการสิทธิ์", - "settings.general.sounds.errors.title": "ข้อผิดพลาด", - "settings.general.sounds.errors.description": "เล่นเสียงเมื่อเกิดข้อผิดพลาด", - - "settings.shortcuts.title": "ทางลัดแป้นพิมพ์", - "settings.shortcuts.reset.button": "รีเซ็ตเป็นค่าเริ่มต้น", - "settings.shortcuts.reset.toast.title": "รีเซ็ตทางลัดแล้ว", - "settings.shortcuts.reset.toast.description": "รีเซ็ตทางลัดแป้นพิมพ์เป็นค่าเริ่มต้นแล้ว", - "settings.shortcuts.conflict.title": "ทางลัดใช้งานอยู่แล้ว", - "settings.shortcuts.conflict.description": "{{keybind}} ถูกกำหนดให้กับ {{titles}} แล้ว", - "settings.shortcuts.unassigned": "ไม่ได้กำหนด", - "settings.shortcuts.pressKeys": "กดปุ่ม", - "settings.shortcuts.search.placeholder": "ค้นหาทางลัด", - "settings.shortcuts.search.empty": "ไม่พบทางลัด", - - "settings.shortcuts.group.general": "ทั่วไป", - "settings.shortcuts.group.session": "เซสชัน", - "settings.shortcuts.group.navigation": "การนำทาง", - "settings.shortcuts.group.modelAndAgent": "โมเดลและเอเจนต์", - "settings.shortcuts.group.terminal": "เทอร์มินัล", - "settings.shortcuts.group.prompt": "พร้อมท์", - - "settings.providers.title": "ผู้ให้บริการ", - "settings.providers.description": "การตั้งค่าผู้ให้บริการจะสามารถกำหนดค่าได้ที่นี่", - "settings.providers.section.connected": "ผู้ให้บริการที่เชื่อมต่อ", - "settings.providers.connected.empty": "ไม่มีผู้ให้บริการที่เชื่อมต่อ", - "settings.providers.section.popular": "ผู้ให้บริการยอดนิยม", - "settings.providers.tag.environment": "สภาพแวดล้อม", - "settings.providers.tag.config": "กำหนดค่า", - "settings.providers.tag.custom": "กำหนดเอง", - "settings.providers.tag.other": "อื่น ๆ", - "settings.models.title": "โมเดล", - "settings.models.description": "การตั้งค่าโมเดลจะสามารถกำหนดค่าได้ที่นี่", - "settings.agents.title": "เอเจนต์", - "settings.agents.description": "การตั้งค่าเอเจนต์จะสามารถกำหนดค่าได้ที่นี่", - "settings.commands.title": "คำสั่ง", - "settings.commands.description": "การตั้งค่าคำสั่งจะสามารถกำหนดค่าได้ที่นี่", - "settings.mcp.title": "MCP", - "settings.mcp.description": "การตั้งค่า MCP จะสามารถกำหนดค่าได้ที่นี่", - - "settings.permissions.title": "สิทธิ์", - "settings.permissions.description": "ควบคุมเครื่องมือที่เซิร์ฟเวอร์สามารถใช้โดยค่าเริ่มต้น", - "settings.permissions.section.tools": "เครื่องมือ", - "settings.permissions.toast.updateFailed.title": "ไม่สามารถอัปเดตสิทธิ์", - - "settings.permissions.action.allow": "อนุญาต", - "settings.permissions.action.ask": "ถาม", - "settings.permissions.action.deny": "ปฏิเสธ", - - "settings.permissions.tool.read.title": "อ่าน", - "settings.permissions.tool.read.description": "อ่านไฟล์ (ตรงกับเส้นทางไฟล์)", - "settings.permissions.tool.edit.title": "แก้ไข", - "settings.permissions.tool.edit.description": "แก้ไขไฟล์ รวมถึงการแก้ไข เขียน แพตช์ และแก้ไขหลายรายการ", - "settings.permissions.tool.glob.title": "Glob", - "settings.permissions.tool.glob.description": "จับคู่ไฟล์โดยใช้รูปแบบ glob", - "settings.permissions.tool.grep.title": "Grep", - "settings.permissions.tool.grep.description": "ค้นหาเนื้อหาไฟล์โดยใช้นิพจน์ทั่วไป", - "settings.permissions.tool.list.title": "รายการ", - "settings.permissions.tool.list.description": "แสดงรายการไฟล์ภายในไดเรกทอรี", - "settings.permissions.tool.bash.title": "Bash", - "settings.permissions.tool.bash.description": "เรียกใช้คำสั่งเชลล์", - "settings.permissions.tool.task.title": "งาน", - "settings.permissions.tool.task.description": "เปิดเอเจนต์ย่อย", - "settings.permissions.tool.skill.title": "ทักษะ", - "settings.permissions.tool.skill.description": "โหลดทักษะตามชื่อ", - "settings.permissions.tool.lsp.title": "LSP", - "settings.permissions.tool.lsp.description": "เรียกใช้การสืบค้นเซิร์ฟเวอร์ภาษา", - "settings.permissions.tool.todowrite.title": "เขียนรายการงาน", - "settings.permissions.tool.todowrite.description": "อัปเดตรายการงาน", - "settings.permissions.tool.webfetch.title": "ดึงข้อมูลจากเว็บ", - "settings.permissions.tool.webfetch.description": "ดึงเนื้อหาจาก URL", - "settings.permissions.tool.websearch.title": "ค้นหาเว็บ", - "settings.permissions.tool.websearch.description": "ค้นหาบนเว็บ", - "settings.permissions.tool.codesearch.title": "ค้นหาโค้ด", - "settings.permissions.tool.codesearch.description": "ค้นหาโค้ดบนเว็บ", - "settings.permissions.tool.external_directory.title": "ไดเรกทอรีภายนอก", - "settings.permissions.tool.external_directory.description": "เข้าถึงไฟล์นอกไดเรกทอรีโปรเจกต์", - "settings.permissions.tool.doom_loop.title": "Doom Loop", - "settings.permissions.tool.doom_loop.description": "ตรวจจับการเรียกเครื่องมือซ้ำด้วยข้อมูลนำเข้าเหมือนกัน", - - "session.delete.failed.title": "ไม่สามารถลบเซสชัน", - "session.delete.title": "ลบเซสชัน", - "session.delete.confirm": 'ลบเซสชัน "{{name}}" หรือไม่?', - "session.delete.button": "ลบเซสชัน", - - "workspace.new": "พื้นที่ทำงานใหม่", - "workspace.type.local": "ในเครื่อง", - "workspace.type.sandbox": "แซนด์บ็อกซ์", - "workspace.create.failed.title": "ไม่สามารถสร้างพื้นที่ทำงาน", - "workspace.delete.failed.title": "ไม่สามารถลบพื้นที่ทำงาน", - "workspace.resetting.title": "กำลังรีเซ็ตพื้นที่ทำงาน", - "workspace.resetting.description": "อาจใช้เวลาประมาณหนึ่งนาที", - "workspace.reset.failed.title": "ไม่สามารถรีเซ็ตพื้นที่ทำงาน", - "workspace.reset.success.title": "รีเซ็ตพื้นที่ทำงานแล้ว", - "workspace.reset.success.description": "พื้นที่ทำงานตรงกับสาขาเริ่มต้นแล้ว", - "workspace.error.stillPreparing": "พื้นที่ทำงานกำลังเตรียมอยู่", - "workspace.status.checking": "กำลังตรวจสอบการเปลี่ยนแปลงที่ไม่ได้ผสาน...", - "workspace.status.error": "ไม่สามารถตรวจสอบสถานะ git", - "workspace.status.clean": "ไม่ตรวจพบการเปลี่ยนแปลงที่ไม่ได้ผสาน", - "workspace.status.dirty": "ตรวจพบการเปลี่ยนแปลงที่ไม่ได้ผสานในพื้นที่ทำงานนี้", - "workspace.delete.title": "ลบพื้นที่ทำงาน", - "workspace.delete.confirm": 'ลบพื้นที่ทำงาน "{{name}}" หรือไม่?', - "workspace.delete.button": "ลบพื้นที่ทำงาน", - "workspace.reset.title": "รีเซ็ตพื้นที่ทำงาน", - "workspace.reset.confirm": 'รีเซ็ตพื้นที่ทำงาน "{{name}}" หรือไม่?', - "workspace.reset.button": "รีเซ็ตพื้นที่ทำงาน", - "workspace.reset.archived.none": "ไม่มีเซสชันที่ใช้งานอยู่จะถูกจัดเก็บ", - "workspace.reset.archived.one": "1 เซสชันจะถูกจัดเก็บ", - "workspace.reset.archived.many": "{{count}} เซสชันจะถูกจัดเก็บ", - "workspace.reset.note": "สิ่งนี้จะรีเซ็ตพื้นที่ทำงานให้ตรงกับสาขาเริ่มต้น", - "common.open": "เปิด", - "dialog.releaseNotes.action.getStarted": "เริ่มต้น", - "dialog.releaseNotes.action.next": "ถัดไป", - "dialog.releaseNotes.action.hideFuture": "ไม่ต้องแสดงสิ่งนี้อีกในอนาคต", - "dialog.releaseNotes.media.alt": "ตัวอย่างรุ่น", - "toast.project.reloadFailed.title": "ไม่สามารถโหลด {{project}} ใหม่ได้", - "error.server.invalidConfiguration": "การกำหนดค่าไม่ถูกต้อง", - "common.moreCountSuffix": " (เพิ่มอีก {{count}})", - "common.time.justNow": "เมื่อสักครู่นี้", - "common.time.minutesAgo.short": "{{count}} นาทีที่แล้ว", - "common.time.hoursAgo.short": "{{count}} ชม. ที่แล้ว", - "common.time.daysAgo.short": "{{count}} วันที่แล้ว", - "settings.providers.connected.environmentDescription": "เชื่อมต่อจากตัวแปรสภาพแวดล้อมของคุณ", - "settings.providers.custom.description": "เพิ่มผู้ให้บริการที่รองรับ OpenAI ด้วย URL หลัก", - - "app.server.unreachable": "ไม่สามารถติดต่อ {{server}}", - "app.server.retrying": "กำลังลองใหม่โดยอัตโนมัติ...", - "app.server.otherServers": "เซิร์ฟเวอร์อื่น ๆ", - "dialog.server.add.usernamePlaceholder": "ชื่อผู้ใช้", - "dialog.server.add.passwordPlaceholder": "รหัสผ่าน", - "server.row.noUsername": "ไม่มีชื่อผู้ใช้", - "session.review.noVcs.createGit.title": "สร้าง Git repository", - "session.review.noVcs.createGit.description": "ติดตาม ตรวจสอบ และเลิกทำสิ่งเปลี่ยนแปลงในโปรเจกต์นี้", - "session.review.noVcs.createGit.actionLoading": "กำลังสร้าง Git repository...", - "session.review.noVcs.createGit.action": "สร้าง Git repository", - "session.todo.progress": "เสร็จสิ้น {{done}} จาก {{total}} รายการ", - "session.question.progress": "{{current}} จาก {{total}} คำถาม", - "session.header.open.finder": "Finder", - "session.header.open.fileExplorer": "File Explorer", - "session.header.open.fileManager": "File Manager", - "session.header.open.app.vscode": "VS Code", - "session.header.open.app.cursor": "Cursor", - "session.header.open.app.zed": "Zed", - "session.header.open.app.textmate": "TextMate", - "session.header.open.app.antigravity": "Antigravity", - "session.header.open.app.terminal": "Terminal", - "session.header.open.app.iterm2": "iTerm2", - "session.header.open.app.ghostty": "Ghostty", - "session.header.open.app.warp": "Warp", - "session.header.open.app.xcode": "Xcode", - "session.header.open.app.androidStudio": "Android Studio", - "session.header.open.app.powershell": "PowerShell", - "session.header.open.app.sublimeText": "Sublime Text", - "debugBar.ariaLabel": "การวินิจฉัยประสิทธิภาพการพัฒนา", - "debugBar.na": "n/a", - "debugBar.nav.label": "NAV", - "debugBar.nav.tip": - "การเปลี่ยนเส้นทางที่เสร็จสมบูรณ์ล่าสุดที่สัมผัสหน้าเซสชัน วัดจากจุดเริ่มต้นเราเตอร์จนถึงการวาดครั้งแรกหลังจากที่นิ่ง", - "debugBar.fps.label": "FPS", - "debugBar.fps.tip": "เฟรมต่อวินาทีแบบต่อเนื่องในช่วง 5 วินาทีที่ผ่านมา", - "debugBar.frame.label": "FRAME", - "debugBar.frame.tip": "เวลาเฟรมที่แย่ที่สุดในช่วง 5 วินาทีที่ผ่านมา", - "debugBar.jank.label": "JANK", - "debugBar.jank.tip": "เฟรมที่เกิน 32ms ในช่วง 5 วินาทีที่ผ่านมา", - "debugBar.long.label": "LONG", - "debugBar.long.tip": "เวลาที่ถูกบล็อกและจำนวนงานยาวในช่วง 5 วินาทีที่ผ่านมา งานสูงสุด: {{max}}", - "debugBar.delay.label": "DELAY", - "debugBar.delay.tip": "ความล่าช้าในการป้อนข้อมูลที่แย่ที่สุดที่สังเกตได้ในช่วง 5 วินาทีที่ผ่านมา", - "debugBar.inp.label": "INP", - "debugBar.inp.tip": - "ระยะเวลาการโต้ตอบโดยประมาณในช่วง 5 วินาทีที่ผ่านมา นี่เป็นเหมือน INP ไม่ใช่ Web Vitals INP อย่างเป็นทางการ", - "debugBar.cls.label": "CLS", - "debugBar.cls.tip": "การเลื่อนเลย์เอาต์สะสมสำหรับอายุการใช้งานของแอปปัจจุบัน", - "debugBar.mem.label": "MEM", - "debugBar.mem.tipUnavailable": "JS heap ที่ใช้เทียบกับขีดจำกัด heap เฉพาะ Chromium", - "debugBar.mem.tip": "JS heap ที่ใช้เทียบกับขีดจำกัด heap {{used}} จาก {{limit}}", - "common.key.ctrl": "Ctrl", - "common.key.alt": "Alt", - "common.key.shift": "Shift", - "common.key.meta": "Meta", - "common.key.space": "Space", - "common.key.backspace": "Backspace", - "common.key.enter": "Enter", - "common.key.tab": "Tab", - "common.key.delete": "Delete", - "common.key.home": "Home", - "common.key.end": "End", - "common.key.pageUp": "Page Up", - "common.key.pageDown": "Page Down", - "common.key.insert": "Insert", - "common.unknown": "ไม่ทราบ", - "error.page.circular": "[วงกลม]", - "error.globalSDK.noServerAvailable": "ไม่มีเซิร์ฟเวอร์", - "error.globalSDK.serverNotAvailable": "เซิร์ฟเวอร์ไม่พร้อมใช้งาน", - "error.childStore.persistedCacheCreateFailed": "ไม่สามารถสร้างแคชถาวร", - "error.childStore.persistedProjectMetadataCreateFailed": "ไม่สามารถสร้างเมตาดาต้าโปรเจกต์ถาวร", - "error.childStore.persistedProjectIconCreateFailed": "ไม่สามารถสร้างไอคอนโปรเจกต์ถาวร", - "error.childStore.storeCreateFailed": "ไม่สามารถสร้างที่เก็บ", - "terminal.connectionLost.abnormalClose": "WebSocket ปิดอย่างผิดปกติ: {{code}}", -} diff --git a/packages/app/src/i18n/tr.ts b/packages/app/src/i18n/tr.ts deleted file mode 100644 index a5c3f4b47b..0000000000 --- a/packages/app/src/i18n/tr.ts +++ /dev/null @@ -1,942 +0,0 @@ -import { dict as en } from "./en" - -type Keys = keyof typeof en - -export const dict = { - "command.category.suggested": "Önerilen", - "command.category.view": "Görünüm", - "command.category.project": "Proje", - "command.category.provider": "Sağlayıcı", - "command.category.server": "Sunucu", - "command.category.session": "Oturum", - "command.category.theme": "Tema", - "command.category.language": "Dil", - "command.category.file": "Dosya", - "command.category.context": "Bağlam", - "command.category.terminal": "Terminal", - "command.category.model": "Model", - "command.category.mcp": "MCP", - "command.category.agent": "Ajan", - "command.category.permissions": "İzinler", - "command.category.workspace": "Çalışma Alanı", - "command.category.settings": "Ayarlar", - - "theme.scheme.system": "Sistem", - "theme.scheme.light": "Açık", - "theme.scheme.dark": "Koyu", - - "command.sidebar.toggle": "Kenar çubuğunu aç/kapat", - "command.project.open": "Proje aç", - "command.provider.connect": "Sağlayıcı bağla", - "command.server.switch": "Sunucu değiştir", - "command.settings.open": "Ayarları aç", - "command.session.previous": "Önceki oturum", - "command.session.next": "Sonraki oturum", - "command.session.previous.unseen": "Önceki okunmamış oturum", - "command.session.next.unseen": "Sonraki okunmamış oturum", - "command.session.archive": "Oturumu arşivle", - - "command.palette": "Komut paleti", - - "command.theme.cycle": "Tema değiştir", - "command.theme.set": "Tema kullan: {{theme}}", - "command.theme.scheme.cycle": "Renk şemasını değiştir", - "command.theme.scheme.set": "Renk şeması kullan: {{scheme}}", - - "command.language.cycle": "Dil değiştir", - "command.language.set": "Dil kullan: {{language}}", - - "command.session.new": "Yeni oturum", - "command.file.open": "Dosya aç", - "command.tab.close": "Sekmeyi kapat", - "command.context.addSelection": "Seçimi bağlama ekle", - "command.context.addSelection.description": "Mevcut dosyadan seçili satırları ekle", - "command.input.focus": "Girişi odakla", - "command.terminal.toggle": "Terminali aç/kapat", - "command.fileTree.toggle": "Dosya ağacını aç/kapat", - "command.review.toggle": "İncelemeyi aç/kapat", - "command.terminal.new": "Yeni terminal", - "command.terminal.new.description": "Yeni bir terminal sekmesi oluştur", - "command.steps.toggle": "Adımları aç/kapat", - "command.steps.toggle.description": "Mevcut mesaj için adımları göster veya gizle", - "command.message.previous": "Önceki mesaj", - "command.message.previous.description": "Önceki kullanıcı mesajına git", - "command.message.next": "Sonraki mesaj", - "command.message.next.description": "Sonraki kullanıcı mesajına git", - "command.model.choose": "Model seç", - "command.model.choose.description": "Farklı bir model seç", - "command.mcp.toggle": "MCP'leri aç/kapat", - "command.mcp.toggle.description": "MCP'leri aç/kapat", - "command.agent.cycle": "Ajan değiştir", - "command.agent.cycle.description": "Sonraki ajana geç", - "command.agent.cycle.reverse": "Ajanı geri değiştir", - "command.agent.cycle.reverse.description": "Önceki ajana geç", - "command.model.variant.cycle": "Düşünme eforu değiştir", - "command.model.variant.cycle.description": "Sonraki efor seviyesine geç", - "command.prompt.mode.shell": "Kabuk", - "command.prompt.mode.normal": "Komut", - "command.permissions.autoaccept.enable": "Düzenlemeleri otomatik kabul et", - "command.permissions.autoaccept.disable": "Otomatik kabulü durdur", - "command.workspace.toggle": "Çalışma alanlarını aç/kapat", - "command.workspace.toggle.description": "Kenar çubuğunda birden fazla çalışma alanını göster veya gizle", - "command.session.undo": "Geri al", - "command.session.undo.description": "Son mesajı geri al", - "command.session.redo": "Yinele", - "command.session.redo.description": "Son geri alınan mesajı yinele", - "command.session.compact": "Oturumu sıkıştır", - "command.session.compact.description": "Bağlam boyutunu azaltmak için oturumu özetle", - "command.session.fork": "Mesajdan dallandır", - "command.session.fork.description": "Önceki bir mesajdan yeni oturum oluştur", - "command.session.share": "Oturumu paylaş", - "command.session.share.description": "Bu oturumu paylaş ve URL'yi panoya kopyala", - "command.session.unshare": "Paylaşımı kaldır", - "command.session.unshare.description": "Bu oturumun paylaşımını durdur", - - "palette.search.placeholder": "Dosya, komut ve oturum ara", - "palette.empty": "Sonuç bulunamadı", - "palette.group.commands": "Komutlar", - "palette.group.files": "Dosyalar", - - "dialog.provider.search.placeholder": "Sağlayıcı ara", - "dialog.provider.empty": "Sağlayıcı bulunamadı", - "dialog.provider.group.popular": "Popüler", - "dialog.provider.group.other": "Diğer", - "dialog.provider.tag.recommended": "Önerilen", - "dialog.provider.opencode.note": "Claude, GPT, Gemini ve daha fazlasını içeren seçilmiş modeller", - "dialog.provider.opencode.tagline": "Güvenilir optimize edilmiş modeller", - "dialog.provider.opencodeGo.tagline": "Herkes için düşük maliyetli abonelik", - "dialog.provider.anthropic.note": "Pro ve Max dahil Claude modellerine doğrudan erişim", - "dialog.provider.copilot.note": "GitHub Copilot üzerinden kodlama yardımı için yapay zekâ modelleri", - "dialog.provider.openai.note": "Hızlı ve yetenekli genel yapay zekâ görevleri için GPT modelleri", - "dialog.provider.google.note": "Hızlı ve yapılandırılmış yanıtlar için Gemini modelleri", - "dialog.provider.openrouter.note": "Tek bir sağlayıcıdan tüm desteklenen modellere eriş", - "dialog.provider.vercel.note": "Akıllı yönlendirme ile yapay zekâ modellerine birleşik erişim", - - "dialog.model.select.title": "Model seç", - "dialog.model.search.placeholder": "Model ara", - "dialog.model.empty": "Model sonucu yok", - "dialog.model.manage": "Modelleri yönet", - "dialog.model.manage.description": "Model seçicide hangi modellerin görüneceğini özelleştirin.", - "dialog.model.manage.provider.toggle": "Tüm {{provider}} modellerini aç/kapat", - - "dialog.model.unpaid.freeModels.title": "Kilo tarafından sunulan ücretsiz modeller", - "dialog.model.unpaid.addMore.title": "Popüler sağlayıcılardan daha fazla model ekleyin", - - "dialog.provider.viewAll": "Daha fazla sağlayıcı göster", - - "provider.connect.title": "{{provider}} bağla", - "provider.connect.title.anthropicProMax": "Claude Pro/Max ile giriş yap", - "provider.connect.selectMethod": "{{provider}} için giriş yöntemini seçin.", - "provider.connect.method.apiKey": "API anahtarı", - "provider.connect.status.inProgress": "Yetkilendirme devam ediyor...", - "provider.connect.status.waiting": "Yetkilendirme bekleniyor...", - "provider.connect.status.failed": "Yetkilendirme başarısız: {{error}}", - "provider.connect.apiKey.description": - "{{provider}} hesabınızı bağlamak ve Kilo'da {{provider}} modellerini kullanmak için {{provider}} API anahtarınızı girin.", - "provider.connect.apiKey.label": "{{provider}} API anahtarı", - "provider.connect.apiKey.placeholder": "API anahtarı", - "provider.connect.apiKey.required": "API anahtarı gerekli", - "provider.connect.opencodeZen.line1": - "OpenCode Zen, kodlama ajanları için seçilmiş güvenilir optimize edilmiş modellere erişim sağlar.", - "provider.connect.opencodeZen.line2": - "Tek bir API anahtarıyla Claude, GPT, Gemini, GLM ve daha fazlası gibi modellere erişebilirsiniz.", - "provider.connect.opencodeZen.visit.prefix": "", - "provider.connect.opencodeZen.visit.link": "opencode.ai/zen", - "provider.connect.opencodeZen.visit.suffix": " adresini ziyaret ederek API anahtarınızı alın.", - "provider.connect.oauth.code.visit.prefix": "Hesabınızı bağlamak ve Kilo'da {{provider}} modellerini kullanmak için ", - "provider.connect.oauth.code.visit.link": "bu bağlantıya", - "provider.connect.oauth.code.visit.suffix": " tıklayarak yetkilendirme kodunuzu alın.", - "provider.connect.oauth.code.label": "{{method}} yetkilendirme kodu", - "provider.connect.oauth.code.placeholder": "Yetkilendirme kodu", - "provider.connect.oauth.code.required": "Yetkilendirme kodu gerekli", - "provider.connect.oauth.code.invalid": "Geçersiz yetkilendirme kodu", - "provider.connect.oauth.auto.visit.prefix": "", - "provider.connect.oauth.auto.visit.link": "Bu bağlantıya", - "provider.connect.oauth.auto.visit.suffix": - " tıklayarak aşağıdaki kodu girin ve hesabınızı bağlayarak Kilo'da {{provider}} modellerini kullanın.", - "provider.connect.oauth.auto.confirmationCode": "Onay kodu", - "provider.connect.toast.connected.title": "{{provider}} bağlandı", - "provider.connect.toast.connected.description": "{{provider}} modelleri artık kullanımda.", - - "provider.custom.title": "Özel sağlayıcı", - "provider.custom.description.prefix": "OpenAI uyumlu bir sağlayıcı yapılandırın. ", - "provider.custom.description.link": "Sağlayıcı yapılandırma dökümanları", - "provider.custom.description.suffix": " sayfasına bakın.", - "provider.custom.field.providerID.label": "Sağlayıcı kimlik", - "provider.custom.field.providerID.placeholder": "saglayicim", - "provider.custom.field.providerID.description": "Küçük harfler, rakamlar, tire veya alt çizgi", - "provider.custom.field.name.label": "Görünen ad", - "provider.custom.field.name.placeholder": "Yapay Zekâ Sağlayıcım", - "provider.custom.field.baseURL.label": "Temel URL", - "provider.custom.field.baseURL.placeholder": "https://api.saglayicim.com/v1", - "provider.custom.field.apiKey.label": "API anahtarı", - "provider.custom.field.apiKey.placeholder": "API anahtarı", - "provider.custom.field.apiKey.description": - "İsteğe bağlı. Kimlik doğrulamayı başlıklar ile yönetiyorsanız boş bırakın.", - "provider.custom.models.label": "Modeller", - "provider.custom.models.id.label": "Kimlik", - "provider.custom.models.id.placeholder": "model-kimlik", - "provider.custom.models.name.label": "Ad", - "provider.custom.models.name.placeholder": "Görünen Ad", - "provider.custom.models.remove": "Modeli kaldır", - "provider.custom.models.add": "Model ekle", - "provider.custom.headers.label": "Başlıklar (isteğe bağlı)", - "provider.custom.headers.key.label": "Başlık", - "provider.custom.headers.key.placeholder": "Başlık-Adı", - "provider.custom.headers.value.label": "Değer", - "provider.custom.headers.value.placeholder": "değer", - "provider.custom.headers.remove": "Başlığı kaldır", - "provider.custom.headers.add": "Başlık ekle", - "provider.custom.error.providerID.required": "Sağlayıcı kimlik gerekli", - "provider.custom.error.providerID.format": "Küçük harf, rakam, tire veya alt çizgi kullanın", - "provider.custom.error.providerID.exists": "Bu sağlayıcı kimlik zaten mevcut", - "provider.custom.error.name.required": "Görünen ad gerekli", - "provider.custom.error.baseURL.required": "Temel URL gerekli", - "provider.custom.error.baseURL.format": "http:// veya https:// ile başlamalı", - "provider.custom.error.required": "Gerekli", - "provider.custom.error.duplicate": "Tekrar", - - "provider.disconnect.toast.disconnected.title": "{{provider}} bağlantısı kesildi", - "provider.disconnect.toast.disconnected.description": "{{provider}} modelleri artık kullanılabilir değil.", - - "model.tag.free": "Ücretsiz", - "model.tag.latest": "En yeni", - "model.provider.anthropic": "Anthropic", - "model.provider.openai": "OpenAI", - "model.provider.google": "Google", - "model.provider.xai": "xAI", - "model.provider.meta": "Meta", - "model.input.text": "metin", - "model.input.image": "görsel", - "model.input.audio": "ses", - "model.input.video": "video", - "model.input.pdf": "pdf", - "model.tooltip.allows": "Kabul eder: {{inputs}}", - "model.tooltip.reasoning.allowed": "Akıl yürütme destekler", - "model.tooltip.reasoning.none": "Akıl yürütme yok", - "model.tooltip.context": "Bağlam limiti {{limit}}", - - "common.search.placeholder": "Ara", - "common.goBack": "Geri git", - "common.goForward": "İleri git", - "common.loading": "Yükleniyor", - "common.loading.ellipsis": "...", - "common.cancel": "İptal", - "common.connect": "Bağlan", - "common.disconnect": "Bağlantı Kes", - "common.continue": "Gönder", - "common.submit": "Gönder", - "common.save": "Kaydet", - "common.saving": "Kaydediliyor...", - "common.default": "Varsayılan", - "common.attachment": "ek", - - "prompt.placeholder.shell": "Kabuk komutu girin... {{example}}", - "prompt.placeholder.normal": 'Bir şeyler sorun... "{{example}}"', - "prompt.placeholder.simple": "Bir şeyler sorun...", - "prompt.placeholder.summarizeComments": "Yorumları özetle…", - "prompt.placeholder.summarizeComment": "Yorumu özetle…", - "prompt.mode.shell": "Kabuk", - "prompt.mode.normal": "Komut", - "prompt.mode.shell.exit": "çıkmak için esc", - - "prompt.example.1": "Kod tabanındaki bir TODO'yu düzelt", - "prompt.example.2": "Bu projenin teknoloji yığını nedir?", - "prompt.example.3": "Bozuk testleri düzelt", - "prompt.example.4": "Kimlik doğrulamanın nasıl çalıştığını açıkla", - "prompt.example.5": "Güvenlik açıkları bul ve düzelt", - "prompt.example.6": "Kullanıcı servisi için birim testleri ekle", - "prompt.example.7": "Bu fonksiyonu daha okunabilir hale getir", - "prompt.example.8": "Bu hata ne anlama geliyor?", - "prompt.example.9": "Bu sorunu ayıklamama yardım et", - "prompt.example.10": "API dokümantasyonu oluştur", - "prompt.example.11": "Veritabanı sorgularını optimize et", - "prompt.example.12": "Girdi doğrulama ekle", - "prompt.example.13": "İçin yeni bir bileşen oluştur...", - "prompt.example.14": "Bu projeyi nasıl dağıtabilirim?", - "prompt.example.15": "Kodumu en iyi uygulamalar için incele", - "prompt.example.16": "Bu fonksiyona hata yönetimi ekle", - "prompt.example.17": "Bu regex kalıbını açıkla", - "prompt.example.18": "Bunu TypeScript'e dönüştür", - "prompt.example.19": "Kod tabanı boyunca loglama ekle", - "prompt.example.20": "Hangi bağımlılıklar güncellenmemiş?", - "prompt.example.21": "Bir göç betiği yazmama yardım et", - "prompt.example.22": "Bu uç nokta için önbellekleme uygula", - "prompt.example.23": "Bu listeye sayfalama ekle", - "prompt.example.24": "İçin bir CLI komutu oluştur...", - "prompt.example.25": "Ortam değişkenleri burada nasıl çalışıyor?", - - "prompt.popover.emptyResults": "Eşleşen sonuç yok", - "prompt.popover.emptyCommands": "Eşleşen komut yok", - "prompt.dropzone.label": "Resimleri, PDF'leri veya metin dosyalarını buraya bırakın", - "prompt.dropzone.file.label": "@bahsetmek için dosyayı bırakın", - "prompt.slash.badge.custom": "özel", - "prompt.slash.badge.skill": "beceri", - "prompt.slash.badge.mcp": "mcp", - "prompt.context.active": "aktif", - "prompt.context.includeActiveFile": "Aktif dosyayı dahil et", - "prompt.context.removeActiveFile": "Aktif dosyayı bağlamdan çıkar", - "prompt.context.removeFile": "Dosyayı bağlamdan çıkar", - "prompt.action.attachFile": "Dosya ekle", - "prompt.attachment.remove": "Eki kaldır", - "prompt.action.send": "Gönder", - "prompt.action.stop": "Durdur", - - "prompt.toast.pasteUnsupported.title": "Desteklenmeyen ek", - "prompt.toast.pasteUnsupported.description": "Buraya yalnızca resimler, PDF'ler veya metin dosyaları eklenebilir.", - "prompt.toast.modelAgentRequired.title": "Bir ajan ve model seçin", - "prompt.toast.modelAgentRequired.description": "Komut göndermeden önce bir ajan ve model seçin.", - "prompt.toast.worktreeCreateFailed.title": "Çalışma ağacı oluşturulamadı", - "prompt.toast.sessionCreateFailed.title": "Oturum oluşturulamadı", - "prompt.toast.shellSendFailed.title": "Kabuk komutu gönderilemedi", - "prompt.toast.commandSendFailed.title": "Komut gönderilemedi", - "prompt.toast.promptSendFailed.title": "Komut gönderilemedi", - "prompt.toast.promptSendFailed.description": "Oturum alınamadı", - - "dialog.mcp.title": "MCP'ler", - "dialog.mcp.description": "{{total}} içerisinden {{enabled}} etkin", - "dialog.mcp.empty": "Yapılandırılmış MCP yok", - - "dialog.lsp.empty": "LSP'ler dosya türlerinden otomatik algılanır", - "dialog.plugins.empty": "Eklentiler opencode.json içinde yapılandırılır", - - "mcp.status.connected": "bağlı", - "mcp.status.failed": "başarısız", - "mcp.status.needs_auth": "kimlik doğrulama gerekli", - "mcp.status.disabled": "devre dışı", - - "dialog.fork.empty": "Dallandırılacak mesaj yok", - - "dialog.directory.search.placeholder": "Klasör ara", - "dialog.directory.empty": "Klasör bulunamadı", - - "dialog.server.title": "Sunucular", - "dialog.server.description": "Bu uygulamanın hangi Kilo sunucusuna bağlanacağını değiştirin.", - "dialog.server.search.placeholder": "Sunucu ara", - "dialog.server.empty": "Henüz sunucu yok", - "dialog.server.add.title": "Sunucu ekle", - "dialog.server.add.url": "Sunucu URL'si", - "dialog.server.add.placeholder": "http://localhost:4096", - "dialog.server.add.error": "Sunucuya bağlanılamadı", - "dialog.server.add.checking": "Kontrol ediliyor...", - "dialog.server.add.button": "Sunucu ekle", - "dialog.server.add.name": "Sunucu adı (isteğe bağlı)", - "dialog.server.add.namePlaceholder": "Localhost", - "dialog.server.add.username": "Kullanıcı adı (isteğe bağlı)", - "dialog.server.add.password": "Şifre (isteğe bağlı)", - "dialog.server.edit.title": "Sunucuyu düzenle", - "dialog.server.default.title": "Varsayılan sunucu", - "dialog.server.default.description": - "Uygulama başlatıldığında yerel sunucu başlatmak yerine bu sunucuya bağlan. Yeniden başlatma gerektirir.", - "dialog.server.default.none": "Sunucu seçilmedi", - "dialog.server.default.set": "Mevcut sunucuyu varsayılan olarak ayarla", - "dialog.server.default.clear": "Temizle", - "dialog.server.action.remove": "Sunucuyu kaldır", - - "dialog.server.menu.edit": "Düzenle", - "dialog.server.menu.default": "Varsayılan olarak ayarla", - "dialog.server.menu.defaultRemove": "Varsayılanı kaldır", - "dialog.server.menu.delete": "Sil", - "dialog.server.current": "Mevcut Sunucu", - "dialog.server.status.default": "Varsayılan", - - "dialog.project.edit.title": "Projeyi düzenle", - "dialog.project.edit.name": "Ad", - "dialog.project.edit.icon": "Simge", - "dialog.project.edit.icon.alt": "Proje simgesi", - "dialog.project.edit.icon.hint": "Tıkla veya bir görsel sürükle", - "dialog.project.edit.icon.recommended": "Önerilen: 128x128px", - "dialog.project.edit.color": "Renk", - "dialog.project.edit.color.select": "{{color}} rengini seç", - "dialog.project.edit.worktree.startup": "Çalışma alanı başlatma betiği", - "dialog.project.edit.worktree.startup.description": "Yeni bir çalışma alanı (worktree) oluşturduktan sonra çalışır.", - "dialog.project.edit.worktree.startup.placeholder": "örneğin bun install", - - "context.breakdown.title": "Bağlam Dökümü", - "context.breakdown.note": 'Girdi tokenlerinin yaklaşık dökümü. "Diğer" araç tanımları ve ek yükleri içerir.', - "context.breakdown.system": "Sistem", - "context.breakdown.user": "Kullanıcı", - "context.breakdown.assistant": "Asistan", - "context.breakdown.tool": "Araç Çağrıları", - "context.breakdown.other": "Diğer", - - "context.systemPrompt.title": "Sistem Komutu", - "context.rawMessages.title": "Ham mesajlar", - - "context.stats.session": "Oturum", - "context.stats.messages": "Mesajlar", - "context.stats.provider": "Sağlayıcı", - "context.stats.model": "Model", - "context.stats.limit": "Bağlam Limiti", - "context.stats.totalTokens": "Toplam Token", - "context.stats.usage": "Kullanım", - "context.stats.inputTokens": "Girdi Tokenleri", - "context.stats.outputTokens": "Çıktı Tokenleri", - "context.stats.reasoningTokens": "Akıl Yürütme Tokenleri", - "context.stats.cacheTokens": "Önbellek Tokenleri (okuma/yazma)", - "context.stats.userMessages": "Kullanıcı Mesajları", - "context.stats.assistantMessages": "Asistan Mesajları", - "context.stats.totalCost": "Toplam Maliyet", - "context.stats.sessionCreated": "Oturum Oluşturulma", - "context.stats.lastActivity": "Son Etkinlik", - - "context.usage.tokens": "Tokenler", - "context.usage.usage": "Kullanım", - "context.usage.cost": "Maliyet", - "context.usage.clickToView": "Bağlamı görüntüle", - "context.usage.view": "Bağlam kullanımını görüntüle", - - "language.en": "English", - "language.zh": "简体中文", - "language.zht": "繁體中文", - "language.ko": "한국어", - "language.de": "Deutsch", - "language.es": "Español", - "language.fr": "Français", - "language.da": "Dansk", - "language.ja": "日本語", - "language.pl": "Polski", - "language.ru": "Русский", - "language.ar": "العربية", - "language.no": "Norsk", - "language.br": "Português (Brasil)", - "language.bs": "Bosanski", - "language.th": "ไทย", - "language.tr": "Türkçe", - - "toast.language.title": "Dil", - "toast.language.description": "{{language}} diline geçildi", - - "toast.theme.title": "Tema değiştirildi", - "toast.scheme.title": "Renk şeması", - - "toast.workspace.enabled.title": "Çalışma alanları etkinleştirildi", - "toast.workspace.enabled.description": "Kenar çubuğunda birden fazla çalışma ağacı gösterilecek", - "toast.workspace.disabled.title": "Çalışma alanları devre dışı bırakıldı", - "toast.workspace.disabled.description": "Kenar çubuğunda yalnızca ana çalışma ağacı gösterilecek", - - "toast.permissions.autoaccept.on.title": "Düzenlemeler otomatik kabul ediliyor", - "toast.permissions.autoaccept.on.description": "Düzenleme ve yazma izinleri otomatik olarak onaylanacak", - "toast.permissions.autoaccept.off.title": "Otomatik kabul durduruldu", - "toast.permissions.autoaccept.off.description": "Düzenleme ve yazma izinleri onay gerektirecek", - - "toast.model.none.title": "Model seçilmedi", - "toast.model.none.description": "Bu oturumu özetlemek için bir sağlayıcı bağlayın", - - "toast.file.loadFailed.title": "Dosya yüklenemedi", - "toast.file.listFailed.title": "Dosyalar listelenemedi", - - "toast.context.noLineSelection.title": "Satır seçimi yok", - "toast.context.noLineSelection.description": "Önce bir dosya sekmesinde satır aralığı seçin.", - - "toast.session.share.copyFailed.title": "URL panoya kopyalanamadı", - "toast.session.share.success.title": "Oturum paylaşıldı", - "toast.session.share.success.description": "Paylaşım URL'si panoya kopyalandı!", - "toast.session.share.failed.title": "Oturum paylaşılamadı", - "toast.session.share.failed.description": "Oturum paylaşılırken bir hata oluştu", - - "toast.session.unshare.success.title": "Oturum paylaşımı kaldırıldı", - "toast.session.unshare.success.description": "Oturum paylaşımı başarıyla kaldırıldı!", - "toast.session.unshare.failed.title": "Oturum paylaşımı kaldırılamadı", - "toast.session.unshare.failed.description": "Oturum paylaşımı kaldırılırken bir hata oluştu", - - "toast.session.listFailed.title": "{{project}} için oturumlar yüklenemedi", - - "toast.update.title": "Güncelleme mevcut", - "toast.update.description": "Kilo'un yeni bir sürümü ({{version}}) yüklemeye hazır.", - "toast.update.action.installRestart": "Yükle ve yeniden başlat", - "toast.update.action.notYet": "Şimdi değil", - - "error.page.title": "Bir şeyler yanlış gitti", - "error.page.description": "Uygulama yüklenirken bir hata oluştu.", - "error.page.details.label": "Hata Detayları", - "error.page.action.restart": "Yeniden Başlat", - "error.page.action.checking": "Kontrol ediliyor...", - "error.page.action.checkUpdates": "Güncellemeleri kontrol et", - "error.page.action.updateTo": "{{version}} sürümüne güncelle", - "error.page.report.prefix": "Lütfen bu hatayı Kilo ekibine bildirin", - "error.page.report.discord": "Discord üzerinden", - "error.page.version": "Sürüm: {{version}}", - - "error.dev.rootNotFound": - "Kök eleman bulunamadı. index.html dosyanıza eklemeyi unuttunuz mu? Ya da id özelliği yanlış mı yazıldı?", - - "error.globalSync.connectFailed": "Sunucuya bağlanılamadı. `{{url}}` adresinde çalışan bir sunucu var mı?", - "directory.error.invalidUrl": "URL'de geçersiz dizin.", - - "error.chain.unknown": "Bilinmeyen hata", - "error.chain.causedBy": "Nedeni:", - "error.chain.apiError": "API hatası", - "error.chain.status": "Durum: {{status}}", - "error.chain.retryable": "Yeniden denenebilir: {{retryable}}", - "error.chain.responseBody": "Yanıt gövdesi:\n{{body}}", - "error.chain.didYouMean": "Bunu mu demek istediniz: {{suggestions}}", - "error.chain.modelNotFound": "Model bulunamadı: {{provider}}/{{model}}", - "error.chain.checkConfig": "Yapılandırma dosyanızı (opencode.json) sağlayıcı/model adlarını kontrol edin", - "error.chain.mcpFailed": - 'MCP sunucusu "{{name}}" başarısız oldu. Not: Kilo henüz MCP kimlik doğrulamasını desteklemiyor.', - "error.chain.providerAuthFailed": "Sağlayıcı kimlik doğrulaması başarısız ({{provider}}): {{message}}", - "error.chain.providerInitFailed": - '"{{provider}}" sağlayıcısı başlatılamadı. Kimlik bilgilerini ve yapılandırmayı kontrol edin.', - "error.chain.configJsonInvalid": "{{path}} adresindeki yapılandırma dosyası geçerli JSON(C) değil", - "error.chain.configJsonInvalidWithMessage": - "{{path}} adresindeki yapılandırma dosyası geçerli JSON(C) değil: {{message}}", - "error.chain.configDirectoryTypo": - '"{{dir}}" dizini {{path}} içinde geçerli değil. Dizini "{{suggestion}}" olarak yeniden adlandırın veya kaldırın. Bu yaygın bir yazım hatasıdır.', - "error.chain.configFrontmatterError": "{{path}} içindeki ön bilgi ayrıştırılamadı:\n{{message}}", - "error.chain.configInvalid": "{{path}} adresindeki yapılandırma dosyası geçersiz", - "error.chain.configInvalidWithMessage": "{{path}} adresindeki yapılandırma dosyası geçersiz: {{message}}", - - "notification.permission.title": "İzin gerekli", - "notification.permission.description": "{{projectName}} içindeki {{sessionTitle}} izin gerektiriyor", - "notification.question.title": "Soru", - "notification.question.description": "{{projectName}} içindeki {{sessionTitle}} bir soru soruyor", - "notification.action.goToSession": "Oturuma git", - - "notification.session.responseReady.title": "Yanıt hazır", - "notification.session.error.title": "Oturum hatası", - "notification.session.error.fallbackDescription": "Bir hata oluştu", - - "home.recentProjects": "Son projeler", - "home.empty.title": "Son proje yok", - "home.empty.description": "Yerel bir proje açarak başlayın", - - "session.tab.session": "Oturum", - "session.tab.review": "İnceleme", - "session.tab.context": "Bağlam", - "session.panel.reviewAndFiles": "İnceleme ve dosyalar", - "session.review.filesChanged": "{{count}} Dosya Değişti", - "session.review.change.one": "Değişiklik", - "session.review.change.other": "Değişiklik", - "session.review.loadingChanges": "Değişiklikler yükleniyor...", - "session.review.empty": "Bu oturumda henüz değişiklik yok", - "session.review.noVcs": "Git VCS algılanamadı, oturum değişiklikleri tespit edilemeyecek", - "session.review.noSnapshot": - "Yapılandırmada anlık görüntü takibi devre dışı bırakıldı, bu nedenle oturum değişiklikleri kullanılamıyor", - "session.review.noChanges": "Değişiklik yok", - - "session.files.selectToOpen": "Açmak için bir dosya seçin", - "session.files.all": "Tüm dosyalar", - "session.files.empty": "Dosya yok", - "session.files.binaryContent": "İkili dosya (içerik görüntülenemiyor)", - - "session.messages.renderEarlier": "Önceki mesajları göster", - "session.messages.loadingEarlier": "Önceki mesajlar yükleniyor...", - "session.messages.loadEarlier": "Önceki mesajları yükle", - "session.messages.loading": "Mesajlar yükleniyor...", - "session.messages.jumpToLatest": "En sona atla", - - "session.context.addToContext": "{{selection}} bağlama ekle", - "session.todo.title": "Görevler", - "session.todo.collapse": "Daralt", - "session.todo.expand": "Genişlet", - "session.followupDock.summary.one": "{{count}} sıradaki mesaj", - "session.followupDock.summary.other": "{{count}} sıradaki mesaj", - "session.followupDock.sendNow": "Şimdi gönder", - "session.followupDock.edit": "Düzenle", - "session.followupDock.collapse": "Sıradaki mesajları daralt", - "session.followupDock.expand": "Sıradaki mesajları genişlet", - "session.revertDock.summary.one": "{{count}} geri alınan mesaj", - "session.revertDock.summary.other": "{{count}} geri alınan mesaj", - "session.revertDock.collapse": "Geri alınan mesajları daralt", - "session.revertDock.expand": "Geri alınan mesajları genişlet", - "session.revertDock.restore": "Mesajı geri yükle", - - "session.new.title": "İstediğini yap", - "session.new.worktree.main": "Ana dal", - "session.new.worktree.mainWithBranch": "Ana dal ({{branch}})", - "session.new.worktree.create": "Yeni çalışma ağacı oluştur", - "session.new.lastModified": "Son değişiklik", - - "session.header.search.placeholder": "{{project}} ara", - "session.header.searchFiles": "Dosya ara", - "session.header.openIn": "Aç", - "session.header.open.action": "{{app}} ile aç", - "session.header.open.ariaLabel": "{{app}} ile aç", - "session.header.open.menu": "Açma seçenekleri", - "session.header.open.copyPath": "Yolu kopyala", - - "status.popover.trigger": "Durum", - "status.popover.ariaLabel": "Sunucu yapılandırmaları", - "status.popover.tab.servers": "Sunucular", - "status.popover.tab.mcp": "MCP", - "status.popover.tab.lsp": "LSP", - "status.popover.tab.plugins": "Eklentiler", - "status.popover.action.manageServers": "Sunucuları yönet", - - "session.share.popover.title": "Web'de yayınla", - "session.share.popover.description.shared": "Bu oturum web'de herkese açıktır. Bağlantıya sahip herkes erişebilir.", - "session.share.popover.description.unshared": - "Oturumu web'de herkese açık olarak paylaşın. Bağlantıya sahip herkes erişebilecek.", - "session.share.action.share": "Paylaş", - "session.share.action.publish": "Yayınla", - "session.share.action.publishing": "Yayınlanıyor...", - "session.share.action.unpublish": "Yayından kaldır", - "session.share.action.unpublishing": "Yayından kaldırılıyor...", - "session.share.action.view": "Görüntüle", - "session.share.copy.copied": "Kopyalandı", - "session.share.copy.copyLink": "Bağlantı kopyala", - - "lsp.tooltip.none": "LSP sunucusu yok", - "lsp.label.connected": "{{count}} LSP", - - "prompt.loading": "Komut yükleniyor...", - "terminal.loading": "Terminal yükleniyor...", - "terminal.title": "Terminal", - "terminal.title.numbered": "Terminal {{number}}", - "terminal.close": "Terminali kapat", - "terminal.connectionLost.title": "Bağlantı Kesildi", - "terminal.connectionLost.description": - "Terminal bağlantısı kesildi. Bu durum sunucu yeniden başladığında oluşabilir.", - - "common.closeTab": "Sekmeyi kapat", - "common.dismiss": "Kapat", - "common.requestFailed": "İstek başarısız", - "common.moreOptions": "Daha fazla seçenek", - "common.learnMore": "Daha fazla bilgi", - "common.rename": "Yeniden adlandır", - "common.reset": "Sıfırla", - "common.archive": "Arşivle", - "common.delete": "Sil", - "common.close": "Kapat", - "common.edit": "Düzenle", - "common.loadMore": "Daha fazla yükle", - "common.key.esc": "ESC", - - "sidebar.menu.toggle": "Menüyü aç/kapat", - "sidebar.nav.projectsAndSessions": "Projeler ve oturumlar", - "sidebar.settings": "Ayarlar", - "sidebar.help": "Yardım", - "sidebar.workspaces.enable": "Çalışma alanlarını etkinleştir", - "sidebar.workspaces.disable": "Çalışma alanlarını devre dışı bırak", - "sidebar.gettingStarted.title": "Başlarken", - "sidebar.gettingStarted.line1": "Kilo ücretsiz modeller içerir, böylece hemen başlayabilirsiniz.", - "sidebar.gettingStarted.line2": "Claude, GPT, Gemini vb. modelleri kullanmak için herhangi bir sağlayıcı bağlayın.", - "sidebar.project.recentSessions": "Son oturumlar", - "sidebar.project.viewAllSessions": "Tüm oturumları görüntüle", - "sidebar.project.clearNotifications": "Bildirimleri temizle", - - "app.name.desktop": "Kilo Masaüstü", - - "settings.section.desktop": "Masaüstü", - "settings.section.server": "Sunucu", - "settings.tab.general": "Genel", - "settings.tab.shortcuts": "Kısayollar", - "settings.desktop.section.wsl": "WSL", - "settings.desktop.wsl.title": "WSL entegrasyonu", - "settings.desktop.wsl.description": "Kilo sunucusunu Windows'ta WSL içinde çalıştırın.", - - "settings.general.section.appearance": "Görünüm", - "settings.general.section.notifications": "Sistem bildirimleri", - "settings.general.section.updates": "Güncellemeler", - "settings.general.section.sounds": "Ses efektleri", - "settings.general.section.feed": "Akış", - "settings.general.section.display": "Ekran", - - "settings.general.row.language.title": "Dil", - "settings.general.row.language.description": "Kilo'un görünüm dilini değiştirin", - "settings.general.row.appearance.title": "Görünüm", - "settings.general.row.appearance.description": "Kilo'un cihazınızdaki görünümünü özelleştirin", - "settings.general.row.colorScheme.title": "Renk şeması", - "settings.general.row.colorScheme.description": "Kilo'un sistem, açık veya koyu temayı takip etip etmeyeceğini seçin", - "settings.general.row.theme.title": "Tema", - "settings.general.row.theme.description": "Kilo'un temasını özelleştirin.", - "settings.general.row.font.title": "Kod Yazı Tipi", - "settings.general.row.font.description": "Kod bloklarında kullanılan yazı tipini özelleştirin", - "settings.general.row.terminalFont.title": "Terminal Font", - "settings.general.row.terminalFont.description": "Customise the font used in the terminal", - "settings.general.row.uiFont.title": "Arayüz Yazı Tipi", - "settings.general.row.uiFont.description": "Arayüz genelinde kullanılan yazı tipini özelleştirin", - "settings.general.row.followup.title": "Takip davranışı", - "settings.general.row.followup.description": - "Takip komutlarının hemen yönlendirilmesini mi yoksa sırada beklemesini mi istediğinizi seçin", - "settings.general.row.followup.option.queue": "Sıra", - "settings.general.row.followup.option.steer": "Yönlendir", - "settings.general.row.reasoningSummaries.title": "Akıl yürütme özetlerini göster", - "settings.general.row.reasoningSummaries.description": "Zaman çizelgesinde model akıl yürütme özetlerini görüntüle", - "settings.general.row.shellToolPartsExpanded.title": "Kabuk araç bileşenlerini genişlet", - "settings.general.row.shellToolPartsExpanded.description": - "Zaman çizelgesinde kabuk araç bileşenlerini varsayılan olarak genişletilmiş göster", - "settings.general.row.editToolPartsExpanded.title": "Düzenleme araç bileşenlerini genişlet", - "settings.general.row.editToolPartsExpanded.description": - "Zaman çizelgesinde düzenleme, yazma ve yama araç bileşenlerini varsayılan olarak genişletilmiş göster", - - "settings.general.row.showSessionProgressBar.title": "Oturum ilerleme çubuğunu göster", - "settings.general.row.showSessionProgressBar.description": - "Ajan çalışırken oturumun üst kısmında animasyonlu ilerleme çubuğunu göster", - - "settings.general.row.wayland.title": "Yerel Wayland kullan", - "settings.general.row.wayland.description": - "Wayland'da X11 geri dönüşünü devre dışı bırak. Yeniden başlatma gerektirir.", - "settings.general.row.wayland.tooltip": - "Karışık yenileme hızlı monitörlere sahip Linux'ta yerel Wayland daha kararlı olabilir.", - - "settings.general.row.releaseNotes.title": "Sürüm notları", - "settings.general.row.releaseNotes.description": "Güncellemelerden sonra Yenilikler bildirimlerini göster", - - "settings.updates.row.startup.title": "Başlangıçta güncellemeleri kontrol et", - "settings.updates.row.startup.description": "Kilo başladığında otomatik güncelleme kontrolü yap", - "settings.updates.row.check.title": "Güncellemeleri kontrol et", - "settings.updates.row.check.description": "Elle güncelleme kontrolü yap ve varsa yükle", - "settings.updates.action.checkNow": "Şimdi kontrol et", - "settings.updates.action.checking": "Kontrol ediliyor...", - "settings.updates.toast.latest.title": "Güncelsiniz", - "settings.updates.toast.latest.description": "Kilo'un en son sürümünü kullanıyorsunuz.", - - "sound.option.none": "Yok", - "sound.option.alert01": "Uyarı 01", - "sound.option.alert02": "Uyarı 02", - "sound.option.alert03": "Uyarı 03", - "sound.option.alert04": "Uyarı 04", - "sound.option.alert05": "Uyarı 05", - "sound.option.alert06": "Uyarı 06", - "sound.option.alert07": "Uyarı 07", - "sound.option.alert08": "Uyarı 08", - "sound.option.alert09": "Uyarı 09", - "sound.option.alert10": "Uyarı 10", - "sound.option.bipbop01": "Bip-bop 01", - "sound.option.bipbop02": "Bip-bop 02", - "sound.option.bipbop03": "Bip-bop 03", - "sound.option.bipbop04": "Bip-bop 04", - "sound.option.bipbop05": "Bip-bop 05", - "sound.option.bipbop06": "Bip-bop 06", - "sound.option.bipbop07": "Bip-bop 07", - "sound.option.bipbop08": "Bip-bop 08", - "sound.option.bipbop09": "Bip-bop 09", - "sound.option.bipbop10": "Bip-bop 10", - "sound.option.staplebops01": "Staplebops 01", - "sound.option.staplebops02": "Staplebops 02", - "sound.option.staplebops03": "Staplebops 03", - "sound.option.staplebops04": "Staplebops 04", - "sound.option.staplebops05": "Staplebops 05", - "sound.option.staplebops06": "Staplebops 06", - "sound.option.staplebops07": "Staplebops 07", - "sound.option.nope01": "Hayır 01", - "sound.option.nope02": "Hayır 02", - "sound.option.nope03": "Hayır 03", - "sound.option.nope04": "Hayır 04", - "sound.option.nope05": "Hayır 05", - "sound.option.nope06": "Hayır 06", - "sound.option.nope07": "Hayır 07", - "sound.option.nope08": "Hayır 08", - "sound.option.nope09": "Hayır 09", - "sound.option.nope10": "Hayır 10", - "sound.option.nope11": "Hayır 11", - "sound.option.nope12": "Hayır 12", - "sound.option.yup01": "Evet 01", - "sound.option.yup02": "Evet 02", - "sound.option.yup03": "Evet 03", - "sound.option.yup04": "Evet 04", - "sound.option.yup05": "Evet 05", - "sound.option.yup06": "Evet 06", - - "settings.general.notifications.agent.title": "Ajan", - "settings.general.notifications.agent.description": - "Ajan tamamlandığında veya dikkat gerektirdiğinde sistem bildirimi göster", - "settings.general.notifications.permissions.title": "İzinler", - "settings.general.notifications.permissions.description": "İzin gerektiğinde sistem bildirimi göster", - "settings.general.notifications.errors.title": "Hatalar", - "settings.general.notifications.errors.description": "Hata oluştuğunda sistem bildirimi göster", - - "settings.general.sounds.agent.title": "Ajan", - "settings.general.sounds.agent.description": "Ajan tamamlandığında veya dikkat gerektirdiğinde ses çal", - "settings.general.sounds.permissions.title": "İzinler", - "settings.general.sounds.permissions.description": "İzin gerektiğinde ses çal", - "settings.general.sounds.errors.title": "Hatalar", - "settings.general.sounds.errors.description": "Hata oluştuğunda ses çal", - - "settings.shortcuts.title": "Klavye kısayolları", - "settings.shortcuts.reset.button": "Varsayılanlara sıfırla", - "settings.shortcuts.reset.toast.title": "Kısayollar sıfırlandı", - "settings.shortcuts.reset.toast.description": "Klavye kısayolları varsayılanlara sıfırlandı.", - "settings.shortcuts.conflict.title": "Kısayol zaten kullanılıyor", - "settings.shortcuts.conflict.description": "{{keybind}} zaten {{titles}} için atanmış.", - "settings.shortcuts.unassigned": "Atanmamış", - "settings.shortcuts.pressKeys": "Tuşlara basın", - "settings.shortcuts.search.placeholder": "Kısayol ara", - "settings.shortcuts.search.empty": "Kısayol bulunamadı", - - "settings.shortcuts.group.general": "Genel", - "settings.shortcuts.group.session": "Oturum", - "settings.shortcuts.group.navigation": "Gezinme", - "settings.shortcuts.group.modelAndAgent": "Model ve ajan", - "settings.shortcuts.group.terminal": "Terminal", - "settings.shortcuts.group.prompt": "Komut", - - "settings.providers.title": "Sağlayıcılar", - "settings.providers.description": "Sağlayıcı ayarları burada yapılandırılabilecek.", - "settings.providers.section.connected": "Bağlı sağlayıcılar", - "settings.providers.connected.empty": "Bağlı sağlayıcı yok", - "settings.providers.section.popular": "Popüler sağlayıcılar", - "settings.providers.tag.environment": "Ortam", - "settings.providers.tag.config": "Yapılandırma", - "settings.providers.tag.custom": "Özel", - "settings.providers.tag.other": "Diğer", - "settings.models.title": "Modeller", - "settings.models.description": "Model ayarları burada yapılandırılabilecek.", - "settings.agents.title": "Ajanlar", - "settings.agents.description": "Ajan ayarları burada yapılandırılabilecek.", - "settings.commands.title": "Komutlar", - "settings.commands.description": "Komut ayarları burada yapılandırılabilecek.", - "settings.mcp.title": "MCP", - "settings.mcp.description": "MCP ayarları burada yapılandırılabilecek.", - - "settings.permissions.title": "İzinler", - "settings.permissions.description": "Sunucunun varsayılan olarak hangi araçları kullanabileceğini kontrol edin.", - "settings.permissions.section.tools": "Araçlar", - "settings.permissions.toast.updateFailed.title": "İzinler güncellenemedi", - - "settings.permissions.action.allow": "İzin Ver", - "settings.permissions.action.ask": "Sor", - "settings.permissions.action.deny": "Reddet", - - "settings.permissions.tool.read.title": "Oku", - "settings.permissions.tool.read.description": "Bir dosyayı okuma (dosya yoluyla eşleşir)", - "settings.permissions.tool.edit.title": "Düzenle", - "settings.permissions.tool.edit.description": "Düzenleme, yazma, yama ve çoklu düzenleme dahil dosyaları değiştir", - "settings.permissions.tool.glob.title": "Glob", - "settings.permissions.tool.glob.description": "Glob kalıpları kullanarak dosyaları eşle", - "settings.permissions.tool.grep.title": "Grep", - "settings.permissions.tool.grep.description": "Düzenli ifadeler kullanarak dosya içerikleri ara", - "settings.permissions.tool.list.title": "Listele", - "settings.permissions.tool.list.description": "Bir dizindeki dosyaları listele", - "settings.permissions.tool.bash.title": "Bash", - "settings.permissions.tool.bash.description": "Kabuk komutları çalıştır", - "settings.permissions.tool.task.title": "Görev", - "settings.permissions.tool.task.description": "Alt ajanlar başlat", - "settings.permissions.tool.skill.title": "Beceri", - "settings.permissions.tool.skill.description": "Ada göre bir beceri yükle", - "settings.permissions.tool.lsp.title": "LSP", - "settings.permissions.tool.lsp.description": "Dil sunucusu sorguları çalıştır", - "settings.permissions.tool.todowrite.title": "Görev Yaz", - "settings.permissions.tool.todowrite.description": "Görev listesini güncelle", - "settings.permissions.tool.webfetch.title": "Web Getir", - "settings.permissions.tool.webfetch.description": "Bir URL'den içerik getir", - "settings.permissions.tool.websearch.title": "Web Ara", - "settings.permissions.tool.websearch.description": "Web'de ara", - "settings.permissions.tool.codesearch.title": "Kod Ara", - "settings.permissions.tool.codesearch.description": "Web'de kod ara", - "settings.permissions.tool.external_directory.title": "Harici Dizin", - "settings.permissions.tool.external_directory.description": "Proje dizini dışındaki dosyalara eriş", - "settings.permissions.tool.doom_loop.title": "Sonsuz Döngü", - "settings.permissions.tool.doom_loop.description": "Aynı girdiyle tekrarlanan araç çağrılarını algıla", - - "session.delete.failed.title": "Oturum silinemedi", - "session.delete.title": "Oturumu sil", - "session.delete.confirm": '"{{name}}" oturumu silinsin mi?', - "session.delete.button": "Oturumu sil", - - "workspace.new": "Yeni çalışma alanı", - "workspace.type.local": "yerel", - "workspace.type.sandbox": "sandbox", - "workspace.create.failed.title": "Çalışma alanı oluşturulamadı", - "workspace.delete.failed.title": "Çalışma alanı silinemedi", - "workspace.resetting.title": "Çalışma alanı sıfırlanıyor", - "workspace.resetting.description": "Bu bir dakika sürebilir.", - "workspace.reset.failed.title": "Çalışma alanı sıfırlanamadı", - "workspace.reset.success.title": "Çalışma alanı sıfırlandı", - "workspace.reset.success.description": "Çalışma alanı artık varsayılan dalla eşleşiyor.", - "workspace.error.stillPreparing": "Çalışma alanı hâlâ hazırlanıyor", - "workspace.status.checking": "Birleşmemiş değişiklikler kontrol ediliyor...", - "workspace.status.error": "Git durumu doğrulanamadı.", - "workspace.status.clean": "Birleşmemiş değişiklik algılanmadı.", - "workspace.status.dirty": "Bu çalışma alanında birleşmemiş değişiklikler algılandı.", - "workspace.delete.title": "Çalışma alanını sil", - "workspace.delete.confirm": '"{{name}}" çalışma alanı silinsin mi?', - "workspace.delete.button": "Çalışma alanını sil", - "workspace.reset.title": "Çalışma alanını sıfırla", - "workspace.reset.confirm": '"{{name}}" çalışma alanı sıfırlansın mı?', - "workspace.reset.button": "Çalışma alanını sıfırla", - "workspace.reset.archived.none": "Arşivlenecek aktif oturum yok.", - "workspace.reset.archived.one": "1 oturum arşivlenecek.", - "workspace.reset.archived.many": "{{count}} oturum arşivlenecek.", - "workspace.reset.note": "Bu işlem çalışma alanını varsayılan dalla eşleşecek şekilde sıfırlayacak.", - "common.open": "Aç", - "dialog.releaseNotes.action.getStarted": "Başla", - "dialog.releaseNotes.action.next": "İleri", - "dialog.releaseNotes.action.hideFuture": "Bunu gelecekte bir daha gösterme", - "dialog.releaseNotes.media.alt": "Sürüm önizlemesi", - "toast.project.reloadFailed.title": "{{project}} yeniden yüklenemedi", - "error.server.invalidConfiguration": "Geçersiz yapılandırma", - "common.moreCountSuffix": " (+{{count}} daha)", - "common.time.justNow": "Şimdi", - "common.time.minutesAgo.short": "{{count}}dk önce", - "common.time.hoursAgo.short": "{{count}}sa önce", - "common.time.daysAgo.short": "{{count}}g önce", - "settings.providers.connected.environmentDescription": "Ortam değişkenlerinizden bağlandı", - "settings.providers.custom.description": "Temel URL üzerinden OpenAI uyumlu bir sağlayıcı ekleyin.", - - "app.server.unreachable": "{{server}} sunucusuna ulaşılamadı", - "app.server.retrying": "Otomatik olarak tekrar deneniyor...", - "app.server.otherServers": "Diğer sunucular", - "dialog.server.add.usernamePlaceholder": "kullanıcı adı", - "dialog.server.add.passwordPlaceholder": "parola", - "server.row.noUsername": "kullanıcı adı yok", - "session.review.noVcs.createGit.title": "Git deposu oluştur", - "session.review.noVcs.createGit.description": "Bu projedeki değişiklikleri takip et, incele ve geri al", - "session.review.noVcs.createGit.actionLoading": "Git deposu oluşturuluyor...", - "session.review.noVcs.createGit.action": "Git deposu oluştur", - "session.todo.progress": "{{total}} görevin {{done}} tanesi tamamlandı", - "session.question.progress": "{{total}} sorunun {{current}} tanesi", - "session.header.open.finder": "Finder", - "session.header.open.fileExplorer": "Dosya Gezgini", - "session.header.open.fileManager": "Dosya Yöneticisi", - "session.header.open.app.vscode": "VS Code", - "session.header.open.app.cursor": "Cursor", - "session.header.open.app.zed": "Zed", - "session.header.open.app.textmate": "TextMate", - "session.header.open.app.antigravity": "Antigravity", - "session.header.open.app.terminal": "Terminal", - "session.header.open.app.iterm2": "iTerm2", - "session.header.open.app.ghostty": "Ghostty", - "session.header.open.app.warp": "Warp", - "session.header.open.app.xcode": "Xcode", - "session.header.open.app.androidStudio": "Android Studio", - "session.header.open.app.powershell": "PowerShell", - "session.header.open.app.sublimeText": "Sublime Text", - "debugBar.ariaLabel": "Geliştirme performansı teşhisi", - "debugBar.na": "yok", - "debugBar.nav.label": "NAV", - "debugBar.nav.tip": - "Yönlendirici başlangıcından yerleşme sonrası ilk boyamaya kadar ölçülen, bir oturum sayfasına dokunan son tamamlanmış rota geçişi.", - "debugBar.fps.label": "FPS", - "debugBar.fps.tip": "Son 5 saniyedeki kayan saniye başına kare sayısı.", - "debugBar.frame.label": "FRAME", - "debugBar.frame.tip": "Son 5 saniyedeki en kötü kare süresi.", - "debugBar.jank.label": "JANK", - "debugBar.jank.tip": "Son 5 saniyede 32ms üzerindeki kareler.", - "debugBar.long.label": "LONG", - "debugBar.long.tip": "Son 5 saniyedeki engellenen süre ve uzun görev sayısı. Maksimum görev: {{max}}.", - "debugBar.delay.label": "DELAY", - "debugBar.delay.tip": "Son 5 saniyede gözlemlenen en kötü giriş gecikmesi.", - "debugBar.inp.label": "INP", - "debugBar.inp.tip": "Son 5 saniyedeki yaklaşık etkileşim süresi. Bu INP benzeridir, resmi Web Vitals INP değildir.", - "debugBar.cls.label": "CLS", - "debugBar.cls.tip": "Mevcut uygulama ömrü için kümülatif düzen kayması.", - "debugBar.mem.label": "MEM", - "debugBar.mem.tipUnavailable": "Kullanılan JS yığını vs yığın sınırı. Yalnızca Chromium.", - "debugBar.mem.tip": "Kullanılan JS yığını vs yığın sınırı. {{limit}} içinde {{used}}.", - "common.key.ctrl": "Ctrl", - "common.key.alt": "Alt", - "common.key.shift": "Shift", - "common.key.meta": "Meta", - "common.key.space": "Boşluk", - "common.key.backspace": "Geri", - "common.key.enter": "Enter", - "common.key.tab": "Tab", - "common.key.delete": "Delete", - "common.key.home": "Home", - "common.key.end": "End", - "common.key.pageUp": "Page Up", - "common.key.pageDown": "Page Down", - "common.key.insert": "Insert", - "common.unknown": "bilinmiyor", - "error.page.circular": "[Döngüsel]", - "error.globalSDK.noServerAvailable": "Sunucu yok", - "error.globalSDK.serverNotAvailable": "Sunucu mevcut değil", - "error.childStore.persistedCacheCreateFailed": "Kalıcı önbellek oluşturulamadı", - "error.childStore.persistedProjectMetadataCreateFailed": "Kalıcı proje meta verileri oluşturulamadı", - "error.childStore.persistedProjectIconCreateFailed": "Kalıcı proje simgesi oluşturulamadı", - "error.childStore.storeCreateFailed": "Depo oluşturulamadı", - "terminal.connectionLost.abnormalClose": "WebSocket anormal şekilde kapandı: {{code}}", -} satisfies Partial> diff --git a/packages/app/src/i18n/zh.ts b/packages/app/src/i18n/zh.ts deleted file mode 100644 index 52c20f9b5e..0000000000 --- a/packages/app/src/i18n/zh.ts +++ /dev/null @@ -1,920 +0,0 @@ -import { dict as en } from "./en" - -type Keys = keyof typeof en - -export const dict = { - "command.category.suggested": "建议", - "command.category.view": "视图", - "command.category.project": "项目", - "command.category.provider": "提供商", - "command.category.server": "服务器", - "command.category.session": "会话", - "command.category.theme": "主题", - "command.category.language": "语言", - "command.category.file": "文件", - "command.category.context": "上下文", - "command.category.terminal": "终端", - "command.category.model": "模型", - "command.category.mcp": "MCP", - "command.category.agent": "智能体", - "command.category.permissions": "权限", - "command.category.workspace": "工作区", - "command.category.settings": "设置", - - "theme.scheme.system": "系统", - "theme.scheme.light": "浅色", - "theme.scheme.dark": "深色", - - "command.sidebar.toggle": "切换侧边栏", - - "command.project.open": "打开项目", - - "command.provider.connect": "连接提供商", - - "command.server.switch": "切换服务器", - - "command.settings.open": "打开设置", - - "command.session.previous": "上一个会话", - "command.session.next": "下一个会话", - "command.session.previous.unseen": "上一个未读会话", - "command.session.next.unseen": "下一个未读会话", - "command.session.archive": "归档会话", - - "command.palette": "命令面板", - - "command.theme.cycle": "切换主题", - "command.theme.set": "使用主题:{{theme}}", - "command.theme.scheme.cycle": "切换配色方案", - "command.theme.scheme.set": "使用配色方案:{{scheme}}", - - "command.language.cycle": "切换语言", - "command.language.set": "使用语言:{{language}}", - - "command.session.new": "新建会话", - - "command.file.open": "打开文件", - - "command.tab.close": "关闭标签页", - - "command.context.addSelection": "将所选内容添加到上下文", - "command.context.addSelection.description": "添加当前文件中选中的行", - - "command.input.focus": "聚焦输入框", - - "command.terminal.toggle": "切换终端", - - "command.fileTree.toggle": "切换文件树", - - "command.review.toggle": "切换审查", - - "command.terminal.new": "新建终端", - "command.terminal.new.description": "创建新的终端标签页", - - "command.steps.toggle": "切换步骤", - "command.steps.toggle.description": "显示或隐藏当前消息的步骤", - - "command.message.previous": "上一条消息", - "command.message.previous.description": "跳转到上一条用户消息", - "command.message.next": "下一条消息", - "command.message.next.description": "跳转到下一条用户消息", - - "command.model.choose": "选择模型", - "command.model.choose.description": "选择不同的模型", - - "command.mcp.toggle": "切换 MCPs", - "command.mcp.toggle.description": "切换 MCPs", - - "command.agent.cycle": "切换智能体", - "command.agent.cycle.description": "切换到下一个智能体", - "command.agent.cycle.reverse": "反向切换智能体", - "command.agent.cycle.reverse.description": "切换到上一个智能体", - - "command.model.variant.cycle": "切换思考强度", - "command.model.variant.cycle.description": "切换到下一个强度等级", - - "command.prompt.mode.shell": "Shell", - "command.prompt.mode.normal": "Prompt", - - "command.permissions.autoaccept.enable": "自动接受权限", - "command.permissions.autoaccept.disable": "停止自动接受权限", - - "command.workspace.toggle": "切换工作区", - "command.workspace.toggle.description": "在侧边栏启用或禁用多个工作区", - - "command.session.undo": "撤销", - "command.session.undo.description": "撤销上一条消息", - "command.session.redo": "重做", - "command.session.redo.description": "重做上一条撤销的消息", - "command.session.compact": "精简会话", - "command.session.compact.description": "总结会话以减少上下文大小", - "command.session.fork": "从消息分叉", - "command.session.fork.description": "从之前的消息创建新会话", - "command.session.share": "分享会话", - "command.session.share.description": "分享此会话并将链接复制到剪贴板", - "command.session.unshare": "取消分享会话", - "command.session.unshare.description": "停止分享此会话", - - "palette.search.placeholder": "搜索文件、命令和会话", - "palette.empty": "未找到结果", - "palette.group.commands": "命令", - "palette.group.files": "文件", - - "dialog.provider.search.placeholder": "搜索提供商", - "dialog.provider.empty": "未找到提供商", - "dialog.provider.group.popular": "热门", - "dialog.provider.group.other": "其他", - "dialog.provider.tag.recommended": "推荐", - "dialog.provider.opencode.note": "使用 OpenCode Zen 或 API 密钥连接", - "dialog.provider.opencode.tagline": "可靠的优化模型", - "dialog.provider.opencodeGo.tagline": "适合所有人的低成本订阅", - "dialog.provider.anthropic.note": "使用 Claude Pro/Max 或 API 密钥连接", - "dialog.provider.copilot.note": "使用 Copilot 或 API 密钥连接", - "dialog.provider.openai.note": "使用 ChatGPT Pro/Plus 或 API 密钥连接", - "dialog.provider.google.note": "使用 Google 账号或 API 密钥连接", - "dialog.provider.openrouter.note": "使用 OpenRouter 账号或 API 密钥连接", - "dialog.provider.vercel.note": "使用 Vercel 账号或 API 密钥连接", - - "dialog.model.select.title": "选择模型", - "dialog.model.search.placeholder": "搜索模型", - "dialog.model.empty": "未找到模型", - "dialog.model.manage": "管理模型", - "dialog.model.manage.description": "自定义模型选择器中显示的模型。", - "dialog.model.manage.provider.toggle": "切换所有 {{provider}} 模型", - "dialog.model.unpaid.freeModels.title": "Kilo 提供的免费模型", - "dialog.model.unpaid.addMore.title": "从热门提供商添加更多模型", - - "dialog.provider.viewAll": "查看更多提供商", - - "provider.connect.title": "连接 {{provider}}", - "provider.connect.title.anthropicProMax": "使用 Claude Pro/Max 登录", - "provider.connect.selectMethod": "选择 {{provider}} 的登录方式。", - "provider.connect.method.apiKey": "API 密钥", - "provider.connect.status.inProgress": "正在授权...", - "provider.connect.status.waiting": "等待授权...", - "provider.connect.status.failed": "授权失败:{{error}}", - "provider.connect.apiKey.description": - "输入你的 {{provider}} API 密钥以连接帐户,并在 Kilo 中使用 {{provider}} 模型。", - "provider.connect.apiKey.label": "{{provider}} API 密钥", - "provider.connect.apiKey.placeholder": "API 密钥", - "provider.connect.apiKey.required": "API 密钥为必填项", - "provider.connect.opencodeZen.line1": "OpenCode Zen 为你提供一组精选的可靠优化模型,用于代码智能体。", - "provider.connect.opencodeZen.line2": "只需一个 API 密钥,你就能使用 Claude、GPT、Gemini、GLM 等模型。", - "provider.connect.opencodeZen.visit.prefix": "访问 ", - "provider.connect.opencodeZen.visit.link": "opencode.ai/zen", - "provider.connect.opencodeZen.visit.suffix": " 获取你的 API 密钥。", - "provider.connect.oauth.code.visit.prefix": "访问 ", - "provider.connect.oauth.code.visit.link": "此链接", - "provider.connect.oauth.code.visit.suffix": " 获取授权码,以连接你的帐户并在 Kilo 中使用 {{provider}} 模型。", - "provider.connect.oauth.code.label": "{{method}} 授权码", - "provider.connect.oauth.code.placeholder": "授权码", - "provider.connect.oauth.code.required": "授权码为必填项", - "provider.connect.oauth.code.invalid": "授权码无效", - "provider.connect.oauth.auto.visit.prefix": "访问 ", - "provider.connect.oauth.auto.visit.link": "此链接", - "provider.connect.oauth.auto.visit.suffix": " 并输入以下代码,以连接你的帐户并在 Kilo 中使用 {{provider}} 模型。", - "provider.connect.oauth.auto.confirmationCode": "确认码", - "provider.connect.toast.connected.title": "{{provider}} 已连接", - "provider.connect.toast.connected.description": "现在可以使用 {{provider}} 模型了。", - - "provider.custom.title": "自定义提供商", - "provider.custom.description.prefix": "配置与 OpenAI 兼容的提供商。请查看", - "provider.custom.description.link": "提供商配置文档", - "provider.custom.description.suffix": "。", - "provider.custom.field.providerID.label": "提供商 ID", - "provider.custom.field.providerID.placeholder": "myprovider", - "provider.custom.field.providerID.description": "使用小写字母、数字、连字符或下划线", - "provider.custom.field.name.label": "显示名称", - "provider.custom.field.name.placeholder": "我的 AI 提供商", - "provider.custom.field.baseURL.label": "基础 URL", - "provider.custom.field.baseURL.placeholder": "https://api.myprovider.com/v1", - "provider.custom.field.apiKey.label": "API 密钥", - "provider.custom.field.apiKey.placeholder": "API 密钥", - "provider.custom.field.apiKey.description": "可选。如果你通过请求头管理认证,可留空。", - "provider.custom.models.label": "模型", - "provider.custom.models.id.label": "ID", - "provider.custom.models.id.placeholder": "model-id", - "provider.custom.models.name.label": "名称", - "provider.custom.models.name.placeholder": "显示名称", - "provider.custom.models.remove": "移除模型", - "provider.custom.models.add": "添加模型", - "provider.custom.headers.label": "请求头(可选)", - "provider.custom.headers.key.label": "请求头", - "provider.custom.headers.key.placeholder": "Header-Name", - "provider.custom.headers.value.label": "值", - "provider.custom.headers.value.placeholder": "value", - "provider.custom.headers.remove": "移除请求头", - "provider.custom.headers.add": "添加请求头", - "provider.custom.error.providerID.required": "提供商 ID 为必填项", - "provider.custom.error.providerID.format": "请使用小写字母、数字、连字符或下划线", - "provider.custom.error.providerID.exists": "该提供商 ID 已存在", - "provider.custom.error.name.required": "显示名称为必填项", - "provider.custom.error.baseURL.required": "基础 URL 为必填项", - "provider.custom.error.baseURL.format": "必须以 http:// 或 https:// 开头", - "provider.custom.error.required": "必填", - "provider.custom.error.duplicate": "重复", - - "provider.disconnect.toast.disconnected.title": "{{provider}} 已断开连接", - "provider.disconnect.toast.disconnected.description": "{{provider}} 模型已不再可用。", - - "model.tag.free": "免费", - "model.tag.latest": "最新", - "model.provider.anthropic": "Anthropic", - "model.provider.openai": "OpenAI", - "model.provider.google": "Google", - "model.provider.xai": "xAI", - "model.provider.meta": "Meta", - "model.input.text": "文本", - "model.input.image": "图像", - "model.input.audio": "音频", - "model.input.video": "视频", - "model.input.pdf": "pdf", - "model.tooltip.allows": "支持:{{inputs}}", - "model.tooltip.reasoning.allowed": "支持推理", - "model.tooltip.reasoning.none": "不支持推理", - "model.tooltip.context": "上下文上限 {{limit}}", - - "common.search.placeholder": "搜索", - "common.goBack": "返回", - "common.goForward": "前进", - "common.loading": "加载中", - "common.loading.ellipsis": "...", - "common.cancel": "取消", - "common.connect": "连接", - "common.disconnect": "断开连接", - "common.continue": "提交", - "common.submit": "提交", - "common.save": "保存", - "common.saving": "保存中...", - "common.default": "默认", - "common.attachment": "附件", - - "prompt.placeholder.shell": "输入 shell 命令... {{example}}", - "prompt.placeholder.normal": '随便问点什么... "{{example}}"', - "prompt.placeholder.simple": "随便问点什么...", - "prompt.placeholder.summarizeComments": "总结评论…", - "prompt.placeholder.summarizeComment": "总结该评论…", - "prompt.mode.shell": "Shell", - "prompt.mode.normal": "Prompt", - "prompt.mode.shell.exit": "按 esc 退出", - "prompt.example.1": "修复代码库中的一个 TODO", - "prompt.example.2": "这个项目的技术栈是什么?", - "prompt.example.3": "修复失败的测试", - "prompt.example.4": "解释认证是如何工作的", - "prompt.example.5": "查找并修复安全漏洞", - "prompt.example.6": "为用户服务添加单元测试", - "prompt.example.7": "重构这个函数,让它更易读", - "prompt.example.8": "这个错误是什么意思?", - "prompt.example.9": "帮我调试这个问题", - "prompt.example.10": "生成 API 文档", - "prompt.example.11": "优化数据库查询", - "prompt.example.12": "添加输入校验", - "prompt.example.13": "创建一个新的组件用于...", - "prompt.example.14": "我该如何部署这个项目?", - "prompt.example.15": "审查我的代码并给出最佳实践建议", - "prompt.example.16": "为这个函数添加错误处理", - "prompt.example.17": "解释这个正则表达式", - "prompt.example.18": "把它转换成 TypeScript", - "prompt.example.19": "在整个代码库中添加日志", - "prompt.example.20": "哪些依赖已经过期?", - "prompt.example.21": "帮我写一个迁移脚本", - "prompt.example.22": "为这个接口实现缓存", - "prompt.example.23": "给这个列表添加分页", - "prompt.example.24": "创建一个 CLI 命令用于...", - "prompt.example.25": "这里的环境变量是怎么工作的?", - "prompt.popover.emptyResults": "没有匹配的结果", - "prompt.popover.emptyCommands": "没有匹配的命令", - "prompt.dropzone.label": "将图片、PDF 或文本文件拖放到此处", - "prompt.dropzone.file.label": "拖放以 @提及文件", - "prompt.slash.badge.custom": "自定义", - "prompt.slash.badge.skill": "技能", - "prompt.slash.badge.mcp": "mcp", - "prompt.context.active": "当前", - "prompt.context.includeActiveFile": "包含当前文件", - "prompt.context.removeActiveFile": "从上下文移除活动文件", - "prompt.context.removeFile": "从上下文移除文件", - "prompt.action.attachFile": "附加文件", - "prompt.attachment.remove": "移除附件", - "prompt.action.send": "发送", - "prompt.action.stop": "停止", - "prompt.toast.pasteUnsupported.title": "不支持的附件", - "prompt.toast.pasteUnsupported.description": "此处仅能附加图片、PDF 或文本文件。", - "prompt.toast.modelAgentRequired.title": "请选择智能体和模型", - "prompt.toast.modelAgentRequired.description": "发送提示前请先选择智能体和模型。", - "prompt.toast.worktreeCreateFailed.title": "创建工作树失败", - "prompt.toast.sessionCreateFailed.title": "创建会话失败", - "prompt.toast.shellSendFailed.title": "发送 shell 命令失败", - "prompt.toast.commandSendFailed.title": "发送命令失败", - "prompt.toast.promptSendFailed.title": "发送提示失败", - "prompt.toast.promptSendFailed.description": "无法获取会话", - - "dialog.mcp.title": "MCPs", - "dialog.mcp.description": "已启用 {{enabled}} / {{total}}", - "dialog.mcp.empty": "未配置 MCPs", - - "dialog.lsp.empty": "已从文件类型自动检测到 LSPs", - - "dialog.plugins.empty": "在 opencode.json 中配置的插件", - - "mcp.status.connected": "已连接", - "mcp.status.failed": "失败", - "mcp.status.needs_auth": "需要授权", - "mcp.status.disabled": "已禁用", - - "dialog.fork.empty": "没有可用于分叉的消息", - - "dialog.directory.search.placeholder": "搜索文件夹", - "dialog.directory.empty": "未找到文件夹", - - "dialog.server.title": "服务器", - "dialog.server.description": "切换此应用连接的 Kilo 服务器。", - "dialog.server.search.placeholder": "搜索服务器", - "dialog.server.empty": "暂无服务器", - "dialog.server.add.title": "添加服务器", - "dialog.server.add.url": "服务器 URL", - "dialog.server.add.placeholder": "http://localhost:4096", - "dialog.server.add.error": "无法连接到服务器", - "dialog.server.add.checking": "检查中...", - "dialog.server.add.button": "添加服务器", - "dialog.server.add.name": "服务器名称(可选)", - "dialog.server.add.namePlaceholder": "Localhost", - "dialog.server.add.username": "用户名(可选)", - "dialog.server.add.password": "密码(可选)", - "dialog.server.edit.title": "编辑服务器", - "dialog.server.default.title": "默认服务器", - "dialog.server.default.description": "应用启动时连接此服务器,而不是启动本地服务器。需要重启。", - "dialog.server.default.none": "未选择服务器", - "dialog.server.default.set": "将当前服务器设为默认", - "dialog.server.default.clear": "清除", - "dialog.server.action.remove": "移除服务器", - "dialog.server.menu.edit": "编辑", - "dialog.server.menu.default": "设为默认", - "dialog.server.menu.defaultRemove": "取消默认", - "dialog.server.menu.delete": "删除", - "dialog.server.current": "当前服务器", - "dialog.server.status.default": "默认", - - "dialog.project.edit.title": "编辑项目", - "dialog.project.edit.name": "名称", - "dialog.project.edit.icon": "图标", - "dialog.project.edit.icon.alt": "项目图标", - "dialog.project.edit.icon.hint": "点击或拖拽图片", - "dialog.project.edit.icon.recommended": "建议:128x128px", - "dialog.project.edit.color": "颜色", - "dialog.project.edit.color.select": "选择{{color}}颜色", - "dialog.project.edit.worktree.startup": "工作区启动脚本", - "dialog.project.edit.worktree.startup.description": "在创建新的工作区 (worktree) 后运行。", - "dialog.project.edit.worktree.startup.placeholder": "例如 bun install", - - "context.breakdown.title": "上下文拆分", - "context.breakdown.note": "输入 token 的大致拆分。“其他”包含工具定义和开销。", - "context.breakdown.system": "系统", - "context.breakdown.user": "用户", - "context.breakdown.assistant": "助手", - "context.breakdown.tool": "工具调用", - "context.breakdown.other": "其他", - "context.systemPrompt.title": "系统提示词", - "context.rawMessages.title": "原始消息", - "context.stats.session": "会话", - "context.stats.messages": "消息数", - "context.stats.provider": "提供商", - "context.stats.model": "模型", - "context.stats.limit": "上下文限制", - "context.stats.totalTokens": "总 token", - "context.stats.usage": "使用率", - "context.stats.inputTokens": "输入 token", - "context.stats.outputTokens": "输出 token", - "context.stats.reasoningTokens": "推理 token", - "context.stats.cacheTokens": "缓存 token(读/写)", - "context.stats.userMessages": "用户消息", - "context.stats.assistantMessages": "助手消息", - "context.stats.totalCost": "总成本", - "context.stats.sessionCreated": "创建时间", - "context.stats.lastActivity": "最后活动", - "context.usage.tokens": "Token", - "context.usage.usage": "使用率", - "context.usage.cost": "成本", - "context.usage.clickToView": "点击查看上下文", - "context.usage.view": "查看上下文用量", - - "language.en": "English", - "language.zh": "简体中文", - "language.zht": "繁體中文", - "language.ko": "한국어", - "language.de": "Deutsch", - "language.es": "Español", - "language.fr": "Français", - "language.da": "Dansk", - "language.ja": "日本語", - "language.pl": "Polski", - "language.ru": "Русский", - "language.ar": "العربية", - "language.no": "Norsk", - "language.br": "Português (Brasil)", - "language.bs": "Bosanski", - "language.th": "ไทย", - "language.tr": "Türkçe", - - "toast.language.title": "语言", - "toast.language.description": "已切换到{{language}}", - "toast.theme.title": "主题已切换", - "toast.scheme.title": "颜色方案", - "toast.workspace.enabled.title": "工作区已启用", - "toast.workspace.enabled.description": "侧边栏现在显示多个工作树", - "toast.workspace.disabled.title": "工作区已禁用", - "toast.workspace.disabled.description": "侧边栏只显示主工作树", - "toast.permissions.autoaccept.on.title": "正在自动接受权限", - "toast.permissions.autoaccept.on.description": "权限请求将被自动批准", - "toast.permissions.autoaccept.off.title": "已停止自动接受权限", - "toast.permissions.autoaccept.off.description": "权限请求将需要批准", - "toast.model.none.title": "未选择模型", - "toast.model.none.description": "请先连接提供商以总结此会话", - "toast.file.loadFailed.title": "加载文件失败", - "toast.file.listFailed.title": "列出文件失败", - "toast.context.noLineSelection.title": "未选择行", - "toast.context.noLineSelection.description": "请先在文件标签中选择行范围。", - "toast.session.share.copyFailed.title": "无法复制链接到剪贴板", - "toast.session.share.success.title": "会话已分享", - "toast.session.share.success.description": "分享链接已复制到剪贴板", - "toast.session.share.failed.title": "分享会话失败", - "toast.session.share.failed.description": "分享会话时发生错误", - "toast.session.unshare.success.title": "已取消分享会话", - "toast.session.unshare.success.description": "会话已成功取消分享", - "toast.session.unshare.failed.title": "取消分享失败", - "toast.session.unshare.failed.description": "取消分享会话时发生错误", - "toast.session.listFailed.title": "无法加载 {{project}} 的会话", - "toast.update.title": "有可用更新", - "toast.update.description": "Kilo 有新版本 ({{version}}) 可安装。", - "toast.update.action.installRestart": "安装并重启", - "toast.update.action.notYet": "稍后", - - "error.page.title": "出了点问题", - "error.page.description": "加载应用程序时发生错误。", - "error.page.details.label": "错误详情", - "error.page.action.restart": "重启", - "error.page.action.checking": "检查中...", - "error.page.action.checkUpdates": "检查更新", - "error.page.action.updateTo": "更新到 {{version}}", - "error.page.report.prefix": "请将此错误报告给 Kilo 团队", - "error.page.report.discord": "在 Discord 上", - "error.page.version": "版本:{{version}}", - "error.dev.rootNotFound": "未找到根元素。你是不是忘了把它添加到 index.html?或者 id 属性拼写错了?", - "error.globalSync.connectFailed": "无法连接到服务器。是否有服务器正在 `{{url}}` 运行?", - - "directory.error.invalidUrl": "URL 中的目录无效。", - - "error.chain.unknown": "未知错误", - "error.chain.causedBy": "原因:", - "error.chain.apiError": "API 错误", - "error.chain.status": "状态:{{status}}", - "error.chain.retryable": "可重试:{{retryable}}", - "error.chain.responseBody": "响应内容:\n{{body}}", - "error.chain.didYouMean": "你是不是想输入:{{suggestions}}", - "error.chain.modelNotFound": "未找到模型:{{provider}}/{{model}}", - "error.chain.checkConfig": "请检查你的配置 (opencode.json) 中的 provider/model 名称", - "error.chain.mcpFailed": 'MCP 服务器 "{{name}}" 启动失败。注意: Kilo 暂不支持 MCP 认证。', - "error.chain.providerAuthFailed": "提供商认证失败({{provider}}):{{message}}", - "error.chain.providerInitFailed": '无法初始化提供商 "{{provider}}"。请检查凭据和配置。', - "error.chain.configJsonInvalid": "配置文件 {{path}} 不是有效的 JSON(C)", - "error.chain.configJsonInvalidWithMessage": "配置文件 {{path}} 不是有效的 JSON(C):{{message}}", - "error.chain.configDirectoryTypo": - '{{path}} 中的目录 "{{dir}}" 无效。请将目录重命名为 "{{suggestion}}" 或移除它。这是一个常见拼写错误。', - "error.chain.configFrontmatterError": "无法解析 {{path}} 中的 frontmatter:\n{{message}}", - "error.chain.configInvalid": "配置文件 {{path}} 无效", - "error.chain.configInvalidWithMessage": "配置文件 {{path}} 无效:{{message}}", - - "notification.permission.title": "需要权限", - "notification.permission.description": "{{sessionTitle}}({{projectName}})需要权限", - "notification.question.title": "问题", - "notification.question.description": "{{sessionTitle}}({{projectName}})有一个问题", - "notification.action.goToSession": "前往会话", - "notification.session.responseReady.title": "回复已就绪", - "notification.session.error.title": "会话错误", - "notification.session.error.fallbackDescription": "发生错误", - - "home.recentProjects": "最近项目", - "home.empty.title": "没有最近项目", - "home.empty.description": "通过打开本地项目开始使用", - - "session.tab.session": "会话", - "session.tab.review": "审查", - "session.tab.context": "上下文", - "session.panel.reviewAndFiles": "审查和文件", - "session.review.filesChanged": "{{count}} 个文件变更", - "session.review.change.one": "更改", - "session.review.change.other": "更改", - "session.review.loadingChanges": "正在加载更改...", - "session.review.empty": "此会话暂无更改", - "session.review.noVcs": "未检测到 Git 版本控制系统,无法显示更改", - "session.review.noSnapshot": "配置中已禁用快照跟踪,因此会话更改不可用", - "session.review.noChanges": "无更改", - "session.files.selectToOpen": "选择要打开的文件", - "session.files.all": "所有文件", - "session.files.empty": "无文件", - "session.files.binaryContent": "二进制文件(无法显示内容)", - "session.messages.renderEarlier": "显示更早的消息", - "session.messages.loadingEarlier": "正在加载更早的消息...", - "session.messages.loadEarlier": "加载更早的消息", - "session.messages.loading": "正在加载消息...", - "session.messages.jumpToLatest": "跳转到最新", - "session.context.addToContext": "将 {{selection}} 添加到上下文", - "session.todo.title": "待办事项", - "session.todo.collapse": "折叠", - "session.todo.expand": "展开", - "session.followupDock.summary.one": "{{count}} 条排队消息", - "session.followupDock.summary.other": "{{count}} 条排队消息", - "session.followupDock.sendNow": "立即发送", - "session.followupDock.edit": "编辑", - "session.followupDock.collapse": "折叠排队消息", - "session.followupDock.expand": "展开排队消息", - "session.revertDock.summary.one": "{{count}} 条已回滚消息", - "session.revertDock.summary.other": "{{count}} 条已回滚消息", - "session.revertDock.collapse": "折叠已回滚消息", - "session.revertDock.expand": "展开已回滚消息", - "session.revertDock.restore": "恢复消息", - "session.new.title": "构建任何东西", - "session.new.worktree.main": "主分支", - "session.new.worktree.mainWithBranch": "主分支({{branch}})", - "session.new.worktree.create": "创建新的 worktree", - "session.new.lastModified": "最后修改", - "session.header.search.placeholder": "搜索 {{project}}", - "session.header.searchFiles": "搜索文件", - "session.header.openIn": "打开方式", - "session.header.open.action": "打开 {{app}}", - "session.header.open.ariaLabel": "在 {{app}} 中打开", - "session.header.open.menu": "打开选项", - "session.header.open.copyPath": "复制路径", - - "status.popover.trigger": "状态", - "status.popover.ariaLabel": "服务器配置", - "status.popover.tab.servers": "服务器", - "status.popover.tab.mcp": "MCP", - "status.popover.tab.lsp": "LSP", - "status.popover.tab.plugins": "插件", - "status.popover.action.manageServers": "管理服务器", - - "session.share.popover.title": "发布到网页", - "session.share.popover.description.shared": "此会话已在网页上公开。任何拥有链接的人都可以访问。", - "session.share.popover.description.unshared": "在网页上公开分享此会话。任何拥有链接的人都可以访问。", - "session.share.action.share": "分享", - "session.share.action.publish": "发布", - "session.share.action.publishing": "正在发布...", - "session.share.action.unpublish": "取消发布", - "session.share.action.unpublishing": "正在取消发布...", - "session.share.action.view": "查看", - "session.share.copy.copied": "已复制", - "session.share.copy.copyLink": "复制链接", - - "lsp.tooltip.none": "没有 LSP 服务器", - "lsp.label.connected": "{{count}} LSP", - - "prompt.loading": "正在加载提示...", - - "terminal.loading": "正在加载终端...", - "terminal.title": "终端", - "terminal.title.numbered": "终端 {{number}}", - "terminal.close": "关闭终端", - "terminal.connectionLost.title": "连接已丢失", - "terminal.connectionLost.description": "终端连接已中断。这可能发生在服务器重启时。", - - "common.closeTab": "关闭标签页", - "common.dismiss": "忽略", - "common.requestFailed": "请求失败", - "common.moreOptions": "更多选项", - "common.learnMore": "了解更多", - "common.rename": "重命名", - "common.reset": "重置", - "common.archive": "归档", - "common.delete": "删除", - "common.close": "关闭", - "common.edit": "编辑", - "common.loadMore": "加载更多", - "common.key.esc": "ESC", - - "sidebar.menu.toggle": "切换菜单", - "sidebar.nav.projectsAndSessions": "项目和会话", - "sidebar.settings": "设置", - "sidebar.help": "帮助", - "sidebar.workspaces.enable": "启用工作区", - "sidebar.workspaces.disable": "禁用工作区", - "sidebar.gettingStarted.title": "入门", - "sidebar.gettingStarted.line1": "Kilo 提供免费模型,你可以立即开始使用。", - "sidebar.gettingStarted.line2": "连接任意提供商即可使用更多模型,如 Claude、GPT、Gemini 等。", - "sidebar.project.recentSessions": "最近会话", - "sidebar.project.viewAllSessions": "查看全部会话", - "sidebar.project.clearNotifications": "清除通知", - - "app.name.desktop": "Kilo Desktop", - - "settings.section.desktop": "桌面", - "settings.section.server": "服务器", - - "settings.tab.general": "通用", - "settings.tab.shortcuts": "快捷键", - - "settings.desktop.section.wsl": "WSL", - "settings.desktop.wsl.title": "WSL 集成", - "settings.desktop.wsl.description": "在 Windows 的 WSL 环境中运行 Kilo 服务器。", - - "settings.general.section.appearance": "外观", - "settings.general.section.notifications": "系统通知", - "settings.general.section.updates": "更新", - "settings.general.section.sounds": "音效", - "settings.general.section.feed": "动态", - "settings.general.section.display": "显示", - "settings.general.row.language.title": "语言", - "settings.general.row.language.description": "更改 Kilo 的显示语言", - "settings.general.row.appearance.title": "外观", - "settings.general.row.appearance.description": "自定义 Kilo 在你的设备上的外观", - "settings.general.row.colorScheme.title": "配色方案", - "settings.general.row.colorScheme.description": "选择 Kilo 跟随系统、浅色或深色主题", - "settings.general.row.theme.title": "主题", - "settings.general.row.theme.description": "自定义 Kilo 的主题。", - "settings.general.row.font.title": "代码字体", - "settings.general.row.font.description": "自定义代码块使用的字体", - "settings.general.row.terminalFont.title": "Terminal Font", - "settings.general.row.terminalFont.description": "Customise the font used in the terminal", - "settings.general.row.uiFont.title": "界面字体", - "settings.general.row.uiFont.description": "自定义整个界面使用的字体", - "settings.general.row.followup.title": "跟进消息行为", - "settings.general.row.followup.description": "选择跟进提示是立即引导还是在队列中等待", - "settings.general.row.followup.option.queue": "排队", - "settings.general.row.followup.option.steer": "引导", - "settings.general.row.reasoningSummaries.title": "显示推理摘要", - "settings.general.row.reasoningSummaries.description": "在时间线中显示模型推理摘要", - "settings.general.row.shellToolPartsExpanded.title": "展开 shell 工具部分", - "settings.general.row.shellToolPartsExpanded.description": "默认在时间线中展开 shell 工具部分", - "settings.general.row.editToolPartsExpanded.title": "展开编辑工具部分", - "settings.general.row.editToolPartsExpanded.description": "默认在时间线中展开 edit、write 和 patch 工具部分", - "settings.general.row.showSessionProgressBar.title": "显示会话进度条", - "settings.general.row.showSessionProgressBar.description": "当智能体正在工作时,在会话顶部显示动画进度条", - "settings.general.row.wayland.title": "使用原生 Wayland", - "settings.general.row.wayland.description": "在 Wayland 上禁用 X11 回退。需要重启。", - "settings.general.row.wayland.tooltip": "在混合刷新率显示器的 Linux 系统上,原生 Wayland 可能更稳定。", - "settings.general.row.releaseNotes.title": "发行说明", - "settings.general.row.releaseNotes.description": "更新后显示“新功能”弹窗", - - "settings.updates.row.startup.title": "启动时检查更新", - "settings.updates.row.startup.description": "在 Kilo 启动时自动检查更新", - "settings.updates.row.check.title": "检查更新", - "settings.updates.row.check.description": "手动检查更新并在有更新时安装", - "settings.updates.action.checkNow": "立即检查", - "settings.updates.action.checking": "正在检查...", - "settings.updates.toast.latest.title": "已是最新版本", - "settings.updates.toast.latest.description": "你正在使用最新版本的 Kilo。", - - "sound.option.none": "无", - "sound.option.alert01": "警报 01", - "sound.option.alert02": "警报 02", - "sound.option.alert03": "警报 03", - "sound.option.alert04": "警报 04", - "sound.option.alert05": "警报 05", - "sound.option.alert06": "警报 06", - "sound.option.alert07": "警报 07", - "sound.option.alert08": "警报 08", - "sound.option.alert09": "警报 09", - "sound.option.alert10": "警报 10", - "sound.option.bipbop01": "哔啵 01", - "sound.option.bipbop02": "哔啵 02", - "sound.option.bipbop03": "哔啵 03", - "sound.option.bipbop04": "哔啵 04", - "sound.option.bipbop05": "哔啵 05", - "sound.option.bipbop06": "哔啵 06", - "sound.option.bipbop07": "哔啵 07", - "sound.option.bipbop08": "哔啵 08", - "sound.option.bipbop09": "哔啵 09", - "sound.option.bipbop10": "哔啵 10", - "sound.option.staplebops01": "斯泰普博普斯 01", - "sound.option.staplebops02": "斯泰普博普斯 02", - "sound.option.staplebops03": "斯泰普博普斯 03", - "sound.option.staplebops04": "斯泰普博普斯 04", - "sound.option.staplebops05": "斯泰普博普斯 05", - "sound.option.staplebops06": "斯泰普博普斯 06", - "sound.option.staplebops07": "斯泰普博普斯 07", - "sound.option.nope01": "否 01", - "sound.option.nope02": "否 02", - "sound.option.nope03": "否 03", - "sound.option.nope04": "否 04", - "sound.option.nope05": "否 05", - "sound.option.nope06": "否 06", - "sound.option.nope07": "否 07", - "sound.option.nope08": "否 08", - "sound.option.nope09": "否 09", - "sound.option.nope10": "否 10", - "sound.option.nope11": "否 11", - "sound.option.nope12": "否 12", - "sound.option.yup01": "是 01", - "sound.option.yup02": "是 02", - "sound.option.yup03": "是 03", - "sound.option.yup04": "是 04", - "sound.option.yup05": "是 05", - "sound.option.yup06": "是 06", - - "settings.general.notifications.agent.title": "智能体", - "settings.general.notifications.agent.description": "当智能体完成或需要注意时显示系统通知", - "settings.general.notifications.permissions.title": "权限", - "settings.general.notifications.permissions.description": "当需要权限时显示系统通知", - "settings.general.notifications.errors.title": "错误", - "settings.general.notifications.errors.description": "发生错误时显示系统通知", - "settings.general.sounds.agent.title": "智能体", - "settings.general.sounds.agent.description": "当智能体完成或需要注意时播放声音", - "settings.general.sounds.permissions.title": "权限", - "settings.general.sounds.permissions.description": "当需要权限时播放声音", - "settings.general.sounds.errors.title": "错误", - "settings.general.sounds.errors.description": "发生错误时播放声音", - - "settings.shortcuts.title": "键盘快捷键", - "settings.shortcuts.reset.button": "重置为默认值", - "settings.shortcuts.reset.toast.title": "快捷键已重置", - "settings.shortcuts.reset.toast.description": "键盘快捷键已重置为默认设置。", - "settings.shortcuts.conflict.title": "快捷键已被占用", - "settings.shortcuts.conflict.description": "{{keybind}} 已分配给 {{titles}}。", - "settings.shortcuts.unassigned": "未设置", - "settings.shortcuts.pressKeys": "按下按键", - "settings.shortcuts.search.placeholder": "搜索快捷键", - "settings.shortcuts.search.empty": "未找到快捷键", - "settings.shortcuts.group.general": "通用", - "settings.shortcuts.group.session": "会话", - "settings.shortcuts.group.navigation": "导航", - "settings.shortcuts.group.modelAndAgent": "模型与智能体", - "settings.shortcuts.group.terminal": "终端", - "settings.shortcuts.group.prompt": "提示", - - "settings.providers.title": "提供商", - "settings.providers.description": "提供商设置将在此处可配置。", - "settings.providers.section.connected": "已连接的提供商", - "settings.providers.connected.empty": "没有已连接的提供商", - "settings.providers.section.popular": "热门提供商", - "settings.providers.tag.environment": "环境", - "settings.providers.tag.config": "配置", - "settings.providers.tag.custom": "自定义", - "settings.providers.tag.other": "其他", - - "settings.models.title": "模型", - "settings.models.description": "模型设置将在此处可配置。", - - "settings.agents.title": "智能体", - "settings.agents.description": "智能体设置将在此处可配置。", - - "settings.commands.title": "命令", - "settings.commands.description": "命令设置将在此处可配置。", - - "settings.mcp.title": "MCP", - "settings.mcp.description": "MCP 设置将在此处可配置。", - - "settings.permissions.title": "权限", - "settings.permissions.description": "控制服务器默认可以使用哪些工具。", - "settings.permissions.section.tools": "工具", - "settings.permissions.toast.updateFailed.title": "更新权限失败", - "settings.permissions.action.allow": "允许", - "settings.permissions.action.ask": "询问", - "settings.permissions.action.deny": "拒绝", - "settings.permissions.tool.read.title": "读取", - "settings.permissions.tool.read.description": "读取文件(匹配文件路径)", - "settings.permissions.tool.edit.title": "编辑", - "settings.permissions.tool.edit.description": "修改文件,包括编辑、写入、补丁和多重编辑", - "settings.permissions.tool.glob.title": "Glob", - "settings.permissions.tool.glob.description": "使用 glob 模式匹配文件", - "settings.permissions.tool.grep.title": "Grep", - "settings.permissions.tool.grep.description": "使用正则表达式搜索文件内容", - "settings.permissions.tool.list.title": "列表", - "settings.permissions.tool.list.description": "列出目录中的文件", - "settings.permissions.tool.bash.title": "Bash", - "settings.permissions.tool.bash.description": "运行 shell 命令", - "settings.permissions.tool.task.title": "任务", - "settings.permissions.tool.task.description": "启动子智能体", - "settings.permissions.tool.skill.title": "技能", - "settings.permissions.tool.skill.description": "按名称加载技能", - "settings.permissions.tool.lsp.title": "LSP", - "settings.permissions.tool.lsp.description": "运行语言服务器查询", - "settings.permissions.tool.todowrite.title": "更新待办", - "settings.permissions.tool.todowrite.description": "更新待办列表", - "settings.permissions.tool.webfetch.title": "网页获取", - "settings.permissions.tool.webfetch.description": "从 URL 获取内容", - "settings.permissions.tool.websearch.title": "网页搜索", - "settings.permissions.tool.websearch.description": "搜索网页", - "settings.permissions.tool.codesearch.title": "代码搜索", - "settings.permissions.tool.codesearch.description": "在网上搜索代码", - "settings.permissions.tool.external_directory.title": "外部目录", - "settings.permissions.tool.external_directory.description": "访问项目目录之外的文件", - "settings.permissions.tool.doom_loop.title": "死循环", - "settings.permissions.tool.doom_loop.description": "检测具有相同输入的重复工具调用", - - "session.delete.failed.title": "删除会话失败", - "session.delete.title": "删除会话", - "session.delete.confirm": '删除会话 "{{name}}"?', - "session.delete.button": "删除会话", - - "workspace.new": "新建工作区", - "workspace.type.local": "本地", - "workspace.type.sandbox": "沙盒", - "workspace.create.failed.title": "创建工作区失败", - "workspace.delete.failed.title": "删除工作区失败", - "workspace.resetting.title": "正在重置工作区", - "workspace.resetting.description": "这可能需要一点时间。", - "workspace.reset.failed.title": "重置工作区失败", - "workspace.reset.success.title": "工作区已重置", - "workspace.reset.success.description": "工作区已与默认分支保持一致。", - "workspace.error.stillPreparing": "工作区仍在准备中", - "workspace.status.checking": "正在检查未合并的更改...", - "workspace.status.error": "无法验证 git 状态。", - "workspace.status.clean": "未检测到未合并的更改。", - "workspace.status.dirty": "检测到未合并的更改。", - "workspace.delete.title": "删除工作区", - "workspace.delete.confirm": '删除工作区 "{{name}}"?', - "workspace.delete.button": "删除工作区", - "workspace.reset.title": "重置工作区", - "workspace.reset.confirm": '重置工作区 "{{name}}"?', - "workspace.reset.button": "重置工作区", - "workspace.reset.archived.none": "不会归档任何活跃会话。", - "workspace.reset.archived.one": "将归档 1 个会话。", - "workspace.reset.archived.many": "将归档 {{count}} 个会话。", - "workspace.reset.note": "这将把工作区重置为与默认分支一致。", - "common.open": "打开", - "dialog.releaseNotes.action.getStarted": "开始", - "dialog.releaseNotes.action.next": "下一步", - "dialog.releaseNotes.action.hideFuture": "不再显示", - "dialog.releaseNotes.media.alt": "发布预览", - "toast.project.reloadFailed.title": "无法重新加载 {{project}}", - "error.server.invalidConfiguration": "配置无效", - "common.moreCountSuffix": " (还有 {{count}} 个)", - "common.time.justNow": "刚刚", - "common.time.minutesAgo.short": "{{count}}分钟前", - "common.time.hoursAgo.short": "{{count}}小时前", - "common.time.daysAgo.short": "{{count}}天前", - "settings.providers.connected.environmentDescription": "已通过环境变量连接", - "settings.providers.custom.description": "通过基础 URL 添加与 OpenAI 兼容的提供商。", - - "app.server.unreachable": "无法连接到 {{server}}", - "app.server.retrying": "正在自动重试...", - "app.server.otherServers": "其他服务器", - "dialog.server.add.usernamePlaceholder": "用户名", - "dialog.server.add.passwordPlaceholder": "密码", - "server.row.noUsername": "无用户名", - "session.review.noVcs.createGit.title": "创建 Git 仓库", - "session.review.noVcs.createGit.description": "在此项目中跟踪、审查和撤消更改", - "session.review.noVcs.createGit.actionLoading": "正在创建 Git 仓库...", - "session.review.noVcs.createGit.action": "创建 Git 仓库", - "session.todo.progress": "已完成 {{done}} 个任务(共 {{total}} 个)", - "session.question.progress": "{{current}}/{{total}} 个问题", - "session.header.open.finder": "访达", - "session.header.open.fileExplorer": "文件资源管理器", - "session.header.open.fileManager": "文件管理器", - "session.header.open.app.vscode": "VS Code", - "session.header.open.app.cursor": "Cursor", - "session.header.open.app.zed": "Zed", - "session.header.open.app.textmate": "TextMate", - "session.header.open.app.antigravity": "Antigravity", - "session.header.open.app.terminal": "终端", - "session.header.open.app.iterm2": "iTerm2", - "session.header.open.app.ghostty": "Ghostty", - "session.header.open.app.warp": "Warp", - "session.header.open.app.xcode": "Xcode", - "session.header.open.app.androidStudio": "Android Studio", - "session.header.open.app.powershell": "PowerShell", - "session.header.open.app.sublimeText": "Sublime Text", - "debugBar.ariaLabel": "开发性能诊断", - "debugBar.na": "不适用", - "debugBar.nav.label": "NAV", - "debugBar.nav.tip": "最后一次完成的涉及会话页面的路由转换,从路由器启动到稳定后的第一次绘制。", - "debugBar.fps.label": "FPS", - "debugBar.fps.tip": "过去 5 秒内的滚动帧率。", - "debugBar.frame.label": "FRAME", - "debugBar.frame.tip": "过去 5 秒内最差的帧时间。", - "debugBar.jank.label": "JANK", - "debugBar.jank.tip": "过去 5 秒内超过 32ms 的帧。", - "debugBar.long.label": "LONG", - "debugBar.long.tip": "过去 5 秒内的阻塞时间和长任务计数。最大任务:{{max}}。", - "debugBar.delay.label": "DELAY", - "debugBar.delay.tip": "过去 5 秒内观察到的最差输入延迟。", - "debugBar.inp.label": "INP", - "debugBar.inp.tip": "过去 5 秒内的近似交互持续时间。这类似于 INP,而非官方的 Web Vitals INP。", - "debugBar.cls.label": "CLS", - "debugBar.cls.tip": "当前应用生命周期的累积布局偏移。", - "debugBar.mem.label": "MEM", - "debugBar.mem.tipUnavailable": "使用的 JS 堆与堆限制。仅限 Chromium。", - "debugBar.mem.tip": "使用的 JS 堆与堆限制。{{used}} / {{limit}}。", - "common.key.ctrl": "Ctrl", - "common.key.alt": "Alt", - "common.key.shift": "Shift", - "common.key.meta": "Meta", - "common.key.space": "空格", - "common.key.backspace": "退格", - "common.key.enter": "回车", - "common.key.tab": "Tab", - "common.key.delete": "Delete", - "common.key.home": "Home", - "common.key.end": "End", - "common.key.pageUp": "Page Up", - "common.key.pageDown": "Page Down", - "common.key.insert": "Insert", - "common.unknown": "未知", - "error.page.circular": "[循环]", - "error.globalSDK.noServerAvailable": "无可用服务器", - "error.globalSDK.serverNotAvailable": "服务器不可用", - "error.childStore.persistedCacheCreateFailed": "创建持久化缓存失败", - "error.childStore.persistedProjectMetadataCreateFailed": "创建持久化项目元数据失败", - "error.childStore.persistedProjectIconCreateFailed": "创建持久化项目图标失败", - "error.childStore.storeCreateFailed": "创建存储失败", - "terminal.connectionLost.abnormalClose": "WebSocket 异常关闭:{{code}}", -} satisfies Partial> diff --git a/packages/app/src/i18n/zht.ts b/packages/app/src/i18n/zht.ts deleted file mode 100644 index 4e94db10ab..0000000000 --- a/packages/app/src/i18n/zht.ts +++ /dev/null @@ -1,915 +0,0 @@ -import { dict as en } from "./en" - -type Keys = keyof typeof en - -export const dict = { - "command.category.suggested": "建議", - "command.category.view": "檢視", - "command.category.project": "專案", - "command.category.provider": "提供者", - "command.category.server": "伺服器", - "command.category.session": "工作階段", - "command.category.theme": "主題", - "command.category.language": "語言", - "command.category.file": "檔案", - "command.category.context": "上下文", - "command.category.terminal": "終端機", - "command.category.model": "模型", - "command.category.mcp": "MCP", - "command.category.agent": "代理程式", - "command.category.permissions": "權限", - "command.category.workspace": "工作區", - - "command.category.settings": "設定", - "theme.scheme.system": "系統", - "theme.scheme.light": "淺色", - "theme.scheme.dark": "深色", - - "command.sidebar.toggle": "切換側邊欄", - "command.project.open": "開啟專案", - "command.provider.connect": "連接提供者", - "command.server.switch": "切換伺服器", - "command.settings.open": "開啟設定", - "command.session.previous": "上一個工作階段", - "command.session.next": "下一個工作階段", - "command.session.previous.unseen": "上一個未讀會話", - "command.session.next.unseen": "下一個未讀會話", - "command.session.archive": "封存工作階段", - - "command.palette": "命令面板", - - "command.theme.cycle": "循環主題", - "command.theme.set": "使用主題: {{theme}}", - "command.theme.scheme.cycle": "循環配色方案", - "command.theme.scheme.set": "使用配色方案: {{scheme}}", - - "command.language.cycle": "循環語言", - "command.language.set": "使用語言: {{language}}", - - "command.session.new": "新增工作階段", - "command.file.open": "開啟檔案", - "command.tab.close": "關閉分頁", - "command.context.addSelection": "將選取內容加入上下文", - "command.context.addSelection.description": "加入目前檔案中選取的行", - "command.input.focus": "聚焦輸入框", - "command.terminal.toggle": "切換終端機", - "command.fileTree.toggle": "切換檔案樹", - "command.review.toggle": "切換審查", - "command.terminal.new": "新增終端機", - "command.terminal.new.description": "建立新的終端機標籤頁", - "command.steps.toggle": "切換步驟", - "command.steps.toggle.description": "顯示或隱藏目前訊息的步驟", - "command.message.previous": "上一則訊息", - "command.message.previous.description": "跳到上一則使用者訊息", - "command.message.next": "下一則訊息", - "command.message.next.description": "跳到下一則使用者訊息", - "command.model.choose": "選擇模型", - "command.model.choose.description": "選擇不同的模型", - "command.mcp.toggle": "切換 MCP", - "command.mcp.toggle.description": "切換 MCP", - "command.agent.cycle": "循環代理程式", - "command.agent.cycle.description": "切換到下一個代理程式", - "command.agent.cycle.reverse": "反向循環代理程式", - "command.agent.cycle.reverse.description": "切換到上一個代理程式", - "command.model.variant.cycle": "循環思考強度", - "command.model.variant.cycle.description": "切換到下一個強度等級", - "command.prompt.mode.shell": "Shell", - "command.prompt.mode.normal": "Prompt", - "command.permissions.autoaccept.enable": "自動接受權限", - "command.permissions.autoaccept.disable": "停止自動接受權限", - "command.workspace.toggle": "切換工作區", - "command.workspace.toggle.description": "在側邊欄啟用或停用多個工作區", - "command.session.undo": "復原", - "command.session.undo.description": "復原上一則訊息", - "command.session.redo": "重做", - "command.session.redo.description": "重做上一則復原的訊息", - "command.session.compact": "精簡工作階段", - "command.session.compact.description": "總結工作階段以減少上下文大小", - "command.session.fork": "從訊息分支", - "command.session.fork.description": "從先前的訊息建立新工作階段", - "command.session.share": "分享工作階段", - "command.session.share.description": "分享此工作階段並將連結複製到剪貼簿", - "command.session.unshare": "取消分享工作階段", - "command.session.unshare.description": "停止分享此工作階段", - - "palette.search.placeholder": "搜尋檔案、命令和工作階段", - "palette.empty": "找不到結果", - "palette.group.commands": "命令", - "palette.group.files": "檔案", - - "dialog.provider.search.placeholder": "搜尋提供者", - "dialog.provider.empty": "找不到提供者", - "dialog.provider.group.popular": "熱門", - "dialog.provider.group.other": "其他", - "dialog.provider.tag.recommended": "推薦", - "dialog.provider.opencode.note": "精選模型,包含 Claude、GPT、Gemini 等等", - "dialog.provider.opencode.tagline": "可靠的優化模型", - "dialog.provider.opencodeGo.tagline": "適合所有人的低成本訂閱", - "dialog.provider.anthropic.note": "使用 Claude Pro/Max 或 API 金鑰連線", - "dialog.provider.openai.note": "使用 ChatGPT Pro/Plus 或 API 金鑰連線", - "dialog.provider.copilot.note": "使用 Copilot 或 API 金鑰連線", - "dialog.provider.google.note": "Gemini 模型,提供快速且結構化的回應", - "dialog.provider.openrouter.note": "從單一提供者存取所有支援的模型", - "dialog.provider.vercel.note": "透過智慧路由統一存取 AI 模型", - - "dialog.model.select.title": "選擇模型", - "dialog.model.search.placeholder": "搜尋模型", - "dialog.model.empty": "找不到模型", - "dialog.model.manage": "管理模型", - "dialog.model.manage.description": "自訂模型選擇器中顯示的模型。", - "dialog.model.manage.provider.toggle": "切換所有 {{provider}} 模型", - - "dialog.model.unpaid.freeModels.title": "Kilo 提供的免費模型", - "dialog.model.unpaid.addMore.title": "從熱門提供者新增更多模型", - - "dialog.provider.viewAll": "查看更多提供者", - - "provider.connect.title": "連線 {{provider}}", - "provider.connect.title.anthropicProMax": "使用 Claude Pro/Max 登入", - "provider.connect.selectMethod": "選擇 {{provider}} 的登入方式。", - "provider.connect.method.apiKey": "API 金鑰", - "provider.connect.status.inProgress": "正在授權...", - "provider.connect.status.waiting": "等待授權...", - "provider.connect.status.failed": "授權失敗: {{error}}", - "provider.connect.apiKey.description": - "輸入你的 {{provider}} API 金鑰以連線帳戶,並在 Kilo 中使用 {{provider}} 模型。", - "provider.connect.apiKey.label": "{{provider}} API 金鑰", - "provider.connect.apiKey.placeholder": "API 金鑰", - "provider.connect.apiKey.required": "API 金鑰為必填", - "provider.connect.opencodeZen.line1": "OpenCode Zen 為你提供一組精選的可靠最佳化模型,用於程式碼代理程式。", - "provider.connect.opencodeZen.line2": "只需一個 API 金鑰,你就能使用 Claude、GPT、Gemini、GLM 等模型。", - "provider.connect.opencodeZen.visit.prefix": "造訪 ", - "provider.connect.opencodeZen.visit.link": "opencode.ai/zen", - "provider.connect.opencodeZen.visit.suffix": " 取得你的 API 金鑰。", - "provider.connect.oauth.code.visit.prefix": "造訪 ", - "provider.connect.oauth.code.visit.link": "此連結", - "provider.connect.oauth.code.visit.suffix": " 取得授權碼,以連線你的帳戶並在 Kilo 中使用 {{provider}} 模型。", - "provider.connect.oauth.code.label": "{{method}} 授權碼", - "provider.connect.oauth.code.placeholder": "授權碼", - "provider.connect.oauth.code.required": "授權碼為必填", - "provider.connect.oauth.code.invalid": "授權碼無效", - "provider.connect.oauth.auto.visit.prefix": "造訪 ", - "provider.connect.oauth.auto.visit.link": "此連結", - "provider.connect.oauth.auto.visit.suffix": " 並輸入以下程式碼,以連線你的帳戶並在 Kilo 中使用 {{provider}} 模型。", - "provider.connect.oauth.auto.confirmationCode": "確認碼", - "provider.connect.toast.connected.title": "{{provider}} 已連線", - "provider.connect.toast.connected.description": "現在可以使用 {{provider}} 模型了。", - - "provider.custom.title": "自訂提供商", - "provider.custom.description.prefix": "設定與 OpenAI 相容的提供商。請參閱", - "provider.custom.description.link": "提供商設定文件", - "provider.custom.description.suffix": "。", - "provider.custom.field.providerID.label": "提供商 ID", - "provider.custom.field.providerID.placeholder": "myprovider", - "provider.custom.field.providerID.description": "使用小寫字母、數字、連字號或底線", - "provider.custom.field.name.label": "顯示名稱", - "provider.custom.field.name.placeholder": "我的 AI 提供商", - "provider.custom.field.baseURL.label": "基礎 URL", - "provider.custom.field.baseURL.placeholder": "https://api.myprovider.com/v1", - "provider.custom.field.apiKey.label": "API 金鑰", - "provider.custom.field.apiKey.placeholder": "API 金鑰", - "provider.custom.field.apiKey.description": "選填。若您透過標頭管理驗證,可留空。", - "provider.custom.models.label": "模型", - "provider.custom.models.id.label": "ID", - "provider.custom.models.id.placeholder": "model-id", - "provider.custom.models.name.label": "名稱", - "provider.custom.models.name.placeholder": "顯示名稱", - "provider.custom.models.remove": "移除模型", - "provider.custom.models.add": "新增模型", - "provider.custom.headers.label": "標頭(選填)", - "provider.custom.headers.key.label": "標頭", - "provider.custom.headers.key.placeholder": "Header-Name", - "provider.custom.headers.value.label": "值", - "provider.custom.headers.value.placeholder": "value", - "provider.custom.headers.remove": "移除標頭", - "provider.custom.headers.add": "新增標頭", - "provider.custom.error.providerID.required": "提供商 ID 為必填", - "provider.custom.error.providerID.format": "請使用小寫字母、數字、連字號或底線", - "provider.custom.error.providerID.exists": "該提供商 ID 已存在", - "provider.custom.error.name.required": "顯示名稱為必填", - "provider.custom.error.baseURL.required": "基礎 URL 為必填", - "provider.custom.error.baseURL.format": "必須以 http:// 或 https:// 開頭", - "provider.custom.error.required": "必填", - "provider.custom.error.duplicate": "重複", - - "provider.disconnect.toast.disconnected.title": "{{provider}} 已中斷連線", - "provider.disconnect.toast.disconnected.description": "{{provider}} 模型已不再可用。", - "model.tag.free": "免費", - "model.tag.latest": "最新", - - "model.provider.anthropic": "Anthropic", - "model.provider.openai": "OpenAI", - "model.provider.google": "Google", - "model.provider.xai": "xAI", - "model.provider.meta": "Meta", - "model.input.text": "文字", - "model.input.image": "圖片", - "model.input.audio": "音訊", - "model.input.video": "影片", - "model.input.pdf": "pdf", - "model.tooltip.allows": "支援: {{inputs}}", - "model.tooltip.reasoning.allowed": "支援推理", - "model.tooltip.reasoning.none": "不支援推理", - "model.tooltip.context": "上下文上限 {{limit}}", - "common.search.placeholder": "搜尋", - "common.goBack": "返回", - "common.goForward": "前進", - "common.loading": "載入中", - "common.loading.ellipsis": "...", - "common.cancel": "取消", - "common.connect": "連線", - "common.disconnect": "中斷連線", - "common.continue": "提交", - "common.submit": "提交", - "common.save": "儲存", - "common.saving": "儲存中...", - "common.default": "預設", - "common.attachment": "附件", - - "prompt.placeholder.shell": "輸入 shell 命令... {{example}}", - "prompt.placeholder.normal": '隨便問點什麼... "{{example}}"', - "prompt.placeholder.simple": "隨便問點什麼...", - "prompt.placeholder.summarizeComments": "摘要評論…", - "prompt.placeholder.summarizeComment": "摘要這則評論…", - "prompt.mode.shell": "Shell", - "prompt.mode.normal": "Prompt", - "prompt.mode.shell.exit": "按 esc 退出", - - "prompt.example.1": "修復程式碼庫中的一個 TODO", - "prompt.example.2": "這個專案的技術堆疊是什麼?", - "prompt.example.3": "修復失敗的測試", - "prompt.example.4": "解釋驗證是如何運作的", - "prompt.example.5": "尋找並修復安全漏洞", - "prompt.example.6": "為使用者服務新增單元測試", - "prompt.example.7": "重構這個函式,讓它更易讀", - "prompt.example.8": "這個錯誤是什麼意思?", - "prompt.example.9": "幫我偵錯這個問題", - "prompt.example.10": "產生 API 文件", - "prompt.example.11": "最佳化資料庫查詢", - "prompt.example.12": "新增輸入驗證", - "prompt.example.13": "建立一個新的元件用於...", - "prompt.example.14": "我該如何部署這個專案?", - "prompt.example.15": "審查我的程式碼並給出最佳實務建議", - "prompt.example.16": "為這個函式新增錯誤處理", - "prompt.example.17": "解釋這個正規表示式", - "prompt.example.18": "把它轉換成 TypeScript", - "prompt.example.19": "在整個程式碼庫中新增日誌", - "prompt.example.20": "哪些相依性已經過期?", - "prompt.example.21": "幫我寫一個遷移腳本", - "prompt.example.22": "為這個端點實作快取", - "prompt.example.23": "給這個清單新增分頁", - "prompt.example.24": "建立一個 CLI 命令用於...", - "prompt.example.25": "這裡的環境變數是怎麼運作的?", - - "prompt.popover.emptyResults": "沒有符合的結果", - "prompt.popover.emptyCommands": "沒有符合的命令", - "prompt.dropzone.label": "將圖片、PDF 或文字檔案拖放到此處", - "prompt.dropzone.file.label": "拖放以 @提及檔案", - "prompt.slash.badge.custom": "自訂", - "prompt.slash.badge.skill": "技能", - "prompt.slash.badge.mcp": "mcp", - "prompt.context.active": "作用中", - "prompt.context.includeActiveFile": "包含作用中檔案", - "prompt.context.removeActiveFile": "從上下文移除目前檔案", - "prompt.context.removeFile": "從上下文移除檔案", - "prompt.action.attachFile": "附加檔案", - "prompt.attachment.remove": "移除附件", - "prompt.action.send": "傳送", - "prompt.action.stop": "停止", - - "prompt.toast.pasteUnsupported.title": "不支援的附件", - "prompt.toast.pasteUnsupported.description": "此處僅能附加圖片、PDF 或文字檔案。", - "prompt.toast.modelAgentRequired.title": "請選擇代理程式和模型", - "prompt.toast.modelAgentRequired.description": "傳送提示前請先選擇代理程式和模型。", - "prompt.toast.worktreeCreateFailed.title": "建立工作樹失敗", - "prompt.toast.sessionCreateFailed.title": "建立工作階段失敗", - "prompt.toast.shellSendFailed.title": "傳送 shell 命令失敗", - "prompt.toast.commandSendFailed.title": "傳送命令失敗", - "prompt.toast.promptSendFailed.title": "傳送提示失敗", - "prompt.toast.promptSendFailed.description": "無法取得工作階段", - - "dialog.mcp.title": "MCP", - "dialog.mcp.description": "已啟用 {{enabled}} / {{total}}", - "dialog.mcp.empty": "未設定 MCP", - - "dialog.lsp.empty": "已從檔案類型自動偵測到 LSPs", - "dialog.plugins.empty": "在 opencode.json 中設定的外掛程式", - - "mcp.status.connected": "已連線", - "mcp.status.failed": "失敗", - "mcp.status.needs_auth": "需要授權", - "mcp.status.disabled": "已停用", - - "dialog.fork.empty": "沒有可用於分支的訊息", - - "dialog.directory.search.placeholder": "搜尋資料夾", - "dialog.directory.empty": "找不到資料夾", - - "dialog.server.title": "伺服器", - "dialog.server.description": "切換此應用程式連線的 Kilo 伺服器。", - "dialog.server.search.placeholder": "搜尋伺服器", - "dialog.server.empty": "暫無伺服器", - "dialog.server.add.title": "新增伺服器", - "dialog.server.add.url": "伺服器 URL", - "dialog.server.add.placeholder": "http://localhost:4096", - "dialog.server.add.error": "無法連線到伺服器", - "dialog.server.add.checking": "檢查中...", - "dialog.server.add.button": "新增伺服器", - "dialog.server.add.name": "伺服器名稱(選填)", - "dialog.server.add.namePlaceholder": "Localhost", - "dialog.server.add.username": "使用者名稱(選填)", - "dialog.server.add.password": "密碼(選填)", - "dialog.server.edit.title": "編輯伺服器", - "dialog.server.default.title": "預設伺服器", - "dialog.server.default.description": "應用程式啟動時連線此伺服器,而不是啟動本地伺服器。需要重新啟動。", - "dialog.server.default.none": "未選擇伺服器", - "dialog.server.default.set": "將目前伺服器設為預設", - "dialog.server.default.clear": "清除", - "dialog.server.action.remove": "移除伺服器", - - "dialog.server.menu.edit": "編輯", - "dialog.server.menu.default": "設為預設", - "dialog.server.menu.defaultRemove": "取消預設", - "dialog.server.menu.delete": "刪除", - "dialog.server.current": "目前伺服器", - "dialog.server.status.default": "預設", - - "dialog.project.edit.title": "編輯專案", - "dialog.project.edit.name": "名稱", - "dialog.project.edit.icon": "圖示", - "dialog.project.edit.icon.alt": "專案圖示", - "dialog.project.edit.icon.hint": "點擊或拖曳圖片", - "dialog.project.edit.icon.recommended": "建議:128x128px", - "dialog.project.edit.color": "顏色", - "dialog.project.edit.color.select": "選擇{{color}}顏色", - - "dialog.project.edit.worktree.startup": "工作區啟動腳本", - "dialog.project.edit.worktree.startup.description": "在建立新的工作區 (worktree) 後執行。", - "dialog.project.edit.worktree.startup.placeholder": "例如 bun install", - "context.breakdown.title": "上下文拆分", - "context.breakdown.note": "輸入 token 的大致拆分。「其他」包含工具定義和額外開銷。", - "context.breakdown.system": "系統", - "context.breakdown.user": "使用者", - "context.breakdown.assistant": "助手", - "context.breakdown.tool": "工具呼叫", - "context.breakdown.other": "其他", - - "context.systemPrompt.title": "系統提示詞", - "context.rawMessages.title": "原始訊息", - - "context.stats.session": "工作階段", - "context.stats.messages": "訊息數", - "context.stats.provider": "提供者", - "context.stats.model": "模型", - "context.stats.limit": "上下文限制", - "context.stats.totalTokens": "總 token", - "context.stats.usage": "使用量", - "context.stats.inputTokens": "輸入 token", - "context.stats.outputTokens": "輸出 token", - "context.stats.reasoningTokens": "推理 token", - "context.stats.cacheTokens": "快取 token(讀/寫)", - "context.stats.userMessages": "使用者訊息", - "context.stats.assistantMessages": "助手訊息", - "context.stats.totalCost": "總成本", - "context.stats.sessionCreated": "建立時間", - "context.stats.lastActivity": "最後活動", - - "context.usage.tokens": "Token", - "context.usage.usage": "使用量", - "context.usage.cost": "成本", - "context.usage.clickToView": "點擊查看上下文", - "context.usage.view": "檢視上下文用量", - - "language.en": "English", - "language.zh": "简体中文", - "language.zht": "繁體中文", - "language.ko": "한국어", - "language.de": "Deutsch", - "language.es": "Español", - "language.fr": "Français", - "language.da": "Dansk", - "language.ja": "日本語", - "language.pl": "Polski", - "language.ru": "Русский", - "language.ar": "العربية", - "language.no": "Norsk", - "language.br": "Português (Brasil)", - "language.bs": "Bosanski", - "language.th": "ไทย", - "language.tr": "Türkçe", - - "toast.language.title": "語言", - "toast.language.description": "已切換到 {{language}}", - - "toast.theme.title": "主題已切換", - "toast.scheme.title": "顏色方案", - - "toast.workspace.enabled.title": "工作區已啟用", - "toast.workspace.enabled.description": "側邊欄現在顯示多個工作樹", - "toast.workspace.disabled.title": "工作區已停用", - "toast.workspace.disabled.description": "側邊欄只顯示主工作樹", - - "toast.permissions.autoaccept.on.title": "正在自動接受權限", - "toast.permissions.autoaccept.on.description": "權限請求將被自動批准", - "toast.permissions.autoaccept.off.title": "已停止自動接受權限", - "toast.permissions.autoaccept.off.description": "權限請求將需要批准", - - "toast.model.none.title": "未選擇模型", - "toast.model.none.description": "請先連線提供者以總結此工作階段", - - "toast.file.loadFailed.title": "載入檔案失敗", - - "toast.file.listFailed.title": "列出檔案失敗", - "toast.context.noLineSelection.title": "未選取行", - "toast.context.noLineSelection.description": "請先在檔案分頁中選取行範圍。", - "toast.session.share.copyFailed.title": "無法複製連結到剪貼簿", - "toast.session.share.success.title": "工作階段已分享", - "toast.session.share.success.description": "分享連結已複製到剪貼簿", - "toast.session.share.failed.title": "分享工作階段失敗", - "toast.session.share.failed.description": "分享工作階段時發生錯誤", - - "toast.session.unshare.success.title": "已取消分享工作階段", - "toast.session.unshare.success.description": "工作階段已成功取消分享", - "toast.session.unshare.failed.title": "取消分享失敗", - "toast.session.unshare.failed.description": "取消分享工作階段時發生錯誤", - - "toast.session.listFailed.title": "無法載入 {{project}} 的工作階段", - - "toast.update.title": "有可用更新", - "toast.update.description": "Kilo 有新版本 ({{version}}) 可安裝。", - "toast.update.action.installRestart": "安裝並重新啟動", - "toast.update.action.notYet": "稍後", - - "error.page.title": "出了點問題", - "error.page.description": "載入應用程式時發生錯誤。", - "error.page.details.label": "錯誤詳情", - "error.page.action.restart": "重新啟動", - "error.page.action.checking": "檢查中...", - "error.page.action.checkUpdates": "檢查更新", - "error.page.action.updateTo": "更新到 {{version}}", - "error.page.report.prefix": "請將此錯誤回報給 Kilo 團隊", - "error.page.report.discord": "在 Discord 上", - "error.page.version": "版本: {{version}}", - - "error.dev.rootNotFound": "找不到根元素。你是不是忘了把它新增到 index.html? 或者 id 屬性拼錯了?", - - "error.globalSync.connectFailed": "無法連線到伺服器。是否有伺服器正在 `{{url}}` 執行?", - "directory.error.invalidUrl": "URL 中的目錄無效。", - - "error.chain.unknown": "未知錯誤", - "error.chain.causedBy": "原因:", - "error.chain.apiError": "API 錯誤", - "error.chain.status": "狀態: {{status}}", - "error.chain.retryable": "可重試: {{retryable}}", - "error.chain.responseBody": "回應內容:\n{{body}}", - "error.chain.didYouMean": "你是不是想輸入: {{suggestions}}", - "error.chain.modelNotFound": "找不到模型: {{provider}}/{{model}}", - "error.chain.checkConfig": "請檢查你的設定 (opencode.json) 中的 provider/model 名稱", - "error.chain.mcpFailed": 'MCP 伺服器 "{{name}}" 啟動失敗。注意: Kilo 暫不支援 MCP 認證。', - "error.chain.providerAuthFailed": "提供者認證失敗 ({{provider}}): {{message}}", - "error.chain.providerInitFailed": '無法初始化提供者 "{{provider}}"。請檢查憑證和設定。', - "error.chain.configJsonInvalid": "設定檔 {{path}} 不是有效的 JSON(C)", - "error.chain.configJsonInvalidWithMessage": "設定檔 {{path}} 不是有效的 JSON(C): {{message}}", - "error.chain.configDirectoryTypo": - '{{path}} 中的目錄 "{{dir}}" 無效。請將目錄重新命名為 "{{suggestion}}" 或移除它。這是一個常見拼寫錯誤。', - "error.chain.configFrontmatterError": "無法解析 {{path}} 中的 frontmatter:\n{{message}}", - "error.chain.configInvalid": "設定檔 {{path}} 無效", - "error.chain.configInvalidWithMessage": "設定檔 {{path}} 無效: {{message}}", - - "notification.permission.title": "需要權限", - "notification.permission.description": "{{sessionTitle}}({{projectName}})需要權限", - "notification.question.title": "問題", - "notification.question.description": "{{sessionTitle}}({{projectName}})有一個問題", - "notification.action.goToSession": "前往工作階段", - - "notification.session.responseReady.title": "回覆已就緒", - "notification.session.error.title": "工作階段錯誤", - "notification.session.error.fallbackDescription": "發生錯誤", - - "home.recentProjects": "最近專案", - "home.empty.title": "沒有最近專案", - "home.empty.description": "透過開啟本地專案開始使用", - - "session.tab.session": "工作階段", - "session.tab.review": "審查", - "session.tab.context": "上下文", - "session.panel.reviewAndFiles": "審查與檔案", - "session.review.filesChanged": "{{count}} 個檔案變更", - "session.review.change.one": "變更", - "session.review.change.other": "變更", - "session.review.loadingChanges": "正在載入變更...", - "session.review.empty": "此工作階段暫無變更", - "session.review.noChanges": "沒有變更", - "session.review.noVcs": "未偵測到 Git 版本控制系統,無法顯示變更", - "session.review.noSnapshot": "設定中已停用快照追蹤,因此無法使用工作階段變更", - "session.files.selectToOpen": "選取要開啟的檔案", - "session.files.all": "所有檔案", - "session.files.empty": "沒有檔案", - "session.files.binaryContent": "二進位檔案(無法顯示內容)", - "session.messages.renderEarlier": "顯示更早的訊息", - "session.messages.loadingEarlier": "正在載入更早的訊息...", - "session.messages.loadEarlier": "載入更早的訊息", - "session.messages.loading": "正在載入訊息...", - - "session.messages.jumpToLatest": "跳到最新", - "session.context.addToContext": "將 {{selection}} 新增到上下文", - "session.todo.title": "待辦事項", - "session.todo.collapse": "折疊", - "session.todo.expand": "展開", - "session.followupDock.summary.one": "{{count}} 則佇列訊息", - "session.followupDock.summary.other": "{{count}} 則佇列訊息", - "session.followupDock.sendNow": "立即傳送", - "session.followupDock.edit": "編輯", - "session.followupDock.collapse": "收合佇列訊息", - "session.followupDock.expand": "展開佇列訊息", - "session.revertDock.summary.one": "{{count}} 則已回復訊息", - "session.revertDock.summary.other": "{{count}} 則已回復訊息", - "session.revertDock.collapse": "收合已回復訊息", - "session.revertDock.expand": "展開已回復訊息", - "session.revertDock.restore": "還原訊息", - - "session.new.title": "建構任何東西", - "session.new.worktree.main": "主分支", - "session.new.worktree.mainWithBranch": "主分支 ({{branch}})", - "session.new.worktree.create": "建立新的 worktree", - "session.new.lastModified": "最後修改", - - "session.header.search.placeholder": "搜尋 {{project}}", - "session.header.searchFiles": "搜尋檔案", - "session.header.openIn": "開啟於", - "session.header.open.action": "開啟 {{app}}", - "session.header.open.ariaLabel": "在 {{app}} 中開啟", - "session.header.open.menu": "開啟選項", - "session.header.open.copyPath": "複製路徑", - - "status.popover.trigger": "狀態", - "status.popover.ariaLabel": "伺服器設定", - "status.popover.tab.servers": "伺服器", - "status.popover.tab.mcp": "MCP", - "status.popover.tab.lsp": "LSP", - "status.popover.tab.plugins": "外掛程式", - "status.popover.action.manageServers": "管理伺服器", - - "session.share.popover.title": "發佈到網頁", - "session.share.popover.description.shared": "此工作階段已在網頁上公開。任何擁有連結的人都可以存取。", - "session.share.popover.description.unshared": "在網頁上公開分享此工作階段。任何擁有連結的人都可以存取。", - "session.share.action.share": "分享", - "session.share.action.publish": "發佈", - "session.share.action.publishing": "正在發佈...", - "session.share.action.unpublish": "取消發佈", - "session.share.action.unpublishing": "正在取消發佈...", - "session.share.action.view": "檢視", - "session.share.copy.copied": "已複製", - "session.share.copy.copyLink": "複製連結", - - "lsp.tooltip.none": "沒有 LSP 伺服器", - "lsp.label.connected": "{{count}} LSP", - - "prompt.loading": "正在載入提示...", - "terminal.loading": "正在載入終端機...", - "terminal.title": "終端機", - "terminal.title.numbered": "終端機 {{number}}", - "terminal.close": "關閉終端機", - - "terminal.connectionLost.title": "連線中斷", - "terminal.connectionLost.description": "終端機連線已中斷。這可能會在伺服器重新啟動時發生。", - "common.closeTab": "關閉標籤頁", - "common.dismiss": "忽略", - "common.requestFailed": "要求失敗", - "common.moreOptions": "更多選項", - "common.learnMore": "深入了解", - "common.rename": "重新命名", - "common.reset": "重設", - "common.archive": "封存", - "common.delete": "刪除", - "common.close": "關閉", - "common.edit": "編輯", - "common.loadMore": "載入更多", - - "common.key.esc": "ESC", - "sidebar.menu.toggle": "切換選單", - "sidebar.nav.projectsAndSessions": "專案與工作階段", - "sidebar.settings": "設定", - "sidebar.help": "說明", - "sidebar.workspaces.enable": "啟用工作區", - "sidebar.workspaces.disable": "停用工作區", - "sidebar.gettingStarted.title": "開始使用", - "sidebar.gettingStarted.line1": "Kilo 提供免費模型,你可以立即開始使用。", - "sidebar.gettingStarted.line2": "連線任意提供者即可使用更多模型,如 Claude、GPT、Gemini 等。", - "sidebar.project.recentSessions": "最近工作階段", - "sidebar.project.viewAllSessions": "查看全部工作階段", - "sidebar.project.clearNotifications": "清除通知", - - "app.name.desktop": "Kilo Desktop", - "settings.section.desktop": "桌面", - "settings.section.server": "伺服器", - "settings.tab.general": "一般", - "settings.tab.shortcuts": "快速鍵", - "settings.desktop.section.wsl": "WSL", - "settings.desktop.wsl.title": "WSL 整合", - "settings.desktop.wsl.description": "在 Windows 上的 WSL 中執行 Kilo 伺服器。", - - "settings.general.section.appearance": "外觀", - "settings.general.section.notifications": "系統通知", - "settings.general.section.updates": "更新", - "settings.general.section.sounds": "音效", - "settings.general.section.feed": "資訊流", - "settings.general.section.display": "顯示", - - "settings.general.row.language.title": "語言", - "settings.general.row.language.description": "變更 Kilo 的顯示語言", - "settings.general.row.appearance.title": "外觀", - "settings.general.row.appearance.description": "自訂 Kilo 在你的裝置上的外觀", - "settings.general.row.colorScheme.title": "配色方案", - "settings.general.row.colorScheme.description": "選擇 Kilo 要跟隨系統、淺色或深色主題", - "settings.general.row.theme.title": "主題", - "settings.general.row.theme.description": "自訂 Kilo 的主題。", - "settings.general.row.font.title": "程式碼字型", - "settings.general.row.font.description": "自訂程式碼區塊使用的字型", - "settings.general.row.terminalFont.title": "Terminal Font", - "settings.general.row.terminalFont.description": "Customise the font used in the terminal", - "settings.general.row.uiFont.title": "介面字型", - "settings.general.row.uiFont.description": "自訂整個介面使用的字型", - "settings.general.row.followup.title": "後續追問行為", - "settings.general.row.followup.description": "選擇後續追問提示是立即引導還是進入佇列等待", - "settings.general.row.followup.option.queue": "佇列", - "settings.general.row.followup.option.steer": "引導", - "settings.general.row.reasoningSummaries.title": "顯示推理摘要", - "settings.general.row.reasoningSummaries.description": "在時間軸中顯示模型推理摘要", - - "settings.general.row.shellToolPartsExpanded.title": "展開 shell 工具區塊", - "settings.general.row.shellToolPartsExpanded.description": "在時間軸中預設展開 shell 工具區塊", - "settings.general.row.editToolPartsExpanded.title": "展開 edit 工具區塊", - "settings.general.row.editToolPartsExpanded.description": "在時間軸中預設展開 edit、write 和 patch 工具區塊", - "settings.general.row.showSessionProgressBar.title": "顯示工作階段進度列", - "settings.general.row.showSessionProgressBar.description": "當代理程式正在運作時,在工作階段頂部顯示動畫進度列", - "settings.general.row.wayland.title": "使用原生 Wayland", - "settings.general.row.wayland.description": "在 Wayland 上停用 X11 後備模式。需要重新啟動。", - "settings.general.row.wayland.tooltip": "在混合更新率螢幕的 Linux 系統上,原生 Wayland 可能更穩定。", - - "settings.general.row.releaseNotes.title": "發行說明", - "settings.general.row.releaseNotes.description": "更新後顯示「新功能」彈出視窗", - - "settings.updates.row.startup.title": "啟動時檢查更新", - "settings.updates.row.startup.description": "在 Kilo 啟動時自動檢查更新", - "settings.updates.row.check.title": "檢查更新", - "settings.updates.row.check.description": "手動檢查更新並在有更新時安裝", - "settings.updates.action.checkNow": "立即檢查", - "settings.updates.action.checking": "檢查中...", - "settings.updates.toast.latest.title": "已是最新版本", - "settings.updates.toast.latest.description": "你正在使用最新版本的 Kilo。", - - "sound.option.none": "無", - "sound.option.alert01": "警報 01", - "sound.option.alert02": "警報 02", - "sound.option.alert03": "警報 03", - "sound.option.alert04": "警報 04", - "sound.option.alert05": "警報 05", - "sound.option.alert06": "警報 06", - "sound.option.alert07": "警報 07", - "sound.option.alert08": "警報 08", - "sound.option.alert09": "警報 09", - "sound.option.alert10": "警報 10", - "sound.option.bipbop01": "嗶啵 01", - "sound.option.bipbop02": "嗶啵 02", - "sound.option.bipbop03": "嗶啵 03", - "sound.option.bipbop04": "嗶啵 04", - "sound.option.bipbop05": "嗶啵 05", - "sound.option.bipbop06": "嗶啵 06", - "sound.option.bipbop07": "嗶啵 07", - "sound.option.bipbop08": "嗶啵 08", - "sound.option.bipbop09": "嗶啵 09", - "sound.option.bipbop10": "嗶啵 10", - "sound.option.staplebops01": "斯泰普博普斯 01", - "sound.option.staplebops02": "斯泰普博普斯 02", - "sound.option.staplebops03": "斯泰普博普斯 03", - "sound.option.staplebops04": "斯泰普博普斯 04", - "sound.option.staplebops05": "斯泰普博普斯 05", - "sound.option.staplebops06": "斯泰普博普斯 06", - "sound.option.staplebops07": "斯泰普博普斯 07", - "sound.option.nope01": "否 01", - "sound.option.nope02": "否 02", - "sound.option.nope03": "否 03", - "sound.option.nope04": "否 04", - "sound.option.nope05": "否 05", - "sound.option.nope06": "否 06", - "sound.option.nope07": "否 07", - "sound.option.nope08": "否 08", - "sound.option.nope09": "否 09", - "sound.option.nope10": "否 10", - "sound.option.nope11": "否 11", - "sound.option.nope12": "否 12", - "sound.option.yup01": "是 01", - "sound.option.yup02": "是 02", - "sound.option.yup03": "是 03", - "sound.option.yup04": "是 04", - "sound.option.yup05": "是 05", - "sound.option.yup06": "是 06", - "settings.general.notifications.agent.title": "代理程式", - "settings.general.notifications.agent.description": "當代理程式完成或需要注意時顯示系統通知", - "settings.general.notifications.permissions.title": "權限", - "settings.general.notifications.permissions.description": "當需要權限時顯示系統通知", - "settings.general.notifications.errors.title": "錯誤", - "settings.general.notifications.errors.description": "發生錯誤時顯示系統通知", - - "settings.general.sounds.agent.title": "代理程式", - "settings.general.sounds.agent.description": "當代理程式完成或需要注意時播放聲音", - "settings.general.sounds.permissions.title": "權限", - "settings.general.sounds.permissions.description": "當需要權限時播放聲音", - "settings.general.sounds.errors.title": "錯誤", - "settings.general.sounds.errors.description": "發生錯誤時播放聲音", - - "settings.shortcuts.title": "鍵盤快速鍵", - "settings.shortcuts.reset.button": "重設為預設值", - "settings.shortcuts.reset.toast.title": "快速鍵已重設", - "settings.shortcuts.reset.toast.description": "鍵盤快速鍵已重設為預設設定。", - "settings.shortcuts.conflict.title": "快速鍵已被占用", - "settings.shortcuts.conflict.description": "{{keybind}} 已分配給 {{titles}}。", - "settings.shortcuts.unassigned": "未設定", - "settings.shortcuts.pressKeys": "按下按鍵", - "settings.shortcuts.search.placeholder": "搜尋快速鍵", - "settings.shortcuts.search.empty": "找不到快速鍵", - - "settings.shortcuts.group.general": "一般", - "settings.shortcuts.group.session": "工作階段", - "settings.shortcuts.group.navigation": "導覽", - "settings.shortcuts.group.modelAndAgent": "模型與代理程式", - "settings.shortcuts.group.terminal": "終端機", - "settings.shortcuts.group.prompt": "提示", - - "settings.providers.title": "提供者", - "settings.providers.description": "提供者設定將在此處可設定。", - "settings.providers.section.connected": "已連線的提供商", - "settings.providers.connected.empty": "沒有已連線的提供商", - "settings.providers.section.popular": "熱門提供商", - "settings.providers.tag.environment": "環境", - "settings.providers.tag.config": "配置", - "settings.providers.tag.custom": "自訂", - "settings.providers.tag.other": "其他", - "settings.models.title": "模型", - "settings.models.description": "模型設定將在此處可設定。", - "settings.agents.title": "代理程式", - "settings.agents.description": "代理程式設定將在此處可設定。", - "settings.commands.title": "命令", - "settings.commands.description": "命令設定將在此處可設定。", - "settings.mcp.title": "MCP", - "settings.mcp.description": "MCP 設定將在此處可設定。", - - "settings.permissions.title": "權限", - "settings.permissions.description": "控制伺服器預設可以使用哪些工具。", - "settings.permissions.section.tools": "工具", - "settings.permissions.toast.updateFailed.title": "更新權限失敗", - - "settings.permissions.action.allow": "允許", - "settings.permissions.action.ask": "詢問", - "settings.permissions.action.deny": "拒絕", - - "settings.permissions.tool.read.title": "讀取", - "settings.permissions.tool.read.description": "讀取檔案(符合檔案路徑)", - "settings.permissions.tool.edit.title": "編輯", - "settings.permissions.tool.edit.description": "修改檔案,包括編輯、寫入、修補和多重編輯", - "settings.permissions.tool.glob.title": "Glob", - "settings.permissions.tool.glob.description": "使用 glob 模式符合檔案", - "settings.permissions.tool.grep.title": "Grep", - "settings.permissions.tool.grep.description": "使用正規表示式搜尋檔案內容", - "settings.permissions.tool.list.title": "清單", - "settings.permissions.tool.list.description": "列出目錄中的檔案", - "settings.permissions.tool.bash.title": "Bash", - "settings.permissions.tool.bash.description": "執行 shell 命令", - "settings.permissions.tool.task.title": "Task", - "settings.permissions.tool.task.description": "啟動子代理程式", - "settings.permissions.tool.skill.title": "Skill", - "settings.permissions.tool.skill.description": "按名稱載入技能", - "settings.permissions.tool.lsp.title": "LSP", - "settings.permissions.tool.lsp.description": "執行語言伺服器查詢", - "settings.permissions.tool.todowrite.title": "更新待辦", - "settings.permissions.tool.todowrite.description": "更新待辦清單", - "settings.permissions.tool.webfetch.title": "Web Fetch", - "settings.permissions.tool.webfetch.description": "從 URL 取得內容", - "settings.permissions.tool.websearch.title": "Web Search", - "settings.permissions.tool.websearch.description": "搜尋網頁", - "settings.permissions.tool.codesearch.title": "Code Search", - "settings.permissions.tool.codesearch.description": "在網路上搜尋程式碼", - "settings.permissions.tool.external_directory.title": "外部目錄", - "settings.permissions.tool.external_directory.description": "存取專案目錄之外的檔案", - "settings.permissions.tool.doom_loop.title": "Doom Loop", - "settings.permissions.tool.doom_loop.description": "偵測具有相同輸入的重複工具呼叫", - - "session.delete.failed.title": "刪除工作階段失敗", - "session.delete.title": "刪除工作階段", - "session.delete.confirm": '刪除工作階段 "{{name}}"?', - "session.delete.button": "刪除工作階段", - - "workspace.new": "新增工作區", - "workspace.type.local": "本地", - "workspace.type.sandbox": "沙盒", - "workspace.create.failed.title": "建立工作區失敗", - "workspace.delete.failed.title": "刪除工作區失敗", - "workspace.resetting.title": "正在重設工作區", - "workspace.resetting.description": "這可能需要一點時間。", - "workspace.reset.failed.title": "重設工作區失敗", - "workspace.reset.success.title": "工作區已重設", - "workspace.reset.success.description": "工作區已與預設分支保持一致。", - "workspace.error.stillPreparing": "工作區仍在準備中", - "workspace.status.checking": "正在檢查未合併的變更...", - "workspace.status.error": "無法驗證 git 狀態。", - "workspace.status.clean": "未偵測到未合併的變更。", - "workspace.status.dirty": "偵測到未合併的變更。", - "workspace.delete.title": "刪除工作區", - "workspace.delete.confirm": '刪除工作區 "{{name}}"?', - "workspace.delete.button": "刪除工作區", - "workspace.reset.title": "重設工作區", - "workspace.reset.confirm": '重設工作區 "{{name}}"?', - "workspace.reset.button": "重設工作區", - "workspace.reset.archived.none": "不會封存任何作用中工作階段。", - "workspace.reset.archived.one": "將封存 1 個工作階段。", - "workspace.reset.archived.many": "將封存 {{count}} 個工作階段。", - "workspace.reset.note": "這將把工作區重設為與預設分支一致。", - "common.open": "打開", - "dialog.releaseNotes.action.getStarted": "開始", - "dialog.releaseNotes.action.next": "下一步", - "dialog.releaseNotes.action.hideFuture": "不再顯示", - "dialog.releaseNotes.media.alt": "發佈預覽", - "toast.project.reloadFailed.title": "無法重新載入 {{project}}", - "error.server.invalidConfiguration": "無效的設定", - "common.moreCountSuffix": " (還有 {{count}} 個)", - "common.time.justNow": "剛剛", - "common.time.minutesAgo.short": "{{count}}分鐘前", - "common.time.hoursAgo.short": "{{count}}小時前", - "common.time.daysAgo.short": "{{count}}天前", - "settings.providers.connected.environmentDescription": "已從環境變數連線", - "settings.providers.custom.description": "透過基本 URL 新增與 OpenAI 相容的提供者。", - - "app.server.unreachable": "無法連線至 {{server}}", - "app.server.retrying": "正在自動重試...", - "app.server.otherServers": "其他伺服器", - "dialog.server.add.usernamePlaceholder": "使用者名稱", - "dialog.server.add.passwordPlaceholder": "密碼", - "server.row.noUsername": "無使用者名稱", - "session.review.noVcs.createGit.title": "建立 Git 儲存庫", - "session.review.noVcs.createGit.description": "追蹤、檢閱及復原此專案中的變更", - "session.review.noVcs.createGit.actionLoading": "正在建立 Git 儲存庫...", - "session.review.noVcs.createGit.action": "建立 Git 儲存庫", - "session.todo.progress": "已完成 {{done}} 個待辦事項(共 {{total}} 個)", - "session.question.progress": "{{current}}/{{total}} 個問題", - "session.header.open.finder": "Finder", - "session.header.open.fileExplorer": "檔案總管", - "session.header.open.fileManager": "檔案管理員", - "session.header.open.app.vscode": "VS Code", - "session.header.open.app.cursor": "Cursor", - "session.header.open.app.zed": "Zed", - "session.header.open.app.textmate": "TextMate", - "session.header.open.app.antigravity": "Antigravity", - "session.header.open.app.terminal": "終端機", - "session.header.open.app.iterm2": "iTerm2", - "session.header.open.app.ghostty": "Ghostty", - "session.header.open.app.warp": "Warp", - "session.header.open.app.xcode": "Xcode", - "session.header.open.app.androidStudio": "Android Studio", - "session.header.open.app.powershell": "PowerShell", - "session.header.open.app.sublimeText": "Sublime Text", - "debugBar.ariaLabel": "開發效能診斷", - "debugBar.na": "不適用", - "debugBar.nav.label": "NAV", - "debugBar.nav.tip": "最後一次完成的涉及工作階段頁面的路由轉換,從路由器啟動到穩定後的第一次繪製。", - "debugBar.fps.label": "FPS", - "debugBar.fps.tip": "過去 5 秒內的滾動幀率。", - "debugBar.frame.label": "FRAME", - "debugBar.frame.tip": "過去 5 秒內最差的幀時間。", - "debugBar.jank.label": "JANK", - "debugBar.jank.tip": "過去 5 秒內超過 32ms 的幀。", - "debugBar.long.label": "LONG", - "debugBar.long.tip": "過去 5 秒內的阻塞時間和長任務計數。最大任務:{{max}}。", - "debugBar.delay.label": "DELAY", - "debugBar.delay.tip": "過去 5 秒內觀察到的最差輸入延遲。", - "debugBar.inp.label": "INP", - "debugBar.inp.tip": "過去 5 秒內的近似互動持續時間。這類似於 INP,而非官方的 Web Vitals INP。", - "debugBar.cls.label": "CLS", - "debugBar.cls.tip": "目前應用程式生命週期的累積版面配置位移。", - "debugBar.mem.label": "MEM", - "debugBar.mem.tipUnavailable": "使用的 JS 堆積與堆積限制。僅限 Chromium。", - "debugBar.mem.tip": "使用的 JS 堆積與堆積限制。{{used}} / {{limit}}。", - "common.key.ctrl": "Ctrl", - "common.key.alt": "Alt", - "common.key.shift": "Shift", - "common.key.meta": "Meta", - "common.key.space": "空白鍵", - "common.key.backspace": "退格鍵", - "common.key.enter": "Enter", - "common.key.tab": "Tab", - "common.key.delete": "Delete", - "common.key.home": "Home", - "common.key.end": "End", - "common.key.pageUp": "Page Up", - "common.key.pageDown": "Page Down", - "common.key.insert": "Insert", - "common.unknown": "未知", - "error.page.circular": "[循環]", - "error.globalSDK.noServerAvailable": "無可用的伺服器", - "error.globalSDK.serverNotAvailable": "伺服器無法使用", - "error.childStore.persistedCacheCreateFailed": "建立持續性快取失敗", - "error.childStore.persistedProjectMetadataCreateFailed": "建立持續性專案中繼資料失敗", - "error.childStore.persistedProjectIconCreateFailed": "建立持續性專案圖示失敗", - "error.childStore.storeCreateFailed": "建立儲存區失敗", - "terminal.connectionLost.abnormalClose": "WebSocket 異常關閉:{{code}}", -} satisfies Partial> diff --git a/packages/app/src/index.css b/packages/app/src/index.css deleted file mode 100644 index 8db576dd83..0000000000 --- a/packages/app/src/index.css +++ /dev/null @@ -1,85 +0,0 @@ -@import "@opencode-ai/ui/styles/tailwind"; - -@font-face { - font-family: "JetBrainsMono Nerd Font Mono"; - src: url("/assets/JetBrainsMonoNerdFontMono-Regular.woff2") format("woff2"); - font-weight: normal; - font-style: normal; -} - -@layer components { - @keyframes session-progress-whip { - 0% { - clip-path: inset(0 100% 0 0 round 999px); - animation-timing-function: cubic-bezier(0.2, 0.8, 0.2, 1); - } - - 48% { - clip-path: inset(0 0 0 0 round 999px); - animation-timing-function: cubic-bezier(0.65, 0, 0.35, 1); - } - - 100% { - clip-path: inset(0 0 0 100% round 999px); - } - } - - [data-component="session-progress"] { - position: absolute; - inset: 0 0 auto; - height: 2px; - overflow: hidden; - pointer-events: none; - opacity: 1; - transition: opacity 220ms ease-out; - } - - [data-component="session-progress"][data-state="hiding"] { - opacity: 0; - } - - [data-component="session-progress-bar"] { - width: 100%; - height: 100%; - border-radius: 999px; - background: var(--session-progress-color); - clip-path: inset(0 100% 0 0 round 999px); - animation: session-progress-whip var(--session-progress-ms, 1800ms) infinite; - will-change: clip-path; - } - - [data-component="getting-started"] { - container-type: inline-size; - container-name: getting-started; - } - - [data-component="getting-started-actions"] { - display: flex; - flex-direction: column; - gap: 0.75rem; /* gap-3 */ - } - - [data-component="getting-started-actions"] > [data-component="button"] { - width: 100%; - } - - @container getting-started (min-width: 17rem) { - [data-component="getting-started-actions"] { - flex-direction: row; - align-items: center; - } - - [data-component="getting-started-actions"] > [data-component="button"] { - width: auto; - } - } - - @keyframes fade-in { - from { - opacity: 0; - } - to { - opacity: 1; - } - } -} diff --git a/packages/app/src/index.ts b/packages/app/src/index.ts deleted file mode 100644 index d80e9fffb0..0000000000 --- a/packages/app/src/index.ts +++ /dev/null @@ -1,7 +0,0 @@ -export { AppBaseProviders, AppInterface } from "./app" -export { ACCEPTED_FILE_EXTENSIONS, ACCEPTED_FILE_TYPES, filePickerFilters } from "./constants/file-picker" -export { useCommand } from "./context/command" -export { loadLocaleDict, normalizeLocale, type Locale } from "./context/language" -export { type DisplayBackend, type Platform, PlatformProvider } from "./context/platform" -export { ServerConnection } from "./context/server" -export { handleNotificationClick } from "./utils/notification-click" diff --git a/packages/app/src/pages/error.tsx b/packages/app/src/pages/error.tsx deleted file mode 100644 index bae8f44e54..0000000000 --- a/packages/app/src/pages/error.tsx +++ /dev/null @@ -1,318 +0,0 @@ -import { TextField } from "@opencode-ai/ui/text-field" -import { Logo } from "@opencode-ai/ui/logo" -import { Button } from "@opencode-ai/ui/button" -import { Component, Show } from "solid-js" -import { createStore } from "solid-js/store" -import { usePlatform } from "@/context/platform" -import { useLanguage } from "@/context/language" -import { Icon } from "@opencode-ai/ui/icon" - -export type InitError = { - name: string - data: Record -} - -type Translator = ReturnType["t"] -const CHAIN_SEPARATOR = "\n" + "─".repeat(40) + "\n" - -function isIssue(value: unknown): value is { message: string; path: string[] } { - if (!value || typeof value !== "object") return false - if (!("message" in value) || !("path" in value)) return false - const message = (value as { message: unknown }).message - const path = (value as { path: unknown }).path - if (typeof message !== "string") return false - if (!Array.isArray(path)) return false - return path.every((part) => typeof part === "string") -} - -function isInitError(error: unknown): error is InitError { - return ( - typeof error === "object" && - error !== null && - "name" in error && - "data" in error && - typeof (error as InitError).data === "object" - ) -} - -function safeJson(value: unknown, circular: string): string { - const seen = new WeakSet() - const json = JSON.stringify( - value, - (_key, val) => { - if (typeof val === "bigint") return val.toString() - if (typeof val === "object" && val) { - if (seen.has(val)) return circular - seen.add(val) - } - return val - }, - 2, - ) - return json ?? String(value) -} - -function formatInitError(error: InitError, t: Translator): string { - const data = error.data - const json = (value: unknown) => safeJson(value, t("error.page.circular")) - switch (error.name) { - case "MCPFailed": { - const name = typeof data.name === "string" ? data.name : "" - return t("error.chain.mcpFailed", { name }) - } - case "ProviderAuthError": { - const providerID = typeof data.providerID === "string" ? data.providerID : t("common.unknown") - const message = typeof data.message === "string" ? data.message : json(data.message) - return t("error.chain.providerAuthFailed", { provider: providerID, message }) - } - case "APIError": { - const message = typeof data.message === "string" ? data.message : t("error.chain.apiError") - const lines: string[] = [message] - - if (typeof data.statusCode === "number") { - lines.push(t("error.chain.status", { status: data.statusCode })) - } - - if (typeof data.isRetryable === "boolean") { - lines.push(t("error.chain.retryable", { retryable: data.isRetryable })) - } - - if (typeof data.responseBody === "string" && data.responseBody) { - lines.push(t("error.chain.responseBody", { body: data.responseBody })) - } - - return lines.join("\n") - } - case "ProviderModelNotFoundError": { - const { providerID, modelID, suggestions } = data as { - providerID: string - modelID: string - suggestions?: string[] - } - - const suggestionsLine = - Array.isArray(suggestions) && suggestions.length - ? [t("error.chain.didYouMean", { suggestions: suggestions.join(", ") })] - : [] - - return [ - t("error.chain.modelNotFound", { provider: providerID, model: modelID }), - ...suggestionsLine, - t("error.chain.checkConfig"), - ].join("\n") - } - case "ProviderInitError": { - const providerID = typeof data.providerID === "string" ? data.providerID : t("common.unknown") - return t("error.chain.providerInitFailed", { provider: providerID }) - } - case "ConfigJsonError": { - const path = typeof data.path === "string" ? data.path : json(data.path) - const message = typeof data.message === "string" ? data.message : "" - if (message) return t("error.chain.configJsonInvalidWithMessage", { path, message }) - return t("error.chain.configJsonInvalid", { path }) - } - case "ConfigDirectoryTypoError": { - const path = typeof data.path === "string" ? data.path : json(data.path) - const dir = typeof data.dir === "string" ? data.dir : json(data.dir) - const suggestion = typeof data.suggestion === "string" ? data.suggestion : json(data.suggestion) - return t("error.chain.configDirectoryTypo", { dir, path, suggestion }) - } - case "ConfigFrontmatterError": { - const path = typeof data.path === "string" ? data.path : json(data.path) - const message = typeof data.message === "string" ? data.message : json(data.message) - return t("error.chain.configFrontmatterError", { path, message }) - } - case "ConfigInvalidError": { - const issues = Array.isArray(data.issues) - ? data.issues.filter(isIssue).map((issue) => "↳ " + issue.message + " " + issue.path.join(".")) - : [] - const message = typeof data.message === "string" ? data.message : "" - const path = typeof data.path === "string" ? data.path : json(data.path) - - const line = message - ? t("error.chain.configInvalidWithMessage", { path, message }) - : t("error.chain.configInvalid", { path }) - - return [line, ...issues].join("\n") - } - case "UnknownError": - return typeof data.message === "string" ? data.message : json(data) - default: - if (typeof data.message === "string") return data.message - return json(data) - } -} - -function formatErrorChain(error: unknown, t: Translator, depth = 0, parentMessage?: string): string { - const json = (value: unknown) => safeJson(value, t("error.page.circular")) - if (!error) return t("error.chain.unknown") - - if (isInitError(error)) { - const message = formatInitError(error, t) - if (depth > 0 && parentMessage === message) return "" - const indent = depth > 0 ? `\n${CHAIN_SEPARATOR}${t("error.chain.causedBy")}\n` : "" - return indent + `${error.name}\n${message}` - } - - if (error instanceof Error) { - const isDuplicate = depth > 0 && parentMessage === error.message - const parts: string[] = [] - const indent = depth > 0 ? `\n${CHAIN_SEPARATOR}${t("error.chain.causedBy")}\n` : "" - - const header = `${error.name}${error.message ? `: ${error.message}` : ""}` - const stack = error.stack?.trim() - - if (stack) { - const startsWithHeader = stack.startsWith(header) - - if (isDuplicate && startsWithHeader) { - const trace = stack.split("\n").slice(1).join("\n").trim() - if (trace) { - parts.push(indent + trace) - } - } - - if (isDuplicate && !startsWithHeader) { - parts.push(indent + stack) - } - - if (!isDuplicate && startsWithHeader) { - parts.push(indent + stack) - } - - if (!isDuplicate && !startsWithHeader) { - parts.push(indent + `${header}\n${stack}`) - } - } - - if (!stack && !isDuplicate) { - parts.push(indent + header) - } - - if (error.cause) { - const causeResult = formatErrorChain(error.cause, t, depth + 1, error.message) - if (causeResult) { - parts.push(causeResult) - } - } - - return parts.join("\n\n") - } - - if (typeof error === "string") { - if (depth > 0 && parentMessage === error) return "" - const indent = depth > 0 ? `\n${CHAIN_SEPARATOR}${t("error.chain.causedBy")}\n` : "" - return indent + error - } - - const indent = depth > 0 ? `\n${CHAIN_SEPARATOR}${t("error.chain.causedBy")}\n` : "" - return indent + json(error) -} - -function formatError(error: unknown, t: Translator): string { - return formatErrorChain(error, t, 0) -} - -interface ErrorPageProps { - error: unknown -} - -export const ErrorPage: Component = (props) => { - const platform = usePlatform() - const language = useLanguage() - const [store, setStore] = createStore({ - checking: false, - version: undefined as string | undefined, - actionError: undefined as string | undefined, - }) - - async function checkForUpdates() { - if (!platform.checkUpdate) return - setStore("checking", true) - await platform - .checkUpdate() - .then((result) => { - setStore("actionError", undefined) - if (result.updateAvailable && result.version) setStore("version", result.version) - }) - .catch((err) => { - setStore("actionError", formatError(err, language.t)) - }) - .finally(() => { - setStore("checking", false) - }) - } - - async function installUpdate() { - if (!platform.updateAndRestart) return - await platform - .updateAndRestart() - .then(() => setStore("actionError", undefined)) - .catch((err) => { - setStore("actionError", formatError(err, language.t)) - }) - } - - return ( -
-
- -
-

{language.t("error.page.title")}

-

{language.t("error.page.description")}

-
- -
- - - - {store.checking - ? language.t("error.page.action.checking") - : language.t("error.page.action.checkUpdates")} - - } - > - - - -
- - {(message) =>

{message()}

} -
-
-
- {language.t("error.page.report.prefix")} - -
- - {(version) => ( -

{language.t("error.page.version", { version: version() })}

- )} -
-
-
-
- ) -} diff --git a/packages/app/src/pages/layout/deep-links.ts b/packages/app/src/pages/layout/deep-links.ts deleted file mode 100644 index 67137f17fa..0000000000 --- a/packages/app/src/pages/layout/deep-links.ts +++ /dev/null @@ -1,50 +0,0 @@ -export const deepLinkEvent = "opencode:deep-link" - -const parseUrl = (input: string) => { - if (!input.startsWith("opencode://")) return - if (typeof URL.canParse === "function" && !URL.canParse(input)) return - try { - return new URL(input) - } catch { - return - } -} - -export const parseDeepLink = (input: string) => { - const url = parseUrl(input) - if (!url) return - if (url.hostname !== "open-project") return - const directory = url.searchParams.get("directory") - if (!directory) return - return directory -} - -export const parseNewSessionDeepLink = (input: string) => { - const url = parseUrl(input) - if (!url) return - if (url.hostname !== "new-session") return - const directory = url.searchParams.get("directory") - if (!directory) return - const prompt = url.searchParams.get("prompt") || undefined - if (!prompt) return { directory } - return { directory, prompt } -} - -export const collectOpenProjectDeepLinks = (urls: string[]) => - urls.map(parseDeepLink).filter((directory): directory is string => !!directory) - -export const collectNewSessionDeepLinks = (urls: string[]) => - urls.map(parseNewSessionDeepLink).filter((link): link is { directory: string; prompt?: string } => !!link) - -type OpenCodeWindow = Window & { - __KILO__?: { - deepLinks?: string[] - } -} - -export const drainPendingDeepLinks = (target: OpenCodeWindow) => { - const pending = target.__KILO__?.deepLinks ?? [] - if (pending.length === 0) return [] - if (target.__KILO__) target.__KILO__.deepLinks = [] - return pending -} diff --git a/packages/app/src/pages/layout/helpers.test.ts b/packages/app/src/pages/layout/helpers.test.ts deleted file mode 100644 index e33afdfc41..0000000000 --- a/packages/app/src/pages/layout/helpers.test.ts +++ /dev/null @@ -1,225 +0,0 @@ -import { describe, expect, test } from "bun:test" -import { - collectNewSessionDeepLinks, - collectOpenProjectDeepLinks, - drainPendingDeepLinks, - parseDeepLink, - parseNewSessionDeepLink, -} from "./deep-links" -import { type Session } from "@kilocode/sdk/v2/client" -import { - childSessionOnPath, - displayName, - effectiveWorkspaceOrder, - errorMessage, - hasProjectPermissions, - latestRootSession, - workspaceKey, -} from "./helpers" - -const session = (input: Partial & Pick) => - ({ - title: "", - version: "v2", - parentID: undefined, - messageCount: 0, - permissions: { session: {}, share: {} }, - time: { created: 0, updated: 0, archived: undefined }, - ...input, - }) as Session - -describe("layout deep links", () => { - test("parses open-project deep links", () => { - expect(parseDeepLink("opencode://open-project?directory=/tmp/demo")).toBe("/tmp/demo") - }) - - test("ignores non-project deep links", () => { - expect(parseDeepLink("opencode://other?directory=/tmp/demo")).toBeUndefined() - expect(parseDeepLink("https://example.com")).toBeUndefined() - }) - - test("ignores malformed deep links safely", () => { - expect(() => parseDeepLink("opencode://open-project/%E0%A4%A%")).not.toThrow() - expect(parseDeepLink("opencode://open-project/%E0%A4%A%")).toBeUndefined() - }) - - test("parses links when URL.canParse is unavailable", () => { - const original = Object.getOwnPropertyDescriptor(URL, "canParse") - Object.defineProperty(URL, "canParse", { configurable: true, value: undefined }) - try { - expect(parseDeepLink("opencode://open-project?directory=/tmp/demo")).toBe("/tmp/demo") - } finally { - if (original) Object.defineProperty(URL, "canParse", original) - if (!original) Reflect.deleteProperty(URL, "canParse") - } - }) - - test("ignores open-project deep links without directory", () => { - expect(parseDeepLink("opencode://open-project")).toBeUndefined() - expect(parseDeepLink("opencode://open-project?directory=")).toBeUndefined() - }) - - test("collects only valid open-project directories", () => { - const result = collectOpenProjectDeepLinks([ - "opencode://open-project?directory=/a", - "opencode://other?directory=/b", - "opencode://open-project?directory=/c", - ]) - expect(result).toEqual(["/a", "/c"]) - }) - - test("parses new-session deep links with optional prompt", () => { - expect(parseNewSessionDeepLink("opencode://new-session?directory=/tmp/demo")).toEqual({ directory: "/tmp/demo" }) - expect(parseNewSessionDeepLink("opencode://new-session?directory=/tmp/demo&prompt=hello%20world")).toEqual({ - directory: "/tmp/demo", - prompt: "hello world", - }) - }) - - test("ignores new-session deep links without directory", () => { - expect(parseNewSessionDeepLink("opencode://new-session")).toBeUndefined() - expect(parseNewSessionDeepLink("opencode://new-session?directory=")).toBeUndefined() - }) - - test("collects only valid new-session deep links", () => { - const result = collectNewSessionDeepLinks([ - "opencode://new-session?directory=/a", - "opencode://open-project?directory=/b", - "opencode://new-session?directory=/c&prompt=ship%20it", - ]) - expect(result).toEqual([{ directory: "/a" }, { directory: "/c", prompt: "ship it" }]) - }) - - test("drains global deep links once", () => { - const target = { - __KILO__: { - deepLinks: ["opencode://open-project?directory=/a"], - }, - } as unknown as Window & { __KILO__?: { deepLinks?: string[] } } - - expect(drainPendingDeepLinks(target)).toEqual(["opencode://open-project?directory=/a"]) - expect(drainPendingDeepLinks(target)).toEqual([]) - }) -}) - -describe("layout workspace helpers", () => { - test("normalizes trailing slash in workspace key", () => { - expect(workspaceKey("/tmp/demo///")).toBe("/tmp/demo") - expect(workspaceKey("C:\\tmp\\demo\\\\")).toBe("C:/tmp/demo") - }) - - test("preserves posix and drive roots in workspace key", () => { - expect(workspaceKey("/")).toBe("/") - expect(workspaceKey("///")).toBe("/") - expect(workspaceKey("C:\\")).toBe("C:/") - expect(workspaceKey("C://")).toBe("C:/") - expect(workspaceKey("C:///")).toBe("C:/") - }) - - test("keeps local first while preserving known order", () => { - const result = effectiveWorkspaceOrder("/root", ["/root", "/b", "/c"], ["/root", "/c", "/a", "/b"]) - expect(result).toEqual(["/root", "/c", "/b"]) - }) - - test("finds the latest root session across workspaces", () => { - const result = latestRootSession( - [ - { - path: { directory: "/root" }, - session: [session({ id: "root", directory: "/root", time: { created: 1, updated: 1, archived: undefined } })], - }, - { - path: { directory: "/workspace" }, - session: [ - session({ - id: "workspace", - directory: "/workspace", - time: { created: 2, updated: 2, archived: undefined }, - }), - ], - }, - ], - 120_000, - ) - - expect(result?.id).toBe("workspace") - }) - - test("detects project permissions with a filter", () => { - const result = hasProjectPermissions( - { - root: [{ id: "perm-root" }, { id: "perm-hidden" }], - child: [{ id: "perm-child" }], - }, - (item) => item.id === "perm-child", - ) - - expect(result).toBe(true) - }) - - test("ignores project permissions filtered out", () => { - const result = hasProjectPermissions( - { - root: [{ id: "perm-root" }], - }, - () => false, - ) - - expect(result).toBe(false) - }) - - test("ignores archived and child sessions when finding latest root session", () => { - const result = latestRootSession( - [ - { - path: { directory: "/workspace" }, - session: [ - session({ - id: "archived", - directory: "/workspace", - time: { created: 10, updated: 10, archived: 10 }, - }), - session({ - id: "child", - directory: "/workspace", - parentID: "parent", - time: { created: 20, updated: 20, archived: undefined }, - }), - session({ - id: "root", - directory: "/workspace", - time: { created: 30, updated: 30, archived: undefined }, - }), - ], - }, - ], - 120_000, - ) - - expect(result?.id).toBe("root") - }) - - test("finds the direct child on the active session path", () => { - const list = [ - session({ id: "root", directory: "/workspace" }), - session({ id: "child", directory: "/workspace", parentID: "root" }), - session({ id: "leaf", directory: "/workspace", parentID: "child" }), - ] - - expect(childSessionOnPath(list, "root", "leaf")?.id).toBe("child") - expect(childSessionOnPath(list, "child", "leaf")?.id).toBe("leaf") - expect(childSessionOnPath(list, "root", "root")).toBeUndefined() - expect(childSessionOnPath(list, "root", "other")).toBeUndefined() - }) - - test("formats fallback project display name", () => { - expect(displayName({ worktree: "/tmp/app" })).toBe("app") - expect(displayName({ worktree: "/tmp/app", name: "My App" })).toBe("My App") - }) - - test("extracts api error message and fallback", () => { - expect(errorMessage({ data: { message: "boom" } }, "fallback")).toBe("boom") - expect(errorMessage(new Error("broken"), "fallback")).toBe("broken") - expect(errorMessage("unknown", "fallback")).toBe("fallback") - }) -}) diff --git a/packages/app/src/pages/layout/inline-editor.tsx b/packages/app/src/pages/layout/inline-editor.tsx deleted file mode 100644 index 4189e4a72a..0000000000 --- a/packages/app/src/pages/layout/inline-editor.tsx +++ /dev/null @@ -1,126 +0,0 @@ -import { createStore } from "solid-js/store" -import { onCleanup, Show, type Accessor } from "solid-js" -import { InlineInput } from "@opencode-ai/ui/inline-input" - -export function createInlineEditorController() { - // This controller intentionally supports one active inline editor at a time. - const [editor, setEditor] = createStore({ - active: "" as string, - value: "", - }) - - const editorOpen = (id: string) => editor.active === id - const editorValue = () => editor.value - const openEditor = (id: string, value: string) => { - if (!id) return - setEditor({ active: id, value }) - } - const closeEditor = () => setEditor({ active: "", value: "" }) - - const saveEditor = (callback: (next: string) => void) => { - const next = editor.value.trim() - if (!next) { - closeEditor() - return - } - closeEditor() - callback(next) - } - - const editorKeyDown = (event: KeyboardEvent, callback: (next: string) => void) => { - if (event.key === "Enter") { - event.preventDefault() - saveEditor(callback) - return - } - if (event.key !== "Escape") return - event.preventDefault() - closeEditor() - } - - const InlineEditor = (props: { - id: string - value: Accessor - onSave: (next: string) => void - class?: string - displayClass?: string - editing?: boolean - stopPropagation?: boolean - openOnDblClick?: boolean - }) => { - let frame: number | undefined - - onCleanup(() => { - if (frame === undefined) return - cancelAnimationFrame(frame) - }) - - const isEditing = () => props.editing ?? editorOpen(props.id) - const stopEvents = () => props.stopPropagation ?? false - const allowDblClick = () => props.openOnDblClick ?? true - const stopPropagation = (event: Event) => { - if (!stopEvents()) return - event.stopPropagation() - } - const handleDblClick = (event: MouseEvent) => { - if (!allowDblClick()) return - stopPropagation(event) - openEditor(props.id, props.value()) - } - - return ( - - {props.value()} - - } - > - { - if (frame !== undefined) cancelAnimationFrame(frame) - frame = requestAnimationFrame(() => { - frame = undefined - if (!el.isConnected) return - el.focus() - }) - }} - value={editorValue()} - class={props.class} - onInput={(event) => setEditor("value", event.currentTarget.value)} - onKeyDown={(event) => { - event.stopPropagation() - editorKeyDown(event, props.onSave) - }} - onBlur={closeEditor} - onPointerDown={stopPropagation} - onClick={stopPropagation} - onDblClick={stopPropagation} - onMouseDown={stopPropagation} - onMouseUp={stopPropagation} - onTouchStart={stopPropagation} - /> - - ) - } - - return { - editor, - editorOpen, - editorValue, - openEditor, - closeEditor, - saveEditor, - editorKeyDown, - setEditor, - InlineEditor, - } -} diff --git a/packages/app/src/pages/layout/sidebar-shell.tsx b/packages/app/src/pages/layout/sidebar-shell.tsx deleted file mode 100644 index ca36af2a42..0000000000 --- a/packages/app/src/pages/layout/sidebar-shell.tsx +++ /dev/null @@ -1,125 +0,0 @@ -import { createEffect, createMemo, For, Show, type Accessor, type JSX } from "solid-js" -import { - DragDropProvider, - DragDropSensors, - DragOverlay, - SortableProvider, - closestCenter, - type DragEvent, -} from "@thisbeyond/solid-dnd" -import { ConstrainDragXAxis } from "@/utils/solid-dnd" -import { IconButton } from "@opencode-ai/ui/icon-button" -import { Tooltip, TooltipKeybind } from "@opencode-ai/ui/tooltip" -import { type LocalProject } from "@/context/layout" - -export const SidebarContent = (props: { - mobile?: boolean - opened: Accessor - aimMove: (event: MouseEvent) => void - projects: Accessor - renderProject: (project: LocalProject) => JSX.Element - handleDragStart: (event: unknown) => void - handleDragEnd: () => void - handleDragOver: (event: DragEvent) => void - openProjectLabel: JSX.Element - openProjectKeybind: Accessor - onOpenProject: () => void - renderProjectOverlay: () => JSX.Element - settingsLabel: Accessor - settingsKeybind: Accessor - onOpenSettings: () => void - helpLabel: Accessor - onOpenHelp: () => void - renderPanel: () => JSX.Element -}): JSX.Element => { - const expanded = createMemo(() => !!props.mobile || props.opened()) - const placement = () => (props.mobile ? "bottom" : "right") - let panel: HTMLDivElement | undefined - - createEffect(() => { - const el = panel - if (!el) return - if (expanded()) { - el.removeAttribute("inert") - return - } - el.setAttribute("inert", "") - }) - - return ( -
-
-
- - - -
- p.worktree)}> - {(project) => props.renderProject(project)} - - - {props.openProjectLabel} - - {props.openProjectKeybind()} - -
- } - > - - -
- {props.renderProjectOverlay()} - -
-
- - - - - - -
-
- -
{ - panel = el - }} - classList={{ "flex-1 flex h-full min-h-0 min-w-0 overflow-hidden": true, "pointer-events-none": !expanded() }} - aria-hidden={!expanded()} - > - {props.renderPanel()} -
- - ) -} diff --git a/packages/app/src/pages/session/composer/index.ts b/packages/app/src/pages/session/composer/index.ts deleted file mode 100644 index b0069de53f..0000000000 --- a/packages/app/src/pages/session/composer/index.ts +++ /dev/null @@ -1,2 +0,0 @@ -export { SessionComposerRegion } from "./session-composer-region" -export { createSessionComposerState } from "./session-composer-state" diff --git a/packages/app/src/pages/session/composer/session-composer-region.tsx b/packages/app/src/pages/session/composer/session-composer-region.tsx deleted file mode 100644 index 60447566ed..0000000000 --- a/packages/app/src/pages/session/composer/session-composer-region.tsx +++ /dev/null @@ -1,289 +0,0 @@ -import { Show, createEffect, createMemo, onCleanup } from "solid-js" -import { createStore } from "solid-js/store" -import { useNavigate } from "@solidjs/router" -import { useSpring } from "@opencode-ai/ui/motion-spring" -import { PromptInput } from "@/components/prompt-input" -import { useLanguage } from "@/context/language" -import { usePrompt } from "@/context/prompt" -import { useSync } from "@/context/sync" -import { getSessionHandoff, setSessionHandoff } from "@/pages/session/handoff" -import { useSessionKey } from "@/pages/session/session-layout" -import { SessionPermissionDock } from "@/pages/session/composer/session-permission-dock" -import { SessionQuestionDock } from "@/pages/session/composer/session-question-dock" -import { SessionFollowupDock } from "@/pages/session/composer/session-followup-dock" -import { SessionRevertDock } from "@/pages/session/composer/session-revert-dock" -import type { SessionComposerState } from "@/pages/session/composer/session-composer-state" -import { SessionTodoDock } from "@/pages/session/composer/session-todo-dock" -import type { FollowupDraft } from "@/components/prompt-input/submit" -import { createResizeObserver } from "@solid-primitives/resize-observer" - -export function SessionComposerRegion(props: { - state: SessionComposerState - ready: boolean - centered: boolean - inputRef: (el: HTMLDivElement) => void - newSessionWorktree: string - onNewSessionWorktreeReset: () => void - onSubmit: () => void - onResponseSubmit: () => void - followup?: { - queue: () => boolean - items: { id: string; text: string }[] - sending?: string - edit?: { id: string; prompt: FollowupDraft["prompt"]; context: FollowupDraft["context"] } - onQueue: (draft: FollowupDraft) => void - onAbort: () => void - onSend: (id: string) => void - onEdit: (id: string) => void - onEditLoaded: () => void - } - revert?: { - items: { id: string; text: string }[] - restoring?: string - disabled?: boolean - onRestore: (id: string) => void - } - setPromptDockRef: (el: HTMLDivElement) => void -}) { - const navigate = useNavigate() - const prompt = usePrompt() - const language = useLanguage() - const route = useSessionKey() - const sync = useSync() - - const handoffPrompt = createMemo(() => getSessionHandoff(route.sessionKey())?.prompt) - const info = createMemo(() => (route.params.id ? sync.session.get(route.params.id) : undefined)) - const parentID = createMemo(() => info()?.parentID) - const child = createMemo(() => !!parentID()) - const showComposer = createMemo(() => !props.state.blocked() || child()) - - const previewPrompt = () => - prompt - .current() - .map((part) => { - if (part.type === "file") return `[file:${part.path}]` - if (part.type === "agent") return `@${part.name}` - if (part.type === "image") return `[image:${part.filename}]` - return part.content - }) - .join("") - .trim() - - createEffect(() => { - if (!prompt.ready()) return - setSessionHandoff(route.sessionKey(), { prompt: previewPrompt() }) - }) - - const [store, setStore] = createStore({ - ready: false, - height: 320, - body: undefined as HTMLDivElement | undefined, - }) - let timer: number | undefined - let frame: number | undefined - - const clear = () => { - if (timer !== undefined) { - window.clearTimeout(timer) - timer = undefined - } - if (frame !== undefined) { - cancelAnimationFrame(frame) - frame = undefined - } - } - - createEffect(() => { - route.sessionKey() - const ready = props.ready - const delay = 140 - - clear() - setStore("ready", false) - if (!ready) return - - frame = requestAnimationFrame(() => { - frame = undefined - timer = window.setTimeout(() => { - setStore("ready", true) - timer = undefined - }, delay) - }) - }) - - onCleanup(clear) - - const open = createMemo(() => store.ready && props.state.dock() && !props.state.closing()) - const progress = useSpring(() => (open() ? 1 : 0), { visualDuration: 0.3, bounce: 0 }) - const value = createMemo(() => Math.max(0, Math.min(1, progress()))) - const dock = createMemo(() => (store.ready && props.state.dock()) || value() > 0.001) - const rolled = createMemo(() => (props.revert?.items.length ? props.revert : undefined)) - const lift = createMemo(() => (rolled() ? 18 : 36 * value())) - const full = createMemo(() => Math.max(78, store.height)) - - const openParent = () => { - const id = parentID() - if (!id) return - navigate(`/${route.params.dir}/session/${id}`) - } - - createEffect(() => { - const el = store.body - if (!el) return - const update = () => setStore("height", el.getBoundingClientRect().height) - createResizeObserver(store.body, update) - update() - }) - - return ( -
-
- - {(request) => ( -
- -
- )} -
- - - {(request) => ( -
- { - props.onResponseSubmit() - props.state.decide(response) - }} - /> -
- )} -
- - - - - {(revert) => ( -
- -
- )} -
-
- {handoffPrompt() || language.t("prompt.loading")} -
- - } - > - -
-
setStore("body", el)}> - -
-
-
- - {(revert) => ( -
- -
- )} -
-
- - - - - - - } - > -
- {language.t("session.child.promptDisabled")} - - - -
- -
-
-
-
-
- ) -} diff --git a/packages/app/src/pages/session/composer/session-composer-state.test.ts b/packages/app/src/pages/session/composer/session-composer-state.test.ts deleted file mode 100644 index 9a1da896ce..0000000000 --- a/packages/app/src/pages/session/composer/session-composer-state.test.ts +++ /dev/null @@ -1,128 +0,0 @@ -import { describe, expect, test } from "bun:test" -import type { PermissionRequest, QuestionRequest, Session } from "@kilocode/sdk/v2/client" -import { todoState } from "./session-composer-state" -import { sessionPermissionRequest, sessionQuestionRequest } from "./session-request-tree" - -const session = (input: { id: string; parentID?: string }) => - ({ - id: input.id, - parentID: input.parentID, - }) as Session - -const permission = (id: string, sessionID: string) => - ({ - id, - sessionID, - }) as PermissionRequest - -const question = (id: string, sessionID: string) => - ({ - id, - sessionID, - questions: [], - }) as QuestionRequest - -describe("sessionPermissionRequest", () => { - test("prefers the current session permission", () => { - const sessions = [session({ id: "root" }), session({ id: "child", parentID: "root" })] - const permissions = { - root: [permission("perm-root", "root")], - child: [permission("perm-child", "child")], - } - - expect(sessionPermissionRequest(sessions, permissions, "root")?.id).toBe("perm-root") - }) - - test("returns a nested child permission", () => { - const sessions = [ - session({ id: "root" }), - session({ id: "child", parentID: "root" }), - session({ id: "grand", parentID: "child" }), - session({ id: "other" }), - ] - const permissions = { - grand: [permission("perm-grand", "grand")], - other: [permission("perm-other", "other")], - } - - expect(sessionPermissionRequest(sessions, permissions, "root")?.id).toBe("perm-grand") - }) - - test("returns undefined without a matching tree permission", () => { - const sessions = [session({ id: "root" }), session({ id: "child", parentID: "root" })] - const permissions = { - other: [permission("perm-other", "other")], - } - - expect(sessionPermissionRequest(sessions, permissions, "root")).toBeUndefined() - }) - - test("skips filtered permissions in the current tree", () => { - const sessions = [session({ id: "root" }), session({ id: "child", parentID: "root" })] - const permissions = { - root: [permission("perm-root", "root")], - child: [permission("perm-child", "child")], - } - - expect(sessionPermissionRequest(sessions, permissions, "root", (item) => item.id !== "perm-root"))?.toMatchObject({ - id: "perm-child", - }) - }) - - test("returns undefined when all tree permissions are filtered out", () => { - const sessions = [session({ id: "root" }), session({ id: "child", parentID: "root" })] - const permissions = { - root: [permission("perm-root", "root")], - child: [permission("perm-child", "child")], - } - - expect(sessionPermissionRequest(sessions, permissions, "root", () => false)).toBeUndefined() - }) -}) - -describe("sessionQuestionRequest", () => { - test("prefers the current session question", () => { - const sessions = [session({ id: "root" }), session({ id: "child", parentID: "root" })] - const questions = { - root: [question("q-root", "root")], - child: [question("q-child", "child")], - } - - expect(sessionQuestionRequest(sessions, questions, "root")?.id).toBe("q-root") - }) - - test("returns a nested child question", () => { - const sessions = [ - session({ id: "root" }), - session({ id: "child", parentID: "root" }), - session({ id: "grand", parentID: "child" }), - ] - const questions = { - grand: [question("q-grand", "grand")], - } - - expect(sessionQuestionRequest(sessions, questions, "root")?.id).toBe("q-grand") - }) -}) - -describe("todoState", () => { - test("hides when there are no todos", () => { - expect(todoState({ count: 0, done: false, live: true })).toBe("hide") - }) - - test("opens while the session is still working", () => { - expect(todoState({ count: 2, done: false, live: true })).toBe("open") - }) - - test("closes completed todos after a running turn", () => { - expect(todoState({ count: 2, done: true, live: true })).toBe("close") - }) - - test("clears stale todos when the turn ends", () => { - expect(todoState({ count: 2, done: false, live: false })).toBe("clear") - }) - - test("clears completed todos when the session is no longer live", () => { - expect(todoState({ count: 2, done: true, live: false })).toBe("clear") - }) -}) diff --git a/packages/app/src/pages/session/composer/session-composer-state.ts b/packages/app/src/pages/session/composer/session-composer-state.ts deleted file mode 100644 index 698d0f8049..0000000000 --- a/packages/app/src/pages/session/composer/session-composer-state.ts +++ /dev/null @@ -1,199 +0,0 @@ -import { createEffect, createMemo, on, onCleanup } from "solid-js" -import { createStore } from "solid-js/store" -import { makeEventListener } from "@solid-primitives/event-listener" -import type { PermissionRequest, QuestionRequest, Todo } from "@kilocode/sdk/v2" -import { useParams } from "@solidjs/router" -import { showToast } from "@opencode-ai/ui/toast" -import { useGlobalSync } from "@/context/global-sync" -import { useLanguage } from "@/context/language" -import { usePermission } from "@/context/permission" -import { useSDK } from "@/context/sdk" -import { useSync } from "@/context/sync" -import { sessionPermissionRequest, sessionQuestionRequest } from "./session-request-tree" - -export const todoState = (input: { - count: number - done: boolean - live: boolean -}): "hide" | "clear" | "open" | "close" => { - if (input.count === 0) return "hide" - if (!input.live) return "clear" - if (!input.done) return "open" - return "close" -} - -const idle = { type: "idle" as const } - -export function createSessionComposerState(options?: { closeMs?: number | (() => number) }) { - const params = useParams() - const sdk = useSDK() - const sync = useSync() - const globalSync = useGlobalSync() - const language = useLanguage() - const permission = usePermission() - - const questionRequest = createMemo((): QuestionRequest | undefined => { - return sessionQuestionRequest(sync.data.session, sync.data.question, params.id) - }) - - const permissionRequest = createMemo((): PermissionRequest | undefined => { - return sessionPermissionRequest(sync.data.session, sync.data.permission, params.id, (item) => { - return !permission.autoResponds(item, sdk.directory) - }) - }) - - const blocked = createMemo(() => { - const id = params.id - if (!id) return false - return !!permissionRequest() || !!questionRequest() - }) - - const todos = createMemo((): Todo[] => { - const id = params.id - if (!id) return [] - return globalSync.data.session_todo[id] ?? [] - }) - - const done = createMemo( - () => todos().length > 0 && todos().every((todo) => todo.status === "completed" || todo.status === "cancelled"), - ) - - const status = createMemo(() => { - const id = params.id - if (!id) return idle - return sync.data.session_status[id] ?? idle - }) - - const busy = createMemo(() => status().type !== "idle") - const live = createMemo(() => busy() || blocked()) - - const [store, setStore] = createStore({ - responding: undefined as string | undefined, - dock: todos().length > 0 && live(), - closing: false, - opening: false, - }) - - const permissionResponding = createMemo(() => { - const perm = permissionRequest() - if (!perm) return false - return store.responding === perm.id - }) - - const decide = (response: "once" | "always" | "reject") => { - const perm = permissionRequest() - if (!perm) return - if (store.responding === perm.id) return - - setStore("responding", perm.id) - sdk.client.permission - .respond({ sessionID: perm.sessionID, permissionID: perm.id, response }) - .catch((err: unknown) => { - const description = err instanceof Error ? err.message : String(err) - showToast({ title: language.t("common.requestFailed"), description }) - }) - .finally(() => { - setStore("responding", (id) => (id === perm.id ? undefined : id)) - }) - } - - let timer: number | undefined - let raf: number | undefined - - const closeMs = () => { - const value = options?.closeMs - if (typeof value === "function") return Math.max(0, value()) - if (typeof value === "number") return Math.max(0, value) - return 400 - } - - const scheduleClose = () => { - if (timer) window.clearTimeout(timer) - timer = window.setTimeout(() => { - setStore({ dock: false, closing: false }) - timer = undefined - }, closeMs()) - } - - // Keep stale turn todos from reopening if the model never clears them. - const clear = () => { - const id = params.id - if (!id) return - globalSync.todo.set(id, []) - sync.set("todo", id, []) - } - - createEffect( - on( - () => [todos().length, done(), live()] as const, - ([count, complete, active]) => { - if (raf) cancelAnimationFrame(raf) - raf = undefined - - const next = todoState({ - count, - done: complete, - live: active, - }) - - if (next === "hide") { - if (timer) window.clearTimeout(timer) - timer = undefined - setStore({ dock: false, closing: false, opening: false }) - return - } - - if (next === "clear") { - if (timer) window.clearTimeout(timer) - timer = undefined - clear() - return - } - - if (next === "open") { - if (timer) window.clearTimeout(timer) - timer = undefined - const hidden = !store.dock || store.closing - setStore({ dock: true, closing: false }) - if (hidden) { - setStore("opening", true) - raf = requestAnimationFrame(() => { - setStore("opening", false) - raf = undefined - }) - return - } - setStore("opening", false) - return - } - - setStore({ dock: true, opening: false, closing: true }) - if (!timer) scheduleClose() - }, - ), - ) - - onCleanup(() => { - if (!timer) return - window.clearTimeout(timer) - }) - - onCleanup(() => { - if (!raf) return - cancelAnimationFrame(raf) - }) - - return { - blocked, - questionRequest, - permissionRequest, - permissionResponding, - decide, - todos, - dock: () => store.dock, - closing: () => store.closing, - opening: () => store.opening, - } -} - -export type SessionComposerState = ReturnType diff --git a/packages/app/src/pages/session/composer/session-followup-dock.tsx b/packages/app/src/pages/session/composer/session-followup-dock.tsx deleted file mode 100644 index 7d744f4e6c..0000000000 --- a/packages/app/src/pages/session/composer/session-followup-dock.tsx +++ /dev/null @@ -1,109 +0,0 @@ -import { For, Show, createMemo } from "solid-js" -import { createStore } from "solid-js/store" -import { Button } from "@opencode-ai/ui/button" -import { DockTray } from "@opencode-ai/ui/dock-surface" -import { IconButton } from "@opencode-ai/ui/icon-button" -import { useLanguage } from "@/context/language" - -export function SessionFollowupDock(props: { - items: { id: string; text: string }[] - sending?: string - onSend: (id: string) => void - onEdit: (id: string) => void -}) { - const language = useLanguage() - const [store, setStore] = createStore({ - collapsed: false, - }) - - const toggle = () => setStore("collapsed", (value) => !value) - const total = createMemo(() => props.items.length) - const label = createMemo(() => - language.t(total() === 1 ? "session.followupDock.summary.one" : "session.followupDock.summary.other", { - count: total(), - }), - ) - const preview = createMemo(() => props.items[0]?.text ?? "") - - return ( - -
{ - if (event.key !== "Enter" && event.key !== " ") return - event.preventDefault() - toggle() - }} - > - {label()} - - {preview()} - -
- { - event.preventDefault() - event.stopPropagation() - }} - onClick={(event) => { - event.stopPropagation() - toggle() - }} - aria-label={ - store.collapsed ? language.t("session.followupDock.expand") : language.t("session.followupDock.collapse") - } - /> -
-
- - - - } - footer={ - <> -
-
- - - -
- - } - > - -
-
-
- - 0}> -
-
-
- - ) -} diff --git a/packages/app/src/pages/session/composer/session-question-dock.tsx b/packages/app/src/pages/session/composer/session-question-dock.tsx deleted file mode 100644 index 92ef05edf9..0000000000 --- a/packages/app/src/pages/session/composer/session-question-dock.tsx +++ /dev/null @@ -1,571 +0,0 @@ -import { For, Show, createMemo, onCleanup, onMount, type Component } from "solid-js" -import { createStore } from "solid-js/store" -import { useMutation } from "@tanstack/solid-query" -import { Button } from "@opencode-ai/ui/button" -import { DockPrompt } from "@opencode-ai/ui/dock-prompt" -import { Icon } from "@opencode-ai/ui/icon" -import { showToast } from "@opencode-ai/ui/toast" -import type { QuestionAnswer, QuestionRequest } from "@kilocode/sdk/v2" -import { useLanguage } from "@/context/language" -import { useSDK } from "@/context/sdk" -import { makeEventListener } from "@solid-primitives/event-listener" -import { createResizeObserver } from "@solid-primitives/resize-observer" - -const cache = new Map() - -function Mark(props: { multi: boolean; picked: boolean; onClick?: (event: MouseEvent) => void }) { - return ( - - ) -} - -function Option(props: { - multi: boolean - picked: boolean - label: string - description?: string - disabled: boolean - ref?: (el: HTMLButtonElement) => void - onFocus?: VoidFunction - onClick: VoidFunction -}) { - return ( - - ) -} - -export const SessionQuestionDock: Component<{ request: QuestionRequest; onSubmit: () => void }> = (props) => { - const sdk = useSDK() - const language = useLanguage() - - const questions = createMemo(() => props.request.questions) - const total = createMemo(() => questions().length) - - const cached = cache.get(props.request.id) - const [store, setStore] = createStore({ - tab: cached?.tab ?? 0, - answers: cached?.answers ?? ([] as QuestionAnswer[]), - custom: cached?.custom ?? ([] as string[]), - customOn: cached?.customOn ?? ([] as boolean[]), - editing: false, - focus: 0, - }) - - let root: HTMLDivElement | undefined - let customRef: HTMLButtonElement | undefined - let optsRef: HTMLButtonElement[] = [] - let replied = false - let focusFrame: number | undefined - - const question = createMemo(() => questions()[store.tab]) - const options = createMemo(() => question()?.options ?? []) - const input = createMemo(() => store.custom[store.tab] ?? "") - const on = createMemo(() => store.customOn[store.tab] === true) - const multi = createMemo(() => question()?.multiple === true) - const count = createMemo(() => options().length + 1) - - const summary = createMemo(() => { - const n = Math.min(store.tab + 1, total()) - return language.t("session.question.progress", { current: n, total: total() }) - }) - - const customLabel = () => language.t("ui.messagePart.option.typeOwnAnswer") - const customPlaceholder = () => language.t("ui.question.custom.placeholder") - - const last = createMemo(() => store.tab >= total() - 1) - - const customUpdate = (value: string, selected: boolean = on()) => { - const prev = input().trim() - const next = value.trim() - - setStore("custom", store.tab, value) - if (!selected) return - - if (multi()) { - setStore("answers", store.tab, (current: QuestionAnswer = []) => { - const removed = prev ? current.filter((item: string) => item.trim() !== prev) : current - if (!next) return removed - if (removed.some((item: string) => item.trim() === next)) return removed - return [...removed, next] - }) - return - } - - setStore("answers", store.tab, next ? [next] : []) - } - - const measure = () => { - if (!root) return - - const scroller = document.querySelector(".scroll-view__viewport") - const head = scroller instanceof HTMLElement ? scroller.firstElementChild : undefined - const top = - head instanceof HTMLElement && head.classList.contains("sticky") ? head.getBoundingClientRect().bottom : 0 - if (!top) { - root.style.removeProperty("--question-prompt-max-height") - return - } - - const dock = root.closest('[data-component="session-prompt-dock"]') - if (!(dock instanceof HTMLElement)) return - - const dockBottom = dock.getBoundingClientRect().bottom - const below = Math.max(0, dockBottom - root.getBoundingClientRect().bottom) - const gap = 8 - const max = Math.max(240, Math.floor(dockBottom - top - gap - below)) - root.style.setProperty("--question-prompt-max-height", `${max}px`) - } - - const clamp = (i: number) => Math.max(0, Math.min(count() - 1, i)) - - const pickFocus = (tab: number = store.tab) => { - const list = questions()[tab]?.options ?? [] - if (store.customOn[tab] === true) return list.length - return Math.max( - 0, - list.findIndex((item) => store.answers[tab]?.includes(item.label) ?? false), - ) - } - - const focus = (i: number) => { - const next = clamp(i) - setStore("focus", next) - if (store.editing) return - if (focusFrame !== undefined) cancelAnimationFrame(focusFrame) - focusFrame = requestAnimationFrame(() => { - focusFrame = undefined - const el = next === options().length ? customRef : optsRef[next] - el?.focus() - }) - } - - onMount(() => { - let raf: number | undefined - const update = () => { - if (raf !== undefined) cancelAnimationFrame(raf) - raf = requestAnimationFrame(() => { - raf = undefined - measure() - }) - } - - update() - - makeEventListener(window, "resize", update) - - const dock = root?.closest('[data-component="session-prompt-dock"]') - const scroller = document.querySelector(".scroll-view__viewport") - createResizeObserver([dock, scroller], update) - - onCleanup(() => { - if (raf !== undefined) cancelAnimationFrame(raf) - }) - - focus(pickFocus()) - }) - - onCleanup(() => { - if (focusFrame !== undefined) cancelAnimationFrame(focusFrame) - if (replied) return - cache.set(props.request.id, { - tab: store.tab, - answers: store.answers.map((a) => (a ? [...a] : [])), - custom: store.custom.map((s) => s ?? ""), - customOn: store.customOn.map((b) => b ?? false), - }) - }) - - const fail = (err: unknown) => { - const message = err instanceof Error ? err.message : String(err) - showToast({ title: language.t("common.requestFailed"), description: message }) - } - - const replyMutation = useMutation(() => ({ - mutationFn: (answers: QuestionAnswer[]) => sdk.client.question.reply({ requestID: props.request.id, answers }), - onMutate: () => { - props.onSubmit() - }, - onSuccess: () => { - replied = true - cache.delete(props.request.id) - }, - onError: fail, - })) - - const rejectMutation = useMutation(() => ({ - mutationFn: () => sdk.client.question.reject({ requestID: props.request.id }), - onMutate: () => { - props.onSubmit() - }, - onSuccess: () => { - replied = true - cache.delete(props.request.id) - }, - onError: fail, - })) - - const sending = createMemo(() => replyMutation.isPending || rejectMutation.isPending) - - const reply = async (answers: QuestionAnswer[]) => { - if (sending()) return - await replyMutation.mutateAsync(answers) - } - - const reject = async () => { - if (sending()) return - await rejectMutation.mutateAsync() - } - - const submit = () => void reply(questions().map((_: unknown, i: number) => store.answers[i] ?? [])) - - const answered = (i: number) => { - if ((store.answers[i]?.length ?? 0) > 0) return true - return store.customOn[i] === true && (store.custom[i] ?? "").trim().length > 0 - } - - const picked = (answer: string) => store.answers[store.tab]?.includes(answer) ?? false - - const pick = (answer: string, custom: boolean = false) => { - setStore("answers", store.tab, [answer]) - if (custom) setStore("custom", store.tab, answer) - if (!custom) setStore("customOn", store.tab, false) - setStore("editing", false) - } - - const toggle = (answer: string) => { - setStore("answers", store.tab, (current: QuestionAnswer = []) => { - if (current.includes(answer)) return current.filter((item: string) => item !== answer) - return [...current, answer] - }) - } - - const customToggle = () => { - if (sending()) return - setStore("focus", options().length) - - if (!multi()) { - setStore("customOn", store.tab, true) - setStore("editing", true) - customUpdate(input(), true) - return - } - - const next = !on() - setStore("customOn", store.tab, next) - if (next) { - setStore("editing", true) - customUpdate(input(), true) - return - } - - const value = input().trim() - if (value) - setStore("answers", store.tab, (current: QuestionAnswer = []) => - current.filter((item: string) => item.trim() !== value), - ) - setStore("editing", false) - focus(options().length) - } - - const customOpen = () => { - if (sending()) return - setStore("focus", options().length) - if (!on()) setStore("customOn", store.tab, true) - setStore("editing", true) - customUpdate(input(), true) - } - - const move = (step: number) => { - if (store.editing || sending()) return - focus(store.focus + step) - } - - const nav = (event: KeyboardEvent) => { - if (event.defaultPrevented) return - - if (event.key === "Escape") { - event.preventDefault() - void reject() - return - } - - const mod = (event.metaKey || event.ctrlKey) && !event.altKey - if (mod && event.key === "Enter") { - if (event.repeat) return - event.preventDefault() - next() - return - } - - const target = - event.target instanceof HTMLElement ? event.target.closest('[data-slot="question-options"]') : undefined - if (store.editing) return - if (!(target instanceof HTMLElement)) return - if (event.altKey || event.ctrlKey || event.metaKey) return - - if (event.key === "ArrowDown" || event.key === "ArrowRight") { - event.preventDefault() - move(1) - return - } - - if (event.key === "ArrowUp" || event.key === "ArrowLeft") { - event.preventDefault() - move(-1) - return - } - - if (event.key === "Home") { - event.preventDefault() - focus(0) - return - } - - if (event.key !== "End") return - event.preventDefault() - focus(count() - 1) - } - - const selectOption = (optIndex: number) => { - if (sending()) return - - if (optIndex === options().length) { - customOpen() - return - } - - const opt = options()[optIndex] - if (!opt) return - if (multi()) { - setStore("editing", false) - toggle(opt.label) - return - } - pick(opt.label) - } - - const commitCustom = () => { - setStore("editing", false) - customUpdate(input()) - focus(options().length) - } - - const resizeInput = (el: HTMLTextAreaElement) => { - el.style.height = "0px" - el.style.height = `${el.scrollHeight}px` - } - - const focusCustom = (el: HTMLTextAreaElement) => { - setTimeout(() => { - el.focus() - resizeInput(el) - }, 0) - } - - const toggleCustomMark = (event: MouseEvent) => { - event.preventDefault() - event.stopPropagation() - customToggle() - } - - const next = () => { - if (sending()) return - if (store.editing) commitCustom() - - if (store.tab >= total() - 1) { - submit() - return - } - - const tab = store.tab + 1 - setStore("tab", tab) - setStore("editing", false) - focus(pickFocus(tab)) - } - - const back = () => { - if (sending()) return - if (store.tab <= 0) return - const tab = store.tab - 1 - setStore("tab", tab) - setStore("editing", false) - focus(pickFocus(tab)) - } - - const jump = (tab: number) => { - if (sending()) return - setStore("tab", tab) - setStore("editing", false) - focus(pickFocus(tab)) - } - - return ( - (root = el)} - onKeyDown={nav} - header={ - <> -
{summary()}
-
- - {(_, i) => ( -
- - } - footer={ - <> - -
- 0}> - - - -
- - } - > -
{question()?.question}
- {language.t("ui.question.singleHint")}
}> -
{language.t("ui.question.multiHint")}
-
-
- - {(opt, i) => ( - - - setStore("focus", options().length)} - onClick={customOpen} - > - - - {customLabel()} - {input() || customPlaceholder()} - - - } - > -
{ - if (sending()) { - e.preventDefault() - return - } - if (e.target instanceof HTMLTextAreaElement) return - const input = e.currentTarget.querySelector('[data-slot="question-custom-input"]') - if (input instanceof HTMLTextAreaElement) input.focus() - }} - onSubmit={(e) => { - e.preventDefault() - commitCustom() - }} - > - - - {customLabel()} -
` - - const focused = focusTerminalById("one") - - expect(focused).toBe(true) - expect(document.activeElement?.tagName).toBe("TEXTAREA") - }) - - test("falls back to terminal element focus", () => { - document.body.innerHTML = `
` - const terminal = document.querySelector('[data-component="terminal"]') as HTMLElement - let pointerDown = false - terminal.addEventListener("pointerdown", () => { - pointerDown = true - }) - - const focused = focusTerminalById("two") - - expect(focused).toBe(true) - expect(document.activeElement).toBe(terminal) - expect(pointerDown).toBe(true) - }) -}) - -describe("shouldFocusTerminalOnKeyDown", () => { - test("skips pure modifier keys", () => { - expect(shouldFocusTerminalOnKeyDown(new KeyboardEvent("keydown", { key: "Meta", metaKey: true }))).toBe(false) - expect(shouldFocusTerminalOnKeyDown(new KeyboardEvent("keydown", { key: "Control", ctrlKey: true }))).toBe(false) - expect(shouldFocusTerminalOnKeyDown(new KeyboardEvent("keydown", { key: "Alt", altKey: true }))).toBe(false) - expect(shouldFocusTerminalOnKeyDown(new KeyboardEvent("keydown", { key: "Shift", shiftKey: true }))).toBe(false) - }) - - test("skips shortcut key combos", () => { - expect(shouldFocusTerminalOnKeyDown(new KeyboardEvent("keydown", { key: "c", metaKey: true }))).toBe(false) - expect(shouldFocusTerminalOnKeyDown(new KeyboardEvent("keydown", { key: "c", ctrlKey: true }))).toBe(false) - expect(shouldFocusTerminalOnKeyDown(new KeyboardEvent("keydown", { key: "ArrowLeft", altKey: true }))).toBe(false) - }) - - test("keeps plain typing focused on terminal", () => { - expect(shouldFocusTerminalOnKeyDown(new KeyboardEvent("keydown", { key: "a" }))).toBe(true) - expect(shouldFocusTerminalOnKeyDown(new KeyboardEvent("keydown", { key: "A", shiftKey: true }))).toBe(true) - }) -}) - -describe("getTabReorderIndex", () => { - test("returns target index for valid drag reorder", () => { - expect(getTabReorderIndex(["a", "b", "c"], "a", "c")).toBe(2) - }) - - test("returns undefined for unknown droppable id", () => { - expect(getTabReorderIndex(["a", "b", "c"], "a", "missing")).toBeUndefined() - }) -}) - -describe("createSessionTabs", () => { - test("normalizes the effective file tab", () => { - createRoot((dispose) => { - const [state] = createStore({ - active: undefined as string | undefined, - all: ["file://src/a.ts", "context"], - }) - const tabs = createMemo(() => ({ active: () => state.active, all: () => state.all })) - const result = createSessionTabs({ - tabs, - pathFromTab: (tab) => (tab.startsWith("file://") ? tab.slice("file://".length) : undefined), - normalizeTab: (tab) => (tab.startsWith("file://") ? `norm:${tab.slice("file://".length)}` : tab), - }) - - expect(result.activeTab()).toBe("norm:src/a.ts") - expect(result.activeFileTab()).toBe("norm:src/a.ts") - expect(result.closableTab()).toBe("norm:src/a.ts") - dispose() - }) - }) - - test("prefers context and review fallbacks when no file tab is active", () => { - createRoot((dispose) => { - const [state] = createStore({ - active: undefined as string | undefined, - all: ["context"], - }) - const tabs = createMemo(() => ({ active: () => state.active, all: () => state.all })) - const result = createSessionTabs({ - tabs, - pathFromTab: () => undefined, - normalizeTab: (tab) => tab, - review: () => true, - hasReview: () => true, - }) - - expect(result.activeTab()).toBe("context") - expect(result.closableTab()).toBe("context") - dispose() - }) - - createRoot((dispose) => { - const [state] = createStore({ - active: undefined as string | undefined, - all: [], - }) - const tabs = createMemo(() => ({ active: () => state.active, all: () => state.all })) - const result = createSessionTabs({ - tabs, - pathFromTab: () => undefined, - normalizeTab: (tab) => tab, - review: () => true, - hasReview: () => true, - }) - - expect(result.activeTab()).toBe("review") - expect(result.activeFileTab()).toBeUndefined() - expect(result.closableTab()).toBeUndefined() - dispose() - }) - }) -}) diff --git a/packages/app/src/pages/session/helpers.ts b/packages/app/src/pages/session/helpers.ts deleted file mode 100644 index e136ba9991..0000000000 --- a/packages/app/src/pages/session/helpers.ts +++ /dev/null @@ -1,194 +0,0 @@ -import { batch, createMemo, onCleanup, onMount, type Accessor } from "solid-js" -import { createStore } from "solid-js/store" -import { makeEventListener } from "@solid-primitives/event-listener" -import { same } from "@/utils/same" - -const emptyTabs: string[] = [] - -type Tabs = { - active: Accessor - all: Accessor -} - -type TabsInput = { - tabs: Accessor - pathFromTab: (tab: string) => string | undefined - normalizeTab: (tab: string) => string - review?: Accessor - hasReview?: Accessor -} - -export const getSessionKey = (dir: string | undefined, id: string | undefined) => `${dir ?? ""}${id ? `/${id}` : ""}` - -export const createSessionTabs = (input: TabsInput) => { - const review = input.review ?? (() => false) - const hasReview = input.hasReview ?? (() => false) - const contextOpen = createMemo(() => input.tabs().active() === "context" || input.tabs().all().includes("context")) - const openedTabs = createMemo( - () => { - const seen = new Set() - return input - .tabs() - .all() - .flatMap((tab) => { - if (tab === "context" || tab === "review") return [] - const value = input.pathFromTab(tab) ? input.normalizeTab(tab) : tab - if (seen.has(value)) return [] - seen.add(value) - return [value] - }) - }, - emptyTabs, - { equals: same }, - ) - const activeTab = createMemo(() => { - const active = input.tabs().active() - if (active === "context") return active - if (active === "review" && review()) return active - if (active && input.pathFromTab(active)) return input.normalizeTab(active) - - const first = openedTabs()[0] - if (first) return first - if (contextOpen()) return "context" - if (review() && hasReview()) return "review" - return "empty" - }) - const activeFileTab = createMemo(() => { - const active = activeTab() - if (!openedTabs().includes(active)) return - return active - }) - const closableTab = createMemo(() => { - const active = activeTab() - if (active === "context") return active - if (!openedTabs().includes(active)) return - return active - }) - - return { - contextOpen, - openedTabs, - activeTab, - activeFileTab, - closableTab, - } -} - -export const focusTerminalById = (id: string) => { - const wrapper = document.getElementById(`terminal-wrapper-${id}`) - const terminal = wrapper?.querySelector('[data-component="terminal"]') - if (!(terminal instanceof HTMLElement)) return false - - const textarea = terminal.querySelector("textarea") - if (textarea instanceof HTMLTextAreaElement) { - textarea.focus() - return true - } - - terminal.focus() - terminal.dispatchEvent( - typeof PointerEvent === "function" - ? new PointerEvent("pointerdown", { bubbles: true, cancelable: true }) - : new MouseEvent("pointerdown", { bubbles: true, cancelable: true }), - ) - return true -} - -const skip = new Set(["Alt", "Control", "Meta", "Shift"]) - -export const shouldFocusTerminalOnKeyDown = (event: Pick) => { - if (skip.has(event.key)) return false - return !(event.ctrlKey || event.metaKey || event.altKey) -} - -export const createOpenReviewFile = (input: { - showAllFiles: () => void - tabForPath: (path: string) => string - openTab: (tab: string) => void - setActive: (tab: string) => void - loadFile: (path: string) => any | Promise -}) => { - return (path: string) => { - batch(() => { - input.showAllFiles() - const maybePromise = input.loadFile(path) - const open = () => { - const tab = input.tabForPath(path) - input.openTab(tab) - input.setActive(tab) - } - if (maybePromise instanceof Promise) void maybePromise.then(open) - else open() - }) - } -} - -export const createOpenSessionFileTab = (input: { - normalizeTab: (tab: string) => string - openTab: (tab: string) => void - pathFromTab: (tab: string) => string | undefined - loadFile: (path: string) => void - openReviewPanel: () => void - setActive: (tab: string) => void -}) => { - return (value: string) => { - const next = input.normalizeTab(value) - input.openTab(next) - - const path = input.pathFromTab(next) - if (!path) return - - input.loadFile(path) - input.openReviewPanel() - input.setActive(next) - } -} - -export const getTabReorderIndex = (tabs: readonly string[], from: string, to: string) => { - const fromIndex = tabs.indexOf(from) - const toIndex = tabs.indexOf(to) - if (fromIndex === -1 || toIndex === -1 || fromIndex === toIndex) return undefined - return toIndex -} - -export const createSizing = () => { - const [state, setState] = createStore({ active: false }) - let t: number | undefined - - const stop = () => { - if (t !== undefined) { - clearTimeout(t) - t = undefined - } - setState("active", false) - } - - const start = () => { - if (t !== undefined) { - clearTimeout(t) - t = undefined - } - setState("active", true) - } - - onMount(() => { - makeEventListener(window, "pointerup", stop) - makeEventListener(window, "pointercancel", stop) - makeEventListener(window, "blur", stop) - }) - - onCleanup(() => { - if (t !== undefined) clearTimeout(t) - }) - - return { - active: () => state.active, - start, - touch() { - start() - t = window.setTimeout(stop, 120) - }, - } -} - -export type Sizing = ReturnType diff --git a/packages/app/src/pages/session/message-gesture.test.ts b/packages/app/src/pages/session/message-gesture.test.ts deleted file mode 100644 index b2af4bb834..0000000000 --- a/packages/app/src/pages/session/message-gesture.test.ts +++ /dev/null @@ -1,62 +0,0 @@ -import { describe, expect, test } from "bun:test" -import { normalizeWheelDelta, shouldMarkBoundaryGesture } from "./message-gesture" - -describe("normalizeWheelDelta", () => { - test("converts line mode to px", () => { - expect(normalizeWheelDelta({ deltaY: 3, deltaMode: 1, rootHeight: 500 })).toBe(120) - }) - - test("converts page mode to container height", () => { - expect(normalizeWheelDelta({ deltaY: -1, deltaMode: 2, rootHeight: 600 })).toBe(-600) - }) - - test("keeps pixel mode unchanged", () => { - expect(normalizeWheelDelta({ deltaY: 16, deltaMode: 0, rootHeight: 600 })).toBe(16) - }) -}) - -describe("shouldMarkBoundaryGesture", () => { - test("marks when nested scroller cannot scroll", () => { - expect( - shouldMarkBoundaryGesture({ - delta: 20, - scrollTop: 0, - scrollHeight: 300, - clientHeight: 300, - }), - ).toBe(true) - }) - - test("marks when scrolling beyond top boundary", () => { - expect( - shouldMarkBoundaryGesture({ - delta: -40, - scrollTop: 10, - scrollHeight: 1000, - clientHeight: 400, - }), - ).toBe(true) - }) - - test("marks when scrolling beyond bottom boundary", () => { - expect( - shouldMarkBoundaryGesture({ - delta: 50, - scrollTop: 580, - scrollHeight: 1000, - clientHeight: 400, - }), - ).toBe(true) - }) - - test("does not mark when nested scroller can consume movement", () => { - expect( - shouldMarkBoundaryGesture({ - delta: 20, - scrollTop: 200, - scrollHeight: 1000, - clientHeight: 400, - }), - ).toBe(false) - }) -}) diff --git a/packages/app/src/pages/session/message-gesture.ts b/packages/app/src/pages/session/message-gesture.ts deleted file mode 100644 index 731cb1bdeb..0000000000 --- a/packages/app/src/pages/session/message-gesture.ts +++ /dev/null @@ -1,21 +0,0 @@ -export const normalizeWheelDelta = (input: { deltaY: number; deltaMode: number; rootHeight: number }) => { - if (input.deltaMode === 1) return input.deltaY * 40 - if (input.deltaMode === 2) return input.deltaY * input.rootHeight - return input.deltaY -} - -export const shouldMarkBoundaryGesture = (input: { - delta: number - scrollTop: number - scrollHeight: number - clientHeight: number -}) => { - const max = input.scrollHeight - input.clientHeight - if (max <= 1) return true - if (!input.delta) return false - - if (input.delta < 0) return input.scrollTop + input.delta <= 0 - - const remaining = max - input.scrollTop - return input.delta > remaining -} diff --git a/packages/app/src/pages/session/message-id-from-hash.ts b/packages/app/src/pages/session/message-id-from-hash.ts deleted file mode 100644 index 2857f4b01d..0000000000 --- a/packages/app/src/pages/session/message-id-from-hash.ts +++ /dev/null @@ -1,6 +0,0 @@ -export const messageIdFromHash = (hash: string) => { - const value = hash.startsWith("#") ? hash.slice(1) : hash - const match = value.match(/^message-(.+)$/) - if (!match) return - return match[1] -} diff --git a/packages/app/src/pages/session/review-tab.tsx b/packages/app/src/pages/session/review-tab.tsx deleted file mode 100644 index 47cfb330cb..0000000000 --- a/packages/app/src/pages/session/review-tab.tsx +++ /dev/null @@ -1,170 +0,0 @@ -import { createEffect, onCleanup, type JSX } from "solid-js" -import { makeEventListener } from "@solid-primitives/event-listener" -import type { SnapshotFileDiff, VcsFileDiff } from "@kilocode/sdk/v2" -import { SessionReview } from "@opencode-ai/ui/session-review" -import type { - SessionReviewCommentActions, - SessionReviewCommentDelete, - SessionReviewCommentUpdate, -} from "@opencode-ai/ui/session-review" -import type { SelectedLineRange } from "@/context/file" -import { useSDK } from "@/context/sdk" -import { useLayout } from "@/context/layout" -import type { LineComment } from "@/context/comments" - -export type DiffStyle = "unified" | "split" - -type ReviewDiff = SnapshotFileDiff | VcsFileDiff - -export interface SessionReviewTabProps { - title?: JSX.Element - empty?: JSX.Element - diffs: () => ReviewDiff[] - view: () => ReturnType["view"]> - diffStyle: DiffStyle - onDiffStyleChange?: (style: DiffStyle) => void - onViewFile?: (file: string) => void - onLineComment?: (comment: { file: string; selection: SelectedLineRange; comment: string; preview?: string }) => void - onLineCommentUpdate?: (comment: SessionReviewCommentUpdate) => void - onLineCommentDelete?: (comment: SessionReviewCommentDelete) => void - lineCommentActions?: SessionReviewCommentActions - comments?: LineComment[] - focusedComment?: { file: string; id: string } | null - onFocusedCommentChange?: (focus: { file: string; id: string } | null) => void - focusedFile?: string - onScrollRef?: (el: HTMLDivElement) => void - commentMentions?: { - items: (query: string) => string[] | Promise - } - classes?: { - root?: string - header?: string - container?: string - } -} - -export function SessionReviewTab(props: SessionReviewTabProps) { - let scroll: HTMLDivElement | undefined - let restoreFrame: number | undefined - let userInteracted = false - let restored: { x: number; y: number } | undefined - - const sdk = useSDK() - const layout = useLayout() - - const readFile = async (path: string) => { - return sdk.client.file - .read({ path }) - .then((x) => x.data) - .catch((error) => { - console.debug("[session-review] failed to read file", { path, error }) - return undefined - }) - } - - const handleInteraction = () => { - userInteracted = true - - if (restoreFrame !== undefined) { - cancelAnimationFrame(restoreFrame) - restoreFrame = undefined - } - } - - const doRestore = () => { - restoreFrame = undefined - const el = scroll - if (!el || !layout.ready() || userInteracted) return - if (el.clientHeight === 0 || el.clientWidth === 0) return - - const s = props.view().scroll("review") - if (!s || (s.x === 0 && s.y === 0)) return - - const maxY = Math.max(0, el.scrollHeight - el.clientHeight) - const maxX = Math.max(0, el.scrollWidth - el.clientWidth) - - const targetY = Math.min(s.y, maxY) - const targetX = Math.min(s.x, maxX) - - if (el.scrollTop === targetY && el.scrollLeft === targetX) return - - if (el.scrollTop !== targetY) el.scrollTop = targetY - if (el.scrollLeft !== targetX) el.scrollLeft = targetX - restored = { x: el.scrollLeft, y: el.scrollTop } - } - - const queueRestore = () => { - if (userInteracted || restoreFrame !== undefined) return - restoreFrame = requestAnimationFrame(doRestore) - } - - const handleScroll = (event: Event & { currentTarget: HTMLDivElement }) => { - const el = event.currentTarget - const prev = restored - if (prev && el.scrollTop === prev.y && el.scrollLeft === prev.x) { - restored = undefined - return - } - - restored = undefined - handleInteraction() - if (!layout.ready()) return - if (el.clientHeight === 0 || el.clientWidth === 0) return - - props.view().setScroll("review", { - x: el.scrollLeft, - y: el.scrollTop, - }) - } - - createEffect(() => { - props.diffs().length - props.diffStyle - if (!layout.ready()) return - queueRestore() - }) - - onCleanup(() => { - if (restoreFrame !== undefined) cancelAnimationFrame(restoreFrame) - }) - - return ( - { - scroll = el - makeEventListener(el, "wheel", handleInteraction, { passive: true, capture: true }) - makeEventListener(el, "mousewheel", handleInteraction, { passive: true, capture: true }) - makeEventListener(el, "pointerdown", handleInteraction, { passive: true, capture: true }) - makeEventListener(el, "touchstart", handleInteraction, { passive: true, capture: true }) - makeEventListener(el, "keydown", handleInteraction, { capture: true }) - props.onScrollRef?.(el) - queueRestore() - }} - onScroll={handleScroll} - onDiffRendered={queueRestore} - open={props.view().review.open()} - onOpenChange={props.view().review.setOpen} - classes={{ - root: props.classes?.root ?? "pr-3", - header: props.classes?.header ?? "px-3", - container: props.classes?.container ?? "pl-3", - }} - diffs={props.diffs()} - diffStyle={props.diffStyle} - onDiffStyleChange={props.onDiffStyleChange} - onViewFile={props.onViewFile} - focusedFile={props.focusedFile} - readFile={readFile} - onLineComment={props.onLineComment} - onLineCommentUpdate={props.onLineCommentUpdate} - onLineCommentDelete={props.onLineCommentDelete} - lineCommentActions={props.lineCommentActions} - lineCommentMention={props.commentMentions} - comments={props.comments} - focusedComment={props.focusedComment} - onFocusedCommentChange={props.onFocusedCommentChange} - /> - ) -} diff --git a/packages/app/src/pages/session/session-layout.ts b/packages/app/src/pages/session/session-layout.ts deleted file mode 100644 index 113411150d..0000000000 --- a/packages/app/src/pages/session/session-layout.ts +++ /dev/null @@ -1,20 +0,0 @@ -import { useParams } from "@solidjs/router" -import { createMemo } from "solid-js" -import { useLayout } from "@/context/layout" - -export const useSessionKey = () => { - const params = useParams() - const sessionKey = createMemo(() => `${params.dir}${params.id ? "/" + params.id : ""}`) - return { params, sessionKey } -} - -export const useSessionLayout = () => { - const layout = useLayout() - const { params, sessionKey } = useSessionKey() - return { - params, - sessionKey, - tabs: createMemo(() => layout.tabs(sessionKey)), - view: createMemo(() => layout.view(sessionKey)), - } -} diff --git a/packages/app/src/pages/session/session-model-helpers.test.ts b/packages/app/src/pages/session/session-model-helpers.test.ts deleted file mode 100644 index 36f0f0772c..0000000000 --- a/packages/app/src/pages/session/session-model-helpers.test.ts +++ /dev/null @@ -1,52 +0,0 @@ -import { describe, expect, test } from "bun:test" -import type { UserMessage } from "@kilocode/sdk/v2" -import { resetSessionModel, syncSessionModel } from "./session-model-helpers" - -const message = (input?: { agent?: string; model?: UserMessage["model"] }) => - ({ - id: "msg", - sessionID: "session", - role: "user", - time: { created: 1 }, - agent: input?.agent ?? "build", - model: input?.model ?? { providerID: "anthropic", modelID: "claude-sonnet-4" }, - }) as UserMessage - -describe("syncSessionModel", () => { - test("restores the last message through session state", () => { - const calls: unknown[] = [] - - syncSessionModel( - { - session: { - restore(value) { - calls.push(value) - }, - reset() {}, - }, - }, - message({ model: { providerID: "anthropic", modelID: "claude-sonnet-4", variant: "high" } }), - ) - - expect(calls).toEqual([ - message({ model: { providerID: "anthropic", modelID: "claude-sonnet-4", variant: "high" } }), - ]) - }) -}) - -describe("resetSessionModel", () => { - test("clears draft session state", () => { - const calls: string[] = [] - - resetSessionModel({ - session: { - reset() { - calls.push("reset") - }, - restore() {}, - }, - }) - - expect(calls).toEqual(["reset"]) - }) -}) diff --git a/packages/app/src/pages/session/session-model-helpers.ts b/packages/app/src/pages/session/session-model-helpers.ts deleted file mode 100644 index 8536f2e9ee..0000000000 --- a/packages/app/src/pages/session/session-model-helpers.ts +++ /dev/null @@ -1,16 +0,0 @@ -import type { UserMessage } from "@kilocode/sdk/v2" - -type Local = { - session: { - reset(): void - restore(msg: UserMessage): void - } -} - -export const resetSessionModel = (local: Local) => { - local.session.reset() -} - -export const syncSessionModel = (local: Local, msg: UserMessage) => { - local.session.restore(msg) -} diff --git a/packages/app/src/pages/session/session-side-panel.tsx b/packages/app/src/pages/session/session-side-panel.tsx deleted file mode 100644 index 77d30ba3a8..0000000000 --- a/packages/app/src/pages/session/session-side-panel.tsx +++ /dev/null @@ -1,453 +0,0 @@ -import { For, Match, Show, Switch, createEffect, createMemo, onCleanup, type JSX } from "solid-js" -import { createStore } from "solid-js/store" -import { createMediaQuery } from "@solid-primitives/media" -import { Tabs } from "@opencode-ai/ui/tabs" -import { IconButton } from "@opencode-ai/ui/icon-button" -import { TooltipKeybind } from "@opencode-ai/ui/tooltip" -import { ResizeHandle } from "@opencode-ai/ui/resize-handle" -import { Mark } from "@opencode-ai/ui/logo" -import { DragDropProvider, DragDropSensors, DragOverlay, SortableProvider, closestCenter } from "@thisbeyond/solid-dnd" -import type { DragEvent } from "@thisbeyond/solid-dnd" -import type { SnapshotFileDiff, VcsFileDiff } from "@kilocode/sdk/v2" -import { ConstrainDragYAxis, getDraggableId } from "@/utils/solid-dnd" -import { useDialog } from "@opencode-ai/ui/context/dialog" - -import FileTree from "@/components/file-tree" -import { SessionContextUsage } from "@/components/session-context-usage" -import { SessionContextTab, SortableTab, FileVisual } from "@/components/session" -import { useCommand } from "@/context/command" -import { useFile, type SelectedLineRange } from "@/context/file" -import { useLanguage } from "@/context/language" -import { useLayout } from "@/context/layout" -import { usePlatform } from "@/context/platform" -import { useSettings } from "@/context/settings" -import { useSync } from "@/context/sync" -import { createFileTabListSync } from "@/pages/session/file-tab-scroll" -import { FileTabContent } from "@/pages/session/file-tabs" -import { createOpenSessionFileTab, createSessionTabs, getTabReorderIndex, type Sizing } from "@/pages/session/helpers" -import { setSessionHandoff } from "@/pages/session/handoff" -import { useSessionLayout } from "@/pages/session/session-layout" - -export function SessionSidePanel(props: { - canReview: () => boolean - diffs: () => (SnapshotFileDiff | VcsFileDiff)[] - diffsReady: () => boolean - empty: () => string - hasReview: () => boolean - reviewCount: () => number - reviewPanel: () => JSX.Element - activeDiff?: string - focusReviewDiff: (path: string) => void - reviewSnap: boolean - size: Sizing -}) { - const layout = useLayout() - const platform = usePlatform() - const settings = useSettings() - const sync = useSync() - const file = useFile() - const language = useLanguage() - const command = useCommand() - const dialog = useDialog() - const { sessionKey, tabs, view } = useSessionLayout() - - const isDesktop = createMediaQuery("(min-width: 768px)") - const shown = createMemo( - () => - platform.platform !== "desktop" || - import.meta.env.VITE_KILO_CHANNEL !== "beta" || - settings.general.showFileTree(), - ) - - const reviewOpen = createMemo(() => isDesktop() && view().reviewPanel.opened()) - const fileOpen = createMemo(() => isDesktop() && shown() && layout.fileTree.opened()) - const open = createMemo(() => reviewOpen() || fileOpen()) - const reviewTab = createMemo(() => isDesktop()) - const panelWidth = createMemo(() => { - if (!open()) return "0px" - if (reviewOpen()) return `calc(100% - ${layout.session.width()}px)` - return `${layout.fileTree.width()}px` - }) - const treeWidth = createMemo(() => (fileOpen() ? `${layout.fileTree.width()}px` : "0px")) - - const diffFiles = createMemo(() => props.diffs().map((d) => d.file)) - const kinds = createMemo(() => { - const merge = (a: "add" | "del" | "mix" | undefined, b: "add" | "del" | "mix") => { - if (!a) return b - if (a === b) return a - return "mix" as const - } - - const normalize = (p: string) => p.replaceAll("\\\\", "/").replace(/\/+$/, "") - - const out = new Map() - for (const diff of props.diffs()) { - const file = normalize(diff.file) - const kind = diff.status === "added" ? "add" : diff.status === "deleted" ? "del" : "mix" - - out.set(file, kind) - - const parts = file.split("/") - for (const [idx] of parts.slice(0, -1).entries()) { - const dir = parts.slice(0, idx + 1).join("/") - if (!dir) continue - out.set(dir, merge(out.get(dir), kind)) - } - } - return out - }) - - const empty = (msg: string) => ( -
-
-
-
{msg}
-
-
- ) - - const nofiles = createMemo(() => { - const state = file.tree.state("") - if (!state?.loaded) return false - return file.tree.children("").length === 0 - }) - - const normalizeTab = (tab: string) => { - if (!tab.startsWith("file://")) return tab - return file.tab(tab) - } - - const openReviewPanel = () => { - if (!view().reviewPanel.opened()) view().reviewPanel.open() - } - - const openTab = createOpenSessionFileTab({ - normalizeTab, - openTab: tabs().open, - pathFromTab: file.pathFromTab, - loadFile: file.load, - openReviewPanel, - setActive: tabs().setActive, - }) - - const tabState = createSessionTabs({ - tabs, - pathFromTab: file.pathFromTab, - normalizeTab, - review: reviewTab, - hasReview: props.canReview, - }) - const contextOpen = tabState.contextOpen - const openedTabs = tabState.openedTabs - const activeTab = tabState.activeTab - const activeFileTab = tabState.activeFileTab - - const fileTreeTab = () => layout.fileTree.tab() - - const setFileTreeTabValue = (value: string) => { - if (value !== "changes" && value !== "all") return - layout.fileTree.setTab(value) - } - - const showAllFiles = () => { - if (fileTreeTab() !== "changes") return - layout.fileTree.setTab("all") - } - - const [store, setStore] = createStore({ - activeDraggable: undefined as string | undefined, - }) - - const handleDragStart = (event: unknown) => { - const id = getDraggableId(event) - if (!id) return - setStore("activeDraggable", id) - } - - const handleDragOver = (event: DragEvent) => { - const { draggable, droppable } = event - if (!draggable || !droppable) return - - const currentTabs = tabs().all() - const toIndex = getTabReorderIndex(currentTabs, draggable.id.toString(), droppable.id.toString()) - if (toIndex === undefined) return - tabs().move(draggable.id.toString(), toIndex) - } - - const handleDragEnd = () => { - setStore("activeDraggable", undefined) - } - - createEffect(() => { - if (!file.ready()) return - - setSessionHandoff(sessionKey(), { - files: tabs() - .all() - .reduce>((acc, tab) => { - const path = file.pathFromTab(tab) - if (!path) return acc - - const selected = file.selectedLines(path) - acc[path] = - selected && typeof selected === "object" && "start" in selected && "end" in selected - ? (selected as SelectedLineRange) - : null - - return acc - }, {}), - }) - }) - - return ( - -
- - - ) -} diff --git a/packages/app/src/pages/session/terminal-label.ts b/packages/app/src/pages/session/terminal-label.ts deleted file mode 100644 index 8a34712e43..0000000000 --- a/packages/app/src/pages/session/terminal-label.ts +++ /dev/null @@ -1,16 +0,0 @@ -import { isDefaultTitle as isDefaultTerminalTitle } from "@/context/terminal-title" - -export const terminalTabLabel = (input: { - title?: string - titleNumber?: number - t: (key: string, vars?: Record) => string -}) => { - const title = input.title ?? "" - const number = input.titleNumber ?? 0 - const isDefaultTitle = Number.isFinite(number) && number > 0 && isDefaultTerminalTitle(title, number) - - if (title && !isDefaultTitle) return title - if (number > 0) return input.t("terminal.title.numbered", { number }) - if (title) return title - return input.t("terminal.title") -} diff --git a/packages/app/src/pages/session/terminal-panel.test.ts b/packages/app/src/pages/session/terminal-panel.test.ts deleted file mode 100644 index 43eeec32f2..0000000000 --- a/packages/app/src/pages/session/terminal-panel.test.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { describe, expect, test } from "bun:test" -import { terminalTabLabel } from "./terminal-label" - -const t = (key: string, vars?: Record) => { - if (key === "terminal.title.numbered") return `Terminal ${vars?.number}` - if (key === "terminal.title") return "Terminal" - return key -} - -describe("terminalTabLabel", () => { - test("returns custom title unchanged", () => { - const label = terminalTabLabel({ title: "server", titleNumber: 3, t }) - expect(label).toBe("server") - }) - - test("normalizes default numbered title", () => { - const label = terminalTabLabel({ title: "Terminal 2", titleNumber: 2, t }) - expect(label).toBe("Terminal 2") - }) - - test("falls back to generic title", () => { - const label = terminalTabLabel({ title: "", titleNumber: 0, t }) - expect(label).toBe("Terminal") - }) -}) diff --git a/packages/app/src/pages/session/terminal-panel.tsx b/packages/app/src/pages/session/terminal-panel.tsx deleted file mode 100644 index 2c2d9817f0..0000000000 --- a/packages/app/src/pages/session/terminal-panel.tsx +++ /dev/null @@ -1,317 +0,0 @@ -import { For, Show, createEffect, createMemo, on, onCleanup, onMount } from "solid-js" -import { createStore } from "solid-js/store" -import { makeEventListener } from "@solid-primitives/event-listener" -import { Tabs } from "@opencode-ai/ui/tabs" -import { ResizeHandle } from "@opencode-ai/ui/resize-handle" -import { IconButton } from "@opencode-ai/ui/icon-button" -import { TooltipKeybind } from "@opencode-ai/ui/tooltip" -import { DragDropProvider, DragDropSensors, DragOverlay, SortableProvider, closestCenter } from "@thisbeyond/solid-dnd" -import type { DragEvent } from "@thisbeyond/solid-dnd" -import { ConstrainDragYAxis, getDraggableId } from "@/utils/solid-dnd" - -import { SortableTerminalTab } from "@/components/session" -import { Terminal } from "@/components/terminal" -import { useCommand } from "@/context/command" -import { useLanguage } from "@/context/language" -import { useLayout } from "@/context/layout" -import { useTerminal } from "@/context/terminal" -import { terminalTabLabel } from "@/pages/session/terminal-label" -import { createSizing, focusTerminalById } from "@/pages/session/helpers" -import { getTerminalHandoff, setTerminalHandoff } from "@/pages/session/handoff" -import { useSessionLayout } from "@/pages/session/session-layout" - -export function TerminalPanel() { - const delays = [120, 240] - const layout = useLayout() - const terminal = useTerminal() - const language = useLanguage() - const command = useCommand() - const { params, view } = useSessionLayout() - - const opened = createMemo(() => view().terminal.opened()) - const size = createSizing() - const height = createMemo(() => layout.terminal.height()) - const close = () => view().terminal.close() - let root: HTMLDivElement | undefined - - const [store, setStore] = createStore({ - autoCreated: false, - activeDraggable: undefined as string | undefined, - view: typeof window === "undefined" ? 1000 : (window.visualViewport?.height ?? window.innerHeight), - }) - - const max = () => store.view * 0.6 - const pane = () => Math.min(height(), max()) - - onMount(() => { - if (typeof window === "undefined") return - - const sync = () => setStore("view", window.visualViewport?.height ?? window.innerHeight) - const port = window.visualViewport - - sync() - makeEventListener(window, "resize", sync) - if (port) makeEventListener(port, "resize", sync) - }) - - createEffect(() => { - if (!opened()) { - setStore("autoCreated", false) - return - } - - if (!terminal.ready() || terminal.all().length !== 0 || store.autoCreated) return - terminal.new() - setStore("autoCreated", true) - }) - - createEffect( - on( - () => terminal.all().length, - (count, prevCount) => { - if (prevCount === undefined || prevCount <= 0 || count !== 0) return - if (!opened()) return - close() - }, - ), - ) - - const focus = (id: string) => { - focusTerminalById(id) - - const frame = requestAnimationFrame(() => { - if (!opened()) return - if (terminal.active() !== id) return - focusTerminalById(id) - }) - - const timers = delays.map((ms) => - window.setTimeout(() => { - if (!opened()) return - if (terminal.active() !== id) return - focusTerminalById(id) - }, ms), - ) - - return () => { - cancelAnimationFrame(frame) - for (const timer of timers) clearTimeout(timer) - } - } - - createEffect( - on( - () => [opened(), terminal.active()] as const, - ([next, id]) => { - if (!next || !id) return - const stop = focus(id) - onCleanup(stop) - }, - ), - ) - - createEffect(() => { - if (opened()) return - const active = document.activeElement - if (!(active instanceof HTMLElement)) return - if (!root?.contains(active)) return - active.blur() - }) - - createEffect(() => { - const dir = params.dir - if (!dir) return - if (!terminal.ready()) return - language.locale() - - setTerminalHandoff( - dir, - terminal.all().map((pty) => - terminalTabLabel({ - title: pty.title, - titleNumber: pty.titleNumber, - t: language.t as (key: string, vars?: Record) => string, - }), - ), - ) - }) - - const handoff = createMemo(() => { - const dir = params.dir - if (!dir) return [] - return getTerminalHandoff(dir) ?? [] - }) - - const all = terminal.all - const ids = createMemo(() => all().map((pty) => pty.id)) - - const handleTerminalDragStart = (event: unknown) => { - const id = getDraggableId(event) - if (!id) return - setStore("activeDraggable", id) - } - - const handleTerminalDragOver = (event: DragEvent) => { - const { draggable, droppable } = event - if (!draggable || !droppable) return - - const terminals = terminal.all() - const fromIndex = terminals.findIndex((t) => t.id === draggable.id.toString()) - const toIndex = terminals.findIndex((t) => t.id === droppable.id.toString()) - if (fromIndex !== -1 && toIndex !== -1 && fromIndex !== toIndex) { - terminal.move(draggable.id.toString(), toIndex) - } - } - - const handleTerminalDragEnd = () => { - setStore("activeDraggable", undefined) - - const activeId = terminal.active() - if (!activeId) return - requestAnimationFrame(() => { - if (terminal.active() !== activeId) return - focusTerminalById(activeId) - }) - } - - return ( -
-
- - -
- - {(title) => ( -
- {title} -
- )} -
-
-
- {language.t("common.loading")} - {language.t("common.loading.ellipsis")} -
-
-
{language.t("terminal.loading")}
-
- } - > - - - -
- terminal.open(id)} - class="!h-auto !flex-none" - > - - - {(pty) => } - -
- - - -
-
-
-
- - {(id) => { - const ops = terminal.bind() - return ( - pty.id === id)}> - {(pty) => ( -
- ops.trim(id)} - onCleanup={ops.update} - onConnectError={() => ops.clone(id)} - /> -
- )} -
- ) - }} -
-
-
- - - {(id) => ( - pty.id === id)}> - {(t) => ( -
- {terminalTabLabel({ - title: t().title, - titleNumber: t().titleNumber, - t: language.t as (key: string, vars?: Record) => string, - })} -
- )} -
- )} -
-
-
-
-
-
- ) -} diff --git a/packages/app/src/pages/session/use-session-hash-scroll.test.ts b/packages/app/src/pages/session/use-session-hash-scroll.test.ts deleted file mode 100644 index 7f3389baaa..0000000000 --- a/packages/app/src/pages/session/use-session-hash-scroll.test.ts +++ /dev/null @@ -1,16 +0,0 @@ -import { describe, expect, test } from "bun:test" -import { messageIdFromHash } from "./message-id-from-hash" - -describe("messageIdFromHash", () => { - test("parses hash with leading #", () => { - expect(messageIdFromHash("#message-abc123")).toBe("abc123") - }) - - test("parses raw hash fragment", () => { - expect(messageIdFromHash("message-42")).toBe("42") - }) - - test("ignores non-message anchors", () => { - expect(messageIdFromHash("#review-panel")).toBeUndefined() - }) -}) diff --git a/packages/app/src/pages/session/use-session-hash-scroll.ts b/packages/app/src/pages/session/use-session-hash-scroll.ts deleted file mode 100644 index 474e2ad321..0000000000 --- a/packages/app/src/pages/session/use-session-hash-scroll.ts +++ /dev/null @@ -1,215 +0,0 @@ -import type { UserMessage } from "@kilocode/sdk/v2" -import { useLocation, useNavigate } from "@solidjs/router" -import { createEffect, createMemo, onCleanup, onMount } from "solid-js" -import { messageIdFromHash } from "./message-id-from-hash" - -export const useSessionHashScroll = (input: { - sessionKey: () => string - sessionID: () => string | undefined - messagesReady: () => boolean - visibleUserMessages: () => UserMessage[] - historyMore: () => boolean - historyLoading: () => boolean - loadMore: (sessionID: string) => Promise - turnStart: () => number - currentMessageId: () => string | undefined - pendingMessage: () => string | undefined - setPendingMessage: (value: string | undefined) => void - setActiveMessage: (message: UserMessage | undefined) => void - setTurnStart: (value: number) => void - autoScroll: { pause: () => void; forceScrollToBottom: () => void } - scroller: () => HTMLDivElement | undefined - anchor: (id: string) => string - scheduleScrollState: (el: HTMLDivElement) => void - consumePendingMessage: (key: string) => string | undefined -}) => { - const visibleUserMessages = createMemo(() => input.visibleUserMessages()) - const messageById = createMemo(() => new Map(visibleUserMessages().map((m) => [m.id, m]))) - const messageIndex = createMemo(() => new Map(visibleUserMessages().map((m, i) => [m.id, i]))) - let pendingKey = "" - let clearing = false - - const location = useLocation() - const navigate = useNavigate() - - const frames = new Set() - const queue = (fn: () => void) => { - const id = requestAnimationFrame(() => { - frames.delete(id) - fn() - }) - frames.add(id) - } - const cancel = () => { - for (const id of frames) cancelAnimationFrame(id) - frames.clear() - } - - const clearMessageHash = () => { - cancel() - input.consumePendingMessage(input.sessionKey()) - if (input.pendingMessage()) input.setPendingMessage(undefined) - if (!location.hash) return - clearing = true - navigate(location.pathname + location.search, { replace: true }) - } - - const updateHash = (id: string) => { - const hash = `#${input.anchor(id)}` - if (location.hash === hash) return - clearing = false - navigate(location.pathname + location.search + hash, { - replace: true, - }) - } - - const scrollToElement = (el: HTMLElement, behavior: ScrollBehavior) => { - const root = input.scroller() - if (!root) return false - - const a = el.getBoundingClientRect() - const b = root.getBoundingClientRect() - const sticky = root.querySelector("[data-session-title]") - const inset = sticky instanceof HTMLElement ? sticky.offsetHeight : 0 - const top = Math.max(0, a.top - b.top + root.scrollTop - inset) - root.scrollTo({ top, behavior }) - return true - } - - const seek = (id: string, behavior: ScrollBehavior, left = 4): boolean => { - const el = document.getElementById(input.anchor(id)) - if (el) return scrollToElement(el, behavior) - if (left <= 0) return false - queue(() => { - seek(id, behavior, left - 1) - }) - return false - } - - const scrollToMessage = (message: UserMessage, behavior: ScrollBehavior = "smooth") => { - cancel() - if (input.currentMessageId() !== message.id) input.setActiveMessage(message) - - const index = messageIndex().get(message.id) ?? -1 - if (index !== -1 && index < input.turnStart()) { - input.setTurnStart(index) - - queue(() => { - seek(message.id, behavior) - }) - - updateHash(message.id) - return - } - - if (seek(message.id, behavior)) { - updateHash(message.id) - return - } - - updateHash(message.id) - } - - const applyHash = (behavior: ScrollBehavior) => { - const hash = location.hash.slice(1) - if (!hash) { - input.autoScroll.forceScrollToBottom() - const el = input.scroller() - if (el) input.scheduleScrollState(el) - return - } - - const messageId = messageIdFromHash(hash) - if (messageId) { - input.autoScroll.pause() - const msg = messageById().get(messageId) - if (msg) { - scrollToMessage(msg, behavior) - return - } - return - } - - const target = document.getElementById(hash) - if (target) { - input.autoScroll.pause() - scrollToElement(target, behavior) - return - } - - input.autoScroll.forceScrollToBottom() - const el = input.scroller() - if (el) input.scheduleScrollState(el) - } - - createEffect(() => { - const hash = location.hash - if (!hash) clearing = false - if (!input.sessionID() || !input.messagesReady()) return - cancel() - queue(() => applyHash("auto")) - }) - - createEffect(() => { - if (!input.sessionID() || !input.messagesReady()) return - - visibleUserMessages() - input.turnStart() - - let targetId = input.pendingMessage() - if (!targetId) { - const key = input.sessionKey() - if (pendingKey !== key) { - pendingKey = key - const next = input.consumePendingMessage(key) - if (next) { - input.setPendingMessage(next) - targetId = next - } - } - } - - if (!targetId && !clearing) targetId = messageIdFromHash(location.hash) - if (!targetId) return - - const pending = input.pendingMessage() === targetId - const msg = messageById().get(targetId) - if (!msg) return - - if (pending) input.setPendingMessage(undefined) - if (input.currentMessageId() === targetId && !pending) return - - input.autoScroll.pause() - cancel() - queue(() => scrollToMessage(msg, "auto")) - }) - - createEffect(() => { - const sessionID = input.sessionID() - if (!sessionID || !input.messagesReady()) return - - visibleUserMessages() - - let targetId = input.pendingMessage() - if (!targetId && !clearing) targetId = messageIdFromHash(location.hash) - if (!targetId) return - if (messageById().has(targetId)) return - if (!input.historyMore() || input.historyLoading()) return - - void input.loadMore(sessionID) - }) - - onMount(() => { - if (typeof window !== "undefined" && "scrollRestoration" in window.history) { - window.history.scrollRestoration = "manual" - } - }) - - onCleanup(cancel) - - return { - clearMessageHash, - scrollToMessage, - applyHash, - } -} diff --git a/packages/app/src/sst-env.d.ts b/packages/app/src/sst-env.d.ts deleted file mode 100644 index 864dce9dc9..0000000000 --- a/packages/app/src/sst-env.d.ts +++ /dev/null @@ -1,10 +0,0 @@ -/* This file is auto-generated by SST. Do not edit. */ -/* tslint:disable */ -/* eslint-disable */ -/* biome-ignore-all lint: auto-generated */ - -/// -interface ImportMetaEnv {} -interface ImportMeta { - readonly env: ImportMetaEnv -} diff --git a/packages/app/src/theme-preload.test.ts b/packages/app/src/theme-preload.test.ts deleted file mode 100644 index 00d7da2394..0000000000 --- a/packages/app/src/theme-preload.test.ts +++ /dev/null @@ -1,46 +0,0 @@ -import { beforeEach, describe, expect, test } from "bun:test" - -const src = await Bun.file(new URL("../public/oc-theme-preload.js", import.meta.url)).text() - -const run = () => Function(src)() - -beforeEach(() => { - document.head.innerHTML = "" - document.documentElement.removeAttribute("data-theme") - document.documentElement.removeAttribute("data-color-scheme") - localStorage.clear() - Object.defineProperty(window, "matchMedia", { - value: () => - ({ - matches: false, - }) as MediaQueryList, - configurable: true, - }) -}) - -describe("theme preload", () => { - test("migrates legacy oc-1 to oc-2 before mount", () => { - localStorage.setItem("opencode-theme-id", "oc-1") - localStorage.setItem("opencode-theme-css-light", "--background-base:#fff;") - localStorage.setItem("opencode-theme-css-dark", "--background-base:#000;") - - run() - - expect(document.documentElement.dataset.theme).toBe("oc-2") - expect(document.documentElement.dataset.colorScheme).toBe("light") - expect(localStorage.getItem("opencode-theme-id")).toBe("oc-2") - expect(localStorage.getItem("opencode-theme-css-light")).toBeNull() - expect(localStorage.getItem("opencode-theme-css-dark")).toBeNull() - expect(document.getElementById("oc-theme-preload")).toBeNull() - }) - - test("keeps cached css for non-default themes", () => { - localStorage.setItem("opencode-theme-id", "nightowl") - localStorage.setItem("opencode-theme-css-light", "--background-base:#fff;") - - run() - - expect(document.documentElement.dataset.theme).toBe("nightowl") - expect(document.getElementById("oc-theme-preload")?.textContent).toContain("--background-base:#fff;") - }) -}) diff --git a/packages/app/src/utils/agent.ts b/packages/app/src/utils/agent.ts deleted file mode 100644 index 59da53af10..0000000000 --- a/packages/app/src/utils/agent.ts +++ /dev/null @@ -1,44 +0,0 @@ -const defaults: Record = { - ask: "var(--icon-agent-ask-base)", - build: "var(--icon-agent-build-base)", - docs: "var(--icon-agent-docs-base)", - plan: "var(--icon-agent-plan-base)", -} - -const palette = [ - "var(--icon-agent-ask-base)", - "var(--icon-agent-build-base)", - "var(--icon-agent-docs-base)", - "var(--icon-agent-plan-base)", - "var(--syntax-info)", - "var(--syntax-success)", - "var(--syntax-warning)", - "var(--syntax-property)", - "var(--syntax-constant)", - "var(--text-diff-add-base)", - "var(--text-diff-delete-base)", - "var(--icon-warning-base)", -] - -function tone(name: string) { - let hash = 0 - for (const char of name) hash = (hash * 31 + char.charCodeAt(0)) >>> 0 - return palette[hash % palette.length] -} - -export function agentColor(name: string, custom?: string) { - if (custom) return custom - return defaults[name] ?? defaults[name.toLowerCase()] ?? tone(name.toLowerCase()) -} - -export function messageAgentColor( - list: readonly { role: string; agent?: string }[] | undefined, - agents: readonly { name: string; color?: string }[], -) { - if (!list) return undefined - for (let i = list.length - 1; i >= 0; i--) { - const item = list[i] - if (item.role !== "user" || !item.agent) continue - return agentColor(item.agent, agents.find((agent) => agent.name === item.agent)?.color) - } -} diff --git a/packages/app/src/utils/aim.ts b/packages/app/src/utils/aim.ts deleted file mode 100644 index 23471959e1..0000000000 --- a/packages/app/src/utils/aim.ts +++ /dev/null @@ -1,138 +0,0 @@ -type Point = { x: number; y: number } - -export function createAim(props: { - enabled: () => boolean - active: () => string | undefined - el: () => HTMLElement | undefined - onActivate: (id: string) => void - delay?: number - max?: number - tolerance?: number - edge?: number -}) { - const state = { - locs: [] as Point[], - timer: undefined as number | undefined, - pending: undefined as string | undefined, - over: undefined as string | undefined, - last: undefined as Point | undefined, - } - - const delay = props.delay ?? 250 - const max = props.max ?? 4 - const tolerance = props.tolerance ?? 80 - const edge = props.edge ?? 18 - - const cancel = () => { - if (state.timer !== undefined) clearTimeout(state.timer) - state.timer = undefined - state.pending = undefined - } - - const reset = () => { - cancel() - state.over = undefined - state.last = undefined - state.locs.length = 0 - } - - const move = (event: MouseEvent) => { - if (!props.enabled()) return - const el = props.el() - if (!el) return - - const rect = el.getBoundingClientRect() - const x = event.clientX - const y = event.clientY - if (x < rect.left || x > rect.right || y < rect.top || y > rect.bottom) return - - state.locs.push({ x, y }) - if (state.locs.length > max) state.locs.shift() - } - - const wait = () => { - if (!props.enabled()) return 0 - if (!props.active()) return 0 - - const el = props.el() - if (!el) return 0 - if (state.locs.length < 2) return 0 - - const rect = el.getBoundingClientRect() - const loc = state.locs[state.locs.length - 1] - if (!loc) return 0 - - const prev = state.locs[0] ?? loc - if (prev.x < rect.left || prev.x > rect.right || prev.y < rect.top || prev.y > rect.bottom) return 0 - if (state.last && loc.x === state.last.x && loc.y === state.last.y) return 0 - - if (rect.right - loc.x <= edge) { - state.last = loc - return delay - } - - const upper = { x: rect.right, y: rect.top - tolerance } - const lower = { x: rect.right, y: rect.bottom + tolerance } - const slope = (a: Point, b: Point) => (b.y - a.y) / (b.x - a.x) - - const decreasing = slope(loc, upper) - const increasing = slope(loc, lower) - const prevDecreasing = slope(prev, upper) - const prevIncreasing = slope(prev, lower) - - if (decreasing < prevDecreasing && increasing > prevIncreasing) { - state.last = loc - return delay - } - - state.last = undefined - return 0 - } - - const activate = (id: string) => { - cancel() - props.onActivate(id) - } - - const request = (id: string) => { - if (!id) return - if (props.active() === id) return - - if (!props.active()) { - activate(id) - return - } - - const ms = wait() - if (ms === 0) { - activate(id) - return - } - - cancel() - state.pending = id - state.timer = window.setTimeout(() => { - state.timer = undefined - if (state.pending !== id) return - state.pending = undefined - if (!props.enabled()) return - if (!props.active()) return - if (state.over !== id) return - props.onActivate(id) - }, ms) - } - - const enter = (id: string, event: MouseEvent) => { - if (!props.enabled()) return - state.over = id - move(event) - request(id) - } - - const leave = (id: string) => { - if (state.over === id) state.over = undefined - if (state.pending === id) cancel() - } - - return { move, enter, leave, activate, request, cancel, reset } -} diff --git a/packages/app/src/utils/comment-note.ts b/packages/app/src/utils/comment-note.ts deleted file mode 100644 index 99e87fc81c..0000000000 --- a/packages/app/src/utils/comment-note.ts +++ /dev/null @@ -1,88 +0,0 @@ -import type { FileSelection } from "@/context/file" - -export type PromptComment = { - path: string - selection?: FileSelection - comment: string - preview?: string - origin?: "review" | "file" -} - -function selection(selection: unknown) { - if (!selection || typeof selection !== "object") return undefined - const startLine = Number((selection as FileSelection).startLine) - const startChar = Number((selection as FileSelection).startChar) - const endLine = Number((selection as FileSelection).endLine) - const endChar = Number((selection as FileSelection).endChar) - if (![startLine, startChar, endLine, endChar].every(Number.isFinite)) return undefined - return { - startLine, - startChar, - endLine, - endChar, - } satisfies FileSelection -} - -export function createCommentMetadata(input: PromptComment) { - return { - opencodeComment: { - path: input.path, - selection: input.selection, - comment: input.comment, - preview: input.preview, - origin: input.origin, - }, - } -} - -export function readCommentMetadata(value: unknown) { - if (!value || typeof value !== "object") return - const meta = (value as { opencodeComment?: unknown }).opencodeComment - if (!meta || typeof meta !== "object") return - const path = (meta as { path?: unknown }).path - const comment = (meta as { comment?: unknown }).comment - if (typeof path !== "string" || typeof comment !== "string") return - const preview = (meta as { preview?: unknown }).preview - const origin = (meta as { origin?: unknown }).origin - return { - path, - selection: selection((meta as { selection?: unknown }).selection), - comment, - preview: typeof preview === "string" ? preview : undefined, - origin: origin === "review" || origin === "file" ? origin : undefined, - } satisfies PromptComment -} - -export function formatCommentNote(input: { path: string; selection?: FileSelection; comment: string }) { - const start = input.selection ? Math.min(input.selection.startLine, input.selection.endLine) : undefined - const end = input.selection ? Math.max(input.selection.startLine, input.selection.endLine) : undefined - const range = - start === undefined || end === undefined - ? "this file" - : start === end - ? `line ${start}` - : `lines ${start} through ${end}` - return `The user made the following comment regarding ${range} of ${input.path}: ${input.comment}` -} - -export function parseCommentNote(text: string) { - const match = text.match( - /^The user made the following comment regarding (this file|line (\d+)|lines (\d+) through (\d+)) of (.+?): ([\s\S]+)$/, - ) - if (!match) return - const start = match[2] ? Number(match[2]) : match[3] ? Number(match[3]) : undefined - const end = match[2] ? Number(match[2]) : match[4] ? Number(match[4]) : undefined - return { - path: match[5], - selection: - start !== undefined && end !== undefined - ? { - startLine: start, - startChar: 0, - endLine: end, - endChar: 0, - } - : undefined, - comment: match[6], - } satisfies PromptComment -} diff --git a/packages/app/src/utils/diffs.test.ts b/packages/app/src/utils/diffs.test.ts deleted file mode 100644 index f3f437ae4f..0000000000 --- a/packages/app/src/utils/diffs.test.ts +++ /dev/null @@ -1,74 +0,0 @@ -import { describe, expect, test } from "bun:test" -import type { SnapshotFileDiff } from "@kilocode/sdk/v2" -import type { Message } from "@kilocode/sdk/v2/client" -import { diffs, message } from "./diffs" - -const item = { - file: "src/app.ts", - patch: "@@ -1 +1 @@\n-old\n+new\n", - additions: 1, - deletions: 1, - status: "modified", -} satisfies SnapshotFileDiff - -describe("diffs", () => { - test("keeps valid arrays", () => { - expect(diffs([item])).toEqual([item]) - }) - - test("wraps a single diff object", () => { - expect(diffs(item)).toEqual([item]) - }) - - test("reads keyed diff objects", () => { - expect(diffs({ a: item })).toEqual([item]) - }) - - test("drops invalid entries", () => { - expect( - diffs([ - item, - { file: "src/bad.ts", additions: 1, deletions: 1 }, - { patch: item.patch, additions: 1, deletions: 1 }, - ]), - ).toEqual([item]) - }) -}) - -describe("message", () => { - test("normalizes user summaries with object diffs", () => { - const input = { - id: "msg_1", - sessionID: "ses_1", - role: "user", - time: { created: 1 }, - agent: "build", - model: { providerID: "openai", modelID: "gpt-5" }, - summary: { - title: "Edit", - diffs: { a: item }, - }, - } as unknown as Message - - expect(message(input)).toMatchObject({ - summary: { - title: "Edit", - diffs: [item], - }, - }) - }) - - test("drops invalid user summaries", () => { - const input = { - id: "msg_1", - sessionID: "ses_1", - role: "user", - time: { created: 1 }, - agent: "build", - model: { providerID: "openai", modelID: "gpt-5" }, - summary: true, - } as unknown as Message - - expect(message(input)).toMatchObject({ summary: undefined }) - }) -}) diff --git a/packages/app/src/utils/diffs.ts b/packages/app/src/utils/diffs.ts deleted file mode 100644 index ccb1c8d3cf..0000000000 --- a/packages/app/src/utils/diffs.ts +++ /dev/null @@ -1,49 +0,0 @@ -import type { SnapshotFileDiff, VcsFileDiff } from "@kilocode/sdk/v2" -import type { Message } from "@kilocode/sdk/v2/client" - -type Diff = SnapshotFileDiff | VcsFileDiff - -function diff(value: unknown): value is Diff { - if (!value || typeof value !== "object" || Array.isArray(value)) return false - if (!("file" in value) || typeof value.file !== "string") return false - if (!("patch" in value) || typeof value.patch !== "string") return false - if (!("additions" in value) || typeof value.additions !== "number") return false - if (!("deletions" in value) || typeof value.deletions !== "number") return false - if (!("status" in value) || value.status === undefined) return true - return value.status === "added" || value.status === "deleted" || value.status === "modified" -} - -function object(value: unknown): value is Record { - return !!value && typeof value === "object" && !Array.isArray(value) -} - -export function diffs(value: unknown): Diff[] { - if (Array.isArray(value) && value.every(diff)) return value - if (Array.isArray(value)) return value.filter(diff) - if (diff(value)) return [value] - if (!object(value)) return [] - return Object.values(value).filter(diff) -} - -export function message(value: Message): Message { - if (value.role !== "user") return value - - const raw = value.summary as unknown - if (raw === undefined) return value - if (!object(raw)) return { ...value, summary: undefined } - - const title = typeof raw.title === "string" ? raw.title : undefined - const body = typeof raw.body === "string" ? raw.body : undefined - const next = diffs(raw.diffs) - - if (title === raw.title && body === raw.body && next === raw.diffs) return value - - return { - ...value, - summary: { - ...(title === undefined ? {} : { title }), - ...(body === undefined ? {} : { body }), - diffs: next, - }, - } -} diff --git a/packages/app/src/utils/id.ts b/packages/app/src/utils/id.ts deleted file mode 100644 index fa27cf4c5f..0000000000 --- a/packages/app/src/utils/id.ts +++ /dev/null @@ -1,99 +0,0 @@ -import z from "zod" - -const prefixes = { - session: "ses", - message: "msg", - permission: "per", - user: "usr", - part: "prt", - pty: "pty", -} as const - -const LENGTH = 26 -let lastTimestamp = 0 -let counter = 0 - -type Prefix = keyof typeof prefixes -export namespace Identifier { - export function schema(prefix: Prefix) { - return z.string().startsWith(prefixes[prefix]) - } - - export function ascending(prefix: Prefix, given?: string) { - return generateID(prefix, false, given) - } - - export function descending(prefix: Prefix, given?: string) { - return generateID(prefix, true, given) - } -} - -function generateID(prefix: Prefix, descending: boolean, given?: string): string { - if (!given) { - return create(prefix, descending) - } - - if (!given.startsWith(prefixes[prefix])) { - throw new Error(`ID ${given} does not start with ${prefixes[prefix]}`) - } - - return given -} - -function create(prefix: Prefix, descending: boolean, timestamp?: number): string { - const currentTimestamp = timestamp ?? Date.now() - - if (currentTimestamp !== lastTimestamp) { - lastTimestamp = currentTimestamp - counter = 0 - } - - counter += 1 - - let now = BigInt(currentTimestamp) * BigInt(0x1000) + BigInt(counter) - - if (descending) { - now = ~now - } - - const timeBytes = new Uint8Array(6) - for (let i = 0; i < 6; i += 1) { - timeBytes[i] = Number((now >> BigInt(40 - 8 * i)) & BigInt(0xff)) - } - - return prefixes[prefix] + "_" + bytesToHex(timeBytes) + randomBase62(LENGTH - 12) -} - -function bytesToHex(bytes: Uint8Array): string { - let hex = "" - for (let i = 0; i < bytes.length; i += 1) { - hex += bytes[i].toString(16).padStart(2, "0") - } - return hex -} - -function randomBase62(length: number): string { - const chars = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz" - const bytes = getRandomBytes(length) - let result = "" - for (let i = 0; i < length; i += 1) { - result += chars[bytes[i] % 62] - } - return result -} - -function getRandomBytes(length: number): Uint8Array { - const bytes = new Uint8Array(length) - const cryptoObj = typeof globalThis !== "undefined" ? globalThis.crypto : undefined - - if (cryptoObj && typeof cryptoObj.getRandomValues === "function") { - cryptoObj.getRandomValues(bytes) - return bytes - } - - for (let i = 0; i < length; i += 1) { - bytes[i] = Math.floor(Math.random() * 256) - } - - return bytes -} diff --git a/packages/app/src/utils/notification-click.test.ts b/packages/app/src/utils/notification-click.test.ts deleted file mode 100644 index fa81b0e025..0000000000 --- a/packages/app/src/utils/notification-click.test.ts +++ /dev/null @@ -1,27 +0,0 @@ -import { afterEach, describe, expect, test } from "bun:test" -import { handleNotificationClick, setNavigate } from "./notification-click" - -describe("notification click", () => { - afterEach(() => { - setNavigate(undefined as any) - }) - - test("navigates via registered navigate function", () => { - const calls: string[] = [] - setNavigate((href) => calls.push(href)) - handleNotificationClick("/abc/session/123") - expect(calls).toEqual(["/abc/session/123"]) - }) - - test("does not navigate when href is missing", () => { - const calls: string[] = [] - setNavigate((href) => calls.push(href)) - handleNotificationClick(undefined) - expect(calls).toEqual([]) - }) - - test("falls back to location.assign without registered navigate", () => { - handleNotificationClick("/abc/session/123") - // falls back to window.location.assign — no error thrown - }) -}) diff --git a/packages/app/src/utils/notification-click.ts b/packages/app/src/utils/notification-click.ts deleted file mode 100644 index 316b278206..0000000000 --- a/packages/app/src/utils/notification-click.ts +++ /dev/null @@ -1,13 +0,0 @@ -let nav: ((href: string) => void) | undefined - -export const setNavigate = (fn: (href: string) => void) => { - nav = fn -} - -export const handleNotificationClick = (href?: string) => { - window.focus() - if (!href) return - if (nav) return nav(href) - console.warn("notification-click: navigate function not set, falling back to window.location.assign") - window.location.assign(href) -} diff --git a/packages/app/src/utils/persist.test.ts b/packages/app/src/utils/persist.test.ts deleted file mode 100644 index 673acd224d..0000000000 --- a/packages/app/src/utils/persist.test.ts +++ /dev/null @@ -1,115 +0,0 @@ -import { beforeAll, beforeEach, describe, expect, mock, test } from "bun:test" - -type PersistTestingType = typeof import("./persist").PersistTesting - -class MemoryStorage implements Storage { - private values = new Map() - readonly events: string[] = [] - readonly calls = { get: 0, set: 0, remove: 0 } - - clear() { - this.values.clear() - } - - get length() { - return this.values.size - } - - key(index: number) { - return Array.from(this.values.keys())[index] ?? null - } - - getItem(key: string) { - this.calls.get += 1 - this.events.push(`get:${key}`) - if (key.startsWith("opencode.throw")) throw new Error("storage get failed") - return this.values.get(key) ?? null - } - - setItem(key: string, value: string) { - this.calls.set += 1 - this.events.push(`set:${key}`) - if (key.startsWith("opencode.quota")) throw new DOMException("quota", "QuotaExceededError") - if (key.startsWith("opencode.throw")) throw new Error("storage set failed") - this.values.set(key, value) - } - - removeItem(key: string) { - this.calls.remove += 1 - this.events.push(`remove:${key}`) - if (key.startsWith("opencode.throw")) throw new Error("storage remove failed") - this.values.delete(key) - } -} - -const storage = new MemoryStorage() - -let persistTesting: PersistTestingType - -beforeAll(async () => { - mock.module("@/context/platform", () => ({ - usePlatform: () => ({ platform: "web" }), - })) - - const mod = await import("./persist") - persistTesting = mod.PersistTesting -}) - -beforeEach(() => { - storage.clear() - storage.events.length = 0 - storage.calls.get = 0 - storage.calls.set = 0 - storage.calls.remove = 0 - Object.defineProperty(globalThis, "localStorage", { - value: storage, - configurable: true, - }) -}) - -describe("persist localStorage resilience", () => { - test("does not cache values as persisted when quota write and eviction fail", () => { - const storageApi = persistTesting.localStorageWithPrefix("opencode.quota.scope") - storageApi.setItem("value", '{"value":1}') - - expect(storage.getItem("opencode.quota.scope:value")).toBeNull() - expect(storageApi.getItem("value")).toBeNull() - }) - - test("disables only the failing scope when storage throws", () => { - const bad = persistTesting.localStorageWithPrefix("opencode.throw.scope") - bad.setItem("value", '{"value":1}') - - const before = storage.calls.set - bad.setItem("value", '{"value":2}') - expect(storage.calls.set).toBe(before) - expect(bad.getItem("value")).toBeNull() - - const healthy = persistTesting.localStorageWithPrefix("opencode.safe.scope") - healthy.setItem("value", '{"value":3}') - expect(storage.getItem("opencode.safe.scope:value")).toBe('{"value":3}') - }) - - test("failing fallback scope does not poison direct storage scope", () => { - const broken = persistTesting.localStorageWithPrefix("opencode.throw.scope2") - broken.setItem("value", '{"value":1}') - - const direct = persistTesting.localStorageDirect() - direct.setItem("direct-value", '{"value":5}') - - expect(storage.getItem("direct-value")).toBe('{"value":5}') - }) - - test("normalizer rejects malformed JSON payloads", () => { - const result = persistTesting.normalize({ value: "ok" }, '{"value":"\\x"}') - expect(result).toBeUndefined() - }) - - test("workspace storage sanitizes Windows filename characters", () => { - const result = persistTesting.workspaceStorage("C:\\Users\\foo") - - expect(result).toStartWith("opencode.workspace.") - expect(result.endsWith(".dat")).toBeTrue() - expect(/[:\\/]/.test(result)).toBeFalse() - }) -}) diff --git a/packages/app/src/utils/prompt.test.ts b/packages/app/src/utils/prompt.test.ts deleted file mode 100644 index 04768dde05..0000000000 --- a/packages/app/src/utils/prompt.test.ts +++ /dev/null @@ -1,44 +0,0 @@ -import { describe, expect, test } from "bun:test" -import type { Part } from "@kilocode/sdk/v2" -import { extractPromptFromParts } from "./prompt" - -describe("extractPromptFromParts", () => { - test("restores multiple uploaded attachments", () => { - const parts = [ - { - id: "text_1", - type: "text", - text: "check these", - sessionID: "ses_1", - messageID: "msg_1", - }, - { - id: "file_1", - type: "file", - mime: "image/png", - url: "data:image/png;base64,AAA", - filename: "a.png", - sessionID: "ses_1", - messageID: "msg_1", - }, - { - id: "file_2", - type: "file", - mime: "application/pdf", - url: "data:application/pdf;base64,BBB", - filename: "b.pdf", - sessionID: "ses_1", - messageID: "msg_1", - }, - ] satisfies Part[] - - const result = extractPromptFromParts(parts) - - expect(result).toHaveLength(3) - expect(result[0]).toMatchObject({ type: "text", content: "check these" }) - expect(result.slice(1)).toMatchObject([ - { type: "image", filename: "a.png", mime: "image/png", dataUrl: "data:image/png;base64,AAA" }, - { type: "image", filename: "b.pdf", mime: "application/pdf", dataUrl: "data:application/pdf;base64,BBB" }, - ]) - }) -}) diff --git a/packages/app/src/utils/prompt.ts b/packages/app/src/utils/prompt.ts deleted file mode 100644 index 2736d0e51a..0000000000 --- a/packages/app/src/utils/prompt.ts +++ /dev/null @@ -1,203 +0,0 @@ -import type { AgentPart as MessageAgentPart, FilePart, Part, TextPart } from "@kilocode/sdk/v2" -import type { AgentPart, FileAttachmentPart, ImageAttachmentPart, Prompt } from "@/context/prompt" - -type Inline = - | { - type: "file" - start: number - end: number - value: string - path: string - selection?: { - startLine: number - endLine: number - startChar: number - endChar: number - } - } - | { - type: "agent" - start: number - end: number - value: string - name: string - } - -function selectionFromFileUrl(url: string): Extract["selection"] { - const queryIndex = url.indexOf("?") - if (queryIndex === -1) return undefined - const params = new URLSearchParams(url.slice(queryIndex + 1)) - const startLine = Number(params.get("start")) - const endLine = Number(params.get("end")) - if (!Number.isFinite(startLine) || !Number.isFinite(endLine)) return undefined - return { - startLine, - endLine, - startChar: 0, - endChar: 0, - } -} - -function textPartValue(parts: Part[]) { - const candidates = parts - .filter((part): part is TextPart => part.type === "text") - .filter((part) => !part.synthetic && !part.ignored) - return candidates.reduce((best: TextPart | undefined, part) => { - if (!best) return part - if (part.text.length > best.text.length) return part - return best - }, undefined) -} - -/** - * Extract prompt content from message parts for restoring into the prompt input. - * This is used by undo to restore the original user prompt. - */ -export function extractPromptFromParts(parts: Part[], opts?: { directory?: string; attachmentName?: string }): Prompt { - const textPart = textPartValue(parts) - const text = textPart?.text ?? "" - const directory = opts?.directory - const attachmentName = opts?.attachmentName ?? "attachment" - - const toRelative = (path: string) => { - if (!directory) return path - - const prefix = directory.endsWith("/") ? directory : directory + "/" - if (path.startsWith(prefix)) return path.slice(prefix.length) - - if (path.startsWith(directory)) { - const next = path.slice(directory.length) - if (next.startsWith("/")) return next.slice(1) - return next - } - - return path - } - - const inline: Inline[] = [] - const images: ImageAttachmentPart[] = [] - - for (const part of parts) { - if (part.type === "file") { - const filePart = part as FilePart - const sourceText = filePart.source?.text - if (sourceText) { - const value = sourceText.value - const start = sourceText.start - const end = sourceText.end - let path = value - if (value.startsWith("@")) path = value.slice(1) - if (!value.startsWith("@") && filePart.source && "path" in filePart.source) { - path = filePart.source.path - } - inline.push({ - type: "file", - start, - end, - value, - path: toRelative(path), - selection: selectionFromFileUrl(filePart.url), - }) - continue - } - - if (filePart.url.startsWith("data:")) { - images.push({ - type: "image", - id: filePart.id, - filename: filePart.filename ?? attachmentName, - mime: filePart.mime, - dataUrl: filePart.url, - }) - } - } - - if (part.type === "agent") { - const agentPart = part as MessageAgentPart - const source = agentPart.source - if (!source) continue - inline.push({ - type: "agent", - start: source.start, - end: source.end, - value: source.value, - name: agentPart.name, - }) - } - } - - inline.sort((a, b) => { - if (a.start !== b.start) return a.start - b.start - return a.end - b.end - }) - - const result: Prompt = [] - let position = 0 - let cursor = 0 - - const pushText = (content: string) => { - if (!content) return - result.push({ - type: "text", - content, - start: position, - end: position + content.length, - }) - position += content.length - } - - const pushFile = (item: Extract) => { - const content = item.value - const attachment: FileAttachmentPart = { - type: "file", - path: item.path, - content, - start: position, - end: position + content.length, - selection: item.selection, - } - result.push(attachment) - position += content.length - } - - const pushAgent = (item: Extract) => { - const content = item.value - const mention: AgentPart = { - type: "agent", - name: item.name, - content, - start: position, - end: position + content.length, - } - result.push(mention) - position += content.length - } - - for (const item of inline) { - if (item.start < 0 || item.end < item.start) continue - - const expected = item.value - if (!expected) continue - - const mismatch = item.end > text.length || item.start < cursor || text.slice(item.start, item.end) !== expected - const start = mismatch ? text.indexOf(expected, cursor) : item.start - if (start === -1) continue - const end = mismatch ? start + expected.length : item.end - - pushText(text.slice(cursor, start)) - - if (item.type === "file") pushFile(item) - if (item.type === "agent") pushAgent(item) - - cursor = end - } - - pushText(text.slice(cursor)) - - if (result.length === 0) { - result.push({ type: "text", content: "", start: 0, end: 0 }) - } - - if (images.length === 0) return result - return [...result, ...images] -} diff --git a/packages/app/src/utils/runtime-adapters.test.ts b/packages/app/src/utils/runtime-adapters.test.ts deleted file mode 100644 index 49552e179c..0000000000 --- a/packages/app/src/utils/runtime-adapters.test.ts +++ /dev/null @@ -1,64 +0,0 @@ -import { describe, expect, test } from "bun:test" -import { - disposeIfDisposable, - getHoveredLinkText, - getSpeechRecognitionCtor, - hasSetOption, - isDisposable, - setOptionIfSupported, -} from "./runtime-adapters" - -describe("runtime adapters", () => { - test("detects and disposes disposable values", () => { - let count = 0 - const value = { - dispose: () => { - count += 1 - }, - } - expect(isDisposable(value)).toBe(true) - disposeIfDisposable(value) - expect(count).toBe(1) - }) - - test("ignores non-disposable values", () => { - expect(isDisposable({ dispose: "nope" })).toBe(false) - expect(() => disposeIfDisposable({ dispose: "nope" })).not.toThrow() - }) - - test("sets options only when setter exists", () => { - const calls: Array<[string, unknown]> = [] - const value = { - setOption: (key: string, next: unknown) => { - calls.push([key, next]) - }, - } - expect(hasSetOption(value)).toBe(true) - setOptionIfSupported(value, "fontFamily", "Berkeley Mono") - expect(calls).toEqual([["fontFamily", "Berkeley Mono"]]) - expect(() => setOptionIfSupported({}, "fontFamily", "Berkeley Mono")).not.toThrow() - }) - - test("reads hovered link text safely", () => { - expect(getHoveredLinkText({ currentHoveredLink: { text: "https://example.com" } })).toBe("https://example.com") - expect(getHoveredLinkText({ currentHoveredLink: { text: 1 } })).toBeUndefined() - expect(getHoveredLinkText(null)).toBeUndefined() - }) - - test("resolves speech recognition constructor with webkit precedence", () => { - // oxlint-disable-next-line no-extraneous-class - class SpeechCtor {} - // oxlint-disable-next-line no-extraneous-class - class WebkitCtor {} - const ctor = getSpeechRecognitionCtor({ - SpeechRecognition: SpeechCtor, - webkitSpeechRecognition: WebkitCtor, - }) - expect(ctor).toBe(WebkitCtor) - }) - - test("returns undefined when no valid speech constructor exists", () => { - expect(getSpeechRecognitionCtor({ SpeechRecognition: "nope" })).toBeUndefined() - expect(getSpeechRecognitionCtor(undefined)).toBeUndefined() - }) -}) diff --git a/packages/app/src/utils/runtime-adapters.ts b/packages/app/src/utils/runtime-adapters.ts deleted file mode 100644 index 4c74da5dc1..0000000000 --- a/packages/app/src/utils/runtime-adapters.ts +++ /dev/null @@ -1,39 +0,0 @@ -type RecordValue = Record - -const isRecord = (value: unknown): value is RecordValue => { - return typeof value === "object" && value !== null -} - -export const isDisposable = (value: unknown): value is { dispose: () => void } => { - return isRecord(value) && typeof value.dispose === "function" -} - -export const disposeIfDisposable = (value: unknown) => { - if (!isDisposable(value)) return - value.dispose() -} - -export const hasSetOption = (value: unknown): value is { setOption: (key: string, next: unknown) => void } => { - return isRecord(value) && typeof value.setOption === "function" -} - -export const setOptionIfSupported = (value: unknown, key: string, next: unknown) => { - if (!hasSetOption(value)) return - value.setOption(key, next) -} - -export const getHoveredLinkText = (value: unknown) => { - if (!isRecord(value)) return - const link = value.currentHoveredLink - if (!isRecord(link)) return - if (typeof link.text !== "string") return - return link.text -} - -export const getSpeechRecognitionCtor = (value: unknown): (new () => T) | undefined => { - if (!isRecord(value)) return - const ctor = - typeof value.webkitSpeechRecognition === "function" ? value.webkitSpeechRecognition : value.SpeechRecognition - if (typeof ctor !== "function") return - return ctor as new () => T -} diff --git a/packages/app/src/utils/same.ts b/packages/app/src/utils/same.ts deleted file mode 100644 index c956f92998..0000000000 --- a/packages/app/src/utils/same.ts +++ /dev/null @@ -1,6 +0,0 @@ -export function same(a: readonly T[] | undefined, b: readonly T[] | undefined) { - if (a === b) return true - if (!a || !b) return false - if (a.length !== b.length) return false - return a.every((x, i) => x === b[i]) -} diff --git a/packages/app/src/utils/scoped-cache.test.ts b/packages/app/src/utils/scoped-cache.test.ts deleted file mode 100644 index 0c6189dafe..0000000000 --- a/packages/app/src/utils/scoped-cache.test.ts +++ /dev/null @@ -1,69 +0,0 @@ -import { describe, expect, test } from "bun:test" -import { createScopedCache } from "./scoped-cache" - -describe("createScopedCache", () => { - test("evicts least-recently-used entry when max is reached", () => { - const disposed: string[] = [] - const cache = createScopedCache((key) => ({ key }), { - maxEntries: 2, - dispose: (value) => disposed.push(value.key), - }) - - const a = cache.get("a") - const b = cache.get("b") - expect(a.key).toBe("a") - expect(b.key).toBe("b") - - cache.get("a") - const c = cache.get("c") - - expect(c.key).toBe("c") - expect(cache.peek("a")?.key).toBe("a") - expect(cache.peek("b")).toBeUndefined() - expect(cache.peek("c")?.key).toBe("c") - expect(disposed).toEqual(["b"]) - }) - - test("disposes entries on delete and clear", () => { - const disposed: string[] = [] - const cache = createScopedCache((key) => ({ key }), { - dispose: (value) => disposed.push(value.key), - }) - - cache.get("a") - cache.get("b") - - const removed = cache.delete("a") - expect(removed?.key).toBe("a") - expect(cache.peek("a")).toBeUndefined() - - cache.clear() - expect(cache.peek("b")).toBeUndefined() - expect(disposed).toEqual(["a", "b"]) - }) - - test("expires stale entries with ttl and recreates on get", () => { - let clock = 0 - let count = 0 - const disposed: string[] = [] - const cache = createScopedCache((key) => ({ key, count: ++count }), { - ttlMs: 10, - now: () => clock, - dispose: (value) => disposed.push(`${value.key}:${value.count}`), - }) - - const first = cache.get("a") - expect(first.count).toBe(1) - - clock = 9 - expect(cache.peek("a")?.count).toBe(1) - - clock = 11 - expect(cache.peek("a")).toBeUndefined() - expect(disposed).toEqual(["a:1"]) - - const second = cache.get("a") - expect(second.count).toBe(2) - expect(disposed).toEqual(["a:1"]) - }) -}) diff --git a/packages/app/src/utils/scoped-cache.ts b/packages/app/src/utils/scoped-cache.ts deleted file mode 100644 index 224c363c1e..0000000000 --- a/packages/app/src/utils/scoped-cache.ts +++ /dev/null @@ -1,104 +0,0 @@ -type ScopedCacheOptions = { - maxEntries?: number - ttlMs?: number - dispose?: (value: T, key: string) => void - now?: () => number -} - -type Entry = { - value: T - touchedAt: number -} - -export function createScopedCache(createValue: (key: string) => T, options: ScopedCacheOptions = {}) { - const store = new Map>() - const now = options.now ?? Date.now - - const dispose = (key: string, entry: Entry) => { - options.dispose?.(entry.value, key) - } - - const expired = (entry: Entry) => { - if (options.ttlMs === undefined) return false - return now() - entry.touchedAt >= options.ttlMs - } - - const sweep = () => { - if (options.ttlMs === undefined) return - for (const [key, entry] of store) { - if (!expired(entry)) continue - store.delete(key) - dispose(key, entry) - } - } - - const touch = (key: string, entry: Entry) => { - entry.touchedAt = now() - store.delete(key) - store.set(key, entry) - } - - const prune = () => { - if (options.maxEntries === undefined) return - while (store.size > options.maxEntries) { - const key = store.keys().next().value - if (!key) return - const entry = store.get(key) - store.delete(key) - if (!entry) continue - dispose(key, entry) - } - } - - const remove = (key: string) => { - const entry = store.get(key) - if (!entry) return - store.delete(key) - dispose(key, entry) - return entry.value - } - - const peek = (key: string) => { - sweep() - const entry = store.get(key) - if (!entry) return - if (!expired(entry)) return entry.value - store.delete(key) - dispose(key, entry) - } - - const get = (key: string) => { - sweep() - const entry = store.get(key) - if (entry && !expired(entry)) { - touch(key, entry) - return entry.value - } - if (entry) { - store.delete(key) - dispose(key, entry) - } - - const created = { - value: createValue(key), - touchedAt: now(), - } - store.set(key, created) - prune() - return created.value - } - - const clear = () => { - for (const [key, entry] of store) { - dispose(key, entry) - } - store.clear() - } - - return { - get, - peek, - delete: remove, - clear, - } -} diff --git a/packages/app/src/utils/server-errors.test.ts b/packages/app/src/utils/server-errors.test.ts deleted file mode 100644 index 1f53bb8cf6..0000000000 --- a/packages/app/src/utils/server-errors.test.ts +++ /dev/null @@ -1,131 +0,0 @@ -import { describe, expect, test } from "bun:test" -import type { ConfigInvalidError, ProviderModelNotFoundError } from "./server-errors" -import { formatServerError, parseReadableConfigInvalidError } from "./server-errors" - -function fill(text: string, vars?: Record) { - if (!vars) return text - return text.replace(/{{\s*(\w+)\s*}}/g, (_, key: string) => { - const value = vars[key] - if (value === undefined) return "" - return String(value) - }) -} - -function useLanguageMock() { - const dict: Record = { - "error.chain.unknown": "Erro desconhecido", - "error.chain.configInvalid": "Arquivo de config em {{path}} invalido", - "error.chain.configInvalidWithMessage": "Arquivo de config em {{path}} invalido: {{message}}", - "error.chain.modelNotFound": "Modelo nao encontrado: {{provider}}/{{model}}", - "error.chain.didYouMean": "Voce quis dizer: {{suggestions}}", - "error.chain.checkConfig": "Revise provider/model no config", - } - return { - t(key: string, vars?: Record) { - const text = dict[key] - if (!text) return key - return fill(text, vars) - }, - } -} - -const language = useLanguageMock() - -describe("parseReadableConfigInvalidError", () => { - test("formats issues with file path", () => { - const error = { - name: "ConfigInvalidError", - data: { - path: "opencode.config.ts", - issues: [ - { path: ["settings", "host"], message: "Required" }, - { path: ["mode"], message: "Invalid" }, - ], - }, - } satisfies ConfigInvalidError - - const result = parseReadableConfigInvalidError(error, language.t) - - expect(result).toBe( - ["Arquivo de config em opencode.config.ts invalido: settings.host: Required", "mode: Invalid"].join("\n"), - ) - }) - - test("uses trimmed message when issues are missing", () => { - const error = { - name: "ConfigInvalidError", - data: { - path: "config", - message: " Bad value ", - }, - } satisfies ConfigInvalidError - - const result = parseReadableConfigInvalidError(error, language.t) - - expect(result).toBe("Arquivo de config em config invalido: Bad value") - }) -}) - -describe("formatServerError", () => { - test("formats config invalid errors", () => { - const error = { - name: "ConfigInvalidError", - data: { - message: "Missing host", - }, - } satisfies ConfigInvalidError - - const result = formatServerError(error, language.t) - - expect(result).toBe("Arquivo de config em config invalido: Missing host") - }) - - test("returns error messages", () => { - expect(formatServerError(new Error("Request failed with status 503"), language.t)).toBe( - "Request failed with status 503", - ) - }) - - test("returns provided string errors", () => { - expect(formatServerError("Failed to connect to server", language.t)).toBe("Failed to connect to server") - }) - - test("uses translated unknown fallback", () => { - expect(formatServerError(0, language.t)).toBe("Erro desconhecido") - }) - - test("falls back for unknown error objects and names", () => { - expect(formatServerError({ name: "ServerTimeoutError", data: { seconds: 30 } }, language.t)).toBe( - "Erro desconhecido", - ) - }) - - test("formats provider model errors using provider/model", () => { - const error = { - name: "ProviderModelNotFoundError", - data: { - providerID: "openai", - modelID: "gpt-4.1", - }, - } satisfies ProviderModelNotFoundError - - expect(formatServerError(error, language.t)).toBe( - ["Modelo nao encontrado: openai/gpt-4.1", "Revise provider/model no config"].join("\n"), - ) - }) - - test("formats provider model suggestions", () => { - const error = { - name: "ProviderModelNotFoundError", - data: { - providerID: "x", - modelID: "y", - suggestions: ["x/y2", "x/y3"], - }, - } satisfies ProviderModelNotFoundError - - expect(formatServerError(error, language.t)).toBe( - ["Modelo nao encontrado: x/y", "Voce quis dizer: x/y2, x/y3", "Revise provider/model no config"].join("\n"), - ) - }) -}) diff --git a/packages/app/src/utils/server-errors.ts b/packages/app/src/utils/server-errors.ts deleted file mode 100644 index 2c3a8c54db..0000000000 --- a/packages/app/src/utils/server-errors.ts +++ /dev/null @@ -1,80 +0,0 @@ -export type ConfigInvalidError = { - name: "ConfigInvalidError" - data: { - path?: string - message?: string - issues?: Array<{ message: string; path: string[] }> - } -} - -export type ProviderModelNotFoundError = { - name: "ProviderModelNotFoundError" - data: { - providerID: string - modelID: string - suggestions?: string[] - } -} - -type Translator = (key: string, vars?: Record) => string - -function tr(translator: Translator | undefined, key: string, text: string, vars?: Record) { - if (!translator) return text - const out = translator(key, vars) - if (!out || out === key) return text - return out -} - -export function formatServerError(error: unknown, translate?: Translator, fallback?: string) { - if (isConfigInvalidErrorLike(error)) return parseReadableConfigInvalidError(error, translate) - if (isProviderModelNotFoundErrorLike(error)) return parseReadableProviderModelNotFoundError(error, translate) - if (error instanceof Error && error.message) return error.message - if (typeof error === "string" && error) return error - if (fallback) return fallback - return tr(translate, "error.chain.unknown", "Unknown error") -} - -function isConfigInvalidErrorLike(error: unknown): error is ConfigInvalidError { - if (typeof error !== "object" || error === null) return false - const o = error as Record - return o.name === "ConfigInvalidError" && typeof o.data === "object" && o.data !== null -} - -function isProviderModelNotFoundErrorLike(error: unknown): error is ProviderModelNotFoundError { - if (typeof error !== "object" || error === null) return false - const o = error as Record - return o.name === "ProviderModelNotFoundError" && typeof o.data === "object" && o.data !== null -} - -export function parseReadableConfigInvalidError(errorInput: ConfigInvalidError, translator?: Translator) { - const file = errorInput.data.path && errorInput.data.path !== "config" ? errorInput.data.path : "config" - const detail = errorInput.data.message?.trim() ?? "" - const issues = (errorInput.data.issues ?? []) - .map((issue) => { - const msg = issue.message.trim() - if (!issue.path.length) return msg - return `${issue.path.join(".")}: ${msg}` - }) - .filter(Boolean) - const msg = issues.length ? issues.join("\n") : detail - if (!msg) return tr(translator, "error.chain.configInvalid", `Config file at ${file} is invalid`, { path: file }) - return tr(translator, "error.chain.configInvalidWithMessage", `Config file at ${file} is invalid: ${msg}`, { - path: file, - message: msg, - }) -} - -function parseReadableProviderModelNotFoundError(errorInput: ProviderModelNotFoundError, translator?: Translator) { - const p = errorInput.data.providerID.trim() - const m = errorInput.data.modelID.trim() - const list = (errorInput.data.suggestions ?? []).map((v) => v.trim()).filter(Boolean) - const body = tr(translator, "error.chain.modelNotFound", `Model not found: ${p}/${m}`, { provider: p, model: m }) - const tail = tr(translator, "error.chain.checkConfig", "Check your config (opencode.json) provider/model names") - if (list.length) { - const suggestions = list.slice(0, 5).join(", ") - return [body, tr(translator, "error.chain.didYouMean", `Did you mean: ${suggestions}`, { suggestions }), tail].join( - "\n", - ) - } - return [body, tail].join("\n") -} diff --git a/packages/app/src/utils/server-health.test.ts b/packages/app/src/utils/server-health.test.ts deleted file mode 100644 index 50082dcf35..0000000000 --- a/packages/app/src/utils/server-health.test.ts +++ /dev/null @@ -1,123 +0,0 @@ -import { describe, expect, test } from "bun:test" -import type { ServerConnection } from "@/context/server" -import { checkServerHealth } from "./server-health" - -const server: ServerConnection.HttpBase = { - url: "http://localhost:4096", -} - -function abortFromInput(input: RequestInfo | URL, init?: RequestInit) { - if (init?.signal) return init.signal - if (input instanceof Request) return input.signal - return undefined -} - -describe("checkServerHealth", () => { - test("returns healthy response with version", async () => { - const fetch = (async () => - new Response(JSON.stringify({ healthy: true, version: "1.2.3" }), { - status: 200, - headers: { "content-type": "application/json" }, - })) as unknown as typeof globalThis.fetch - - const result = await checkServerHealth(server, fetch) - - expect(result).toEqual({ healthy: true, version: "1.2.3" }) - }) - - test("returns unhealthy when request fails", async () => { - const fetch = (async () => { - throw new Error("network") - }) as unknown as typeof globalThis.fetch - - const result = await checkServerHealth(server, fetch) - - expect(result).toEqual({ healthy: false }) - }) - - test("uses timeout fallback when AbortSignal.timeout is unavailable", async () => { - const timeout = Object.getOwnPropertyDescriptor(AbortSignal, "timeout") - Object.defineProperty(AbortSignal, "timeout", { - configurable: true, - value: undefined, - }) - - let aborted = false - const fetch = ((input: RequestInfo | URL, init?: RequestInit) => - new Promise((_resolve, reject) => { - const signal = abortFromInput(input, init) - signal?.addEventListener( - "abort", - () => { - aborted = true - reject(new DOMException("Aborted", "AbortError")) - }, - { once: true }, - ) - })) as unknown as typeof globalThis.fetch - - const result = await checkServerHealth(server, fetch, { - timeoutMs: 10, - }).finally(() => { - if (timeout) Object.defineProperty(AbortSignal, "timeout", timeout) - if (!timeout) Reflect.deleteProperty(AbortSignal, "timeout") - }) - - expect(aborted).toBe(true) - expect(result).toEqual({ healthy: false }) - }) - - test("uses provided abort signal", async () => { - let signal: AbortSignal | undefined - const fetch = (async (input: RequestInfo | URL, init?: RequestInit) => { - signal = abortFromInput(input, init) - return new Response(JSON.stringify({ healthy: true, version: "1.2.3" }), { - status: 200, - headers: { "content-type": "application/json" }, - }) - }) as unknown as typeof globalThis.fetch - - const abort = new AbortController() - await checkServerHealth(server, fetch, { - signal: abort.signal, - }) - - expect(signal).toBe(abort.signal) - }) - - test("retries transient failures and eventually succeeds", async () => { - let count = 0 - const fetch = (async () => { - count += 1 - if (count < 3) throw new TypeError("network") - return new Response(JSON.stringify({ healthy: true, version: "1.2.3" }), { - status: 200, - headers: { "content-type": "application/json" }, - }) - }) as unknown as typeof globalThis.fetch - - const result = await checkServerHealth(server, fetch, { - retryCount: 2, - retryDelayMs: 1, - }) - - expect(count).toBe(3) - expect(result).toEqual({ healthy: true, version: "1.2.3" }) - }) - - test("returns unhealthy when retries are exhausted", async () => { - let count = 0 - const fetch = (async () => { - count += 1 - throw new TypeError("network") - }) as unknown as typeof globalThis.fetch - - const result = await checkServerHealth(server, fetch, { - retryCount: 2, - retryDelayMs: 1, - }) - - expect(count).toBe(3) - expect(result).toEqual({ healthy: false }) - }) -}) diff --git a/packages/app/src/utils/server-health.ts b/packages/app/src/utils/server-health.ts deleted file mode 100644 index a13fd34ef7..0000000000 --- a/packages/app/src/utils/server-health.ts +++ /dev/null @@ -1,113 +0,0 @@ -import { usePlatform } from "@/context/platform" -import type { ServerConnection } from "@/context/server" -import { createSdkForServer } from "./server" - -export type ServerHealth = { healthy: boolean; version?: string } - -interface CheckServerHealthOptions { - timeoutMs?: number - signal?: AbortSignal - retryCount?: number - retryDelayMs?: number -} - -const defaultTimeoutMs = 3000 -const defaultRetryCount = 2 -const defaultRetryDelayMs = 100 -const cacheMs = 750 -const healthCache = new Map< - string, - { at: number; done: boolean; fetch: typeof globalThis.fetch; promise: Promise } ->() - -function cacheKey(server: ServerConnection.HttpBase) { - return `${server.url}\n${server.username ?? ""}\n${server.password ?? ""}` -} - -function timeoutSignal(timeoutMs: number) { - const timeout = (AbortSignal as unknown as { timeout?: (ms: number) => AbortSignal }).timeout - if (timeout) { - try { - return { - signal: timeout.call(AbortSignal, timeoutMs), - clear: undefined as (() => void) | undefined, - } - } catch {} - } - const controller = new AbortController() - const timer = setTimeout(() => controller.abort(), timeoutMs) - return { signal: controller.signal, clear: () => clearTimeout(timer) } -} - -function wait(ms: number, signal?: AbortSignal) { - return new Promise((resolve, reject) => { - if (signal?.aborted) { - reject(new DOMException("Aborted", "AbortError")) - return - } - const timer = setTimeout(() => { - signal?.removeEventListener("abort", onAbort) - resolve() - }, ms) - const onAbort = () => { - clearTimeout(timer) - reject(new DOMException("Aborted", "AbortError")) - } - signal?.addEventListener("abort", onAbort, { once: true }) - }) -} - -function retryable(error: unknown, signal?: AbortSignal) { - if (signal?.aborted) return false - if (!(error instanceof Error)) return false - if (error.name === "AbortError" || error.name === "TimeoutError") return false - if (error instanceof TypeError) return true - return /network|fetch|econnreset|econnrefused|enotfound|timedout/i.test(error.message) -} - -export async function checkServerHealth( - server: ServerConnection.HttpBase, - fetch: typeof globalThis.fetch, - opts?: CheckServerHealthOptions, -): Promise { - const timeout = opts?.signal ? undefined : timeoutSignal(opts?.timeoutMs ?? defaultTimeoutMs) - const signal = opts?.signal ?? timeout?.signal - const retryCount = opts?.retryCount ?? defaultRetryCount - const retryDelayMs = opts?.retryDelayMs ?? defaultRetryDelayMs - const next = (count: number, error: unknown) => { - if (count >= retryCount || !retryable(error, signal)) return Promise.resolve({ healthy: false } as const) - return wait(retryDelayMs * (count + 1), signal) - .then(() => attempt(count + 1)) - .catch(() => ({ healthy: false })) - } - const attempt = (count: number): Promise => - createSdkForServer({ - server, - fetch, - signal, - }) - .global.health() - .then((x) => (x.error ? next(count, x.error) : { healthy: x.data?.healthy === true, version: x.data?.version })) - .catch((error) => next(count, error)) - return attempt(0).finally(() => timeout?.clear?.()) -} - -export function useCheckServerHealth() { - const platform = usePlatform() - const fetcher = platform.fetch ?? globalThis.fetch - - return (http: ServerConnection.HttpBase) => { - const key = cacheKey(http) - const hit = healthCache.get(key) - const now = Date.now() - if (hit && hit.fetch === fetcher && (!hit.done || now - hit.at < cacheMs)) return hit.promise - const promise = checkServerHealth(http, fetcher).finally(() => { - const next = healthCache.get(key) - if (!next || next.promise !== promise) return - next.done = true - next.at = Date.now() - }) - healthCache.set(key, { at: now, done: false, fetch: fetcher, promise }) - return promise - } -} diff --git a/packages/app/src/utils/server.ts b/packages/app/src/utils/server.ts deleted file mode 100644 index 139641882c..0000000000 --- a/packages/app/src/utils/server.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { createKiloClient } from "@kilocode/sdk/v2/client" -import type { ServerConnection } from "@/context/server" - -export function createSdkForServer({ - server, - ...config -}: Omit[0]>, "baseUrl"> & { - server: ServerConnection.HttpBase -}) { - const auth = (() => { - if (!server.password) return - return { - Authorization: `Basic ${btoa(`${server.username ?? "kilo"}:${server.password}`)}`, - } - })() - - return createKiloClient({ - ...config, - headers: { - ...(config.headers instanceof Headers ? Object.fromEntries(config.headers.entries()) : config.headers), - ...auth, - }, - baseUrl: server.url, - }) -} diff --git a/packages/app/src/utils/session-title.ts b/packages/app/src/utils/session-title.ts deleted file mode 100644 index ca04c01047..0000000000 --- a/packages/app/src/utils/session-title.ts +++ /dev/null @@ -1,7 +0,0 @@ -const pattern = /^(New session|Child session) - \d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}\.\d{3}Z$/ - -export function sessionTitle(title?: string) { - if (!title) return title - const match = title.match(pattern) - return match?.[1] ?? title -} diff --git a/packages/app/src/utils/solid-dnd.tsx b/packages/app/src/utils/solid-dnd.tsx deleted file mode 100644 index 8e30a033ae..0000000000 --- a/packages/app/src/utils/solid-dnd.tsx +++ /dev/null @@ -1,49 +0,0 @@ -import { useDragDropContext } from "@thisbeyond/solid-dnd" -import type { Transformer } from "@thisbeyond/solid-dnd" -import { createRoot, onCleanup, type JSXElement } from "solid-js" - -type DragEvent = { draggable?: { id?: unknown } } - -const isDragEvent = (event: unknown): event is DragEvent => { - if (typeof event !== "object" || event === null) return false - return "draggable" in event -} - -export const getDraggableId = (event: unknown): string | undefined => { - if (!isDragEvent(event)) return undefined - const draggable = event.draggable - if (!draggable) return undefined - return typeof draggable.id === "string" ? draggable.id : undefined -} - -const createTransformer = (id: string, axis: "x" | "y"): Transformer => ({ - id, - order: 100, - callback: (transform) => (axis === "x" ? { ...transform, x: 0 } : { ...transform, y: 0 }), -}) - -const createAxisConstraint = (axis: "x" | "y", transformerId: string) => (): JSXElement => { - const context = useDragDropContext() - if (!context) return null - const [, { onDragStart, onDragEnd, addTransformer, removeTransformer }] = context - const transformer = createTransformer(transformerId, axis) - const dispose = createRoot((dispose) => { - onDragStart((event) => { - const id = getDraggableId(event) - if (!id) return - addTransformer("draggables", id, transformer) - }) - onDragEnd((event) => { - const id = getDraggableId(event) - if (!id) return - removeTransformer("draggables", id, transformer.id) - }) - return dispose - }) - onCleanup(dispose) - return null -} - -export const ConstrainDragXAxis = createAxisConstraint("x", "constrain-x-axis") - -export const ConstrainDragYAxis = createAxisConstraint("y", "constrain-y-axis") diff --git a/packages/app/src/utils/sound.ts b/packages/app/src/utils/sound.ts deleted file mode 100644 index 78e5a0c565..0000000000 --- a/packages/app/src/utils/sound.ts +++ /dev/null @@ -1,102 +0,0 @@ -let files: Record Promise> | undefined -let loads: Record Promise> | undefined - -function getFiles() { - if (files) return files - files = import.meta.glob("../../../ui/src/assets/audio/*.aac", { import: "default" }) as Record< - string, - () => Promise - > - return files -} - -export const SOUND_OPTIONS = [ - { id: "alert-01", label: "sound.option.alert01" }, - { id: "alert-02", label: "sound.option.alert02" }, - { id: "alert-03", label: "sound.option.alert03" }, - { id: "alert-04", label: "sound.option.alert04" }, - { id: "alert-05", label: "sound.option.alert05" }, - { id: "alert-06", label: "sound.option.alert06" }, - { id: "alert-07", label: "sound.option.alert07" }, - { id: "alert-08", label: "sound.option.alert08" }, - { id: "alert-09", label: "sound.option.alert09" }, - { id: "alert-10", label: "sound.option.alert10" }, - { id: "bip-bop-01", label: "sound.option.bipbop01" }, - { id: "bip-bop-02", label: "sound.option.bipbop02" }, - { id: "bip-bop-03", label: "sound.option.bipbop03" }, - { id: "bip-bop-04", label: "sound.option.bipbop04" }, - { id: "bip-bop-05", label: "sound.option.bipbop05" }, - { id: "bip-bop-06", label: "sound.option.bipbop06" }, - { id: "bip-bop-07", label: "sound.option.bipbop07" }, - { id: "bip-bop-08", label: "sound.option.bipbop08" }, - { id: "bip-bop-09", label: "sound.option.bipbop09" }, - { id: "bip-bop-10", label: "sound.option.bipbop10" }, - { id: "staplebops-01", label: "sound.option.staplebops01" }, - { id: "staplebops-02", label: "sound.option.staplebops02" }, - { id: "staplebops-03", label: "sound.option.staplebops03" }, - { id: "staplebops-04", label: "sound.option.staplebops04" }, - { id: "staplebops-05", label: "sound.option.staplebops05" }, - { id: "staplebops-06", label: "sound.option.staplebops06" }, - { id: "staplebops-07", label: "sound.option.staplebops07" }, - { id: "nope-01", label: "sound.option.nope01" }, - { id: "nope-02", label: "sound.option.nope02" }, - { id: "nope-03", label: "sound.option.nope03" }, - { id: "nope-04", label: "sound.option.nope04" }, - { id: "nope-05", label: "sound.option.nope05" }, - { id: "nope-06", label: "sound.option.nope06" }, - { id: "nope-07", label: "sound.option.nope07" }, - { id: "nope-08", label: "sound.option.nope08" }, - { id: "nope-09", label: "sound.option.nope09" }, - { id: "nope-10", label: "sound.option.nope10" }, - { id: "nope-11", label: "sound.option.nope11" }, - { id: "nope-12", label: "sound.option.nope12" }, - { id: "yup-01", label: "sound.option.yup01" }, - { id: "yup-02", label: "sound.option.yup02" }, - { id: "yup-03", label: "sound.option.yup03" }, - { id: "yup-04", label: "sound.option.yup04" }, - { id: "yup-05", label: "sound.option.yup05" }, - { id: "yup-06", label: "sound.option.yup06" }, -] as const - -export type SoundOption = (typeof SOUND_OPTIONS)[number] -export type SoundID = SoundOption["id"] - -function getLoads() { - if (loads) return loads - loads = Object.fromEntries( - Object.entries(getFiles()).flatMap(([path, load]) => { - const file = path.split("/").at(-1) - if (!file) return [] - return [[file.replace(/\.aac$/, ""), load] as const] - }), - ) as Record Promise> - return loads -} - -const cache = new Map>() - -export function soundSrc(id: string | undefined) { - const loads = getLoads() - if (!id || !(id in loads)) return Promise.resolve(undefined) - const key = id as SoundID - const hit = cache.get(key) - if (hit) return hit - const next = loads[key]().catch(() => undefined) - cache.set(key, next) - return next -} - -export function playSound(src: string | undefined) { - if (typeof Audio === "undefined") return - if (!src) return - const audio = new Audio(src) - audio.play().catch(() => undefined) - return () => { - audio.pause() - audio.currentTime = 0 - } -} - -export function playSoundById(id: string | undefined) { - return soundSrc(id).then((src) => playSound(src)) -} diff --git a/packages/app/src/utils/terminal-writer.test.ts b/packages/app/src/utils/terminal-writer.test.ts deleted file mode 100644 index c49702e39b..0000000000 --- a/packages/app/src/utils/terminal-writer.test.ts +++ /dev/null @@ -1,64 +0,0 @@ -import { describe, expect, test } from "bun:test" -import { terminalWriter } from "./terminal-writer" - -describe("terminalWriter", () => { - test("buffers and flushes once per schedule", () => { - const calls: string[] = [] - const scheduled: VoidFunction[] = [] - const writer = terminalWriter( - (data, done) => { - calls.push(data) - done?.() - }, - (flush) => scheduled.push(flush), - ) - - writer.push("a") - writer.push("b") - writer.push("c") - - expect(calls).toEqual([]) - expect(scheduled).toHaveLength(1) - - scheduled[0]?.() - expect(calls).toEqual(["abc"]) - }) - - test("flush is a no-op when empty", () => { - const calls: string[] = [] - const writer = terminalWriter( - (data, done) => { - calls.push(data) - done?.() - }, - (flush) => flush(), - ) - writer.flush() - expect(calls).toEqual([]) - }) - - test("flush waits for pending write completion", () => { - const calls: string[] = [] - let done: VoidFunction | undefined - const writer = terminalWriter( - (data, finish) => { - calls.push(data) - done = finish - }, - (flush) => flush(), - ) - - writer.push("a") - - let settled = false - writer.flush(() => { - settled = true - }) - - expect(calls).toEqual(["a"]) - expect(settled).toBe(false) - - done?.() - expect(settled).toBe(true) - }) -}) diff --git a/packages/app/src/utils/terminal-writer.ts b/packages/app/src/utils/terminal-writer.ts deleted file mode 100644 index 083f51de47..0000000000 --- a/packages/app/src/utils/terminal-writer.ts +++ /dev/null @@ -1,65 +0,0 @@ -export function terminalWriter( - write: (data: string, done?: VoidFunction) => void, - schedule: (flush: VoidFunction) => void = queueMicrotask, -) { - let chunks: string[] | undefined - let waits: VoidFunction[] | undefined - let scheduled = false - let writing = false - - const settle = () => { - if (scheduled || writing || chunks?.length) return - const list = waits - if (!list?.length) return - waits = undefined - for (const fn of list) { - fn() - } - } - - const run = () => { - if (writing) return - scheduled = false - const items = chunks - if (!items?.length) { - settle() - return - } - chunks = undefined - writing = true - write(items.join(""), () => { - writing = false - if (chunks?.length) { - if (scheduled) return - scheduled = true - schedule(run) - return - } - settle() - }) - } - - const push = (data: string) => { - if (!data) return - if (chunks) chunks.push(data) - else chunks = [data] - - if (scheduled || writing) return - scheduled = true - schedule(run) - } - - const flush = (done?: VoidFunction) => { - if (!scheduled && !writing && !chunks?.length) { - done?.() - return - } - if (done) { - if (waits) waits.push(done) - else waits = [done] - } - run() - } - - return { push, flush } -} diff --git a/packages/app/src/utils/time.ts b/packages/app/src/utils/time.ts deleted file mode 100644 index d183e10807..0000000000 --- a/packages/app/src/utils/time.ts +++ /dev/null @@ -1,22 +0,0 @@ -type TimeKey = - | "common.time.justNow" - | "common.time.minutesAgo.short" - | "common.time.hoursAgo.short" - | "common.time.daysAgo.short" - -type Translate = (key: TimeKey, params?: Record) => string - -export function getRelativeTime(dateString: string, t: Translate): string { - const date = new Date(dateString) - const now = new Date() - const diffMs = now.getTime() - date.getTime() - const diffSeconds = Math.floor(diffMs / 1000) - const diffMinutes = Math.floor(diffSeconds / 60) - const diffHours = Math.floor(diffMinutes / 60) - const diffDays = Math.floor(diffHours / 24) - - if (diffSeconds < 60) return t("common.time.justNow") - if (diffMinutes < 60) return t("common.time.minutesAgo.short", { count: diffMinutes }) - if (diffHours < 24) return t("common.time.hoursAgo.short", { count: diffHours }) - return t("common.time.daysAgo.short", { count: diffDays }) -} diff --git a/packages/app/src/utils/uuid.test.ts b/packages/app/src/utils/uuid.test.ts deleted file mode 100644 index e6b4e28240..0000000000 --- a/packages/app/src/utils/uuid.test.ts +++ /dev/null @@ -1,78 +0,0 @@ -import { afterEach, describe, expect, test } from "bun:test" -import { uuid } from "./uuid" - -const cryptoDescriptor = Object.getOwnPropertyDescriptor(globalThis, "crypto") -const secureDescriptor = Object.getOwnPropertyDescriptor(globalThis, "isSecureContext") -const randomDescriptor = Object.getOwnPropertyDescriptor(Math, "random") - -const setCrypto = (value: Partial) => { - Object.defineProperty(globalThis, "crypto", { - configurable: true, - value: value as Crypto, - }) -} - -const setSecure = (value: boolean) => { - Object.defineProperty(globalThis, "isSecureContext", { - configurable: true, - value, - }) -} - -const setRandom = (value: () => number) => { - Object.defineProperty(Math, "random", { - configurable: true, - value, - }) -} - -afterEach(() => { - if (cryptoDescriptor) { - Object.defineProperty(globalThis, "crypto", cryptoDescriptor) - } - - if (secureDescriptor) { - Object.defineProperty(globalThis, "isSecureContext", secureDescriptor) - } - - if (!secureDescriptor) { - delete (globalThis as { isSecureContext?: boolean }).isSecureContext - } - - if (randomDescriptor) { - Object.defineProperty(Math, "random", randomDescriptor) - } -}) - -describe("uuid", () => { - test("uses randomUUID in secure contexts", () => { - setCrypto({ randomUUID: () => "00000000-0000-0000-0000-000000000000" }) - setSecure(true) - expect(uuid()).toBe("00000000-0000-0000-0000-000000000000") - }) - - test("falls back in insecure contexts", () => { - setCrypto({ randomUUID: () => "00000000-0000-0000-0000-000000000000" }) - setSecure(false) - setRandom(() => 0.5) - expect(uuid()).toBe("8") - }) - - test("falls back when randomUUID throws", () => { - setCrypto({ - randomUUID: () => { - throw new DOMException("Failed", "OperationError") - }, - }) - setSecure(true) - setRandom(() => 0.5) - expect(uuid()).toBe("8") - }) - - test("falls back when randomUUID is unavailable", () => { - setCrypto({}) - setSecure(true) - setRandom(() => 0.5) - expect(uuid()).toBe("8") - }) -}) diff --git a/packages/app/src/utils/uuid.ts b/packages/app/src/utils/uuid.ts deleted file mode 100644 index 7b964068c8..0000000000 --- a/packages/app/src/utils/uuid.ts +++ /dev/null @@ -1,12 +0,0 @@ -const fallback = () => Math.random().toString(16).slice(2) - -export function uuid() { - const c = globalThis.crypto - if (!c || typeof c.randomUUID !== "function") return fallback() - if (typeof globalThis.isSecureContext === "boolean" && !globalThis.isSecureContext) return fallback() - try { - return c.randomUUID() - } catch { - return fallback() - } -} diff --git a/packages/app/src/utils/worktree.test.ts b/packages/app/src/utils/worktree.test.ts deleted file mode 100644 index 8161e7ad83..0000000000 --- a/packages/app/src/utils/worktree.test.ts +++ /dev/null @@ -1,46 +0,0 @@ -import { describe, expect, test } from "bun:test" -import { Worktree } from "./worktree" - -const dir = (name: string) => `/tmp/opencode-worktree-${name}-${crypto.randomUUID()}` - -describe("Worktree", () => { - test("normalizes trailing slashes", () => { - const key = dir("normalize") - Worktree.ready(`${key}/`) - - expect(Worktree.get(key)).toEqual({ status: "ready" }) - }) - - test("pending does not overwrite a terminal state", () => { - const key = dir("pending") - Worktree.failed(key, "boom") - Worktree.pending(key) - - expect(Worktree.get(key)).toEqual({ status: "failed", message: "boom" }) - }) - - test("wait resolves shared pending waiter when ready", async () => { - const key = dir("wait-ready") - Worktree.pending(key) - - const a = Worktree.wait(key) - const b = Worktree.wait(`${key}/`) - - expect(a).toBe(b) - - Worktree.ready(key) - - expect(await a).toEqual({ status: "ready" }) - expect(await b).toEqual({ status: "ready" }) - }) - - test("wait resolves with failure message", async () => { - const key = dir("wait-failed") - const waiting = Worktree.wait(key) - - Worktree.failed(key, "permission denied") - - expect(await waiting).toEqual({ status: "failed", message: "permission denied" }) - expect(await Worktree.wait(key)).toEqual({ status: "failed", message: "permission denied" }) - }) -}) diff --git a/packages/app/src/utils/worktree.ts b/packages/app/src/utils/worktree.ts deleted file mode 100644 index 581afd5535..0000000000 --- a/packages/app/src/utils/worktree.ts +++ /dev/null @@ -1,73 +0,0 @@ -const normalize = (directory: string) => directory.replace(/[\\/]+$/, "") - -type State = - | { - status: "pending" - } - | { - status: "ready" - } - | { - status: "failed" - message: string - } - -const state = new Map() -const waiters = new Map< - string, - { - promise: Promise - resolve: (state: State) => void - } ->() - -function deferred() { - const box = { resolve: (_: State) => {} } - const promise = new Promise((resolve) => { - box.resolve = resolve - }) - return { promise, resolve: box.resolve } -} - -export const Worktree = { - get(directory: string) { - return state.get(normalize(directory)) - }, - pending(directory: string) { - const key = normalize(directory) - const current = state.get(key) - if (current && current.status !== "pending") return - state.set(key, { status: "pending" }) - }, - ready(directory: string) { - const key = normalize(directory) - const next = { status: "ready" } as const - state.set(key, next) - const waiter = waiters.get(key) - if (!waiter) return - waiters.delete(key) - waiter.resolve(next) - }, - failed(directory: string, message: string) { - const key = normalize(directory) - const next = { status: "failed", message } as const - state.set(key, next) - const waiter = waiters.get(key) - if (!waiter) return - waiters.delete(key) - waiter.resolve(next) - }, - wait(directory: string) { - const key = normalize(directory) - const current = state.get(key) - if (current && current.status !== "pending") return Promise.resolve(current) - - const existing = waiters.get(key) - if (existing) return existing.promise - - const waiter = deferred() - - waiters.set(key, waiter) - return waiter.promise - }, -} diff --git a/packages/app/sst-env.d.ts b/packages/app/sst-env.d.ts deleted file mode 100644 index f25b971455..0000000000 --- a/packages/app/sst-env.d.ts +++ /dev/null @@ -1,10 +0,0 @@ -/* This file is auto-generated by SST. Do not edit. */ -/* tslint:disable */ -/* eslint-disable */ -/* deno-fmt-ignore-file */ -/* biome-ignore-all lint: auto-generated */ - -/// - -import "sst" -export {} diff --git a/packages/app/vite.config.ts b/packages/app/vite.config.ts deleted file mode 100644 index 6a29ae6345..0000000000 --- a/packages/app/vite.config.ts +++ /dev/null @@ -1,15 +0,0 @@ -import { defineConfig } from "vite" -import desktopPlugin from "./vite" - -export default defineConfig({ - plugins: [desktopPlugin] as any, - server: { - host: "0.0.0.0", - allowedHosts: true, - port: 3000, - }, - build: { - target: "esnext", - // sourcemap: true, - }, -}) diff --git a/packages/app/vite.js b/packages/app/vite.js deleted file mode 100644 index bdbf32c091..0000000000 --- a/packages/app/vite.js +++ /dev/null @@ -1,38 +0,0 @@ -import { readFileSync } from "node:fs" -import solidPlugin from "vite-plugin-solid" -import tailwindcss from "@tailwindcss/vite" -import { fileURLToPath } from "url" - -const theme = fileURLToPath(new URL("./public/oc-theme-preload.js", import.meta.url)) - -/** - * @type {import("vite").PluginOption} - */ -export default [ - { - name: "kilo-desktop:config", - config() { - return { - resolve: { - alias: { - "@": fileURLToPath(new URL("./src", import.meta.url)), - }, - }, - worker: { - format: "es", - }, - } - }, - }, - { - name: "opencode-desktop:theme-preload", - transformIndexHtml(html) { - return html.replace( - '', - ``, - ) - }, - }, - tailwindcss(), - solidPlugin(), -] diff --git a/packages/core/package.json b/packages/core/package.json index a2c05b7bad..3fc50a9ce7 100644 --- a/packages/core/package.json +++ b/packages/core/package.json @@ -1,6 +1,6 @@ { "$schema": "https://json.schemastore.org/package.json", - "version": "7.2.34", + "version": "7.2.49", "name": "@opencode-ai/core", "type": "module", "license": "MIT", @@ -47,5 +47,6 @@ }, "overrides": { "drizzle-orm": "catalog:" - } + }, + "peerDependencies": {} } diff --git a/packages/core/src/filesystem.ts b/packages/core/src/filesystem.ts index 8faaf95f76..bc149d586e 100644 --- a/packages/core/src/filesystem.ts +++ b/packages/core/src/filesystem.ts @@ -7,6 +7,28 @@ import { Effect, FileSystem, Layer, Schema, Context } from "effect" import type { PlatformError } from "effect/PlatformError" import { Glob } from "./util/glob" +// kilocode_change start - Windows-resilient mkdir -p. +// fs.mkdir(dir, { recursive: true }) should be idempotent, but on Windows +// with NTFS reparse points (OneDrive), directory junctions, or WSL-served +// paths, libuv can still throw EEXIST. This wrapper catches that specific +// error so callers get the promised directory-exists semantics. +// +// https://github.com/Kilo-Org/kilocode/issues/9618 +// https://github.com/Kilo-Org/kilocode/issues/9755 +function isEexist(err: unknown): boolean { + return typeof err === "object" && err !== null && "code" in err && (err as NodeJS.ErrnoException).code === "EEXIST" +} + +async function mkdirSafe(dir: string): Promise { + try { + await NFS.mkdir(dir, { recursive: true }) + } catch (err: unknown) { + if (isEexist(err)) return + throw err + } +} +// kilocode_change end + export namespace AppFileSystem { export class FileSystemError extends Schema.TaggedErrorClass()("FileSystemError", { method: Schema.String, @@ -84,7 +106,12 @@ export namespace AppFileSystem { }) const ensureDir = Effect.fn("FileSystem.ensureDir")(function* (path: string) { - yield* fs.makeDirectory(path, { recursive: true }) + // kilocode_change start - use mkdirSafe to tolerate Windows EEXIST + yield* Effect.tryPromise({ + try: () => mkdirSafe(path), + catch: (cause) => new FileSystemError({ method: "ensureDir", cause }), + }) + // kilocode_change end }) const writeWithDirs = Effect.fn("FileSystem.writeWithDirs")(function* ( @@ -99,7 +126,12 @@ export namespace AppFileSystem { (e) => e.reason._tag === "NotFound", () => Effect.gen(function* () { - yield* fs.makeDirectory(dirname(path), { recursive: true }) + // kilocode_change start - use mkdirSafe to tolerate Windows EEXIST + yield* Effect.tryPromise({ + try: () => mkdirSafe(dirname(path)), + catch: (cause) => new FileSystemError({ method: "writeWithDirs:mkdir", cause }), + }) + // kilocode_change end yield* write }), ), diff --git a/packages/core/src/global.ts b/packages/core/src/global.ts index 8ebf1cd9b1..74eca1386a 100644 --- a/packages/core/src/global.ts +++ b/packages/core/src/global.ts @@ -1,10 +1,11 @@ import path from "path" -import fs from "fs/promises" import { xdgData, xdgCache, xdgConfig, xdgState } from "xdg-basedir" import os from "os" import { Context, Effect, Layer } from "effect" import { Flock } from "./util/flock" import { markNoIndex } from "./kilocode/spotlight" // kilocode_change +import { ensureRealDir } from "./kilocode/global" // kilocode_change +import { Flag } from "./flag/flag" const app = "czcode" // czcode_change // kilocode_change start @@ -21,6 +22,7 @@ const cache = path.join(clean(xdgCache)!, app) const config = path.join(clean(xdgConfig)!, app) const state = path.join(clean(xdgState)!, app) // kilocode_change end +const tmp = path.join(os.tmpdir(), app) const paths = { // Allow override via CZCODE_TEST_HOME for test isolation @@ -33,6 +35,7 @@ const paths = { cache, config, state, + tmp, } export const Path = paths @@ -40,11 +43,12 @@ export const Path = paths Flock.setGlobal({ state }) await Promise.all([ - fs.mkdir(Path.data, { recursive: true }), - fs.mkdir(Path.config, { recursive: true }), - fs.mkdir(Path.state, { recursive: true }), - fs.mkdir(Path.log, { recursive: true }), - fs.mkdir(Path.bin, { recursive: true }), + ensureRealDir(Path.data), // kilocode_change + ensureRealDir(Path.config), // kilocode_change + ensureRealDir(Path.state), // kilocode_change + ensureRealDir(Path.tmp), // kilocode_change + ensureRealDir(Path.log), // kilocode_change + ensureRealDir(Path.bin), // kilocode_change ]) // kilocode_change start - keep generated Kilo data out of macOS Spotlight @@ -59,23 +63,34 @@ export interface Interface { readonly cache: string readonly config: string readonly state: string + readonly tmp: string readonly bin: string readonly log: string } +export function make(input: Partial = {}): Interface { + return { + home: Path.home, + data: Path.data, + cache: Path.cache, + config: Flag.KILO_CONFIG_DIR ?? Path.config, + state: Path.state, + tmp: Path.tmp, + bin: Path.bin, + log: Path.log, + ...input, + } +} + export const layer = Layer.effect( Service, - Effect.gen(function* () { - return Service.of({ - home: Path.home, - data: Path.data, - cache: Path.cache, - config: Path.config, - state: Path.state, - bin: Path.bin, - log: Path.log, - }) - }), + Effect.sync(() => Service.of(make())), ) +export const layerWith = (input: Partial) => + Layer.effect( + Service, + Effect.sync(() => Service.of(make(input))), + ) + export * as Global from "./global" diff --git a/packages/core/src/kilocode/global.ts b/packages/core/src/kilocode/global.ts new file mode 100644 index 0000000000..eb5b5d06b5 --- /dev/null +++ b/packages/core/src/kilocode/global.ts @@ -0,0 +1,20 @@ +import fs from "fs/promises" + +/** + * Like `fs.mkdir({ recursive: true })` but also repairs broken symlinks and + * junctions whose target no longer exists (a Windows edge-case where the user + * had a junction at e.g. `~/.kilocode` pointing to a deleted directory). + * + * `fs.mkdir({ recursive: true })` silently no-ops when a junction exists even + * if its target is gone, so subsequent writes inside that path fail with ENOENT. + * We detect this by calling `fs.stat` (which follows the symlink/junction) after + * mkdir: if stat fails the entry is broken and we remove + recreate it. + */ +export async function ensureRealDir(p: string) { + await fs.mkdir(p, { recursive: true }) + const ok = await fs.stat(p).then(() => true).catch(() => false) + if (!ok) { + await fs.rm(p, { force: true }) + await fs.mkdir(p, { recursive: true }) + } +} diff --git a/packages/core/src/npm-config.ts b/packages/core/src/npm-config.ts index 896bb84872..3525be7f2f 100644 --- a/packages/core/src/npm-config.ts +++ b/packages/core/src/npm-config.ts @@ -7,7 +7,7 @@ import Config from "@npmcli/config" import { definitions, flatten, nerfDarts, shorthands } from "@npmcli/config/lib/definitions/index.js" import { Effect } from "effect" -const npmPath = fileURLToPath(new URL("..", import.meta.url)) +const npmPath = fileURLToPath(new URL("..", import.meta.url) as unknown as string) // czcode_change - fix URL type mismatch export const load = (dir: string) => Effect.tryPromise({ diff --git a/packages/core/src/npm.ts b/packages/core/src/npm.ts index 92e4042768..8dac8faf01 100644 --- a/packages/core/src/npm.ts +++ b/packages/core/src/npm.ts @@ -120,13 +120,17 @@ export const layer = Layer.effect( } })() - if (yield* afs.existsSafe(dir)) { + if (yield* afs.existsSafe(path.join(dir, "node_modules", name))) { return resolveEntryPoint(name, path.join(dir, "node_modules", name)) } const tree = yield* reify({ dir, add: [pkg] }) const first = tree.edgesOut.values().next().value?.to - if (!first) return yield* new InstallFailedError({ add: [pkg], dir }) + if (!first) { + const result = resolveEntryPoint(name, path.join(dir, "node_modules", name)) + if (Option.isSome(result.entrypoint)) return result + return yield* new InstallFailedError({ add: [pkg], dir }) + } return resolveEntryPoint(first.name, first.path) }, Effect.scoped) diff --git a/packages/core/test/fixture/effect-flock-worker.ts b/packages/core/test/fixture/effect-flock-worker.ts index 3dc3ee2c8b..c442a62cf5 100644 --- a/packages/core/test/fixture/effect-flock-worker.ts +++ b/packages/core/test/fixture/effect-flock-worker.ts @@ -18,20 +18,17 @@ function sleep(ms: number) { return new Promise((resolve) => setTimeout(resolve, ms)) } -const msg: Msg = JSON.parse(process.argv[2]!) - -const testGlobal = Layer.succeed( - Global.Service, - Global.Service.of({ - home: os.homedir(), - data: os.tmpdir(), - cache: os.tmpdir(), - config: os.tmpdir(), - state: os.tmpdir(), - bin: os.tmpdir(), - log: os.tmpdir(), - }), -) +const msg: Msg = JSON.parse(process.argv[2]) + +const testGlobal = Global.layerWith({ + home: os.homedir(), + data: os.tmpdir(), + cache: os.tmpdir(), + config: os.tmpdir(), + state: os.tmpdir(), + bin: os.tmpdir(), + log: os.tmpdir(), +}) const testLayer = EffectFlock.layer.pipe(Layer.provide(testGlobal), Layer.provide(AppFileSystem.defaultLayer)) diff --git a/packages/core/test/global.test.ts b/packages/core/test/global.test.ts new file mode 100644 index 0000000000..4e13e88424 --- /dev/null +++ b/packages/core/test/global.test.ts @@ -0,0 +1,16 @@ +import { describe, expect, test } from "bun:test" +import fs from "fs/promises" +import os from "os" +import path from "path" +import { Global } from "@opencode-ai/core/global" + +describe("global paths", () => { + test("tmp path is under the system temp directory", () => { + expect(Global.Path.tmp).toBe(path.join(os.tmpdir(), "opencode")) + expect(Global.make().tmp).toBe(Global.Path.tmp) + }) + + test("tmp path is created on module load", async () => { + expect((await fs.stat(Global.Path.tmp)).isDirectory()).toBe(true) + }) +}) diff --git a/packages/core/test/npm.test.ts b/packages/core/test/npm.test.ts index 3e94a08692..3d0767aaff 100644 --- a/packages/core/test/npm.test.ts +++ b/packages/core/test/npm.test.ts @@ -1,7 +1,12 @@ import fs from "fs/promises" import path from "path" import { describe, expect, test } from "bun:test" +import { NodeFileSystem } from "@effect/platform-node" +import { Effect, Layer, Option } from "effect" +import { AppFileSystem } from "@opencode-ai/core/filesystem" +import { Global } from "@opencode-ai/core/global" import { Npm } from "@opencode-ai/core/npm" +import { EffectFlock } from "@opencode-ai/core/util/effect-flock" import { tmpdir } from "./fixture/tmpdir" const win = process.platform === "win32" @@ -15,6 +20,14 @@ const writePackage = (dir: string, pkg: Record) => }), ) +const npmLayer = (cache: string) => + Npm.layer.pipe( + Layer.provide(EffectFlock.layer), + Layer.provide(AppFileSystem.layer), + Layer.provide(Global.layerWith({ cache, state: path.join(cache, "state") })), + Layer.provide(NodeFileSystem.layer), + ) + describe("Npm.sanitize", () => { test("keeps normal scoped package specs unchanged", () => { expect(Npm.sanitize("@opencode/acme")).toBe("@opencode/acme") @@ -29,6 +42,28 @@ describe("Npm.sanitize", () => { }) }) +describe("Npm.add", () => { + test("reifies when package cache directory exists without the package installed", async () => { + await using tmp = await tmpdir() + await fs.mkdir(path.join(tmp.path, "fixture-provider")) + await writePackage(path.join(tmp.path, "fixture-provider"), { + name: "fixture-provider", + main: "index.js", + }) + await Bun.write(path.join(tmp.path, "fixture-provider", "index.js"), "export const fixture = true\n") + + const spec = `fixture-provider@file:${path.join(tmp.path, "fixture-provider")}` + await fs.mkdir(path.join(tmp.path, "cache", "packages", Npm.sanitize(spec)), { recursive: true }) + + const entry = await Effect.gen(function* () { + const npm = yield* Npm.Service + return yield* npm.add(spec) + }).pipe(Effect.scoped, Effect.provide(npmLayer(path.join(tmp.path, "cache"))), Effect.runPromise) + + expect(Option.isSome(entry.entrypoint)).toBe(true) + }) +}) + describe("Npm.install", () => { test("respects omit from project .npmrc", async () => { await using tmp = await tmpdir() diff --git a/packages/core/test/util/effect-flock.test.ts b/packages/core/test/util/effect-flock.test.ts index 9e8bc24ace..76cee4f8e0 100644 --- a/packages/core/test/util/effect-flock.test.ts +++ b/packages/core/test/util/effect-flock.test.ts @@ -93,18 +93,15 @@ async function waitForFile(file: string, timeout = 3_000) { // Test layer // --------------------------------------------------------------------------- -const testGlobal = Layer.succeed( - Global.Service, - Global.Service.of({ - home: os.homedir(), - data: os.tmpdir(), - cache: os.tmpdir(), - config: os.tmpdir(), - state: os.tmpdir(), - bin: os.tmpdir(), - log: os.tmpdir(), - }), -) +const testGlobal = Global.layerWith({ + home: os.homedir(), + data: os.tmpdir(), + cache: os.tmpdir(), + config: os.tmpdir(), + state: os.tmpdir(), + bin: os.tmpdir(), + log: os.tmpdir(), +}) const testLayer = EffectFlock.layer.pipe(Layer.provide(testGlobal), Layer.provide(AppFileSystem.defaultLayer)) diff --git a/packages/core/tsconfig.json b/packages/core/tsconfig.json index 160c631887..4bb417f5b4 100644 --- a/packages/core/tsconfig.json +++ b/packages/core/tsconfig.json @@ -3,13 +3,6 @@ "extends": "@tsconfig/bun/tsconfig.json", "compilerOptions": { "noUncheckedIndexedAccess": false, - "types": ["bun"], - "plugins": [ - { - "name": "@effect/language-service", - "transform": "@effect/language-service/transform", - "namespaceImportPackages": ["effect", "@effect/*"] - } - ] + "types": ["bun"] } } diff --git a/packages/desktop-electron/.gitignore b/packages/desktop-electron/.gitignore deleted file mode 100644 index ac9d8db969..0000000000 --- a/packages/desktop-electron/.gitignore +++ /dev/null @@ -1,28 +0,0 @@ -# Logs -logs -*.log -npm-debug.log* -yarn-debug.log* -yarn-error.log* -pnpm-debug.log* -lerna-debug.log* - -node_modules -dist -dist-ssr -*.local - -# Editor directories and files -.vscode/* -!.vscode/extensions.json -.idea -.DS_Store -*.suo -*.ntvs* -*.njsproj -*.sln -*.sw? -out/ - -resources/opencode-cli* -resources/icons diff --git a/packages/desktop-electron/AGENTS.md b/packages/desktop-electron/AGENTS.md deleted file mode 100644 index 7805ea835f..0000000000 --- a/packages/desktop-electron/AGENTS.md +++ /dev/null @@ -1,4 +0,0 @@ -# Desktop package notes - -- Renderer process should only call `window.api` from `src/preload`. -- Main process should register IPC handlers in `src/main/ipc.ts`. diff --git a/packages/desktop-electron/README.md b/packages/desktop-electron/README.md deleted file mode 100644 index ebaf488223..0000000000 --- a/packages/desktop-electron/README.md +++ /dev/null @@ -1,32 +0,0 @@ -# OpenCode Desktop - -Native OpenCode desktop app, built with Tauri v2. - -## Development - -From the repo root: - -```bash -bun install -bun run --cwd packages/desktop tauri dev -``` - -This starts the Vite dev server on http://localhost:1420 and opens the native window. - -If you only want the web dev server (no native shell): - -```bash -bun run --cwd packages/desktop dev -``` - -## Build - -To create a production `dist/` and build the native app bundle: - -```bash -bun run --cwd packages/desktop tauri build -``` - -## Prerequisites - -Running the desktop app requires additional Tauri dependencies (Rust toolchain, platform-specific libraries). See the [Tauri prerequisites](https://v2.tauri.app/start/prerequisites/) for setup instructions. diff --git a/packages/desktop-electron/electron-builder.config.ts b/packages/desktop-electron/electron-builder.config.ts deleted file mode 100644 index 5eff37547f..0000000000 --- a/packages/desktop-electron/electron-builder.config.ts +++ /dev/null @@ -1,115 +0,0 @@ -import { execFile } from "node:child_process" -import path from "node:path" -import { fileURLToPath } from "node:url" -import { promisify } from "node:util" - -import type { Configuration } from "electron-builder" - -const execFileAsync = promisify(execFile) -const rootDir = path.resolve(path.dirname(fileURLToPath(import.meta.url)), "../..") -const signScript = path.join(rootDir, "script", "sign-windows.ps1") - -async function signWindows(configuration: { path: string }) { - if (process.platform !== "win32") return - if (process.env.GITHUB_ACTIONS !== "true") return - - await execFileAsync( - "pwsh", - ["-NoLogo", "-NoProfile", "-ExecutionPolicy", "Bypass", "-File", signScript, configuration.path], - { cwd: rootDir }, - ) -} - -const channel = (() => { - const raw = process.env.KILO_CHANNEL - if (raw === "dev" || raw === "beta" || raw === "prod") return raw - return "dev" -})() - -const getBase = (): Configuration => ({ - artifactName: "opencode-electron-${os}-${arch}.${ext}", - directories: { - output: "dist", - buildResources: "resources", - }, - files: ["out/**/*", "resources/**/*"], - extraResources: [ - { - from: "native/", - to: "native/", - filter: ["index.js", "index.d.ts", "build/Release/mac_window.node", "swift-build/**"], - }, - ], - mac: { - category: "public.app-category.developer-tools", - icon: `resources/icons/icon.icns`, - hardenedRuntime: true, - gatekeeperAssess: false, - entitlements: "resources/entitlements.plist", - entitlementsInherit: "resources/entitlements.plist", - notarize: true, - target: ["dmg", "zip"], - }, - dmg: { - sign: true, - }, - protocols: { - name: "OpenCode", - schemes: ["opencode"], - }, - win: { - icon: `resources/icons/icon.ico`, - signtoolOptions: { - sign: signWindows, - }, - target: ["nsis"], - }, - nsis: { - oneClick: false, - allowToChangeInstallationDirectory: true, - installerIcon: `resources/icons/icon.ico`, - installerHeaderIcon: `resources/icons/icon.ico`, - }, - linux: { - icon: `resources/icons`, - category: "Development", - target: ["AppImage", "deb", "rpm"], - }, -}) - -function getConfig() { - const base = getBase() - - switch (channel) { - case "dev": { - return { - ...base, - appId: "ai.opencode.desktop.dev", - productName: "OpenCode Dev", - rpm: { packageName: "opencode-dev" }, - } - } - case "beta": { - return { - ...base, - appId: "ai.opencode.desktop.beta", - productName: "OpenCode Beta", - protocols: { name: "OpenCode Beta", schemes: ["opencode"] }, - publish: { provider: "github", owner: "anomalyco", repo: "opencode-beta", channel: "latest" }, - rpm: { packageName: "opencode-beta" }, - } - } - case "prod": { - return { - ...base, - appId: "ai.opencode.desktop", - productName: "OpenCode", - protocols: { name: "OpenCode", schemes: ["opencode"] }, - publish: { provider: "github", owner: "anomalyco", repo: "opencode", channel: "latest" }, - rpm: { packageName: "opencode" }, - } - } - } -} - -export default getConfig() diff --git a/packages/desktop-electron/electron.vite.config.ts b/packages/desktop-electron/electron.vite.config.ts deleted file mode 100644 index 22eedeada6..0000000000 --- a/packages/desktop-electron/electron.vite.config.ts +++ /dev/null @@ -1,79 +0,0 @@ -import { defineConfig } from "electron-vite" -import appPlugin from "@opencode-ai/app/vite" -import * as fs from "node:fs/promises" - -const channel = (() => { - const raw = process.env.KILO_CHANNEL - if (raw === "dev" || raw === "beta" || raw === "prod") return raw - return "dev" -})() - -const KILO_SERVER_DIST = "../opencode/dist/node" - -const nodePtyPkg = `@lydell/node-pty-${process.platform}-${process.arch}` - -export default defineConfig({ - main: { - define: { - "import.meta.env.KILO_CHANNEL": JSON.stringify(channel), - }, - build: { - rollupOptions: { - input: { index: "src/main/index.ts" }, - }, - externalizeDeps: { include: [nodePtyPkg] }, - }, - plugins: [ - { - name: "opencode:node-pty-narrower", - enforce: "pre", - resolveId(s) { - if (s === "@lydell/node-pty") return nodePtyPkg - }, - }, - { - name: "opencode:virtual-server-module", - enforce: "pre", - resolveId(id) { - if (id === "virtual:opencode-server") return this.resolve(`${KILO_SERVER_DIST}/node.js`) - }, - }, - { - name: "opencode:copy-server-assets", - async writeBundle() { - for (const l of await fs.readdir(KILO_SERVER_DIST)) { - if (!l.endsWith(".wasm")) continue - await fs.writeFile(`./out/main/chunks/${l}`, await fs.readFile(`${KILO_SERVER_DIST}/${l}`)) - } - }, - }, - ], - }, - preload: { - build: { - rollupOptions: { - input: { index: "src/preload/index.ts" }, - output: { - format: "cjs", - entryFileNames: "[name].js", - }, - }, - }, - }, - renderer: { - plugins: [appPlugin], - publicDir: "../../../app/public", - root: "src/renderer", - define: { - "import.meta.env.VITE_KILO_CHANNEL": JSON.stringify(channel), - }, - build: { - rollupOptions: { - input: { - main: "src/renderer/index.html", - loading: "src/renderer/loading.html", - }, - }, - }, - }, -}) diff --git a/packages/desktop-electron/icons/README.md b/packages/desktop-electron/icons/README.md deleted file mode 100644 index cf2f8e24c5..0000000000 --- a/packages/desktop-electron/icons/README.md +++ /dev/null @@ -1,14 +0,0 @@ -# Tauri Icons - -Here's the process I've been using to create icons: - -- Save source image as `app-icon.png` in `packages/desktop` -- `cd` to `packages/desktop` -- Run `bun tauri icon -o src-tauri/icons/{environment}` -- Use [Image2Icon](https://img2icnsapp.com/)'s 'Big Sur Icon' preset to generate an `icon.icns` file and place it in the appropriate icons folder - -The Image2Icon step is necessary as the `icon.icns` generated by `app-icon.png` does not apply the shadow/padding expected by macOS, -so app icons appear larger than expected. - -For unpackaged Electron on macOS, `app.dock.setIcon()` should use a PNG. Keep `dock.png` in each channel folder synced with the -extracted `icon_128x128@2x.png` from that channel's `icon.icns` so the dev Dock icon matches the packaged app inset. diff --git a/packages/desktop-electron/icons/beta/128x128.png b/packages/desktop-electron/icons/beta/128x128.png deleted file mode 100644 index 751e80f1fd..0000000000 Binary files a/packages/desktop-electron/icons/beta/128x128.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/128x128@2x.png b/packages/desktop-electron/icons/beta/128x128@2x.png deleted file mode 100644 index fe330df419..0000000000 Binary files a/packages/desktop-electron/icons/beta/128x128@2x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/32x32.png b/packages/desktop-electron/icons/beta/32x32.png deleted file mode 100644 index 2703048eed..0000000000 Binary files a/packages/desktop-electron/icons/beta/32x32.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/64x64.png b/packages/desktop-electron/icons/beta/64x64.png deleted file mode 100644 index ecd7fe3142..0000000000 Binary files a/packages/desktop-electron/icons/beta/64x64.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/Square107x107Logo.png b/packages/desktop-electron/icons/beta/Square107x107Logo.png deleted file mode 100644 index e6ea73f4da..0000000000 Binary files a/packages/desktop-electron/icons/beta/Square107x107Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/Square142x142Logo.png b/packages/desktop-electron/icons/beta/Square142x142Logo.png deleted file mode 100644 index 74ae729c42..0000000000 Binary files a/packages/desktop-electron/icons/beta/Square142x142Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/Square150x150Logo.png b/packages/desktop-electron/icons/beta/Square150x150Logo.png deleted file mode 100644 index 0b109b8f4a..0000000000 Binary files a/packages/desktop-electron/icons/beta/Square150x150Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/Square284x284Logo.png b/packages/desktop-electron/icons/beta/Square284x284Logo.png deleted file mode 100644 index 0261ded42c..0000000000 Binary files a/packages/desktop-electron/icons/beta/Square284x284Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/Square30x30Logo.png b/packages/desktop-electron/icons/beta/Square30x30Logo.png deleted file mode 100644 index 34158f10a4..0000000000 Binary files a/packages/desktop-electron/icons/beta/Square30x30Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/Square310x310Logo.png b/packages/desktop-electron/icons/beta/Square310x310Logo.png deleted file mode 100644 index f18bfada4c..0000000000 Binary files a/packages/desktop-electron/icons/beta/Square310x310Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/Square44x44Logo.png b/packages/desktop-electron/icons/beta/Square44x44Logo.png deleted file mode 100644 index 6d1cc06c08..0000000000 Binary files a/packages/desktop-electron/icons/beta/Square44x44Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/Square71x71Logo.png b/packages/desktop-electron/icons/beta/Square71x71Logo.png deleted file mode 100644 index a26084dc2f..0000000000 Binary files a/packages/desktop-electron/icons/beta/Square71x71Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/Square89x89Logo.png b/packages/desktop-electron/icons/beta/Square89x89Logo.png deleted file mode 100644 index 58b0eb6053..0000000000 Binary files a/packages/desktop-electron/icons/beta/Square89x89Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/StoreLogo.png b/packages/desktop-electron/icons/beta/StoreLogo.png deleted file mode 100644 index 648fd2114d..0000000000 Binary files a/packages/desktop-electron/icons/beta/StoreLogo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/android/mipmap-anydpi-v26/ic_launcher.xml b/packages/desktop-electron/icons/beta/android/mipmap-anydpi-v26/ic_launcher.xml deleted file mode 100644 index 2ffbf24b68..0000000000 --- a/packages/desktop-electron/icons/beta/android/mipmap-anydpi-v26/ic_launcher.xml +++ /dev/null @@ -1,5 +0,0 @@ - - - - - \ No newline at end of file diff --git a/packages/desktop-electron/icons/beta/android/mipmap-hdpi/ic_launcher.png b/packages/desktop-electron/icons/beta/android/mipmap-hdpi/ic_launcher.png deleted file mode 100644 index 39d1dd0d51..0000000000 Binary files a/packages/desktop-electron/icons/beta/android/mipmap-hdpi/ic_launcher.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/android/mipmap-hdpi/ic_launcher_foreground.png b/packages/desktop-electron/icons/beta/android/mipmap-hdpi/ic_launcher_foreground.png deleted file mode 100644 index 84908e71c1..0000000000 Binary files a/packages/desktop-electron/icons/beta/android/mipmap-hdpi/ic_launcher_foreground.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/android/mipmap-hdpi/ic_launcher_round.png b/packages/desktop-electron/icons/beta/android/mipmap-hdpi/ic_launcher_round.png deleted file mode 100644 index a6b8cb6162..0000000000 Binary files a/packages/desktop-electron/icons/beta/android/mipmap-hdpi/ic_launcher_round.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/android/mipmap-mdpi/ic_launcher.png b/packages/desktop-electron/icons/beta/android/mipmap-mdpi/ic_launcher.png deleted file mode 100644 index 6522e0fba8..0000000000 Binary files a/packages/desktop-electron/icons/beta/android/mipmap-mdpi/ic_launcher.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/android/mipmap-mdpi/ic_launcher_foreground.png b/packages/desktop-electron/icons/beta/android/mipmap-mdpi/ic_launcher_foreground.png deleted file mode 100644 index b3449bd4f3..0000000000 Binary files a/packages/desktop-electron/icons/beta/android/mipmap-mdpi/ic_launcher_foreground.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/android/mipmap-mdpi/ic_launcher_round.png b/packages/desktop-electron/icons/beta/android/mipmap-mdpi/ic_launcher_round.png deleted file mode 100644 index 7aa97d8276..0000000000 Binary files a/packages/desktop-electron/icons/beta/android/mipmap-mdpi/ic_launcher_round.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/android/mipmap-xhdpi/ic_launcher.png b/packages/desktop-electron/icons/beta/android/mipmap-xhdpi/ic_launcher.png deleted file mode 100644 index 82bc9d22a6..0000000000 Binary files a/packages/desktop-electron/icons/beta/android/mipmap-xhdpi/ic_launcher.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/android/mipmap-xhdpi/ic_launcher_foreground.png b/packages/desktop-electron/icons/beta/android/mipmap-xhdpi/ic_launcher_foreground.png deleted file mode 100644 index 6b031ce851..0000000000 Binary files a/packages/desktop-electron/icons/beta/android/mipmap-xhdpi/ic_launcher_foreground.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/android/mipmap-xhdpi/ic_launcher_round.png b/packages/desktop-electron/icons/beta/android/mipmap-xhdpi/ic_launcher_round.png deleted file mode 100644 index 34859de5ef..0000000000 Binary files a/packages/desktop-electron/icons/beta/android/mipmap-xhdpi/ic_launcher_round.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/android/mipmap-xxhdpi/ic_launcher.png b/packages/desktop-electron/icons/beta/android/mipmap-xxhdpi/ic_launcher.png deleted file mode 100644 index 4cdb71d62b..0000000000 Binary files a/packages/desktop-electron/icons/beta/android/mipmap-xxhdpi/ic_launcher.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/android/mipmap-xxhdpi/ic_launcher_foreground.png b/packages/desktop-electron/icons/beta/android/mipmap-xxhdpi/ic_launcher_foreground.png deleted file mode 100644 index a64be6ada1..0000000000 Binary files a/packages/desktop-electron/icons/beta/android/mipmap-xxhdpi/ic_launcher_foreground.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/android/mipmap-xxhdpi/ic_launcher_round.png b/packages/desktop-electron/icons/beta/android/mipmap-xxhdpi/ic_launcher_round.png deleted file mode 100644 index 2de3c27342..0000000000 Binary files a/packages/desktop-electron/icons/beta/android/mipmap-xxhdpi/ic_launcher_round.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/android/mipmap-xxxhdpi/ic_launcher.png b/packages/desktop-electron/icons/beta/android/mipmap-xxxhdpi/ic_launcher.png deleted file mode 100644 index 0ead288664..0000000000 Binary files a/packages/desktop-electron/icons/beta/android/mipmap-xxxhdpi/ic_launcher.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/android/mipmap-xxxhdpi/ic_launcher_foreground.png b/packages/desktop-electron/icons/beta/android/mipmap-xxxhdpi/ic_launcher_foreground.png deleted file mode 100644 index bdd1748258..0000000000 Binary files a/packages/desktop-electron/icons/beta/android/mipmap-xxxhdpi/ic_launcher_foreground.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/android/mipmap-xxxhdpi/ic_launcher_round.png b/packages/desktop-electron/icons/beta/android/mipmap-xxxhdpi/ic_launcher_round.png deleted file mode 100644 index 69f74758ec..0000000000 Binary files a/packages/desktop-electron/icons/beta/android/mipmap-xxxhdpi/ic_launcher_round.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/android/values/ic_launcher_background.xml b/packages/desktop-electron/icons/beta/android/values/ic_launcher_background.xml deleted file mode 100644 index ea9c223a6c..0000000000 --- a/packages/desktop-electron/icons/beta/android/values/ic_launcher_background.xml +++ /dev/null @@ -1,4 +0,0 @@ - - - #fff - \ No newline at end of file diff --git a/packages/desktop-electron/icons/beta/dock.png b/packages/desktop-electron/icons/beta/dock.png deleted file mode 100644 index f274ef6459..0000000000 Binary files a/packages/desktop-electron/icons/beta/dock.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/icon.icns b/packages/desktop-electron/icons/beta/icon.icns deleted file mode 100644 index f98de5da88..0000000000 Binary files a/packages/desktop-electron/icons/beta/icon.icns and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/icon.ico b/packages/desktop-electron/icons/beta/icon.ico deleted file mode 100644 index df8588c8e4..0000000000 Binary files a/packages/desktop-electron/icons/beta/icon.ico and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/icon.png b/packages/desktop-electron/icons/beta/icon.png deleted file mode 100644 index 5313049562..0000000000 Binary files a/packages/desktop-electron/icons/beta/icon.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/ios/AppIcon-20x20@1x.png b/packages/desktop-electron/icons/beta/ios/AppIcon-20x20@1x.png deleted file mode 100644 index e8ebb28efe..0000000000 Binary files a/packages/desktop-electron/icons/beta/ios/AppIcon-20x20@1x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/ios/AppIcon-20x20@2x-1.png b/packages/desktop-electron/icons/beta/ios/AppIcon-20x20@2x-1.png deleted file mode 100644 index 50c8015dea..0000000000 Binary files a/packages/desktop-electron/icons/beta/ios/AppIcon-20x20@2x-1.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/ios/AppIcon-20x20@2x.png b/packages/desktop-electron/icons/beta/ios/AppIcon-20x20@2x.png deleted file mode 100644 index 50c8015dea..0000000000 Binary files a/packages/desktop-electron/icons/beta/ios/AppIcon-20x20@2x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/ios/AppIcon-20x20@3x.png b/packages/desktop-electron/icons/beta/ios/AppIcon-20x20@3x.png deleted file mode 100644 index 6e290dbc68..0000000000 Binary files a/packages/desktop-electron/icons/beta/ios/AppIcon-20x20@3x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/ios/AppIcon-29x29@1x.png b/packages/desktop-electron/icons/beta/ios/AppIcon-29x29@1x.png deleted file mode 100644 index 4ef554b4de..0000000000 Binary files a/packages/desktop-electron/icons/beta/ios/AppIcon-29x29@1x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/ios/AppIcon-29x29@2x-1.png b/packages/desktop-electron/icons/beta/ios/AppIcon-29x29@2x-1.png deleted file mode 100644 index b9ddfd47c8..0000000000 Binary files a/packages/desktop-electron/icons/beta/ios/AppIcon-29x29@2x-1.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/ios/AppIcon-29x29@2x.png b/packages/desktop-electron/icons/beta/ios/AppIcon-29x29@2x.png deleted file mode 100644 index b9ddfd47c8..0000000000 Binary files a/packages/desktop-electron/icons/beta/ios/AppIcon-29x29@2x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/ios/AppIcon-29x29@3x.png b/packages/desktop-electron/icons/beta/ios/AppIcon-29x29@3x.png deleted file mode 100644 index 052322d682..0000000000 Binary files a/packages/desktop-electron/icons/beta/ios/AppIcon-29x29@3x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/ios/AppIcon-40x40@1x.png b/packages/desktop-electron/icons/beta/ios/AppIcon-40x40@1x.png deleted file mode 100644 index 50c8015dea..0000000000 Binary files a/packages/desktop-electron/icons/beta/ios/AppIcon-40x40@1x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/ios/AppIcon-40x40@2x-1.png b/packages/desktop-electron/icons/beta/ios/AppIcon-40x40@2x-1.png deleted file mode 100644 index 9317b25001..0000000000 Binary files a/packages/desktop-electron/icons/beta/ios/AppIcon-40x40@2x-1.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/ios/AppIcon-40x40@2x.png b/packages/desktop-electron/icons/beta/ios/AppIcon-40x40@2x.png deleted file mode 100644 index 9317b25001..0000000000 Binary files a/packages/desktop-electron/icons/beta/ios/AppIcon-40x40@2x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/ios/AppIcon-40x40@3x.png b/packages/desktop-electron/icons/beta/ios/AppIcon-40x40@3x.png deleted file mode 100644 index 6b921a17e3..0000000000 Binary files a/packages/desktop-electron/icons/beta/ios/AppIcon-40x40@3x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/ios/AppIcon-512@2x.png b/packages/desktop-electron/icons/beta/ios/AppIcon-512@2x.png deleted file mode 100644 index b83131d64b..0000000000 Binary files a/packages/desktop-electron/icons/beta/ios/AppIcon-512@2x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/ios/AppIcon-60x60@2x.png b/packages/desktop-electron/icons/beta/ios/AppIcon-60x60@2x.png deleted file mode 100644 index 6b921a17e3..0000000000 Binary files a/packages/desktop-electron/icons/beta/ios/AppIcon-60x60@2x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/ios/AppIcon-60x60@3x.png b/packages/desktop-electron/icons/beta/ios/AppIcon-60x60@3x.png deleted file mode 100644 index 685004995c..0000000000 Binary files a/packages/desktop-electron/icons/beta/ios/AppIcon-60x60@3x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/ios/AppIcon-76x76@1x.png b/packages/desktop-electron/icons/beta/ios/AppIcon-76x76@1x.png deleted file mode 100644 index 1ffceb752a..0000000000 Binary files a/packages/desktop-electron/icons/beta/ios/AppIcon-76x76@1x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/ios/AppIcon-76x76@2x.png b/packages/desktop-electron/icons/beta/ios/AppIcon-76x76@2x.png deleted file mode 100644 index 81c4178c91..0000000000 Binary files a/packages/desktop-electron/icons/beta/ios/AppIcon-76x76@2x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/beta/ios/AppIcon-83.5x83.5@2x.png b/packages/desktop-electron/icons/beta/ios/AppIcon-83.5x83.5@2x.png deleted file mode 100644 index d5453adffb..0000000000 Binary files a/packages/desktop-electron/icons/beta/ios/AppIcon-83.5x83.5@2x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/128x128.png b/packages/desktop-electron/icons/dev/128x128.png deleted file mode 100644 index d7fc4db149..0000000000 Binary files a/packages/desktop-electron/icons/dev/128x128.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/128x128@2x.png b/packages/desktop-electron/icons/dev/128x128@2x.png deleted file mode 100644 index 5918823064..0000000000 Binary files a/packages/desktop-electron/icons/dev/128x128@2x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/32x32.png b/packages/desktop-electron/icons/dev/32x32.png deleted file mode 100644 index 53925cc4f5..0000000000 Binary files a/packages/desktop-electron/icons/dev/32x32.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/64x64.png b/packages/desktop-electron/icons/dev/64x64.png deleted file mode 100644 index a88ef15c64..0000000000 Binary files a/packages/desktop-electron/icons/dev/64x64.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/Square107x107Logo.png b/packages/desktop-electron/icons/dev/Square107x107Logo.png deleted file mode 100644 index 0de29ec828..0000000000 Binary files a/packages/desktop-electron/icons/dev/Square107x107Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/Square142x142Logo.png b/packages/desktop-electron/icons/dev/Square142x142Logo.png deleted file mode 100644 index af62e8e1e9..0000000000 Binary files a/packages/desktop-electron/icons/dev/Square142x142Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/Square150x150Logo.png b/packages/desktop-electron/icons/dev/Square150x150Logo.png deleted file mode 100644 index 2b19dc39cc..0000000000 Binary files a/packages/desktop-electron/icons/dev/Square150x150Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/Square284x284Logo.png b/packages/desktop-electron/icons/dev/Square284x284Logo.png deleted file mode 100644 index eda6d9901f..0000000000 Binary files a/packages/desktop-electron/icons/dev/Square284x284Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/Square30x30Logo.png b/packages/desktop-electron/icons/dev/Square30x30Logo.png deleted file mode 100644 index dad821ba84..0000000000 Binary files a/packages/desktop-electron/icons/dev/Square30x30Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/Square310x310Logo.png b/packages/desktop-electron/icons/dev/Square310x310Logo.png deleted file mode 100644 index 555b3b1979..0000000000 Binary files a/packages/desktop-electron/icons/dev/Square310x310Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/Square44x44Logo.png b/packages/desktop-electron/icons/dev/Square44x44Logo.png deleted file mode 100644 index 9f8ad001f7..0000000000 Binary files a/packages/desktop-electron/icons/dev/Square44x44Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/Square71x71Logo.png b/packages/desktop-electron/icons/dev/Square71x71Logo.png deleted file mode 100644 index 43feb78488..0000000000 Binary files a/packages/desktop-electron/icons/dev/Square71x71Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/Square89x89Logo.png b/packages/desktop-electron/icons/dev/Square89x89Logo.png deleted file mode 100644 index 628cc597f0..0000000000 Binary files a/packages/desktop-electron/icons/dev/Square89x89Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/StoreLogo.png b/packages/desktop-electron/icons/dev/StoreLogo.png deleted file mode 100644 index 8d3aa53cff..0000000000 Binary files a/packages/desktop-electron/icons/dev/StoreLogo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/android/mipmap-anydpi-v26/ic_launcher.xml b/packages/desktop-electron/icons/dev/android/mipmap-anydpi-v26/ic_launcher.xml deleted file mode 100644 index 2ffbf24b68..0000000000 --- a/packages/desktop-electron/icons/dev/android/mipmap-anydpi-v26/ic_launcher.xml +++ /dev/null @@ -1,5 +0,0 @@ - - - - - \ No newline at end of file diff --git a/packages/desktop-electron/icons/dev/android/mipmap-hdpi/ic_launcher.png b/packages/desktop-electron/icons/dev/android/mipmap-hdpi/ic_launcher.png deleted file mode 100644 index b355e37fea..0000000000 Binary files a/packages/desktop-electron/icons/dev/android/mipmap-hdpi/ic_launcher.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/android/mipmap-hdpi/ic_launcher_foreground.png b/packages/desktop-electron/icons/dev/android/mipmap-hdpi/ic_launcher_foreground.png deleted file mode 100644 index c33f8713bc..0000000000 Binary files a/packages/desktop-electron/icons/dev/android/mipmap-hdpi/ic_launcher_foreground.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/android/mipmap-hdpi/ic_launcher_round.png b/packages/desktop-electron/icons/dev/android/mipmap-hdpi/ic_launcher_round.png deleted file mode 100644 index 04e37aa654..0000000000 Binary files a/packages/desktop-electron/icons/dev/android/mipmap-hdpi/ic_launcher_round.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/android/mipmap-mdpi/ic_launcher.png b/packages/desktop-electron/icons/dev/android/mipmap-mdpi/ic_launcher.png deleted file mode 100644 index 98e53cd220..0000000000 Binary files a/packages/desktop-electron/icons/dev/android/mipmap-mdpi/ic_launcher.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/android/mipmap-mdpi/ic_launcher_foreground.png b/packages/desktop-electron/icons/dev/android/mipmap-mdpi/ic_launcher_foreground.png deleted file mode 100644 index 40fe6e3786..0000000000 Binary files a/packages/desktop-electron/icons/dev/android/mipmap-mdpi/ic_launcher_foreground.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/android/mipmap-mdpi/ic_launcher_round.png b/packages/desktop-electron/icons/dev/android/mipmap-mdpi/ic_launcher_round.png deleted file mode 100644 index 4814f1ddf5..0000000000 Binary files a/packages/desktop-electron/icons/dev/android/mipmap-mdpi/ic_launcher_round.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/android/mipmap-xhdpi/ic_launcher.png b/packages/desktop-electron/icons/dev/android/mipmap-xhdpi/ic_launcher.png deleted file mode 100644 index 608493283e..0000000000 Binary files a/packages/desktop-electron/icons/dev/android/mipmap-xhdpi/ic_launcher.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/android/mipmap-xhdpi/ic_launcher_foreground.png b/packages/desktop-electron/icons/dev/android/mipmap-xhdpi/ic_launcher_foreground.png deleted file mode 100644 index 898066a3fc..0000000000 Binary files a/packages/desktop-electron/icons/dev/android/mipmap-xhdpi/ic_launcher_foreground.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/android/mipmap-xhdpi/ic_launcher_round.png b/packages/desktop-electron/icons/dev/android/mipmap-xhdpi/ic_launcher_round.png deleted file mode 100644 index 64035c0f3c..0000000000 Binary files a/packages/desktop-electron/icons/dev/android/mipmap-xhdpi/ic_launcher_round.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/android/mipmap-xxhdpi/ic_launcher.png b/packages/desktop-electron/icons/dev/android/mipmap-xxhdpi/ic_launcher.png deleted file mode 100644 index f47691bf42..0000000000 Binary files a/packages/desktop-electron/icons/dev/android/mipmap-xxhdpi/ic_launcher.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/android/mipmap-xxhdpi/ic_launcher_foreground.png b/packages/desktop-electron/icons/dev/android/mipmap-xxhdpi/ic_launcher_foreground.png deleted file mode 100644 index dba6f5635b..0000000000 Binary files a/packages/desktop-electron/icons/dev/android/mipmap-xxhdpi/ic_launcher_foreground.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/android/mipmap-xxhdpi/ic_launcher_round.png b/packages/desktop-electron/icons/dev/android/mipmap-xxhdpi/ic_launcher_round.png deleted file mode 100644 index 764702604e..0000000000 Binary files a/packages/desktop-electron/icons/dev/android/mipmap-xxhdpi/ic_launcher_round.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/android/mipmap-xxxhdpi/ic_launcher.png b/packages/desktop-electron/icons/dev/android/mipmap-xxxhdpi/ic_launcher.png deleted file mode 100644 index 2e8430a604..0000000000 Binary files a/packages/desktop-electron/icons/dev/android/mipmap-xxxhdpi/ic_launcher.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/android/mipmap-xxxhdpi/ic_launcher_foreground.png b/packages/desktop-electron/icons/dev/android/mipmap-xxxhdpi/ic_launcher_foreground.png deleted file mode 100644 index db953d128c..0000000000 Binary files a/packages/desktop-electron/icons/dev/android/mipmap-xxxhdpi/ic_launcher_foreground.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/android/mipmap-xxxhdpi/ic_launcher_round.png b/packages/desktop-electron/icons/dev/android/mipmap-xxxhdpi/ic_launcher_round.png deleted file mode 100644 index d5c9ba6a8d..0000000000 Binary files a/packages/desktop-electron/icons/dev/android/mipmap-xxxhdpi/ic_launcher_round.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/android/values/ic_launcher_background.xml b/packages/desktop-electron/icons/dev/android/values/ic_launcher_background.xml deleted file mode 100644 index ea9c223a6c..0000000000 --- a/packages/desktop-electron/icons/dev/android/values/ic_launcher_background.xml +++ /dev/null @@ -1,4 +0,0 @@ - - - #fff - \ No newline at end of file diff --git a/packages/desktop-electron/icons/dev/dock.png b/packages/desktop-electron/icons/dev/dock.png deleted file mode 100644 index 4953d5531e..0000000000 Binary files a/packages/desktop-electron/icons/dev/dock.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/icon.icns b/packages/desktop-electron/icons/dev/icon.icns deleted file mode 100644 index d73a94904a..0000000000 Binary files a/packages/desktop-electron/icons/dev/icon.icns and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/icon.ico b/packages/desktop-electron/icons/dev/icon.ico deleted file mode 100644 index bec385d9aa..0000000000 Binary files a/packages/desktop-electron/icons/dev/icon.ico and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/icon.png b/packages/desktop-electron/icons/dev/icon.png deleted file mode 100644 index 6de37ea294..0000000000 Binary files a/packages/desktop-electron/icons/dev/icon.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/ios/AppIcon-20x20@1x.png b/packages/desktop-electron/icons/dev/ios/AppIcon-20x20@1x.png deleted file mode 100644 index 0e823043e7..0000000000 Binary files a/packages/desktop-electron/icons/dev/ios/AppIcon-20x20@1x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/ios/AppIcon-20x20@2x-1.png b/packages/desktop-electron/icons/dev/ios/AppIcon-20x20@2x-1.png deleted file mode 100644 index 54e4b2aaca..0000000000 Binary files a/packages/desktop-electron/icons/dev/ios/AppIcon-20x20@2x-1.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/ios/AppIcon-20x20@2x.png b/packages/desktop-electron/icons/dev/ios/AppIcon-20x20@2x.png deleted file mode 100644 index 54e4b2aaca..0000000000 Binary files a/packages/desktop-electron/icons/dev/ios/AppIcon-20x20@2x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/ios/AppIcon-20x20@3x.png b/packages/desktop-electron/icons/dev/ios/AppIcon-20x20@3x.png deleted file mode 100644 index 645b01561a..0000000000 Binary files a/packages/desktop-electron/icons/dev/ios/AppIcon-20x20@3x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/ios/AppIcon-29x29@1x.png b/packages/desktop-electron/icons/dev/ios/AppIcon-29x29@1x.png deleted file mode 100644 index 054225c6e9..0000000000 Binary files a/packages/desktop-electron/icons/dev/ios/AppIcon-29x29@1x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/ios/AppIcon-29x29@2x-1.png b/packages/desktop-electron/icons/dev/ios/AppIcon-29x29@2x-1.png deleted file mode 100644 index 0b1b2e0b7f..0000000000 Binary files a/packages/desktop-electron/icons/dev/ios/AppIcon-29x29@2x-1.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/ios/AppIcon-29x29@2x.png b/packages/desktop-electron/icons/dev/ios/AppIcon-29x29@2x.png deleted file mode 100644 index 0b1b2e0b7f..0000000000 Binary files a/packages/desktop-electron/icons/dev/ios/AppIcon-29x29@2x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/ios/AppIcon-29x29@3x.png b/packages/desktop-electron/icons/dev/ios/AppIcon-29x29@3x.png deleted file mode 100644 index d2c42592b3..0000000000 Binary files a/packages/desktop-electron/icons/dev/ios/AppIcon-29x29@3x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/ios/AppIcon-40x40@1x.png b/packages/desktop-electron/icons/dev/ios/AppIcon-40x40@1x.png deleted file mode 100644 index 54e4b2aaca..0000000000 Binary files a/packages/desktop-electron/icons/dev/ios/AppIcon-40x40@1x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/ios/AppIcon-40x40@2x-1.png b/packages/desktop-electron/icons/dev/ios/AppIcon-40x40@2x-1.png deleted file mode 100644 index 471ed2eec3..0000000000 Binary files a/packages/desktop-electron/icons/dev/ios/AppIcon-40x40@2x-1.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/ios/AppIcon-40x40@2x.png b/packages/desktop-electron/icons/dev/ios/AppIcon-40x40@2x.png deleted file mode 100644 index 471ed2eec3..0000000000 Binary files a/packages/desktop-electron/icons/dev/ios/AppIcon-40x40@2x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/ios/AppIcon-40x40@3x.png b/packages/desktop-electron/icons/dev/ios/AppIcon-40x40@3x.png deleted file mode 100644 index 1a490cbf16..0000000000 Binary files a/packages/desktop-electron/icons/dev/ios/AppIcon-40x40@3x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/ios/AppIcon-512@2x.png b/packages/desktop-electron/icons/dev/ios/AppIcon-512@2x.png deleted file mode 100644 index f53b404e5f..0000000000 Binary files a/packages/desktop-electron/icons/dev/ios/AppIcon-512@2x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/ios/AppIcon-60x60@2x.png b/packages/desktop-electron/icons/dev/ios/AppIcon-60x60@2x.png deleted file mode 100644 index 1a490cbf16..0000000000 Binary files a/packages/desktop-electron/icons/dev/ios/AppIcon-60x60@2x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/ios/AppIcon-60x60@3x.png b/packages/desktop-electron/icons/dev/ios/AppIcon-60x60@3x.png deleted file mode 100644 index bdc759eefe..0000000000 Binary files a/packages/desktop-electron/icons/dev/ios/AppIcon-60x60@3x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/ios/AppIcon-76x76@1x.png b/packages/desktop-electron/icons/dev/ios/AppIcon-76x76@1x.png deleted file mode 100644 index d22096a2df..0000000000 Binary files a/packages/desktop-electron/icons/dev/ios/AppIcon-76x76@1x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/ios/AppIcon-76x76@2x.png b/packages/desktop-electron/icons/dev/ios/AppIcon-76x76@2x.png deleted file mode 100644 index d675773d17..0000000000 Binary files a/packages/desktop-electron/icons/dev/ios/AppIcon-76x76@2x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/dev/ios/AppIcon-83.5x83.5@2x.png b/packages/desktop-electron/icons/dev/ios/AppIcon-83.5x83.5@2x.png deleted file mode 100644 index 31698afce2..0000000000 Binary files a/packages/desktop-electron/icons/dev/ios/AppIcon-83.5x83.5@2x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/128x128.png b/packages/desktop-electron/icons/prod/128x128.png deleted file mode 100644 index caf7b02eb3..0000000000 Binary files a/packages/desktop-electron/icons/prod/128x128.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/128x128@2x.png b/packages/desktop-electron/icons/prod/128x128@2x.png deleted file mode 100644 index 47fe4c61ea..0000000000 Binary files a/packages/desktop-electron/icons/prod/128x128@2x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/32x32.png b/packages/desktop-electron/icons/prod/32x32.png deleted file mode 100644 index 5868bcc933..0000000000 Binary files a/packages/desktop-electron/icons/prod/32x32.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/64x64.png b/packages/desktop-electron/icons/prod/64x64.png deleted file mode 100644 index 1ed7425d85..0000000000 Binary files a/packages/desktop-electron/icons/prod/64x64.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/Square107x107Logo.png b/packages/desktop-electron/icons/prod/Square107x107Logo.png deleted file mode 100644 index 1db249bf72..0000000000 Binary files a/packages/desktop-electron/icons/prod/Square107x107Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/Square142x142Logo.png b/packages/desktop-electron/icons/prod/Square142x142Logo.png deleted file mode 100644 index 1961c34081..0000000000 Binary files a/packages/desktop-electron/icons/prod/Square142x142Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/Square150x150Logo.png b/packages/desktop-electron/icons/prod/Square150x150Logo.png deleted file mode 100644 index abc507347e..0000000000 Binary files a/packages/desktop-electron/icons/prod/Square150x150Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/Square284x284Logo.png b/packages/desktop-electron/icons/prod/Square284x284Logo.png deleted file mode 100644 index 51e2a1b9fe..0000000000 Binary files a/packages/desktop-electron/icons/prod/Square284x284Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/Square30x30Logo.png b/packages/desktop-electron/icons/prod/Square30x30Logo.png deleted file mode 100644 index 066a1fd0c8..0000000000 Binary files a/packages/desktop-electron/icons/prod/Square30x30Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/Square310x310Logo.png b/packages/desktop-electron/icons/prod/Square310x310Logo.png deleted file mode 100644 index 2a85c8e952..0000000000 Binary files a/packages/desktop-electron/icons/prod/Square310x310Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/Square44x44Logo.png b/packages/desktop-electron/icons/prod/Square44x44Logo.png deleted file mode 100644 index c855b80632..0000000000 Binary files a/packages/desktop-electron/icons/prod/Square44x44Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/Square71x71Logo.png b/packages/desktop-electron/icons/prod/Square71x71Logo.png deleted file mode 100644 index c8168f7111..0000000000 Binary files a/packages/desktop-electron/icons/prod/Square71x71Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/Square89x89Logo.png b/packages/desktop-electron/icons/prod/Square89x89Logo.png deleted file mode 100644 index 19ec1777de..0000000000 Binary files a/packages/desktop-electron/icons/prod/Square89x89Logo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/StoreLogo.png b/packages/desktop-electron/icons/prod/StoreLogo.png deleted file mode 100644 index 3fd053d349..0000000000 Binary files a/packages/desktop-electron/icons/prod/StoreLogo.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/android/mipmap-anydpi-v26/ic_launcher.xml b/packages/desktop-electron/icons/prod/android/mipmap-anydpi-v26/ic_launcher.xml deleted file mode 100644 index 2ffbf24b68..0000000000 --- a/packages/desktop-electron/icons/prod/android/mipmap-anydpi-v26/ic_launcher.xml +++ /dev/null @@ -1,5 +0,0 @@ - - - - - \ No newline at end of file diff --git a/packages/desktop-electron/icons/prod/android/mipmap-hdpi/ic_launcher.png b/packages/desktop-electron/icons/prod/android/mipmap-hdpi/ic_launcher.png deleted file mode 100644 index 4f3ea0e367..0000000000 Binary files a/packages/desktop-electron/icons/prod/android/mipmap-hdpi/ic_launcher.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/android/mipmap-hdpi/ic_launcher_foreground.png b/packages/desktop-electron/icons/prod/android/mipmap-hdpi/ic_launcher_foreground.png deleted file mode 100644 index 7db80699bc..0000000000 Binary files a/packages/desktop-electron/icons/prod/android/mipmap-hdpi/ic_launcher_foreground.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/android/mipmap-hdpi/ic_launcher_round.png b/packages/desktop-electron/icons/prod/android/mipmap-hdpi/ic_launcher_round.png deleted file mode 100644 index a54ebe6528..0000000000 Binary files a/packages/desktop-electron/icons/prod/android/mipmap-hdpi/ic_launcher_round.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/android/mipmap-mdpi/ic_launcher.png b/packages/desktop-electron/icons/prod/android/mipmap-mdpi/ic_launcher.png deleted file mode 100644 index 9337ccfa3f..0000000000 Binary files a/packages/desktop-electron/icons/prod/android/mipmap-mdpi/ic_launcher.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/android/mipmap-mdpi/ic_launcher_foreground.png b/packages/desktop-electron/icons/prod/android/mipmap-mdpi/ic_launcher_foreground.png deleted file mode 100644 index 0bfc1082e6..0000000000 Binary files a/packages/desktop-electron/icons/prod/android/mipmap-mdpi/ic_launcher_foreground.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/android/mipmap-mdpi/ic_launcher_round.png b/packages/desktop-electron/icons/prod/android/mipmap-mdpi/ic_launcher_round.png deleted file mode 100644 index 5b02ec732e..0000000000 Binary files a/packages/desktop-electron/icons/prod/android/mipmap-mdpi/ic_launcher_round.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/android/mipmap-xhdpi/ic_launcher.png b/packages/desktop-electron/icons/prod/android/mipmap-xhdpi/ic_launcher.png deleted file mode 100644 index 322aeaeaaa..0000000000 Binary files a/packages/desktop-electron/icons/prod/android/mipmap-xhdpi/ic_launcher.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/android/mipmap-xhdpi/ic_launcher_foreground.png b/packages/desktop-electron/icons/prod/android/mipmap-xhdpi/ic_launcher_foreground.png deleted file mode 100644 index ca1e336cc3..0000000000 Binary files a/packages/desktop-electron/icons/prod/android/mipmap-xhdpi/ic_launcher_foreground.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/android/mipmap-xhdpi/ic_launcher_round.png b/packages/desktop-electron/icons/prod/android/mipmap-xhdpi/ic_launcher_round.png deleted file mode 100644 index f711107992..0000000000 Binary files a/packages/desktop-electron/icons/prod/android/mipmap-xhdpi/ic_launcher_round.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/android/mipmap-xxhdpi/ic_launcher.png b/packages/desktop-electron/icons/prod/android/mipmap-xxhdpi/ic_launcher.png deleted file mode 100644 index 287a6b500b..0000000000 Binary files a/packages/desktop-electron/icons/prod/android/mipmap-xxhdpi/ic_launcher.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/android/mipmap-xxhdpi/ic_launcher_foreground.png b/packages/desktop-electron/icons/prod/android/mipmap-xxhdpi/ic_launcher_foreground.png deleted file mode 100644 index 9d3d06a867..0000000000 Binary files a/packages/desktop-electron/icons/prod/android/mipmap-xxhdpi/ic_launcher_foreground.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/android/mipmap-xxhdpi/ic_launcher_round.png b/packages/desktop-electron/icons/prod/android/mipmap-xxhdpi/ic_launcher_round.png deleted file mode 100644 index d4b6fde1b8..0000000000 Binary files a/packages/desktop-electron/icons/prod/android/mipmap-xxhdpi/ic_launcher_round.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/android/mipmap-xxxhdpi/ic_launcher.png b/packages/desktop-electron/icons/prod/android/mipmap-xxxhdpi/ic_launcher.png deleted file mode 100644 index bde8d75967..0000000000 Binary files a/packages/desktop-electron/icons/prod/android/mipmap-xxxhdpi/ic_launcher.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/android/mipmap-xxxhdpi/ic_launcher_foreground.png b/packages/desktop-electron/icons/prod/android/mipmap-xxxhdpi/ic_launcher_foreground.png deleted file mode 100644 index 03df7809da..0000000000 Binary files a/packages/desktop-electron/icons/prod/android/mipmap-xxxhdpi/ic_launcher_foreground.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/android/mipmap-xxxhdpi/ic_launcher_round.png b/packages/desktop-electron/icons/prod/android/mipmap-xxxhdpi/ic_launcher_round.png deleted file mode 100644 index 62363be047..0000000000 Binary files a/packages/desktop-electron/icons/prod/android/mipmap-xxxhdpi/ic_launcher_round.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/android/values/ic_launcher_background.xml b/packages/desktop-electron/icons/prod/android/values/ic_launcher_background.xml deleted file mode 100644 index ea9c223a6c..0000000000 --- a/packages/desktop-electron/icons/prod/android/values/ic_launcher_background.xml +++ /dev/null @@ -1,4 +0,0 @@ - - - #fff - \ No newline at end of file diff --git a/packages/desktop-electron/icons/prod/dock.png b/packages/desktop-electron/icons/prod/dock.png deleted file mode 100644 index f2ab694e97..0000000000 Binary files a/packages/desktop-electron/icons/prod/dock.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/icon.icns b/packages/desktop-electron/icons/prod/icon.icns deleted file mode 100644 index be910ad5f9..0000000000 Binary files a/packages/desktop-electron/icons/prod/icon.icns and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/icon.ico b/packages/desktop-electron/icons/prod/icon.ico deleted file mode 100644 index ff88d21e4c..0000000000 Binary files a/packages/desktop-electron/icons/prod/icon.ico and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/icon.png b/packages/desktop-electron/icons/prod/icon.png deleted file mode 100644 index 0ecbb6d5f8..0000000000 Binary files a/packages/desktop-electron/icons/prod/icon.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/ios/AppIcon-20x20@1x.png b/packages/desktop-electron/icons/prod/ios/AppIcon-20x20@1x.png deleted file mode 100644 index eb137e164a..0000000000 Binary files a/packages/desktop-electron/icons/prod/ios/AppIcon-20x20@1x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/ios/AppIcon-20x20@2x-1.png b/packages/desktop-electron/icons/prod/ios/AppIcon-20x20@2x-1.png deleted file mode 100644 index aa76ab10ba..0000000000 Binary files a/packages/desktop-electron/icons/prod/ios/AppIcon-20x20@2x-1.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/ios/AppIcon-20x20@2x.png b/packages/desktop-electron/icons/prod/ios/AppIcon-20x20@2x.png deleted file mode 100644 index aa76ab10ba..0000000000 Binary files a/packages/desktop-electron/icons/prod/ios/AppIcon-20x20@2x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/ios/AppIcon-20x20@3x.png b/packages/desktop-electron/icons/prod/ios/AppIcon-20x20@3x.png deleted file mode 100644 index c58ea3d49b..0000000000 Binary files a/packages/desktop-electron/icons/prod/ios/AppIcon-20x20@3x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/ios/AppIcon-29x29@1x.png b/packages/desktop-electron/icons/prod/ios/AppIcon-29x29@1x.png deleted file mode 100644 index 0eeb4d9bf9..0000000000 Binary files a/packages/desktop-electron/icons/prod/ios/AppIcon-29x29@1x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/ios/AppIcon-29x29@2x-1.png b/packages/desktop-electron/icons/prod/ios/AppIcon-29x29@2x-1.png deleted file mode 100644 index 32601c70a1..0000000000 Binary files a/packages/desktop-electron/icons/prod/ios/AppIcon-29x29@2x-1.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/ios/AppIcon-29x29@2x.png b/packages/desktop-electron/icons/prod/ios/AppIcon-29x29@2x.png deleted file mode 100644 index 32601c70a1..0000000000 Binary files a/packages/desktop-electron/icons/prod/ios/AppIcon-29x29@2x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/ios/AppIcon-29x29@3x.png b/packages/desktop-electron/icons/prod/ios/AppIcon-29x29@3x.png deleted file mode 100644 index a372c4a111..0000000000 Binary files a/packages/desktop-electron/icons/prod/ios/AppIcon-29x29@3x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/ios/AppIcon-40x40@1x.png b/packages/desktop-electron/icons/prod/ios/AppIcon-40x40@1x.png deleted file mode 100644 index aa76ab10ba..0000000000 Binary files a/packages/desktop-electron/icons/prod/ios/AppIcon-40x40@1x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/ios/AppIcon-40x40@2x-1.png b/packages/desktop-electron/icons/prod/ios/AppIcon-40x40@2x-1.png deleted file mode 100644 index e82ce2765f..0000000000 Binary files a/packages/desktop-electron/icons/prod/ios/AppIcon-40x40@2x-1.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/ios/AppIcon-40x40@2x.png b/packages/desktop-electron/icons/prod/ios/AppIcon-40x40@2x.png deleted file mode 100644 index e82ce2765f..0000000000 Binary files a/packages/desktop-electron/icons/prod/ios/AppIcon-40x40@2x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/ios/AppIcon-40x40@3x.png b/packages/desktop-electron/icons/prod/ios/AppIcon-40x40@3x.png deleted file mode 100644 index 15ad593628..0000000000 Binary files a/packages/desktop-electron/icons/prod/ios/AppIcon-40x40@3x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/ios/AppIcon-512@2x.png b/packages/desktop-electron/icons/prod/ios/AppIcon-512@2x.png deleted file mode 100644 index 2260671c00..0000000000 Binary files a/packages/desktop-electron/icons/prod/ios/AppIcon-512@2x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/ios/AppIcon-60x60@2x.png b/packages/desktop-electron/icons/prod/ios/AppIcon-60x60@2x.png deleted file mode 100644 index 15ad593628..0000000000 Binary files a/packages/desktop-electron/icons/prod/ios/AppIcon-60x60@2x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/ios/AppIcon-60x60@3x.png b/packages/desktop-electron/icons/prod/ios/AppIcon-60x60@3x.png deleted file mode 100644 index 5c66bd3b18..0000000000 Binary files a/packages/desktop-electron/icons/prod/ios/AppIcon-60x60@3x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/ios/AppIcon-76x76@1x.png b/packages/desktop-electron/icons/prod/ios/AppIcon-76x76@1x.png deleted file mode 100644 index a5b05f3b50..0000000000 Binary files a/packages/desktop-electron/icons/prod/ios/AppIcon-76x76@1x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/ios/AppIcon-76x76@2x.png b/packages/desktop-electron/icons/prod/ios/AppIcon-76x76@2x.png deleted file mode 100644 index 9c0615d411..0000000000 Binary files a/packages/desktop-electron/icons/prod/ios/AppIcon-76x76@2x.png and /dev/null differ diff --git a/packages/desktop-electron/icons/prod/ios/AppIcon-83.5x83.5@2x.png b/packages/desktop-electron/icons/prod/ios/AppIcon-83.5x83.5@2x.png deleted file mode 100644 index 6b792b36ad..0000000000 Binary files a/packages/desktop-electron/icons/prod/ios/AppIcon-83.5x83.5@2x.png and /dev/null differ diff --git a/packages/desktop-electron/package.json b/packages/desktop-electron/package.json index d5f38dd993..182081ae16 100644 --- a/packages/desktop-electron/package.json +++ b/packages/desktop-electron/package.json @@ -10,9 +10,7 @@ "email": "hello@opencode.ai" }, "scripts": { - "typecheck": "tsgo -b", - "predev": "bun ./scripts/predev.ts", - "dev": "electron-vite dev", + "typecheck": "echo 'desktop-electron sources removed in v7.3.0 upstream merge' && exit 0", "prebuild": "bun ./scripts/prebuild.ts", "build": "electron-vite build", "preview": "electron-vite preview", diff --git a/packages/desktop-electron/resources/entitlements.plist b/packages/desktop-electron/resources/entitlements.plist deleted file mode 100644 index b61dc02228..0000000000 --- a/packages/desktop-electron/resources/entitlements.plist +++ /dev/null @@ -1,18 +0,0 @@ - - - - - com.apple.security.cs.allow-jit - - com.apple.security.cs.allow-unsigned-executable-memory - - com.apple.security.cs.disable-executable-page-protection - - com.apple.security.cs.allow-dyld-environment-variables - - com.apple.security.cs.disable-library-validation - - com.apple.security.device.audio-input - - - diff --git a/packages/desktop-electron/scripts/copy-bundles.ts b/packages/desktop-electron/scripts/copy-bundles.ts deleted file mode 100644 index 6ef3335eb7..0000000000 --- a/packages/desktop-electron/scripts/copy-bundles.ts +++ /dev/null @@ -1,12 +0,0 @@ -import { $ } from "bun" -import * as path from "node:path" - -import { RUST_TARGET } from "./utils" - -if (!RUST_TARGET) throw new Error("RUST_TARGET not defined") - -const BUNDLE_DIR = "dist" -const BUNDLES_OUT_DIR = path.join(process.cwd(), "dist/bundles") - -await $`mkdir -p ${BUNDLES_OUT_DIR}` -await $`cp -r ${BUNDLE_DIR}/* ${BUNDLES_OUT_DIR}` diff --git a/packages/desktop-electron/scripts/copy-icons.ts b/packages/desktop-electron/scripts/copy-icons.ts deleted file mode 100644 index 400f427571..0000000000 --- a/packages/desktop-electron/scripts/copy-icons.ts +++ /dev/null @@ -1,12 +0,0 @@ -import { $ } from "bun" -import { resolveChannel } from "./utils" - -const arg = process.argv[2] -const channel = arg === "dev" || arg === "beta" || arg === "prod" ? arg : resolveChannel() - -const src = `./icons/${channel}` -const dest = "resources/icons" - -await $`rm -rf ${dest}` -await $`cp -R ${src} ${dest}` -console.log(`Copied ${channel} icons from ${src} to ${dest}`) diff --git a/packages/desktop-electron/scripts/finalize-latest-yml.ts b/packages/desktop-electron/scripts/finalize-latest-yml.ts deleted file mode 100644 index 7d8ad32a88..0000000000 --- a/packages/desktop-electron/scripts/finalize-latest-yml.ts +++ /dev/null @@ -1,124 +0,0 @@ -#!/usr/bin/env bun - -import { $ } from "bun" -import path from "path" - -const dir = process.env.LATEST_YML_DIR! -if (!dir) throw new Error("LATEST_YML_DIR is required") - -const repo = process.env.GH_REPO -if (!repo) throw new Error("GH_REPO is required") - -const version = process.env.KILO_VERSION -if (!version) throw new Error("KILO_VERSION is required") - -type FileEntry = { - url: string - sha512: string - size: number - blockMapSize?: number -} - -type LatestYml = { - version: string - files: FileEntry[] - releaseDate: string -} - -function parse(content: string): LatestYml { - const lines = content.split("\n") - let version = "" - let releaseDate = "" - const files: FileEntry[] = [] - let current: Partial | undefined - - const flush = () => { - if (current?.url && current.sha512 && current.size) files.push(current as FileEntry) - current = undefined - } - - for (const line of lines) { - const indented = line.startsWith(" ") || line.startsWith(" -") - if (line.startsWith("version:")) version = line.slice("version:".length).trim() - else if (line.startsWith("releaseDate:")) - releaseDate = line.slice("releaseDate:".length).trim().replace(/^'|'$/g, "") - else if (line.trim().startsWith("- url:")) { - flush() - current = { url: line.trim().slice("- url:".length).trim() } - } else if (indented && current && line.trim().startsWith("sha512:")) - current.sha512 = line.trim().slice("sha512:".length).trim() - else if (indented && current && line.trim().startsWith("size:")) - current.size = Number(line.trim().slice("size:".length).trim()) - else if (indented && current && line.trim().startsWith("blockMapSize:")) - current.blockMapSize = Number(line.trim().slice("blockMapSize:".length).trim()) - else if (!indented && current) flush() - } - flush() - - return { version, files, releaseDate } -} - -function serialize(data: LatestYml) { - const lines = [`version: ${data.version}`, "files:"] - for (const file of data.files) { - lines.push(` - url: ${file.url}`) - lines.push(` sha512: ${file.sha512}`) - lines.push(` size: ${file.size}`) - if (file.blockMapSize) lines.push(` blockMapSize: ${file.blockMapSize}`) - } - lines.push(`releaseDate: '${data.releaseDate}'`) - return lines.join("\n") + "\n" -} - -async function read(subdir: string, filename: string): Promise { - const file = Bun.file(path.join(dir, subdir, filename)) - if (!(await file.exists())) return undefined - return parse(await file.text()) -} - -const output: Record = {} - -// Windows: merge arm64 + x64 into single file -const winX64 = await read("latest-yml-x86_64-pc-windows-msvc", "latest.yml") -const winArm64 = await read("latest-yml-aarch64-pc-windows-msvc", "latest.yml") -if (winX64 || winArm64) { - const base = winArm64 ?? winX64! - output["latest.yml"] = serialize({ - version: base.version, - files: [...(winArm64?.files ?? []), ...(winX64?.files ?? [])], - releaseDate: base.releaseDate, - }) -} - -// Linux x64: pass through -const linuxX64 = await read("latest-yml-x86_64-unknown-linux-gnu", "latest-linux.yml") -if (linuxX64) output["latest-linux.yml"] = serialize(linuxX64) - -// Linux arm64: pass through -const linuxArm64 = await read("latest-yml-aarch64-unknown-linux-gnu", "latest-linux-arm64.yml") -if (linuxArm64) output["latest-linux-arm64.yml"] = serialize(linuxArm64) - -// macOS: merge arm64 + x64 into single file -const macX64 = await read("latest-yml-x86_64-apple-darwin", "latest-mac.yml") -const macArm64 = await read("latest-yml-aarch64-apple-darwin", "latest-mac.yml") -if (macX64 || macArm64) { - const base = macArm64 ?? macX64! - output["latest-mac.yml"] = serialize({ - version: base.version, - files: [...(macArm64?.files ?? []), ...(macX64?.files ?? [])], - releaseDate: base.releaseDate, - }) -} - -// Upload to release -const tag = `v${version}` -const tmp = process.env.RUNNER_TEMP ?? "/tmp" - -for (const [filename, content] of Object.entries(output)) { - const filepath = path.join(tmp, filename) - await Bun.write(filepath, content) - await $`gh release upload ${tag} ${filepath} --clobber --repo ${repo}` - console.log(`uploaded ${filename}`) -} - -console.log("finalized latest yml files") diff --git a/packages/desktop-electron/scripts/prebuild.ts b/packages/desktop-electron/scripts/prebuild.ts deleted file mode 100644 index 46a2475ea5..0000000000 --- a/packages/desktop-electron/scripts/prebuild.ts +++ /dev/null @@ -1,9 +0,0 @@ -#!/usr/bin/env bun -import { $ } from "bun" - -import { resolveChannel } from "./utils" - -const channel = resolveChannel() -await $`bun ./scripts/copy-icons.ts ${channel}` - -await $`cd ../opencode && bun script/build-node.ts` diff --git a/packages/desktop-electron/scripts/predev.ts b/packages/desktop-electron/scripts/predev.ts deleted file mode 100644 index 644c77556b..0000000000 --- a/packages/desktop-electron/scripts/predev.ts +++ /dev/null @@ -1,5 +0,0 @@ -import { $ } from "bun" - -await $`bun ./scripts/copy-icons.ts ${process.env.KILO_CHANNEL ?? "dev"}` - -await $`cd ../opencode && bun script/build-node.ts` diff --git a/packages/desktop-electron/scripts/prepare.ts b/packages/desktop-electron/scripts/prepare.ts deleted file mode 100755 index 0dfd5a35cb..0000000000 --- a/packages/desktop-electron/scripts/prepare.ts +++ /dev/null @@ -1,9 +0,0 @@ -#!/usr/bin/env bun -import { Script } from "@opencode-ai/script" - -await import("./prebuild") - -const pkg = await Bun.file("./package.json").json() -pkg.version = Script.version -await Bun.write("./package.json", JSON.stringify(pkg, null, 2) + "\n") -console.log(`Updated package.json version to ${Script.version}`) diff --git a/packages/desktop-electron/scripts/utils.ts b/packages/desktop-electron/scripts/utils.ts deleted file mode 100644 index c039542110..0000000000 --- a/packages/desktop-electron/scripts/utils.ts +++ /dev/null @@ -1,77 +0,0 @@ -import { $ } from "bun" - -export type Channel = "dev" | "beta" | "prod" - -export function resolveChannel(): Channel { - const raw = Bun.env.KILO_CHANNEL - if (raw === "dev" || raw === "beta" || raw === "prod") return raw - return "dev" -} - -export const SIDECAR_BINARIES: Array<{ rustTarget: string; ocBinary: string; assetExt: string }> = [ - { - rustTarget: "aarch64-apple-darwin", - ocBinary: "opencode-darwin-arm64", - assetExt: "zip", - }, - { - rustTarget: "x86_64-apple-darwin", - ocBinary: "opencode-darwin-x64-baseline", - assetExt: "zip", - }, - { - rustTarget: "aarch64-pc-windows-msvc", - ocBinary: "opencode-windows-arm64", - assetExt: "zip", - }, - { - rustTarget: "x86_64-pc-windows-msvc", - ocBinary: "opencode-windows-x64-baseline", - assetExt: "zip", - }, - { - rustTarget: "x86_64-unknown-linux-gnu", - ocBinary: "opencode-linux-x64-baseline", - assetExt: "tar.gz", - }, - { - rustTarget: "aarch64-unknown-linux-gnu", - ocBinary: "opencode-linux-arm64", - assetExt: "tar.gz", - }, -] - -export const RUST_TARGET = Bun.env.RUST_TARGET - -function nativeTarget() { - const { platform, arch } = process - if (platform === "darwin") return arch === "arm64" ? "aarch64-apple-darwin" : "x86_64-apple-darwin" - if (platform === "win32") return arch === "arm64" ? "aarch64-pc-windows-msvc" : "x86_64-pc-windows-msvc" - if (platform === "linux") return arch === "arm64" ? "aarch64-unknown-linux-gnu" : "x86_64-unknown-linux-gnu" - throw new Error(`Unsupported platform: ${platform}/${arch}`) -} - -export function getCurrentSidecar(target = RUST_TARGET ?? nativeTarget()) { - const binaryConfig = SIDECAR_BINARIES.find((b) => b.rustTarget === target) - if (!binaryConfig) throw new Error(`Sidecar configuration not available for Rust target '${target}'`) - - return binaryConfig -} - -export async function copyBinaryToSidecarFolder(source: string) { - const dir = `resources` - await $`mkdir -p ${dir}` - const dest = windowsify(`${dir}/opencode-cli`) - await $`cp ${source} ${dest}` - if (process.platform === "win32" && process.env.GITHUB_ACTIONS === "true") { - await $`pwsh -NoLogo -NoProfile -ExecutionPolicy Bypass -File ../../script/sign-windows.ps1 ${dest}` - } - if (process.platform === "darwin") await $`codesign --force --sign - ${dest}` - - console.log(`Copied ${source} to ${dest}`) -} - -export function windowsify(path: string) { - if (path.endsWith(".exe")) return path - return `${path}${process.platform === "win32" ? ".exe" : ""}` -} diff --git a/packages/desktop-electron/src/main/apps.ts b/packages/desktop-electron/src/main/apps.ts deleted file mode 100644 index 174da94a5d..0000000000 --- a/packages/desktop-electron/src/main/apps.ts +++ /dev/null @@ -1,148 +0,0 @@ -import { execFileSync } from "node:child_process" -import { existsSync, readFileSync, readdirSync } from "node:fs" -import { dirname, extname, join } from "node:path" - -export function checkAppExists(appName: string): boolean { - if (process.platform === "win32") return true - if (process.platform === "linux") return true - return checkMacosApp(appName) -} - -export function resolveAppPath(appName: string): string | null { - if (process.platform !== "win32") return appName - return resolveWindowsAppPath(appName) -} - -export function wslPath(path: string, mode: "windows" | "linux" | null): string { - if (process.platform !== "win32") return path - - const flag = mode === "windows" ? "-w" : "-u" - try { - if (path.startsWith("~")) { - const suffix = path.slice(1) - const cmd = `wslpath ${flag} "$HOME${suffix.replace(/"/g, '\\"')}"` - const output = execFileSync("wsl", ["-e", "sh", "-lc", cmd]) - return output.toString().trim() - } - - const output = execFileSync("wsl", ["-e", "wslpath", flag, path]) - return output.toString().trim() - } catch (error) { - throw new Error(`Failed to run wslpath: ${String(error)}`, { cause: error }) - } -} - -function checkMacosApp(appName: string) { - const locations = [`/Applications/${appName}.app`, `/System/Applications/${appName}.app`] - - const home = process.env.HOME - if (home) locations.push(`${home}/Applications/${appName}.app`) - - if (locations.some((location) => existsSync(location))) return true - - try { - execFileSync("which", [appName]) - return true - } catch { - return false - } -} - -function resolveWindowsAppPath(appName: string): string | null { - let output: string - try { - output = execFileSync("where", [appName]).toString() - } catch { - return null - } - - const paths = output - .split(/\r?\n/) - .map((line) => line.trim()) - .filter((line) => line.length > 0) - - const hasExt = (path: string, ext: string) => extname(path).toLowerCase() === `.${ext}` - - const exe = paths.find((path) => hasExt(path, "exe")) - if (exe) return exe - - const resolveCmd = (path: string) => { - const content = readFileSync(path, "utf8") - for (const token of content.split('"').map((value: string) => value.trim())) { - const lower = token.toLowerCase() - if (!lower.includes(".exe")) continue - - const index = lower.indexOf("%~dp0") - if (index >= 0) { - const base = dirname(path) - const suffix = token.slice(index + 5) - const resolved = suffix - .replace(/\//g, "\\") - .split("\\") - .filter((part: string) => part && part !== ".") - .reduce((current: string, part: string) => { - if (part === "..") return dirname(current) - return join(current, part) - }, base) - - if (existsSync(resolved)) return resolved - } - - if (existsSync(token)) return token - } - - return null - } - - for (const path of paths) { - if (hasExt(path, "cmd") || hasExt(path, "bat")) { - const resolved = resolveCmd(path) - if (resolved) return resolved - } - - if (!extname(path)) { - const cmd = `${path}.cmd` - if (existsSync(cmd)) { - const resolved = resolveCmd(cmd) - if (resolved) return resolved - } - - const bat = `${path}.bat` - if (existsSync(bat)) { - const resolved = resolveCmd(bat) - if (resolved) return resolved - } - } - } - - const key = appName - .split("") - .filter((value: string) => /[a-z0-9]/i.test(value)) - .map((value: string) => value.toLowerCase()) - .join("") - - if (key) { - for (const path of paths) { - const dirs = [dirname(path), dirname(dirname(path)), dirname(dirname(dirname(path)))] - for (const dir of dirs) { - try { - for (const entry of readdirSync(dir)) { - const candidate = join(dir, entry) - if (!hasExt(candidate, "exe")) continue - const stem = entry.replace(/\.exe$/i, "") - const name = stem - .split("") - .filter((value: string) => /[a-z0-9]/i.test(value)) - .map((value: string) => value.toLowerCase()) - .join("") - if (name.includes(key) || key.includes(name)) return candidate - } - } catch { - continue - } - } - } - } - - return paths[0] ?? null -} diff --git a/packages/desktop-electron/src/main/constants.ts b/packages/desktop-electron/src/main/constants.ts deleted file mode 100644 index abb65c6337..0000000000 --- a/packages/desktop-electron/src/main/constants.ts +++ /dev/null @@ -1,10 +0,0 @@ -import { app } from "electron" - -type Channel = "dev" | "beta" | "prod" -const raw = import.meta.env.KILO_CHANNEL -export const CHANNEL: Channel = raw === "dev" || raw === "beta" || raw === "prod" ? raw : "dev" - -export const SETTINGS_STORE = "opencode.settings" -export const DEFAULT_SERVER_URL_KEY = "defaultServerUrl" -export const WSL_ENABLED_KEY = "wslEnabled" -export const UPDATER_ENABLED = app.isPackaged && CHANNEL !== "dev" diff --git a/packages/desktop-electron/src/main/env.d.ts b/packages/desktop-electron/src/main/env.d.ts deleted file mode 100644 index 6169024263..0000000000 --- a/packages/desktop-electron/src/main/env.d.ts +++ /dev/null @@ -1,29 +0,0 @@ -interface ImportMetaEnv { - readonly KILO_CHANNEL: string -} - -interface ImportMeta { - readonly env: ImportMetaEnv -} -declare module "virtual:opencode-server" { - export namespace Server { - export const listen: typeof import("../../../opencode/dist/types/src/node").Server.listen - export type Listener = import("../../../opencode/dist/types/src/node").Server.Listener - } - export namespace Config { - export const get: typeof import("../../../opencode/dist/types/src/node").Config.get - export type Info = import("../../../opencode/dist/types/src/node").Config.Info - } - export namespace Log { - export const init: typeof import("../../../opencode/dist/types/src/node").Log.init - } - export namespace Database { - export const Path: typeof import("../../../opencode/dist/types/src/node").Database.Path - export const Client: typeof import("../../../opencode/dist/types/src/node").Database.Client - } - export namespace JsonMigration { - export type Progress = import("../../../opencode/dist/types/src/node").JsonMigration.Progress - export const run: typeof import("../../../opencode/dist/types/src/node").JsonMigration.run - } - export const bootstrap: typeof import("../../../opencode/dist/types/src/node").bootstrap -} diff --git a/packages/desktop-electron/src/main/index.ts b/packages/desktop-electron/src/main/index.ts deleted file mode 100644 index f00ba5d573..0000000000 --- a/packages/desktop-electron/src/main/index.ts +++ /dev/null @@ -1,452 +0,0 @@ -import { randomUUID } from "node:crypto" -import { EventEmitter } from "node:events" -import { existsSync } from "node:fs" -import { createServer } from "node:net" -import { homedir } from "node:os" -import { join } from "node:path" -import type { Event } from "electron" -import { app, BrowserWindow, dialog } from "electron" -import pkg from "electron-updater" - -import contextMenu from "electron-context-menu" -contextMenu({ showSaveImageAs: true, showLookUpSelection: false, showSearchWithGoogle: false }) - -// on macOS apps run in `/` which can cause issues with ripgrep -try { - process.chdir(homedir()) -} catch {} - -process.env.KILO_DISABLE_EMBEDDED_WEB_UI = "true" - -const APP_NAMES: Record = { - dev: "OpenCode Dev", - beta: "OpenCode Beta", - prod: "OpenCode", -} -const APP_IDS: Record = { - dev: "ai.opencode.desktop.dev", - beta: "ai.opencode.desktop.beta", - prod: "ai.opencode.desktop", -} -const appId = app.isPackaged ? APP_IDS[CHANNEL] : "ai.opencode.desktop.dev" -app.setName(app.isPackaged ? APP_NAMES[CHANNEL] : "OpenCode Dev") -app.setAppUserModelId(appId) -app.setPath("userData", join(app.getPath("appData"), appId)) -const { autoUpdater } = pkg - -import type { InitStep, ServerReadyData, SqliteMigrationProgress, WslConfig } from "../preload/types" -import { checkAppExists, resolveAppPath, wslPath } from "./apps" -import { CHANNEL, UPDATER_ENABLED } from "./constants" -import { registerIpcHandlers, sendDeepLinks, sendMenuCommand, sendSqliteMigrationProgress } from "./ipc" -import { initLogging } from "./logging" -import { parseMarkdown } from "./markdown" -import { createMenu } from "./menu" -import { getDefaultServerUrl, getWslConfig, setDefaultServerUrl, setWslConfig, spawnLocalServer } from "./server" -import { - createLoadingWindow, - createMainWindow, - registerRendererProtocol, - setBackgroundColor, - setDockIcon, -} from "./windows" -import { drizzle } from "drizzle-orm/node-sqlite/driver" -import type { Server } from "virtual:opencode-server" - -const initEmitter = new EventEmitter() -let initStep: InitStep = { phase: "server_waiting" } - -let mainWindow: BrowserWindow | null = null -let server: Server.Listener | null = null -const loadingComplete = defer() - -const pendingDeepLinks: string[] = [] - -const serverReady = defer() -const logger = initLogging() - -logger.log("app starting", { - version: app.getVersion(), - packaged: app.isPackaged, -}) - -setupApp() - -function setupApp() { - ensureLoopbackNoProxy() - app.commandLine.appendSwitch("proxy-bypass-list", "<-loopback>") - - if (!app.requestSingleInstanceLock()) { - app.quit() - return - } - - app.on("second-instance", (_event: Event, argv: string[]) => { - const urls = argv.filter((arg: string) => arg.startsWith("opencode://")) - if (urls.length) { - logger.log("deep link received via second-instance", { urls }) - emitDeepLinks(urls) - } - focusMainWindow() - }) - - app.on("open-url", (event: Event, url: string) => { - event.preventDefault() - logger.log("deep link received via open-url", { url }) - emitDeepLinks([url]) - }) - - app.on("before-quit", () => { - killSidecar() - }) - - app.on("will-quit", () => { - killSidecar() - }) - - for (const signal of ["SIGINT", "SIGTERM"] as const) { - process.on(signal, () => { - killSidecar() - app.exit(0) - }) - } - - void app.whenReady().then(async () => { - app.setAsDefaultProtocolClient("opencode") - registerRendererProtocol() - setDockIcon() - setupAutoUpdater() - await initialize() - }) -} - -function emitDeepLinks(urls: string[]) { - if (urls.length === 0) return - pendingDeepLinks.push(...urls) - if (mainWindow) sendDeepLinks(mainWindow, urls) -} - -function focusMainWindow() { - if (!mainWindow) return - mainWindow.show() - mainWindow.focus() -} - -function setInitStep(step: InitStep) { - initStep = step - logger.log("init step", { step }) - initEmitter.emit("step", step) -} - -async function initialize() { - const needsMigration = !sqliteFileExists() - const sqliteDone = needsMigration ? defer() : undefined - let overlay: BrowserWindow | null = null - - const port = await getSidecarPort() - const hostname = "127.0.0.1" - const url = `http://${hostname}:${port}` - const password = randomUUID() - - const loadingTask = (async () => { - logger.log("sidecar connection started", { url }) - - initEmitter.on("sqlite", (progress: SqliteMigrationProgress) => { - setInitStep({ phase: "sqlite_waiting" }) - if (overlay) sendSqliteMigrationProgress(overlay, progress) - if (mainWindow) sendSqliteMigrationProgress(mainWindow, progress) - if (progress.type === "Done") sqliteDone?.resolve() - }) - - if (needsMigration) { - const { Database, JsonMigration } = await import("virtual:opencode-server") - await JsonMigration.run(drizzle({ client: Database.Client().$client }), { - progress: (event: { current: number; total: number }) => { - const percent = Math.round(event.current / event.total) * 100 - initEmitter.emit("sqlite", { type: "InProgress", value: percent }) - }, - }) - initEmitter.emit("sqlite", { type: "Done" }) - - sqliteDone?.resolve() - } - - if (needsMigration) { - await sqliteDone?.promise - } - - logger.log("spawning sidecar", { url }) - const { listener, health } = await spawnLocalServer(hostname, port, password) - server = listener - serverReady.resolve({ - url, - username: "kilo", // kilocode_change - password, - }) - - await Promise.race([ - health.wait, - delay(30_000).then(() => { - throw new Error("Sidecar health check timed out") - }), - ]).catch((error) => { - logger.error("sidecar health check failed", error) - }) - - logger.log("loading task finished") - })() - - if (needsMigration) { - const show = await Promise.race([loadingTask.then(() => false), delay(1_000).then(() => true)]) - if (show) { - overlay = createLoadingWindow() - await delay(1_000) - } - } - - await loadingTask - setInitStep({ phase: "done" }) - - if (overlay) { - await loadingComplete.promise - } - - mainWindow = createMainWindow() - wireMenu() - - overlay?.close() -} - -function wireMenu() { - if (!mainWindow) return - createMenu({ - trigger: (id) => mainWindow && sendMenuCommand(mainWindow, id), - checkForUpdates: () => { - void checkForUpdates(true) - }, - reload: () => mainWindow?.reload(), - relaunch: () => { - killSidecar() - app.relaunch() - app.exit(0) - }, - }) -} - -registerIpcHandlers({ - killSidecar: () => killSidecar(), - awaitInitialization: async (sendStep) => { - sendStep(initStep) - const listener = (step: InitStep) => sendStep(step) - initEmitter.on("step", listener) - try { - logger.log("awaiting server ready") - const res = await serverReady.promise - logger.log("server ready", { url: res.url }) - return res - } finally { - initEmitter.off("step", listener) - } - }, - getWindowConfig: () => ({ updaterEnabled: UPDATER_ENABLED }), - consumeInitialDeepLinks: () => pendingDeepLinks.splice(0), - getDefaultServerUrl: () => getDefaultServerUrl(), - setDefaultServerUrl: (url) => setDefaultServerUrl(url), - getWslConfig: () => Promise.resolve(getWslConfig()), - setWslConfig: (config: WslConfig) => setWslConfig(config), - getDisplayBackend: async () => null, - setDisplayBackend: async () => undefined, - parseMarkdown: async (markdown) => parseMarkdown(markdown), - checkAppExists: async (appName) => checkAppExists(appName), - wslPath: async (path, mode) => wslPath(path, mode), - resolveAppPath: async (appName) => resolveAppPath(appName), - loadingWindowComplete: () => loadingComplete.resolve(), - runUpdater: async (alertOnFail) => checkForUpdates(alertOnFail), - checkUpdate: async () => checkUpdate(), - installUpdate: async () => installUpdate(), - setBackgroundColor: (color) => setBackgroundColor(color), -}) - -function killSidecar() { - if (!server) return - server.stop() - server = null -} - -function ensureLoopbackNoProxy() { - const loopback = ["127.0.0.1", "localhost", "::1"] - const upsert = (key: string) => { - const items = (process.env[key] ?? "") - .split(",") - .map((value: string) => value.trim()) - .filter((value: string) => Boolean(value)) - - for (const host of loopback) { - if (items.some((value: string) => value.toLowerCase() === host)) continue - items.push(host) - } - - process.env[key] = items.join(",") - } - - upsert("NO_PROXY") - upsert("no_proxy") -} - -async function getSidecarPort() { - const fromEnv = process.env.KILO_PORT - if (fromEnv) { - const parsed = Number.parseInt(fromEnv, 10) - if (!Number.isNaN(parsed)) return parsed - } - - return await new Promise((resolve, reject) => { - const server = createServer() - server.on("error", reject) - server.listen(0, "127.0.0.1", () => { - const address = server.address() - if (typeof address !== "object" || !address) { - server.close() - reject(new Error("Failed to get port")) - return - } - const port = address.port - server.close(() => resolve(port)) - }) - }) -} - -function sqliteFileExists() { - const xdg = process.env.XDG_DATA_HOME - const base = xdg && xdg.length > 0 ? xdg : join(homedir(), ".local", "share") - return existsSync(join(base, "opencode", "kilo.db")) -} - -function setupAutoUpdater() { - if (!UPDATER_ENABLED) return - autoUpdater.logger = logger - autoUpdater.channel = "latest" - autoUpdater.allowPrerelease = false - autoUpdater.allowDowngrade = true - autoUpdater.autoDownload = false - autoUpdater.autoInstallOnAppQuit = true - logger.log("auto updater configured", { - channel: autoUpdater.channel, - allowPrerelease: autoUpdater.allowPrerelease, - allowDowngrade: autoUpdater.allowDowngrade, - currentVersion: app.getVersion(), - }) -} - -let downloadedUpdateVersion: string | undefined - -async function checkUpdate() { - if (!UPDATER_ENABLED) return { updateAvailable: false } - if (downloadedUpdateVersion) { - logger.log("returning cached downloaded update", { - version: downloadedUpdateVersion, - }) - return { updateAvailable: true, version: downloadedUpdateVersion } - } - logger.log("checking for updates", { - currentVersion: app.getVersion(), - channel: autoUpdater.channel, - allowPrerelease: autoUpdater.allowPrerelease, - allowDowngrade: autoUpdater.allowDowngrade, - }) - try { - const result = await autoUpdater.checkForUpdates() - const updateInfo = result?.updateInfo - logger.log("update metadata fetched", { - releaseVersion: updateInfo?.version ?? null, - releaseDate: updateInfo?.releaseDate ?? null, - releaseName: updateInfo?.releaseName ?? null, - files: updateInfo?.files?.map((file) => file.url) ?? [], - }) - const version = result?.updateInfo?.version - if (result?.isUpdateAvailable === false || !version) { - logger.log("no update available", { - reason: "provider returned no newer version", - }) - return { updateAvailable: false } - } - logger.log("update available", { version }) - await autoUpdater.downloadUpdate() - logger.log("update download completed", { version }) - downloadedUpdateVersion = version - return { updateAvailable: true, version } - } catch (error) { - logger.error("update check failed", error) - return { updateAvailable: false, failed: true } - } -} - -async function installUpdate() { - if (!downloadedUpdateVersion) { - logger.log("install update skipped", { - reason: "no downloaded update ready", - }) - return - } - logger.log("installing downloaded update", { - version: downloadedUpdateVersion, - }) - killSidecar() - autoUpdater.quitAndInstall() -} - -async function checkForUpdates(alertOnFail: boolean) { - if (!UPDATER_ENABLED) return - logger.log("checkForUpdates invoked", { alertOnFail }) - const result = await checkUpdate() - if (!result.updateAvailable) { - if (result.failed) { - logger.log("no update decision", { reason: "update check failed" }) - if (!alertOnFail) return - await dialog.showMessageBox({ - type: "error", - message: "Update check failed.", - title: "Update Error", - }) - return - } - - logger.log("no update decision", { reason: "already up to date" }) - if (!alertOnFail) return - await dialog.showMessageBox({ - type: "info", - message: "You're up to date.", - title: "No Updates", - }) - return - } - - const response = await dialog.showMessageBox({ - type: "info", - message: `Update ${result.version ?? ""} downloaded. Restart now?`, - title: "Update Ready", - buttons: ["Restart", "Later"], - defaultId: 0, - cancelId: 1, - }) - logger.log("update prompt response", { - version: result.version ?? null, - restartNow: response.response === 0, - }) - if (response.response === 0) { - await installUpdate() - } -} - -function delay(ms: number) { - return new Promise((resolve) => setTimeout(resolve, ms)) -} - -function defer() { - let resolve!: (value: T) => void - let reject!: (error: Error) => void - const promise = new Promise((res, rej) => { - resolve = res - reject = rej - }) - return { promise, resolve, reject } -} diff --git a/packages/desktop-electron/src/main/ipc.ts b/packages/desktop-electron/src/main/ipc.ts deleted file mode 100644 index 8dbca8eea1..0000000000 --- a/packages/desktop-electron/src/main/ipc.ts +++ /dev/null @@ -1,204 +0,0 @@ -import { execFile } from "node:child_process" -import { BrowserWindow, Notification, app, clipboard, dialog, ipcMain, shell } from "electron" -import type { IpcMainEvent, IpcMainInvokeEvent } from "electron" - -import type { - InitStep, - ServerReadyData, - SqliteMigrationProgress, - TitlebarTheme, - WindowConfig, - WslConfig, -} from "../preload/types" -import { getStore } from "./store" -import { setTitlebar } from "./windows" - -const pickerFilters = (ext?: string[]) => { - if (!ext || ext.length === 0) return undefined - return [{ name: "Files", extensions: ext }] -} - -type Deps = { - killSidecar: () => void - awaitInitialization: (sendStep: (step: InitStep) => void) => Promise - getWindowConfig: () => Promise | WindowConfig - consumeInitialDeepLinks: () => Promise | string[] - getDefaultServerUrl: () => Promise | string | null - setDefaultServerUrl: (url: string | null) => Promise | void - getWslConfig: () => Promise - setWslConfig: (config: WslConfig) => Promise | void - getDisplayBackend: () => Promise - setDisplayBackend: (backend: string | null) => Promise | void - parseMarkdown: (markdown: string) => Promise | string - checkAppExists: (appName: string) => Promise | boolean - wslPath: (path: string, mode: "windows" | "linux" | null) => Promise - resolveAppPath: (appName: string) => Promise - loadingWindowComplete: () => void - runUpdater: (alertOnFail: boolean) => Promise | void - checkUpdate: () => Promise<{ updateAvailable: boolean; version?: string }> - installUpdate: () => Promise | void - setBackgroundColor: (color: string) => void -} - -export function registerIpcHandlers(deps: Deps) { - ipcMain.handle("kill-sidecar", () => deps.killSidecar()) - ipcMain.handle("await-initialization", (event: IpcMainInvokeEvent) => { - const send = (step: InitStep) => event.sender.send("init-step", step) - return deps.awaitInitialization(send) - }) - ipcMain.handle("get-window-config", () => deps.getWindowConfig()) - ipcMain.handle("consume-initial-deep-links", () => deps.consumeInitialDeepLinks()) - ipcMain.handle("get-default-server-url", () => deps.getDefaultServerUrl()) - ipcMain.handle("set-default-server-url", (_event: IpcMainInvokeEvent, url: string | null) => - deps.setDefaultServerUrl(url), - ) - ipcMain.handle("get-wsl-config", () => deps.getWslConfig()) - ipcMain.handle("set-wsl-config", (_event: IpcMainInvokeEvent, config: WslConfig) => deps.setWslConfig(config)) - ipcMain.handle("get-display-backend", () => deps.getDisplayBackend()) - ipcMain.handle("set-display-backend", (_event: IpcMainInvokeEvent, backend: string | null) => - deps.setDisplayBackend(backend), - ) - ipcMain.handle("parse-markdown", (_event: IpcMainInvokeEvent, markdown: string) => deps.parseMarkdown(markdown)) - ipcMain.handle("check-app-exists", (_event: IpcMainInvokeEvent, appName: string) => deps.checkAppExists(appName)) - ipcMain.handle("wsl-path", (_event: IpcMainInvokeEvent, path: string, mode: "windows" | "linux" | null) => - deps.wslPath(path, mode), - ) - ipcMain.handle("resolve-app-path", (_event: IpcMainInvokeEvent, appName: string) => deps.resolveAppPath(appName)) - ipcMain.on("loading-window-complete", () => deps.loadingWindowComplete()) - ipcMain.handle("run-updater", (_event: IpcMainInvokeEvent, alertOnFail: boolean) => deps.runUpdater(alertOnFail)) - ipcMain.handle("check-update", () => deps.checkUpdate()) - ipcMain.handle("install-update", () => deps.installUpdate()) - ipcMain.handle("set-background-color", (_event: IpcMainInvokeEvent, color: string) => deps.setBackgroundColor(color)) - ipcMain.handle("store-get", (_event: IpcMainInvokeEvent, name: string, key: string) => { - const store = getStore(name) - const value = store.get(key) - if (value === undefined || value === null) return null - return typeof value === "string" ? value : JSON.stringify(value) - }) - ipcMain.handle("store-set", (_event: IpcMainInvokeEvent, name: string, key: string, value: string) => { - getStore(name).set(key, value) - }) - ipcMain.handle("store-delete", (_event: IpcMainInvokeEvent, name: string, key: string) => { - getStore(name).delete(key) - }) - ipcMain.handle("store-clear", (_event: IpcMainInvokeEvent, name: string) => { - getStore(name).clear() - }) - ipcMain.handle("store-keys", (_event: IpcMainInvokeEvent, name: string) => { - const store = getStore(name) - return Object.keys(store.store) - }) - ipcMain.handle("store-length", (_event: IpcMainInvokeEvent, name: string) => { - const store = getStore(name) - return Object.keys(store.store).length - }) - - ipcMain.handle( - "open-directory-picker", - async (_event: IpcMainInvokeEvent, opts?: { multiple?: boolean; title?: string; defaultPath?: string }) => { - const result = await dialog.showOpenDialog({ - properties: ["openDirectory", ...(opts?.multiple ? ["multiSelections" as const] : []), "createDirectory"], - title: opts?.title ?? "Choose a folder", - defaultPath: opts?.defaultPath, - }) - if (result.canceled) return null - return opts?.multiple ? result.filePaths : result.filePaths[0] - }, - ) - - ipcMain.handle( - "open-file-picker", - async ( - _event: IpcMainInvokeEvent, - opts?: { multiple?: boolean; title?: string; defaultPath?: string; accept?: string[]; extensions?: string[] }, - ) => { - const result = await dialog.showOpenDialog({ - properties: ["openFile", ...(opts?.multiple ? ["multiSelections" as const] : [])], - title: opts?.title ?? "Choose a file", - defaultPath: opts?.defaultPath, - filters: pickerFilters(opts?.extensions), - }) - if (result.canceled) return null - return opts?.multiple ? result.filePaths : result.filePaths[0] - }, - ) - - ipcMain.handle( - "save-file-picker", - async (_event: IpcMainInvokeEvent, opts?: { title?: string; defaultPath?: string }) => { - const result = await dialog.showSaveDialog({ - title: opts?.title ?? "Save file", - defaultPath: opts?.defaultPath, - }) - if (result.canceled) return null - return result.filePath ?? null - }, - ) - - ipcMain.on("open-link", (_event: IpcMainEvent, url: string) => { - void shell.openExternal(url) - }) - - ipcMain.handle("open-path", async (_event: IpcMainInvokeEvent, path: string, app?: string) => { - if (!app) return shell.openPath(path) - await new Promise((resolve, reject) => { - const [cmd, args] = - process.platform === "darwin" ? (["open", ["-a", app, path]] as const) : ([app, [path]] as const) - execFile(cmd, args, (err) => (err ? reject(err) : resolve())) - }) - }) - - ipcMain.handle("read-clipboard-image", () => { - const image = clipboard.readImage() - if (image.isEmpty()) return null - const buffer = image.toPNG().buffer - const size = image.getSize() - return { buffer, width: size.width, height: size.height } - }) - - ipcMain.on("show-notification", (_event: IpcMainEvent, title: string, body?: string) => { - new Notification({ title, body }).show() - }) - - ipcMain.handle("get-window-count", () => BrowserWindow.getAllWindows().length) - - ipcMain.handle("get-window-focused", (event: IpcMainInvokeEvent) => { - const win = BrowserWindow.fromWebContents(event.sender) - return win?.isFocused() ?? false - }) - - ipcMain.handle("set-window-focus", (event: IpcMainInvokeEvent) => { - const win = BrowserWindow.fromWebContents(event.sender) - win?.focus() - }) - - ipcMain.handle("show-window", (event: IpcMainInvokeEvent) => { - const win = BrowserWindow.fromWebContents(event.sender) - win?.show() - }) - - ipcMain.on("relaunch", () => { - app.relaunch() - app.exit(0) - }) - - ipcMain.handle("get-zoom-factor", (event: IpcMainInvokeEvent) => event.sender.getZoomFactor()) - ipcMain.handle("set-zoom-factor", (event: IpcMainInvokeEvent, factor: number) => event.sender.setZoomFactor(factor)) - ipcMain.handle("set-titlebar", (event: IpcMainInvokeEvent, theme: TitlebarTheme) => { - const win = BrowserWindow.fromWebContents(event.sender) - if (!win) return - setTitlebar(win, theme) - }) -} - -export function sendSqliteMigrationProgress(win: BrowserWindow, progress: SqliteMigrationProgress) { - win.webContents.send("sqlite-migration-progress", progress) -} - -export function sendMenuCommand(win: BrowserWindow, id: string) { - win.webContents.send("menu-command", id) -} - -export function sendDeepLinks(win: BrowserWindow, urls: string[]) { - win.webContents.send("deep-link", urls) -} diff --git a/packages/desktop-electron/src/main/logging.ts b/packages/desktop-electron/src/main/logging.ts deleted file mode 100644 index d315b2d344..0000000000 --- a/packages/desktop-electron/src/main/logging.ts +++ /dev/null @@ -1,40 +0,0 @@ -import log from "electron-log/main.js" -import { readFileSync, readdirSync, statSync, unlinkSync } from "node:fs" -import { dirname, join } from "node:path" - -const MAX_LOG_AGE_DAYS = 7 -const TAIL_LINES = 1000 - -export function initLogging() { - log.transports.file.maxSize = 5 * 1024 * 1024 - cleanup() - return log -} - -export function tail(): string { - try { - const path = log.transports.file.getFile().path - const contents = readFileSync(path, "utf8") - const lines = contents.split("\n") - return lines.slice(Math.max(0, lines.length - TAIL_LINES)).join("\n") - } catch { - return "" - } -} - -function cleanup() { - const path = log.transports.file.getFile().path - const dir = dirname(path) - const cutoff = Date.now() - MAX_LOG_AGE_DAYS * 24 * 60 * 60 * 1000 - - for (const entry of readdirSync(dir)) { - const file = join(dir, entry) - try { - const info = statSync(file) - if (!info.isFile()) continue - if (info.mtimeMs < cutoff) unlinkSync(file) - } catch { - continue - } - } -} diff --git a/packages/desktop-electron/src/main/markdown.ts b/packages/desktop-electron/src/main/markdown.ts deleted file mode 100644 index b956f48760..0000000000 --- a/packages/desktop-electron/src/main/markdown.ts +++ /dev/null @@ -1,16 +0,0 @@ -import { marked, type Tokens } from "marked" - -const renderer = new marked.Renderer() - -renderer.link = ({ href, title, text }: Tokens.Link) => { - const titleAttr = title ? ` title="${title}"` : "" - return `${text}` -} - -export function parseMarkdown(input: string) { - return marked(input, { - renderer, - breaks: false, - gfm: true, - }) -} diff --git a/packages/desktop-electron/src/main/menu.ts b/packages/desktop-electron/src/main/menu.ts deleted file mode 100644 index 0d9a697fa9..0000000000 --- a/packages/desktop-electron/src/main/menu.ts +++ /dev/null @@ -1,136 +0,0 @@ -import { Menu, shell } from "electron" - -import { UPDATER_ENABLED } from "./constants" -import { createMainWindow } from "./windows" - -type Deps = { - trigger: (id: string) => void - checkForUpdates: () => void - reload: () => void - relaunch: () => void -} - -export function createMenu(deps: Deps) { - if (process.platform !== "darwin") return - - const template: Electron.MenuItemConstructorOptions[] = [ - { - label: "OpenCode", - submenu: [ - { role: "about" }, - { - label: "Check for Updates...", - enabled: UPDATER_ENABLED, - click: () => deps.checkForUpdates(), - }, - { - label: "Reload Webview", - click: () => deps.reload(), - }, - { - label: "Restart", - click: () => deps.relaunch(), - }, - { type: "separator" }, - { role: "hide" }, - { role: "hideOthers" }, - { role: "unhide" }, - { type: "separator" }, - { role: "quit" }, - ], - }, - { - label: "File", - submenu: [ - { label: "New Session", accelerator: "Shift+Cmd+S", click: () => deps.trigger("session.new") }, - { label: "Open Project...", accelerator: "Cmd+O", click: () => deps.trigger("project.open") }, - { - label: "New Window", - accelerator: "Cmd+Shift+N", - click: () => createMainWindow(), - }, - { type: "separator" }, - { role: "close" }, - ], - }, - { - label: "Edit", - submenu: [ - { role: "undo" }, - { role: "redo" }, - { type: "separator" }, - { role: "cut" }, - { role: "copy" }, - { role: "paste" }, - { role: "selectAll" }, - ], - }, - { - label: "View", - submenu: [ - { label: "Toggle Sidebar", accelerator: "Cmd+B", click: () => deps.trigger("sidebar.toggle") }, - { label: "Toggle Terminal", accelerator: "Ctrl+`", click: () => deps.trigger("terminal.toggle") }, - { label: "Toggle File Tree", click: () => deps.trigger("fileTree.toggle") }, - { type: "separator" }, - { role: "reload" }, - { role: "toggleDevTools" }, - { type: "separator" }, - { role: "resetZoom" }, - { role: "zoomIn" }, - { role: "zoomOut" }, - { type: "separator" }, - { role: "togglefullscreen" }, - ], - }, - { - label: "Go", - submenu: [ - { label: "Back", accelerator: "Cmd+[", click: () => deps.trigger("common.goBack") }, - { label: "Forward", accelerator: "Cmd+]", click: () => deps.trigger("common.goForward") }, - { type: "separator" }, - { - label: "Previous Session", - accelerator: "Option+Up", - click: () => deps.trigger("session.previous"), - }, - { - label: "Next Session", - accelerator: "Option+Down", - click: () => deps.trigger("session.next"), - }, - { type: "separator" }, - { - label: "Previous Project", - accelerator: "Cmd+Option+Up", - click: () => deps.trigger("project.previous"), - }, - { - label: "Next Project", - accelerator: "Cmd+Option+Down", - click: () => deps.trigger("project.next"), - }, - ], - }, - { role: "windowMenu" }, - { - label: "Help", - submenu: [ - { label: "OpenCode Documentation", click: () => shell.openExternal("https://opencode.ai/docs") }, - { label: "Support Forum", click: () => shell.openExternal("https://discord.com/invite/opencode") }, - { type: "separator" }, - { type: "separator" }, - { - label: "Share Feedback", - click: () => - shell.openExternal("https://github.com/anomalyco/opencode/issues/new?template=feature_request.yml"), - }, - { - label: "Report a Bug", - click: () => shell.openExternal("https://github.com/anomalyco/opencode/issues/new?template=bug_report.yml"), - }, - ], - }, - ] - - Menu.setApplicationMenu(Menu.buildFromTemplate(template)) -} diff --git a/packages/desktop-electron/src/main/migrate.ts b/packages/desktop-electron/src/main/migrate.ts deleted file mode 100644 index 70e3dc9c75..0000000000 --- a/packages/desktop-electron/src/main/migrate.ts +++ /dev/null @@ -1,91 +0,0 @@ -import { app } from "electron" -import log from "electron-log/main.js" -import { existsSync, readdirSync, readFileSync } from "node:fs" -import { homedir } from "node:os" -import { join } from "node:path" -import { CHANNEL } from "./constants" -import { getStore } from "./store" - -const TAURI_MIGRATED_KEY = "tauriMigrated" - -// Resolve the directory where Tauri stored its .dat files for the given app identifier. -// Mirrors Tauri's AppLocalData / AppData resolution per OS. -function tauriDir(id: string) { - switch (process.platform) { - case "darwin": - return join(homedir(), "Library", "Application Support", id) - case "win32": - return join(process.env.APPDATA ?? join(homedir(), "AppData", "Roaming"), id) - default: - return join(process.env.XDG_DATA_HOME ?? join(homedir(), ".local", "share"), id) - } -} - -// The Tauri app identifier changes between dev/beta/prod builds. -const TAURI_APP_IDS: Record = { - dev: "ai.opencode.desktop.dev", - beta: "ai.opencode.desktop.beta", - prod: "ai.opencode.desktop", -} -function tauriAppId() { - return app.isPackaged ? TAURI_APP_IDS[CHANNEL] : "ai.opencode.desktop.dev" -} - -// Migrate a single Tauri .dat file into the corresponding electron-store. -// `opencode.settings.dat` is special: it maps to the `opencode.settings` store -// (the electron-store name without the `.dat` extension). All other .dat files -// keep their full filename as the electron-store name so they match what the -// renderer already passes via IPC (e.g. `"default.dat"`, `"opencode.global.dat"`). -function migrateFile(datPath: string, filename: string) { - let data: Record - try { - data = JSON.parse(readFileSync(datPath, "utf-8")) - } catch (err) { - log.warn("tauri migration: failed to parse", filename, err) - return - } - - // opencode.settings.dat → the electron settings store ("opencode.settings"). - // All other .dat files keep their full filename as the store name so they match - // what the renderer passes via IPC (e.g. "default.dat", "opencode.global.dat"). - const storeName = filename === "opencode.settings.dat" ? "opencode.settings" : filename - const target = getStore(storeName) - const migrated: string[] = [] - const skipped: string[] = [] - - for (const [key, value] of Object.entries(data)) { - // Don't overwrite values the user has already set in the Electron app. - if (target.has(key)) { - skipped.push(key) - continue - } - target.set(key, value) - migrated.push(key) - } - - log.log("tauri migration: migrated", filename, "→", storeName, { migrated, skipped }) -} - -export function migrate() { - if (getStore().get(TAURI_MIGRATED_KEY)) { - log.log("tauri migration: already done, skipping") - return - } - - const dir = tauriDir(tauriAppId()) - log.log("tauri migration: starting", { dir }) - - if (!existsSync(dir)) { - log.log("tauri migration: no tauri data directory found, nothing to migrate") - getStore().set(TAURI_MIGRATED_KEY, true) - return - } - - for (const filename of readdirSync(dir)) { - if (!filename.endsWith(".dat")) continue - migrateFile(join(dir, filename), filename) - } - - log.log("tauri migration: complete") - getStore().set(TAURI_MIGRATED_KEY, true) -} diff --git a/packages/desktop-electron/src/main/server.ts b/packages/desktop-electron/src/main/server.ts deleted file mode 100644 index 8c8f0895e9..0000000000 --- a/packages/desktop-electron/src/main/server.ts +++ /dev/null @@ -1,101 +0,0 @@ -import { app } from "electron" -import { DEFAULT_SERVER_URL_KEY, WSL_ENABLED_KEY } from "./constants" -import { getUserShell, loadShellEnv } from "./shell-env" -import { getStore } from "./store" - -export type WslConfig = { enabled: boolean } - -export type HealthCheck = { wait: Promise } - -export function getDefaultServerUrl(): string | null { - const value = getStore().get(DEFAULT_SERVER_URL_KEY) - return typeof value === "string" ? value : null -} - -export function setDefaultServerUrl(url: string | null) { - if (url) { - getStore().set(DEFAULT_SERVER_URL_KEY, url) - return - } - - getStore().delete(DEFAULT_SERVER_URL_KEY) -} - -export function getWslConfig(): WslConfig { - const value = getStore().get(WSL_ENABLED_KEY) - return { enabled: typeof value === "boolean" ? value : false } -} - -export function setWslConfig(config: WslConfig) { - getStore().set(WSL_ENABLED_KEY, config.enabled) -} - -export async function spawnLocalServer(hostname: string, port: number, password: string) { - prepareServerEnv(password) - const { Log, Server } = await import("virtual:opencode-server") - await Log.init({ level: "WARN" }) - const listener = await Server.listen({ - port, - hostname, - username: "opencode", - password, - cors: ["oc://renderer"], - }) - - const wait = (async () => { - const url = `http://${hostname}:${port}` - - const ready = async () => { - while (true) { - await new Promise((resolve) => setTimeout(resolve, 100)) - if (await checkHealth(url, password)) return - } - } - - await ready() - })() - - return { listener, health: { wait } } -} - -function prepareServerEnv(password: string) { - const shell = process.platform === "win32" ? null : getUserShell() - const shellEnv = shell ? (loadShellEnv(shell) ?? {}) : {} - const env = { - ...process.env, - ...shellEnv, - KILO_EXPERIMENTAL_ICON_DISCOVERY: "true", - KILO_EXPERIMENTAL_FILEWATCHER: "true", - KILO_CLIENT: "desktop", - KILO_SERVER_USERNAME: "opencode", - KILO_SERVER_PASSWORD: password, - XDG_STATE_HOME: app.getPath("userData"), - } - Object.assign(process.env, env) -} - -export async function checkHealth(url: string, password?: string | null): Promise { - let healthUrl: URL - try { - healthUrl = new URL("/global/health", url) - } catch { - return false - } - - const headers = new Headers() - if (password) { - const auth = Buffer.from(`opencode:${password}`).toString("base64") - headers.set("authorization", `Basic ${auth}`) - } - - try { - const res = await fetch(healthUrl, { - method: "GET", - headers, - signal: AbortSignal.timeout(3000), - }) - return res.ok - } catch { - return false - } -} diff --git a/packages/desktop-electron/src/main/shell-env.test.ts b/packages/desktop-electron/src/main/shell-env.test.ts deleted file mode 100644 index 7499cd1219..0000000000 --- a/packages/desktop-electron/src/main/shell-env.test.ts +++ /dev/null @@ -1,43 +0,0 @@ -import { describe, expect, test } from "bun:test" - -import { isNushell, mergeShellEnv, parseShellEnv } from "./shell-env" - -describe("shell env", () => { - test("parseShellEnv supports null-delimited pairs", () => { - const env = parseShellEnv(Buffer.from("PATH=/usr/bin:/bin\0FOO=bar=baz\0\0")) - - expect(env.PATH).toBe("/usr/bin:/bin") - expect(env.FOO).toBe("bar=baz") - }) - - test("parseShellEnv ignores invalid entries", () => { - const env = parseShellEnv(Buffer.from("INVALID\0=empty\0OK=1\0")) - - expect(Object.keys(env).length).toBe(1) - expect(env.OK).toBe("1") - }) - - test("mergeShellEnv keeps explicit overrides", () => { - const env = mergeShellEnv( - { - PATH: "/shell/path", - HOME: "/tmp/home", - }, - { - PATH: "/desktop/path", - KILO_CLIENT: "desktop", - }, - ) - - expect(env.PATH).toBe("/desktop/path") - expect(env.HOME).toBe("/tmp/home") - expect(env.KILO_CLIENT).toBe("desktop") - }) - - test("isNushell handles path and binary name", () => { - expect(isNushell("nu")).toBe(true) - expect(isNushell("/opt/homebrew/bin/nu")).toBe(true) - expect(isNushell("C:\\Program Files\\nu.exe")).toBe(true) - expect(isNushell("/bin/zsh")).toBe(false) - }) -}) diff --git a/packages/desktop-electron/src/main/shell-env.ts b/packages/desktop-electron/src/main/shell-env.ts deleted file mode 100644 index f57677323c..0000000000 --- a/packages/desktop-electron/src/main/shell-env.ts +++ /dev/null @@ -1,88 +0,0 @@ -import { spawnSync } from "node:child_process" -import { basename } from "node:path" - -const TIMEOUT = 5_000 - -type Probe = { type: "Loaded"; value: Record } | { type: "Timeout" } | { type: "Unavailable" } - -export function getUserShell() { - return process.env.SHELL || "/bin/sh" -} - -export function parseShellEnv(out: Buffer) { - const env: Record = {} - for (const line of out.toString("utf8").split("\0")) { - if (!line) continue - const ix = line.indexOf("=") - if (ix <= 0) continue - env[line.slice(0, ix)] = line.slice(ix + 1) - } - return env -} - -function probe(shell: string, mode: "-il" | "-l"): Probe { - const out = spawnSync(shell, [mode, "-c", "env -0"], { - stdio: ["ignore", "pipe", "ignore"], - timeout: TIMEOUT, - windowsHide: true, - }) - - const err = out.error as NodeJS.ErrnoException | undefined - if (err) { - if (err.code === "ETIMEDOUT") return { type: "Timeout" } - console.log(`[server] Shell env probe failed for ${shell} ${mode}: ${err.message}`) - return { type: "Unavailable" } - } - - if (out.status !== 0) { - console.log(`[server] Shell env probe exited with non-zero status for ${shell} ${mode}`) - return { type: "Unavailable" } - } - - const env = parseShellEnv(out.stdout) - if (Object.keys(env).length === 0) { - console.log(`[server] Shell env probe returned empty env for ${shell} ${mode}`) - return { type: "Unavailable" } - } - - return { type: "Loaded", value: env } -} - -export function isNushell(shell: string) { - const name = basename(shell).toLowerCase() - const raw = shell.toLowerCase() - return name === "nu" || name === "nu.exe" || raw.endsWith("\\nu.exe") -} - -export function loadShellEnv(shell: string) { - if (isNushell(shell)) { - console.log(`[server] Skipping shell env probe for nushell: ${shell}`) - return null - } - - const interactive = probe(shell, "-il") - if (interactive.type === "Loaded") { - console.log(`[server] Loaded shell environment with -il (${Object.keys(interactive.value).length} vars)`) - return interactive.value - } - if (interactive.type === "Timeout") { - console.warn(`[server] Interactive shell env probe timed out: ${shell}`) - return null - } - - const login = probe(shell, "-l") - if (login.type === "Loaded") { - console.log(`[server] Loaded shell environment with -l (${Object.keys(login.value).length} vars)`) - return login.value - } - - console.warn(`[server] Falling back to app environment: ${shell}`) - return null -} - -export function mergeShellEnv(shell: Record | null, env: Record) { - return { - ...shell, - ...env, - } -} diff --git a/packages/desktop-electron/src/main/store.ts b/packages/desktop-electron/src/main/store.ts deleted file mode 100644 index 61f0c0a493..0000000000 --- a/packages/desktop-electron/src/main/store.ts +++ /dev/null @@ -1,17 +0,0 @@ -import Store from "electron-store" - -import { SETTINGS_STORE } from "./constants" - -const cache = new Map() - -// We cannot instantiate the electron-store at module load time because -// module import hoisting causes this to run before app.setPath("userData", ...) -// in index.ts has executed, which would result in files being written to the default directory -// (e.g. bad: %APPDATA%\@opencode-ai\desktop-electron\opencode.settings vs good: %APPDATA%\ai.opencode.desktop.dev\opencode.settings). -export function getStore(name = SETTINGS_STORE) { - const cached = cache.get(name) - if (cached) return cached - const next = new Store({ name, fileExtension: "", accessPropertiesByDotNotation: false }) - cache.set(name, next) - return next -} diff --git a/packages/desktop-electron/src/main/windows.ts b/packages/desktop-electron/src/main/windows.ts deleted file mode 100644 index 337e1ca0bc..0000000000 --- a/packages/desktop-electron/src/main/windows.ts +++ /dev/null @@ -1,206 +0,0 @@ -import windowState from "electron-window-state" -import { app, BrowserWindow, net, nativeImage, nativeTheme, protocol } from "electron" -import { dirname, isAbsolute, join, relative, resolve } from "node:path" -import { fileURLToPath, pathToFileURL } from "node:url" -import type { TitlebarTheme } from "../preload/types" - -const root = dirname(fileURLToPath(import.meta.url)) -const rendererRoot = join(root, "../renderer") -const rendererProtocol = "oc" -const rendererHost = "renderer" - -protocol.registerSchemesAsPrivileged([ - { - scheme: rendererProtocol, - privileges: { - secure: true, - standard: true, - supportFetchAPI: true, - }, - }, -]) - -let backgroundColor: string | undefined - -export function setBackgroundColor(color: string) { - backgroundColor = color -} - -export function getBackgroundColor(): string | undefined { - return backgroundColor -} - -function iconsDir() { - return app.isPackaged ? join(process.resourcesPath, "icons") : join(root, "../../resources/icons") -} - -function iconPath() { - const ext = process.platform === "win32" ? "ico" : "png" - return join(iconsDir(), `icon.${ext}`) -} - -function tone() { - return nativeTheme.shouldUseDarkColors ? "dark" : "light" -} - -function overlay(theme: Partial = {}) { - const mode = theme.mode ?? tone() - return { - color: "#00000000", - symbolColor: mode === "dark" ? "white" : "black", - height: 40, - } -} - -export function setTitlebar(win: BrowserWindow, theme: Partial = {}) { - if (process.platform !== "win32") return - win.setTitleBarOverlay(overlay(theme)) -} - -export function setDockIcon() { - if (process.platform !== "darwin") return - const icon = nativeImage.createFromPath(join(iconsDir(), "dock.png")) - if (!icon.isEmpty()) app.dock?.setIcon(icon) -} - -export function createMainWindow() { - const state = windowState({ - defaultWidth: 1280, - defaultHeight: 800, - }) - - const mode = tone() - const win = new BrowserWindow({ - x: state.x, - y: state.y, - width: state.width, - height: state.height, - show: false, - title: "OpenCode", - icon: iconPath(), - backgroundColor, - ...(process.platform === "darwin" - ? { - titleBarStyle: "hidden" as const, - trafficLightPosition: { x: 12, y: 14 }, - } - : {}), - ...(process.platform === "win32" - ? { - frame: false, - titleBarStyle: "hidden" as const, - titleBarOverlay: overlay({ mode }), - } - : {}), - webPreferences: { - preload: join(root, "../preload/index.js"), - contextIsolation: true, - nodeIntegration: false, - sandbox: true, - }, - }) - - win.webContents.session.webRequest.onBeforeSendHeaders((details, callback) => { - const { requestHeaders } = details - upsertKeyValue(requestHeaders, "Access-Control-Allow-Origin", ["*"]) - callback({ requestHeaders }) - }) - - win.webContents.session.webRequest.onHeadersReceived((details, callback) => { - const { responseHeaders = {} } = details - upsertKeyValue(responseHeaders, "Access-Control-Allow-Origin", ["*"]) - upsertKeyValue(responseHeaders, "Access-Control-Allow-Headers", ["*"]) - callback({ responseHeaders }) - }) - - state.manage(win) - loadWindow(win, "index.html") - wireZoom(win) - - win.once("ready-to-show", () => { - win.show() - }) - - return win -} - -export function createLoadingWindow() { - const mode = tone() - const win = new BrowserWindow({ - width: 640, - height: 480, - resizable: false, - center: true, - show: true, - icon: iconPath(), - backgroundColor, - ...(process.platform === "darwin" ? { titleBarStyle: "hidden" as const } : {}), - ...(process.platform === "win32" - ? { - frame: false, - titleBarStyle: "hidden" as const, - titleBarOverlay: overlay({ mode }), - } - : {}), - webPreferences: { - preload: join(root, "../preload/index.js"), - contextIsolation: true, - nodeIntegration: false, - sandbox: true, - }, - }) - - loadWindow(win, "loading.html") - - return win -} - -export function registerRendererProtocol() { - if (protocol.isProtocolHandled(rendererProtocol)) return - - protocol.handle(rendererProtocol, (request) => { - const url = new URL(request.url) - if (url.host !== rendererHost) { - return new Response("Not found", { status: 404 }) - } - - const file = resolve(rendererRoot, `.${decodeURIComponent(url.pathname)}`) - const rel = relative(rendererRoot, file) - if (rel.startsWith("..") || isAbsolute(rel)) { - return new Response("Not found", { status: 404 }) - } - - return net.fetch(pathToFileURL(file).toString()) - }) -} - -function loadWindow(win: BrowserWindow, html: string) { - const devUrl = process.env.ELECTRON_RENDERER_URL - if (devUrl) { - const url = new URL(html, devUrl) - void win.loadURL(url.toString()) - return - } - - void win.loadURL(`${rendererProtocol}://${rendererHost}/${html}`) -} -function wireZoom(win: BrowserWindow) { - win.webContents.setZoomFactor(1) - win.webContents.on("zoom-changed", () => { - win.webContents.setZoomFactor(1) - }) -} - -function upsertKeyValue(obj: Record, keyToChange: string, value: any) { - const keyToChangeLower = keyToChange.toLowerCase() - for (const key of Object.keys(obj)) { - if (key.toLowerCase() === keyToChangeLower) { - // Reassign old key - obj[key] = value - // Done - return - } - } - // Insert at end instead - obj[keyToChange] = value -} diff --git a/packages/desktop-electron/src/preload/index.ts b/packages/desktop-electron/src/preload/index.ts deleted file mode 100644 index 6261419ca5..0000000000 --- a/packages/desktop-electron/src/preload/index.ts +++ /dev/null @@ -1,71 +0,0 @@ -import { contextBridge, ipcRenderer } from "electron" -import type { ElectronAPI, InitStep, SqliteMigrationProgress } from "./types" - -const api: ElectronAPI = { - killSidecar: () => ipcRenderer.invoke("kill-sidecar"), - installCli: () => ipcRenderer.invoke("install-cli"), - awaitInitialization: (onStep) => { - const handler = (_: unknown, step: InitStep) => onStep(step) - ipcRenderer.on("init-step", handler) - return ipcRenderer.invoke("await-initialization").finally(() => { - ipcRenderer.removeListener("init-step", handler) - }) - }, - getWindowConfig: () => ipcRenderer.invoke("get-window-config"), - consumeInitialDeepLinks: () => ipcRenderer.invoke("consume-initial-deep-links"), - getDefaultServerUrl: () => ipcRenderer.invoke("get-default-server-url"), - setDefaultServerUrl: (url) => ipcRenderer.invoke("set-default-server-url", url), - getWslConfig: () => ipcRenderer.invoke("get-wsl-config"), - setWslConfig: (config) => ipcRenderer.invoke("set-wsl-config", config), - getDisplayBackend: () => ipcRenderer.invoke("get-display-backend"), - setDisplayBackend: (backend) => ipcRenderer.invoke("set-display-backend", backend), - parseMarkdownCommand: (markdown) => ipcRenderer.invoke("parse-markdown", markdown), - checkAppExists: (appName) => ipcRenderer.invoke("check-app-exists", appName), - wslPath: (path, mode) => ipcRenderer.invoke("wsl-path", path, mode), - resolveAppPath: (appName) => ipcRenderer.invoke("resolve-app-path", appName), - storeGet: (name, key) => ipcRenderer.invoke("store-get", name, key), - storeSet: (name, key, value) => ipcRenderer.invoke("store-set", name, key, value), - storeDelete: (name, key) => ipcRenderer.invoke("store-delete", name, key), - storeClear: (name) => ipcRenderer.invoke("store-clear", name), - storeKeys: (name) => ipcRenderer.invoke("store-keys", name), - storeLength: (name) => ipcRenderer.invoke("store-length", name), - - getWindowCount: () => ipcRenderer.invoke("get-window-count"), - onSqliteMigrationProgress: (cb) => { - const handler = (_: unknown, progress: SqliteMigrationProgress) => cb(progress) - ipcRenderer.on("sqlite-migration-progress", handler) - return () => ipcRenderer.removeListener("sqlite-migration-progress", handler) - }, - onMenuCommand: (cb) => { - const handler = (_: unknown, id: string) => cb(id) - ipcRenderer.on("menu-command", handler) - return () => ipcRenderer.removeListener("menu-command", handler) - }, - onDeepLink: (cb) => { - const handler = (_: unknown, urls: string[]) => cb(urls) - ipcRenderer.on("deep-link", handler) - return () => ipcRenderer.removeListener("deep-link", handler) - }, - - openDirectoryPicker: (opts) => ipcRenderer.invoke("open-directory-picker", opts), - openFilePicker: (opts) => ipcRenderer.invoke("open-file-picker", opts), - saveFilePicker: (opts) => ipcRenderer.invoke("save-file-picker", opts), - openLink: (url) => ipcRenderer.send("open-link", url), - openPath: (path, app) => ipcRenderer.invoke("open-path", path, app), - readClipboardImage: () => ipcRenderer.invoke("read-clipboard-image"), - showNotification: (title, body) => ipcRenderer.send("show-notification", title, body), - getWindowFocused: () => ipcRenderer.invoke("get-window-focused"), - setWindowFocus: () => ipcRenderer.invoke("set-window-focus"), - showWindow: () => ipcRenderer.invoke("show-window"), - relaunch: () => ipcRenderer.send("relaunch"), - getZoomFactor: () => ipcRenderer.invoke("get-zoom-factor"), - setZoomFactor: (factor) => ipcRenderer.invoke("set-zoom-factor", factor), - setTitlebar: (theme) => ipcRenderer.invoke("set-titlebar", theme), - loadingWindowComplete: () => ipcRenderer.send("loading-window-complete"), - runUpdater: (alertOnFail) => ipcRenderer.invoke("run-updater", alertOnFail), - checkUpdate: () => ipcRenderer.invoke("check-update"), - installUpdate: () => ipcRenderer.invoke("install-update"), - setBackgroundColor: (color: string) => ipcRenderer.invoke("set-background-color", color), -} - -contextBridge.exposeInMainWorld("api", api) diff --git a/packages/desktop-electron/src/preload/types.ts b/packages/desktop-electron/src/preload/types.ts deleted file mode 100644 index 6e22954d18..0000000000 --- a/packages/desktop-electron/src/preload/types.ts +++ /dev/null @@ -1,79 +0,0 @@ -export type InitStep = { phase: "server_waiting" } | { phase: "sqlite_waiting" } | { phase: "done" } - -export type ServerReadyData = { - url: string - username: string | null - password: string | null -} - -export type SqliteMigrationProgress = { type: "InProgress"; value: number } | { type: "Done" } - -export type WslConfig = { enabled: boolean } - -export type LinuxDisplayBackend = "wayland" | "auto" -export type TitlebarTheme = { - mode: "light" | "dark" -} - -export type WindowConfig = { - updaterEnabled: boolean -} - -export type ElectronAPI = { - killSidecar: () => Promise - installCli: () => Promise - awaitInitialization: (onStep: (step: InitStep) => void) => Promise - getWindowConfig: () => Promise - consumeInitialDeepLinks: () => Promise - getDefaultServerUrl: () => Promise - setDefaultServerUrl: (url: string | null) => Promise - getWslConfig: () => Promise - setWslConfig: (config: WslConfig) => Promise - getDisplayBackend: () => Promise - setDisplayBackend: (backend: LinuxDisplayBackend | null) => Promise - parseMarkdownCommand: (markdown: string) => Promise - checkAppExists: (appName: string) => Promise - wslPath: (path: string, mode: "windows" | "linux" | null) => Promise - resolveAppPath: (appName: string) => Promise - storeGet: (name: string, key: string) => Promise - storeSet: (name: string, key: string, value: string) => Promise - storeDelete: (name: string, key: string) => Promise - storeClear: (name: string) => Promise - storeKeys: (name: string) => Promise - storeLength: (name: string) => Promise - - getWindowCount: () => Promise - onSqliteMigrationProgress: (cb: (progress: SqliteMigrationProgress) => void) => () => void - onMenuCommand: (cb: (id: string) => void) => () => void - onDeepLink: (cb: (urls: string[]) => void) => () => void - - openDirectoryPicker: (opts?: { - multiple?: boolean - title?: string - defaultPath?: string - }) => Promise - openFilePicker: (opts?: { - multiple?: boolean - title?: string - defaultPath?: string - accept?: string[] - extensions?: string[] - }) => Promise - saveFilePicker: (opts?: { title?: string; defaultPath?: string }) => Promise - openLink: (url: string) => void - openPath: (path: string, app?: string) => Promise - readClipboardImage: () => Promise<{ buffer: ArrayBuffer; width: number; height: number } | null> - showNotification: (title: string, body?: string) => void - getWindowFocused: () => Promise - setWindowFocus: () => Promise - showWindow: () => Promise - relaunch: () => void - getZoomFactor: () => Promise - setZoomFactor: (factor: number) => Promise - setTitlebar: (theme: TitlebarTheme) => Promise - loadingWindowComplete: () => void - runUpdater: (alertOnFail: boolean) => Promise - checkUpdate: () => Promise<{ updateAvailable: boolean; version?: string }> - installUpdate: () => Promise - setBackgroundColor: (color: string) => Promise -} diff --git a/packages/desktop-electron/src/renderer/cli.ts b/packages/desktop-electron/src/renderer/cli.ts deleted file mode 100644 index 11d3c1f1b0..0000000000 --- a/packages/desktop-electron/src/renderer/cli.ts +++ /dev/null @@ -1,12 +0,0 @@ -import { initI18n, t } from "./i18n" - -export async function installCli(): Promise { - await initI18n() - - try { - const path = await window.api.installCli() - window.alert(t("desktop.cli.installed.message", { path })) - } catch (e) { - window.alert(t("desktop.cli.failed.message", { error: String(e) })) - } -} diff --git a/packages/desktop-electron/src/renderer/env.d.ts b/packages/desktop-electron/src/renderer/env.d.ts deleted file mode 100644 index 6dff3baf1c..0000000000 --- a/packages/desktop-electron/src/renderer/env.d.ts +++ /dev/null @@ -1,10 +0,0 @@ -import type { ElectronAPI } from "../preload/types" - -declare global { - interface Window { - api: ElectronAPI - __OPENCODE__?: { - deepLinks?: string[] - } - } -} diff --git a/packages/desktop-electron/src/renderer/html.test.ts b/packages/desktop-electron/src/renderer/html.test.ts deleted file mode 100644 index 1fc5c87178..0000000000 --- a/packages/desktop-electron/src/renderer/html.test.ts +++ /dev/null @@ -1,62 +0,0 @@ -import { describe, expect, test } from "bun:test" -import { join, dirname, resolve } from "node:path" -import { existsSync } from "node:fs" -import { fileURLToPath } from "node:url" - -const dir = dirname(fileURLToPath(import.meta.url)) -const root = resolve(dir, "../..") - -const html = async (name: string) => Bun.file(join(dir, name)).text() - -/** - * Packaged Electron windows load renderer HTML via the privileged `oc://` - * protocol. Root-relative asset paths like `src="/foo.js"` would resolve from - * the protocol origin root instead of relative to the current HTML entrypoint. - * - * All local resource references must use relative paths (`./`). - */ -describe("electron renderer html", () => { - for (const name of ["index.html", "loading.html"]) { - describe(name, () => { - test("script src attributes use relative paths", async () => { - const content = await html(name) - const srcs = [...content.matchAll(/\bsrc=["']([^"']+)["']/g)].map((m) => m[1]) - for (const src of srcs) { - expect(src).not.toMatch(/^\/[^/]/) - } - }) - - test("link href attributes use relative paths", async () => { - const content = await html(name) - const hrefs = [...content.matchAll(/]+href=["']([^"']+)["']/g)].map((m) => m[1]) - for (const href of hrefs) { - expect(href).not.toMatch(/^\/[^/]/) - } - }) - - test("no web manifest link (not applicable in Electron)", async () => { - const content = await html(name) - expect(content).not.toContain('rel="manifest"') - }) - }) - } -}) - -/** - * Vite resolves `publicDir` relative to `root`, not the config file. - * This test reads the actual values from electron.vite.config.ts to catch - * regressions where the publicDir path no longer resolves correctly - * after the renderer root is accounted for. - */ -describe("electron vite publicDir", () => { - test("configured publicDir resolves to a directory with oc-theme-preload.js", async () => { - const config = await Bun.file(join(root, "electron.vite.config.ts")).text() - const pub = config.match(/publicDir:\s*["']([^"']+)["']/) - const rendererRoot = config.match(/root:\s*["']([^"']+)["']/) - expect(pub).not.toBeNull() - expect(rendererRoot).not.toBeNull() - const resolved = resolve(root, rendererRoot![1], pub![1]) - expect(existsSync(resolved)).toBe(true) - expect(existsSync(join(resolved, "oc-theme-preload.js"))).toBe(true) - }) -}) diff --git a/packages/desktop-electron/src/renderer/i18n/ar.ts b/packages/desktop-electron/src/renderer/i18n/ar.ts deleted file mode 100644 index fdbf0a8047..0000000000 --- a/packages/desktop-electron/src/renderer/i18n/ar.ts +++ /dev/null @@ -1,26 +0,0 @@ -export const dict = { - "desktop.menu.checkForUpdates": "التحقق من وجود تحديثات...", - "desktop.menu.installCli": "تثبيت CLI...", - "desktop.menu.reloadWebview": "إعادة تحميل Webview", - "desktop.menu.restart": "إعادة تشغيل", - - "desktop.dialog.chooseFolder": "اختر مجلدًا", - "desktop.dialog.chooseFile": "اختر ملفًا", - "desktop.dialog.saveFile": "حفظ ملف", - - "desktop.updater.checkFailed.title": "فشل التحقق من التحديثات", - "desktop.updater.checkFailed.message": "فشل التحقق من وجود تحديثات", - "desktop.updater.none.title": "لا توجد تحديثات متاحة", - "desktop.updater.none.message": "أنت تستخدم بالفعل أحدث إصدار من OpenCode", - "desktop.updater.downloadFailed.title": "فشل التحديث", - "desktop.updater.downloadFailed.message": "فشل تنزيل التحديث", - "desktop.updater.downloaded.title": "تم تنزيل التحديث", - "desktop.updater.downloaded.prompt": "تم تنزيل إصدار {{version}} من OpenCode، هل ترغب في تثبيته وإعادة تشغيله؟", - "desktop.updater.installFailed.title": "فشل التحديث", - "desktop.updater.installFailed.message": "فشل تثبيت التحديث", - - "desktop.cli.installed.title": "تم تثبيت CLI", - "desktop.cli.installed.message": "تم تثبيت CLI في {{path}}\n\nأعد تشغيل الطرفية لاستخدام الأمر 'opencode'.", - "desktop.cli.failed.title": "فشل التثبيت", - "desktop.cli.failed.message": "فشل تثبيت CLI: {{error}}", -} diff --git a/packages/desktop-electron/src/renderer/i18n/br.ts b/packages/desktop-electron/src/renderer/i18n/br.ts deleted file mode 100644 index 75fe2dc32b..0000000000 --- a/packages/desktop-electron/src/renderer/i18n/br.ts +++ /dev/null @@ -1,27 +0,0 @@ -export const dict = { - "desktop.menu.checkForUpdates": "Verificar atualizações...", - "desktop.menu.installCli": "Instalar CLI...", - "desktop.menu.reloadWebview": "Recarregar Webview", - "desktop.menu.restart": "Reiniciar", - - "desktop.dialog.chooseFolder": "Escolher uma pasta", - "desktop.dialog.chooseFile": "Escolher um arquivo", - "desktop.dialog.saveFile": "Salvar arquivo", - - "desktop.updater.checkFailed.title": "Falha ao verificar atualizações", - "desktop.updater.checkFailed.message": "Falha ao verificar atualizações", - "desktop.updater.none.title": "Nenhuma atualização disponível", - "desktop.updater.none.message": "Você já está usando a versão mais recente do OpenCode", - "desktop.updater.downloadFailed.title": "Falha na atualização", - "desktop.updater.downloadFailed.message": "Falha ao baixar a atualização", - "desktop.updater.downloaded.title": "Atualização baixada", - "desktop.updater.downloaded.prompt": - "A versão {{version}} do OpenCode foi baixada. Você gostaria de instalá-la e reiniciar?", - "desktop.updater.installFailed.title": "Falha na atualização", - "desktop.updater.installFailed.message": "Falha ao instalar a atualização", - - "desktop.cli.installed.title": "CLI instalada", - "desktop.cli.installed.message": "CLI instalada em {{path}}\n\nReinicie seu terminal para usar o comando 'opencode'.", - "desktop.cli.failed.title": "Falha na instalação", - "desktop.cli.failed.message": "Falha ao instalar a CLI: {{error}}", -} diff --git a/packages/desktop-electron/src/renderer/i18n/bs.ts b/packages/desktop-electron/src/renderer/i18n/bs.ts deleted file mode 100644 index 58c266f530..0000000000 --- a/packages/desktop-electron/src/renderer/i18n/bs.ts +++ /dev/null @@ -1,28 +0,0 @@ -export const dict = { - "desktop.menu.checkForUpdates": "Provjeri ažuriranja...", - "desktop.menu.installCli": "Instaliraj CLI...", - "desktop.menu.reloadWebview": "Ponovo učitavanje webview-a", - "desktop.menu.restart": "Restartuj", - - "desktop.dialog.chooseFolder": "Odaberi folder", - "desktop.dialog.chooseFile": "Odaberi datoteku", - "desktop.dialog.saveFile": "Sačuvaj datoteku", - - "desktop.updater.checkFailed.title": "Provjera ažuriranja nije uspjela", - "desktop.updater.checkFailed.message": "Nije moguće provjeriti ažuriranja", - "desktop.updater.none.title": "Nema dostupnog ažuriranja", - "desktop.updater.none.message": "Već koristiš najnoviju verziju OpenCode-a", - "desktop.updater.downloadFailed.title": "Ažuriranje nije uspjelo", - "desktop.updater.downloadFailed.message": "Neuspjelo preuzimanje ažuriranja", - "desktop.updater.downloaded.title": "Ažuriranje preuzeto", - "desktop.updater.downloaded.prompt": - "Verzija {{version}} OpenCode-a je preuzeta. Želiš li da je instaliraš i ponovo pokreneš aplikaciju?", - "desktop.updater.installFailed.title": "Ažuriranje nije uspjelo", - "desktop.updater.installFailed.message": "Neuspjela instalacija ažuriranja", - - "desktop.cli.installed.title": "CLI instaliran", - "desktop.cli.installed.message": - "CLI je instaliran u {{path}}\n\nRestartuj terminal da bi koristio komandu 'opencode'.", - "desktop.cli.failed.title": "Instalacija nije uspjela", - "desktop.cli.failed.message": "Neuspjela instalacija CLI-a: {{error}}", -} diff --git a/packages/desktop-electron/src/renderer/i18n/da.ts b/packages/desktop-electron/src/renderer/i18n/da.ts deleted file mode 100644 index 2109495f76..0000000000 --- a/packages/desktop-electron/src/renderer/i18n/da.ts +++ /dev/null @@ -1,28 +0,0 @@ -export const dict = { - "desktop.menu.checkForUpdates": "Tjek for opdateringer...", - "desktop.menu.installCli": "Installer CLI...", - "desktop.menu.reloadWebview": "Genindlæs Webview", - "desktop.menu.restart": "Genstart", - - "desktop.dialog.chooseFolder": "Vælg en mappe", - "desktop.dialog.chooseFile": "Vælg en fil", - "desktop.dialog.saveFile": "Gem fil", - - "desktop.updater.checkFailed.title": "Opdateringstjek mislykkedes", - "desktop.updater.checkFailed.message": "Kunne ikke tjekke for opdateringer", - "desktop.updater.none.title": "Ingen opdatering tilgængelig", - "desktop.updater.none.message": "Du bruger allerede den nyeste version af OpenCode", - "desktop.updater.downloadFailed.title": "Opdatering mislykkedes", - "desktop.updater.downloadFailed.message": "Kunne ikke downloade opdateringen", - "desktop.updater.downloaded.title": "Opdatering downloadet", - "desktop.updater.downloaded.prompt": - "Version {{version}} af OpenCode er blevet downloadet. Vil du installere den og genstarte?", - "desktop.updater.installFailed.title": "Opdatering mislykkedes", - "desktop.updater.installFailed.message": "Kunne ikke installere opdateringen", - - "desktop.cli.installed.title": "CLI installeret", - "desktop.cli.installed.message": - "CLI installeret i {{path}}\n\nGenstart din terminal for at bruge 'opencode'-kommandoen.", - "desktop.cli.failed.title": "Installation mislykkedes", - "desktop.cli.failed.message": "Kunne ikke installere CLI: {{error}}", -} diff --git a/packages/desktop-electron/src/renderer/i18n/de.ts b/packages/desktop-electron/src/renderer/i18n/de.ts deleted file mode 100644 index 38ad8096e3..0000000000 --- a/packages/desktop-electron/src/renderer/i18n/de.ts +++ /dev/null @@ -1,28 +0,0 @@ -export const dict = { - "desktop.menu.checkForUpdates": "Nach Updates suchen...", - "desktop.menu.installCli": "CLI installieren...", - "desktop.menu.reloadWebview": "Webview neu laden", - "desktop.menu.restart": "Neustart", - - "desktop.dialog.chooseFolder": "Ordner auswählen", - "desktop.dialog.chooseFile": "Datei auswählen", - "desktop.dialog.saveFile": "Datei speichern", - - "desktop.updater.checkFailed.title": "Updateprüfung fehlgeschlagen", - "desktop.updater.checkFailed.message": "Updates konnten nicht geprüft werden", - "desktop.updater.none.title": "Kein Update verfügbar", - "desktop.updater.none.message": "Sie verwenden bereits die neueste Version von OpenCode", - "desktop.updater.downloadFailed.title": "Update fehlgeschlagen", - "desktop.updater.downloadFailed.message": "Update konnte nicht heruntergeladen werden", - "desktop.updater.downloaded.title": "Update heruntergeladen", - "desktop.updater.downloaded.prompt": - "Version {{version}} von OpenCode wurde heruntergeladen. Möchten Sie sie installieren und neu starten?", - "desktop.updater.installFailed.title": "Update fehlgeschlagen", - "desktop.updater.installFailed.message": "Update konnte nicht installiert werden", - - "desktop.cli.installed.title": "CLI installiert", - "desktop.cli.installed.message": - "CLI wurde in {{path}} installiert\n\nStarten Sie Ihr Terminal neu, um den Befehl 'opencode' zu verwenden.", - "desktop.cli.failed.title": "Installation fehlgeschlagen", - "desktop.cli.failed.message": "CLI konnte nicht installiert werden: {{error}}", -} diff --git a/packages/desktop-electron/src/renderer/i18n/en.ts b/packages/desktop-electron/src/renderer/i18n/en.ts deleted file mode 100644 index 4c30380d56..0000000000 --- a/packages/desktop-electron/src/renderer/i18n/en.ts +++ /dev/null @@ -1,27 +0,0 @@ -export const dict = { - "desktop.menu.checkForUpdates": "Check for Updates...", - "desktop.menu.installCli": "Install CLI...", - "desktop.menu.reloadWebview": "Reload Webview", - "desktop.menu.restart": "Restart", - - "desktop.dialog.chooseFolder": "Choose a folder", - "desktop.dialog.chooseFile": "Choose a file", - "desktop.dialog.saveFile": "Save file", - - "desktop.updater.checkFailed.title": "Update Check Failed", - "desktop.updater.checkFailed.message": "Failed to check for updates", - "desktop.updater.none.title": "No Update Available", - "desktop.updater.none.message": "You are already using the latest version of OpenCode", - "desktop.updater.downloadFailed.title": "Update Failed", - "desktop.updater.downloadFailed.message": "Failed to download update", - "desktop.updater.downloaded.title": "Update Downloaded", - "desktop.updater.downloaded.prompt": - "Version {{version}} of OpenCode has been downloaded, would you like to install it and relaunch?", - "desktop.updater.installFailed.title": "Update Failed", - "desktop.updater.installFailed.message": "Failed to install update", - - "desktop.cli.installed.title": "CLI Installed", - "desktop.cli.installed.message": "CLI installed to {{path}}\n\nRestart your terminal to use the 'opencode' command.", - "desktop.cli.failed.title": "Installation Failed", - "desktop.cli.failed.message": "Failed to install CLI: {{error}}", -} diff --git a/packages/desktop-electron/src/renderer/i18n/es.ts b/packages/desktop-electron/src/renderer/i18n/es.ts deleted file mode 100644 index 80504a8f24..0000000000 --- a/packages/desktop-electron/src/renderer/i18n/es.ts +++ /dev/null @@ -1,27 +0,0 @@ -export const dict = { - "desktop.menu.checkForUpdates": "Buscar actualizaciones...", - "desktop.menu.installCli": "Instalar CLI...", - "desktop.menu.reloadWebview": "Recargar Webview", - "desktop.menu.restart": "Reiniciar", - - "desktop.dialog.chooseFolder": "Elegir una carpeta", - "desktop.dialog.chooseFile": "Elegir un archivo", - "desktop.dialog.saveFile": "Guardar archivo", - - "desktop.updater.checkFailed.title": "Comprobación de actualizaciones fallida", - "desktop.updater.checkFailed.message": "No se pudieron buscar actualizaciones", - "desktop.updater.none.title": "No hay actualizaciones disponibles", - "desktop.updater.none.message": "Ya estás usando la versión más reciente de OpenCode", - "desktop.updater.downloadFailed.title": "Actualización fallida", - "desktop.updater.downloadFailed.message": "No se pudo descargar la actualización", - "desktop.updater.downloaded.title": "Actualización descargada", - "desktop.updater.downloaded.prompt": - "Se ha descargado la versión {{version}} de OpenCode. ¿Quieres instalarla y reiniciar?", - "desktop.updater.installFailed.title": "Actualización fallida", - "desktop.updater.installFailed.message": "No se pudo instalar la actualización", - - "desktop.cli.installed.title": "CLI instalada", - "desktop.cli.installed.message": "CLI instalada en {{path}}\n\nReinicia tu terminal para usar el comando 'opencode'.", - "desktop.cli.failed.title": "Instalación fallida", - "desktop.cli.failed.message": "No se pudo instalar la CLI: {{error}}", -} diff --git a/packages/desktop-electron/src/renderer/i18n/fr.ts b/packages/desktop-electron/src/renderer/i18n/fr.ts deleted file mode 100644 index 4f0bb2b16c..0000000000 --- a/packages/desktop-electron/src/renderer/i18n/fr.ts +++ /dev/null @@ -1,28 +0,0 @@ -export const dict = { - "desktop.menu.checkForUpdates": "Vérifier les mises à jour...", - "desktop.menu.installCli": "Installer la CLI...", - "desktop.menu.reloadWebview": "Recharger la Webview", - "desktop.menu.restart": "Redémarrer", - - "desktop.dialog.chooseFolder": "Choisir un dossier", - "desktop.dialog.chooseFile": "Choisir un fichier", - "desktop.dialog.saveFile": "Enregistrer le fichier", - - "desktop.updater.checkFailed.title": "Échec de la vérification des mises à jour", - "desktop.updater.checkFailed.message": "Impossible de vérifier les mises à jour", - "desktop.updater.none.title": "Aucune mise à jour disponible", - "desktop.updater.none.message": "Vous utilisez déjà la dernière version d'OpenCode", - "desktop.updater.downloadFailed.title": "Échec de la mise à jour", - "desktop.updater.downloadFailed.message": "Impossible de télécharger la mise à jour", - "desktop.updater.downloaded.title": "Mise à jour téléchargée", - "desktop.updater.downloaded.prompt": - "La version {{version}} d'OpenCode a été téléchargée. Voulez-vous l'installer et redémarrer ?", - "desktop.updater.installFailed.title": "Échec de la mise à jour", - "desktop.updater.installFailed.message": "Impossible d'installer la mise à jour", - - "desktop.cli.installed.title": "CLI installée", - "desktop.cli.installed.message": - "CLI installée dans {{path}}\n\nRedémarrez votre terminal pour utiliser la commande 'opencode'.", - "desktop.cli.failed.title": "Échec de l'installation", - "desktop.cli.failed.message": "Impossible d'installer la CLI : {{error}}", -} diff --git a/packages/desktop-electron/src/renderer/i18n/index.ts b/packages/desktop-electron/src/renderer/i18n/index.ts deleted file mode 100644 index 2a7a7b188e..0000000000 --- a/packages/desktop-electron/src/renderer/i18n/index.ts +++ /dev/null @@ -1,194 +0,0 @@ -import * as i18n from "@solid-primitives/i18n" - -import { dict as desktopEn } from "./en" -import { dict as desktopZh } from "./zh" -import { dict as desktopZht } from "./zht" -import { dict as desktopKo } from "./ko" -import { dict as desktopDe } from "./de" -import { dict as desktopEs } from "./es" -import { dict as desktopFr } from "./fr" -import { dict as desktopDa } from "./da" -import { dict as desktopJa } from "./ja" -import { dict as desktopPl } from "./pl" -import { dict as desktopRu } from "./ru" -import { dict as desktopAr } from "./ar" -import { dict as desktopNo } from "./no" -import { dict as desktopBr } from "./br" -import { dict as desktopBs } from "./bs" -import { dict as desktopNl } from "./nl" - -import { dict as appEn } from "../../../../app/src/i18n/en" -import { dict as appZh } from "../../../../app/src/i18n/zh" -import { dict as appZht } from "../../../../app/src/i18n/zht" -import { dict as appKo } from "../../../../app/src/i18n/ko" -import { dict as appDe } from "../../../../app/src/i18n/de" -import { dict as appEs } from "../../../../app/src/i18n/es" -import { dict as appFr } from "../../../../app/src/i18n/fr" -import { dict as appDa } from "../../../../app/src/i18n/da" -import { dict as appJa } from "../../../../app/src/i18n/ja" -import { dict as appPl } from "../../../../app/src/i18n/pl" -import { dict as appRu } from "../../../../app/src/i18n/ru" -import { dict as appAr } from "../../../../app/src/i18n/ar" -import { dict as appNo } from "../../../../app/src/i18n/no" -import { dict as appBr } from "../../../../app/src/i18n/br" -import { dict as appBs } from "../../../../app/src/i18n/bs" -import { dict as appNl } from "../../../../app/src/i18n/nl" - -export type Locale = - | "en" - | "zh" - | "zht" - | "ko" - | "de" - | "es" - | "fr" - | "da" - | "ja" - | "pl" - | "ru" - | "ar" - | "no" - | "br" - | "bs" - | "nl" - -type RawDictionary = typeof appEn & typeof desktopEn -type Dictionary = i18n.Flatten - -const LOCALES: readonly Locale[] = [ - "en", - "zh", - "zht", - "ko", - "de", - "es", - "fr", - "da", - "ja", - "pl", - "ru", - "bs", - "nl", - "ar", - "no", - "br", -] - -function detectLocale(): Locale { - if (typeof navigator !== "object") return "en" - - const languages = navigator.languages?.length ? navigator.languages : [navigator.language] - for (const language of languages) { - if (!language) continue - if (language.toLowerCase().startsWith("en")) return "en" - if (language.toLowerCase().startsWith("zh")) { - if (language.toLowerCase().includes("hant")) return "zht" - return "zh" - } - if (language.toLowerCase().startsWith("ko")) return "ko" - if (language.toLowerCase().startsWith("de")) return "de" - if (language.toLowerCase().startsWith("es")) return "es" - if (language.toLowerCase().startsWith("fr")) return "fr" - if (language.toLowerCase().startsWith("da")) return "da" - if (language.toLowerCase().startsWith("ja")) return "ja" - if (language.toLowerCase().startsWith("pl")) return "pl" - if (language.toLowerCase().startsWith("ru")) return "ru" - if (language.toLowerCase().startsWith("ar")) return "ar" - if ( - language.toLowerCase().startsWith("no") || - language.toLowerCase().startsWith("nb") || - language.toLowerCase().startsWith("nn") - ) - return "no" - if (language.toLowerCase().startsWith("pt")) return "br" - if (language.toLowerCase().startsWith("bs")) return "bs" - if (language.toLowerCase().startsWith("nl")) return "nl" - } - - return "en" -} - -function parseLocale(value: unknown): Locale | null { - if (!value) return null - if (typeof value !== "string") return null - if ((LOCALES as readonly string[]).includes(value)) return value as Locale - return null -} - -function parseRecord(value: unknown) { - if (!value || typeof value !== "object") return null - if (Array.isArray(value)) return null - return value as Record -} - -function parseStored(value: unknown) { - if (typeof value !== "string") return value - try { - return JSON.parse(value) as unknown - } catch { - return value - } -} - -function pickLocale(value: unknown): Locale | null { - const direct = parseLocale(value) - if (direct) return direct - - const record = parseRecord(value) - if (!record) return null - - return parseLocale(record.locale) -} - -const base = i18n.flatten({ ...appEn, ...desktopEn }) - -function build(locale: Locale): Dictionary { - if (locale === "en") return base - if (locale === "zh") return { ...base, ...i18n.flatten(appZh), ...i18n.flatten(desktopZh) } - if (locale === "zht") return { ...base, ...i18n.flatten(appZht), ...i18n.flatten(desktopZht) } - if (locale === "de") return { ...base, ...i18n.flatten(appDe), ...i18n.flatten(desktopDe) } - if (locale === "es") return { ...base, ...i18n.flatten(appEs), ...i18n.flatten(desktopEs) } - if (locale === "fr") return { ...base, ...i18n.flatten(appFr), ...i18n.flatten(desktopFr) } - if (locale === "da") return { ...base, ...i18n.flatten(appDa), ...i18n.flatten(desktopDa) } - if (locale === "ja") return { ...base, ...i18n.flatten(appJa), ...i18n.flatten(desktopJa) } - if (locale === "pl") return { ...base, ...i18n.flatten(appPl), ...i18n.flatten(desktopPl) } - if (locale === "ru") return { ...base, ...i18n.flatten(appRu), ...i18n.flatten(desktopRu) } - if (locale === "ar") return { ...base, ...i18n.flatten(appAr), ...i18n.flatten(desktopAr) } - if (locale === "no") return { ...base, ...i18n.flatten(appNo), ...i18n.flatten(desktopNo) } - if (locale === "br") return { ...base, ...i18n.flatten(appBr), ...i18n.flatten(desktopBr) } - if (locale === "bs") return { ...base, ...i18n.flatten(appBs), ...i18n.flatten(desktopBs) } - if (locale === "nl") return { ...base, ...i18n.flatten(appNl), ...i18n.flatten(desktopNl) } - return { ...base, ...i18n.flatten(appKo), ...i18n.flatten(desktopKo) } -} - -const state = { - locale: detectLocale(), - dict: base as Dictionary, - init: undefined as Promise | undefined, -} - -state.dict = build(state.locale) - -const translate = i18n.translator(() => state.dict, i18n.resolveTemplate) - -export function t(key: keyof Dictionary, params?: Record) { - return translate(key, params) -} - -export function initI18n(): Promise { - const cached = state.init - if (cached) return cached - - const promise = (async () => { - const raw = await window.api.storeGet("opencode.global.dat", "language").catch(() => null) - const value = parseStored(raw) - const next = pickLocale(value) ?? state.locale - - state.locale = next - state.dict = build(next) - return next - })().catch(() => state.locale) - - state.init = promise - return promise -} diff --git a/packages/desktop-electron/src/renderer/i18n/ja.ts b/packages/desktop-electron/src/renderer/i18n/ja.ts deleted file mode 100644 index fc485c6f40..0000000000 --- a/packages/desktop-electron/src/renderer/i18n/ja.ts +++ /dev/null @@ -1,28 +0,0 @@ -export const dict = { - "desktop.menu.checkForUpdates": "アップデートを確認...", - "desktop.menu.installCli": "CLI をインストール...", - "desktop.menu.reloadWebview": "Webview を再読み込み", - "desktop.menu.restart": "再起動", - - "desktop.dialog.chooseFolder": "フォルダーを選択", - "desktop.dialog.chooseFile": "ファイルを選択", - "desktop.dialog.saveFile": "ファイルを保存", - - "desktop.updater.checkFailed.title": "アップデートの確認に失敗しました", - "desktop.updater.checkFailed.message": "アップデートを確認できませんでした", - "desktop.updater.none.title": "利用可能なアップデートはありません", - "desktop.updater.none.message": "すでに最新バージョンの OpenCode を使用しています", - "desktop.updater.downloadFailed.title": "アップデートに失敗しました", - "desktop.updater.downloadFailed.message": "アップデートをダウンロードできませんでした", - "desktop.updater.downloaded.title": "アップデートをダウンロードしました", - "desktop.updater.downloaded.prompt": - "OpenCode のバージョン {{version}} がダウンロードされました。インストールして再起動しますか?", - "desktop.updater.installFailed.title": "アップデートに失敗しました", - "desktop.updater.installFailed.message": "アップデートをインストールできませんでした", - - "desktop.cli.installed.title": "CLI をインストールしました", - "desktop.cli.installed.message": - "CLI を {{path}} にインストールしました\n\nターミナルを再起動して 'opencode' コマンドを使用してください。", - "desktop.cli.failed.title": "インストールに失敗しました", - "desktop.cli.failed.message": "CLI のインストールに失敗しました: {{error}}", -} diff --git a/packages/desktop-electron/src/renderer/i18n/ko.ts b/packages/desktop-electron/src/renderer/i18n/ko.ts deleted file mode 100644 index be27cec86a..0000000000 --- a/packages/desktop-electron/src/renderer/i18n/ko.ts +++ /dev/null @@ -1,27 +0,0 @@ -export const dict = { - "desktop.menu.checkForUpdates": "업데이트 확인...", - "desktop.menu.installCli": "CLI 설치...", - "desktop.menu.reloadWebview": "Webview 새로고침", - "desktop.menu.restart": "다시 시작", - - "desktop.dialog.chooseFolder": "폴더 선택", - "desktop.dialog.chooseFile": "파일 선택", - "desktop.dialog.saveFile": "파일 저장", - - "desktop.updater.checkFailed.title": "업데이트 확인 실패", - "desktop.updater.checkFailed.message": "업데이트를 확인하지 못했습니다", - "desktop.updater.none.title": "사용 가능한 업데이트 없음", - "desktop.updater.none.message": "이미 최신 버전의 OpenCode를 사용하고 있습니다", - "desktop.updater.downloadFailed.title": "업데이트 실패", - "desktop.updater.downloadFailed.message": "업데이트를 다운로드하지 못했습니다", - "desktop.updater.downloaded.title": "업데이트 다운로드 완료", - "desktop.updater.downloaded.prompt": "OpenCode {{version}} 버전을 다운로드했습니다. 설치하고 다시 실행할까요?", - "desktop.updater.installFailed.title": "업데이트 실패", - "desktop.updater.installFailed.message": "업데이트를 설치하지 못했습니다", - - "desktop.cli.installed.title": "CLI 설치됨", - "desktop.cli.installed.message": - "CLI가 {{path}}에 설치되었습니다\n\n터미널을 다시 시작하여 'opencode' 명령을 사용하세요.", - "desktop.cli.failed.title": "설치 실패", - "desktop.cli.failed.message": "CLI 설치 실패: {{error}}", -} diff --git a/packages/desktop-electron/src/renderer/i18n/nl.ts b/packages/desktop-electron/src/renderer/i18n/nl.ts deleted file mode 100644 index 2f8ca9fd42..0000000000 --- a/packages/desktop-electron/src/renderer/i18n/nl.ts +++ /dev/null @@ -1,28 +0,0 @@ -export const dict = { - "desktop.menu.checkForUpdates": "Controleren op updates...", - "desktop.menu.installCli": "CLI installeren...", - "desktop.menu.reloadWebview": "Webview herladen", - "desktop.menu.restart": "Opnieuw opstarten", - - "desktop.dialog.chooseFolder": "Kies een map", - "desktop.dialog.chooseFile": "Kies een bestand", - "desktop.dialog.saveFile": "Bestand opslaan", - - "desktop.updater.checkFailed.title": "Controleren op updates mislukt", - "desktop.updater.checkFailed.message": "Kan niet controleren op updates", - "desktop.updater.none.title": "Geen update beschikbaar", - "desktop.updater.none.message": "Je gebruikt al de nieuwste versie van OpenCode", - "desktop.updater.downloadFailed.title": "Update mislukt", - "desktop.updater.downloadFailed.message": "Downloaden van update mislukt", - "desktop.updater.downloaded.title": "Update gedownload", - "desktop.updater.downloaded.prompt": - "Versie {{version}} van OpenCode is gedownload. Wil je deze installeren en opnieuw opstarten?", - "desktop.updater.installFailed.title": "Update mislukt", - "desktop.updater.installFailed.message": "Installeren van update mislukt", - - "desktop.cli.installed.title": "CLI geïnstalleerd", - "desktop.cli.installed.message": - "CLI geïnstalleerd in {{path}}\n\nHerstart je terminal om de opdracht 'opencode' te gebruiken.", - "desktop.cli.failed.title": "Installatie mislukt", - "desktop.cli.failed.message": "Installeren van CLI mislukt: {{error}}", -} diff --git a/packages/desktop-electron/src/renderer/i18n/no.ts b/packages/desktop-electron/src/renderer/i18n/no.ts deleted file mode 100644 index e39bd7f3b4..0000000000 --- a/packages/desktop-electron/src/renderer/i18n/no.ts +++ /dev/null @@ -1,28 +0,0 @@ -export const dict = { - "desktop.menu.checkForUpdates": "Se etter oppdateringer...", - "desktop.menu.installCli": "Installer CLI...", - "desktop.menu.reloadWebview": "Last inn Webview på nytt", - "desktop.menu.restart": "Start på nytt", - - "desktop.dialog.chooseFolder": "Velg en mappe", - "desktop.dialog.chooseFile": "Velg en fil", - "desktop.dialog.saveFile": "Lagre fil", - - "desktop.updater.checkFailed.title": "Oppdateringssjekk mislyktes", - "desktop.updater.checkFailed.message": "Kunne ikke se etter oppdateringer", - "desktop.updater.none.title": "Ingen oppdatering tilgjengelig", - "desktop.updater.none.message": "Du bruker allerede den nyeste versjonen av OpenCode", - "desktop.updater.downloadFailed.title": "Oppdatering mislyktes", - "desktop.updater.downloadFailed.message": "Kunne ikke laste ned oppdateringen", - "desktop.updater.downloaded.title": "Oppdatering lastet ned", - "desktop.updater.downloaded.prompt": - "Versjon {{version}} av OpenCode er lastet ned. Vil du installere den og starte på nytt?", - "desktop.updater.installFailed.title": "Oppdatering mislyktes", - "desktop.updater.installFailed.message": "Kunne ikke installere oppdateringen", - - "desktop.cli.installed.title": "CLI installert", - "desktop.cli.installed.message": - "CLI installert til {{path}}\n\nStart terminalen på nytt for å bruke 'opencode'-kommandoen.", - "desktop.cli.failed.title": "Installasjon mislyktes", - "desktop.cli.failed.message": "Kunne ikke installere CLI: {{error}}", -} diff --git a/packages/desktop-electron/src/renderer/i18n/pl.ts b/packages/desktop-electron/src/renderer/i18n/pl.ts deleted file mode 100644 index d3ad7ce64f..0000000000 --- a/packages/desktop-electron/src/renderer/i18n/pl.ts +++ /dev/null @@ -1,28 +0,0 @@ -export const dict = { - "desktop.menu.checkForUpdates": "Sprawdź aktualizacje...", - "desktop.menu.installCli": "Zainstaluj CLI...", - "desktop.menu.reloadWebview": "Przeładuj Webview", - "desktop.menu.restart": "Restartuj", - - "desktop.dialog.chooseFolder": "Wybierz folder", - "desktop.dialog.chooseFile": "Wybierz plik", - "desktop.dialog.saveFile": "Zapisz plik", - - "desktop.updater.checkFailed.title": "Nie udało się sprawdzić aktualizacji", - "desktop.updater.checkFailed.message": "Nie udało się sprawdzić aktualizacji", - "desktop.updater.none.title": "Brak dostępnych aktualizacji", - "desktop.updater.none.message": "Korzystasz już z najnowszej wersji OpenCode", - "desktop.updater.downloadFailed.title": "Aktualizacja nie powiodła się", - "desktop.updater.downloadFailed.message": "Nie udało się pobrać aktualizacji", - "desktop.updater.downloaded.title": "Aktualizacja pobrana", - "desktop.updater.downloaded.prompt": - "Pobrano wersję {{version}} OpenCode. Czy chcesz ją zainstalować i uruchomić ponownie?", - "desktop.updater.installFailed.title": "Aktualizacja nie powiodła się", - "desktop.updater.installFailed.message": "Nie udało się zainstalować aktualizacji", - - "desktop.cli.installed.title": "CLI zainstalowane", - "desktop.cli.installed.message": - "CLI zainstalowane w {{path}}\n\nUruchom ponownie terminal, aby użyć polecenia 'opencode'.", - "desktop.cli.failed.title": "Instalacja nie powiodła się", - "desktop.cli.failed.message": "Nie udało się zainstalować CLI: {{error}}", -} diff --git a/packages/desktop-electron/src/renderer/i18n/ru.ts b/packages/desktop-electron/src/renderer/i18n/ru.ts deleted file mode 100644 index 8e09cc45b4..0000000000 --- a/packages/desktop-electron/src/renderer/i18n/ru.ts +++ /dev/null @@ -1,27 +0,0 @@ -export const dict = { - "desktop.menu.checkForUpdates": "Проверить обновления...", - "desktop.menu.installCli": "Установить CLI...", - "desktop.menu.reloadWebview": "Перезагрузить Webview", - "desktop.menu.restart": "Перезапустить", - - "desktop.dialog.chooseFolder": "Выберите папку", - "desktop.dialog.chooseFile": "Выберите файл", - "desktop.dialog.saveFile": "Сохранить файл", - - "desktop.updater.checkFailed.title": "Не удалось проверить обновления", - "desktop.updater.checkFailed.message": "Не удалось проверить обновления", - "desktop.updater.none.title": "Обновлений нет", - "desktop.updater.none.message": "Вы уже используете последнюю версию OpenCode", - "desktop.updater.downloadFailed.title": "Обновление не удалось", - "desktop.updater.downloadFailed.message": "Не удалось скачать обновление", - "desktop.updater.downloaded.title": "Обновление загружено", - "desktop.updater.downloaded.prompt": "Версия OpenCode {{version}} загружена. Хотите установить и перезапустить?", - "desktop.updater.installFailed.title": "Обновление не удалось", - "desktop.updater.installFailed.message": "Не удалось установить обновление", - - "desktop.cli.installed.title": "CLI установлен", - "desktop.cli.installed.message": - "CLI установлен в {{path}}\n\nПерезапустите терминал, чтобы использовать команду 'opencode'.", - "desktop.cli.failed.title": "Ошибка установки", - "desktop.cli.failed.message": "Не удалось установить CLI: {{error}}", -} diff --git a/packages/desktop-electron/src/renderer/i18n/zh.ts b/packages/desktop-electron/src/renderer/i18n/zh.ts deleted file mode 100644 index aeb3a54e03..0000000000 --- a/packages/desktop-electron/src/renderer/i18n/zh.ts +++ /dev/null @@ -1,26 +0,0 @@ -export const dict = { - "desktop.menu.checkForUpdates": "检查更新...", - "desktop.menu.installCli": "安装 CLI...", - "desktop.menu.reloadWebview": "重新加载 Webview", - "desktop.menu.restart": "重启", - - "desktop.dialog.chooseFolder": "选择文件夹", - "desktop.dialog.chooseFile": "选择文件", - "desktop.dialog.saveFile": "保存文件", - - "desktop.updater.checkFailed.title": "检查更新失败", - "desktop.updater.checkFailed.message": "无法检查更新", - "desktop.updater.none.title": "没有可用更新", - "desktop.updater.none.message": "你已经在使用最新版本的 OpenCode", - "desktop.updater.downloadFailed.title": "更新失败", - "desktop.updater.downloadFailed.message": "无法下载更新", - "desktop.updater.downloaded.title": "更新已下载", - "desktop.updater.downloaded.prompt": "已下载 OpenCode {{version}} 版本,是否安装并重启?", - "desktop.updater.installFailed.title": "更新失败", - "desktop.updater.installFailed.message": "无法安装更新", - - "desktop.cli.installed.title": "CLI 已安装", - "desktop.cli.installed.message": "CLI 已安装到 {{path}}\n\n重启终端以使用 'opencode' 命令。", - "desktop.cli.failed.title": "安装失败", - "desktop.cli.failed.message": "无法安装 CLI: {{error}}", -} diff --git a/packages/desktop-electron/src/renderer/i18n/zht.ts b/packages/desktop-electron/src/renderer/i18n/zht.ts deleted file mode 100644 index 7fd677aca4..0000000000 --- a/packages/desktop-electron/src/renderer/i18n/zht.ts +++ /dev/null @@ -1,26 +0,0 @@ -export const dict = { - "desktop.menu.checkForUpdates": "檢查更新...", - "desktop.menu.installCli": "安裝 CLI...", - "desktop.menu.reloadWebview": "重新載入 Webview", - "desktop.menu.restart": "重新啟動", - - "desktop.dialog.chooseFolder": "選擇資料夾", - "desktop.dialog.chooseFile": "選擇檔案", - "desktop.dialog.saveFile": "儲存檔案", - - "desktop.updater.checkFailed.title": "檢查更新失敗", - "desktop.updater.checkFailed.message": "無法檢查更新", - "desktop.updater.none.title": "沒有可用更新", - "desktop.updater.none.message": "你已在使用最新版的 OpenCode", - "desktop.updater.downloadFailed.title": "更新失敗", - "desktop.updater.downloadFailed.message": "無法下載更新", - "desktop.updater.downloaded.title": "更新已下載", - "desktop.updater.downloaded.prompt": "已下載 OpenCode {{version}} 版本,是否安裝並重新啟動?", - "desktop.updater.installFailed.title": "更新失敗", - "desktop.updater.installFailed.message": "無法安裝更新", - - "desktop.cli.installed.title": "CLI 已安裝", - "desktop.cli.installed.message": "CLI 已安裝到 {{path}}\n\n重新啟動終端機以使用 'opencode' 命令。", - "desktop.cli.failed.title": "安裝失敗", - "desktop.cli.failed.message": "無法安裝 CLI: {{error}}", -} diff --git a/packages/desktop-electron/src/renderer/index.html b/packages/desktop-electron/src/renderer/index.html deleted file mode 100644 index dd8675ee6b..0000000000 --- a/packages/desktop-electron/src/renderer/index.html +++ /dev/null @@ -1,22 +0,0 @@ - - - - - - OpenCode - - - - - - - - - - - - -
- - - diff --git a/packages/desktop-electron/src/renderer/index.tsx b/packages/desktop-electron/src/renderer/index.tsx deleted file mode 100644 index a1a11adf19..0000000000 --- a/packages/desktop-electron/src/renderer/index.tsx +++ /dev/null @@ -1,357 +0,0 @@ -// @refresh reload - -import { - ACCEPTED_FILE_EXTENSIONS, - ACCEPTED_FILE_TYPES, - AppBaseProviders, - AppInterface, - handleNotificationClick, - loadLocaleDict, - normalizeLocale, - type Locale, - type Platform, - PlatformProvider, - ServerConnection, - useCommand, -} from "@opencode-ai/app" -import type { AsyncStorage } from "@solid-primitives/storage" -import { MemoryRouter } from "@solidjs/router" -import { createEffect, createResource, onCleanup, onMount, Show } from "solid-js" -import { render } from "solid-js/web" -import pkg from "../../package.json" -import { initI18n, t } from "./i18n" -import { webviewZoom } from "./webview-zoom" -import "./styles.css" -import { useTheme } from "@opencode-ai/ui/theme" - -const root = document.getElementById("root") -if (import.meta.env.DEV && !(root instanceof HTMLElement)) { - throw new Error(t("error.dev.rootNotFound")) -} - -void initI18n() - -const deepLinkEvent = "opencode:deep-link" - -const emitDeepLinks = (urls: string[]) => { - if (urls.length === 0) return - window.__KILO__ ??= {} - const pending = window.__KILO__.deepLinks ?? [] - window.__KILO__.deepLinks = [...pending, ...urls] - window.dispatchEvent(new CustomEvent(deepLinkEvent, { detail: { urls } })) -} - -const listenForDeepLinks = () => { - void window.api.consumeInitialDeepLinks().then((urls) => emitDeepLinks(urls)) - return window.api.onDeepLink((urls) => emitDeepLinks(urls)) -} - -const createPlatform = (): Platform => { - const os = (() => { - const ua = navigator.userAgent - if (ua.includes("Mac")) return "macos" - if (ua.includes("Windows")) return "windows" - if (ua.includes("Linux")) return "linux" - return undefined - })() - - const isWslEnabled = async () => { - if (os !== "windows") return false - return window.api - .getWslConfig() - .then((config) => config.enabled) - .catch(() => false) - } - - const wslHome = async () => { - if (!(await isWslEnabled())) return undefined - return window.api.wslPath("~", "windows").catch(() => undefined) - } - - const handleWslPicker = async (result: T | null): Promise => { - if (!result || !(await isWslEnabled())) return result - if (Array.isArray(result)) { - return Promise.all(result.map((path) => window.api.wslPath(path, "linux").catch(() => path))) as any - } - return window.api.wslPath(result, "linux").catch(() => result) as any - } - - const storage = (() => { - const cache = new Map() - - const createStorage = (name: string) => { - const api: AsyncStorage = { - getItem: (key: string) => window.api.storeGet(name, key), - setItem: (key: string, value: string) => window.api.storeSet(name, key, value), - removeItem: (key: string) => window.api.storeDelete(name, key), - clear: () => window.api.storeClear(name), - key: async (index: number) => (await window.api.storeKeys(name))[index], - getLength: () => window.api.storeLength(name), - get length() { - return api.getLength() - }, - } - return api - } - - return (name = "default.dat") => { - const cached = cache.get(name) - if (cached) return cached - const api = createStorage(name) - cache.set(name, api) - return api - } - })() - - return { - platform: "desktop", - os, - version: pkg.version, - - async openDirectoryPickerDialog(opts) { - const defaultPath = await wslHome() - const result = await window.api.openDirectoryPicker({ - multiple: opts?.multiple ?? false, - title: opts?.title ?? t("desktop.dialog.chooseFolder"), - defaultPath, - }) - return await handleWslPicker(result) - }, - - async openFilePickerDialog(opts) { - const result = await window.api.openFilePicker({ - multiple: opts?.multiple ?? false, - title: opts?.title ?? t("desktop.dialog.chooseFile"), - accept: opts?.accept ?? ACCEPTED_FILE_TYPES, - extensions: opts?.extensions ?? ACCEPTED_FILE_EXTENSIONS, - }) - return handleWslPicker(result) - }, - - async saveFilePickerDialog(opts) { - const result = await window.api.saveFilePicker({ - title: opts?.title ?? t("desktop.dialog.saveFile"), - defaultPath: opts?.defaultPath, - }) - return handleWslPicker(result) - }, - - openLink(url: string) { - window.api.openLink(url) - }, - async openPath(path: string, app?: string) { - if (os === "windows") { - const resolvedApp = app ? await window.api.resolveAppPath(app).catch(() => null) : null - const resolvedPath = await (async () => { - if (await isWslEnabled()) { - const converted = await window.api.wslPath(path, "windows").catch(() => null) - if (converted) return converted - } - return path - })() - return window.api.openPath(resolvedPath, resolvedApp ?? undefined) - } - return window.api.openPath(path, app) - }, - - back() { - window.history.back() - }, - - forward() { - window.history.forward() - }, - - storage, - - checkUpdate: async () => { - const config = await window.api.getWindowConfig().catch(() => ({ updaterEnabled: false })) - if (!config.updaterEnabled) return { updateAvailable: false } - return window.api.checkUpdate() - }, - - updateAndRestart: async () => { - const config = await window.api.getWindowConfig().catch(() => ({ updaterEnabled: false })) - if (!config.updaterEnabled) return - await window.api.installUpdate() - }, - - restart: async () => { - await window.api.killSidecar().catch(() => undefined) - window.api.relaunch() - }, - - notify: async (title, description, href) => { - const focused = await window.api.getWindowFocused().catch(() => document.hasFocus()) - if (focused) return - - const notification = new Notification(title, { - body: description ?? "", - icon: "https://opencode.ai/favicon-96x96-v3.png", - }) - notification.onclick = () => { - void window.api.showWindow() - void window.api.setWindowFocus() - handleNotificationClick(href) - notification.close() - } - }, - - fetch: (input, init) => { - if (input instanceof Request) return fetch(input) - return fetch(input, init) - }, - - getWslEnabled: () => isWslEnabled(), - - setWslEnabled: async (enabled) => { - await window.api.setWslConfig({ enabled }) - }, - - getDefaultServer: async () => { - const url = await window.api.getDefaultServerUrl().catch(() => null) - if (!url) return null - return ServerConnection.Key.make(url) - }, - - setDefaultServer: async (url: string | null) => { - await window.api.setDefaultServerUrl(url) - }, - - getDisplayBackend: async () => { - return window.api.getDisplayBackend().catch(() => null) - }, - - setDisplayBackend: async (backend) => { - await window.api.setDisplayBackend(backend) - }, - - parseMarkdown: (markdown: string) => window.api.parseMarkdownCommand(markdown), - - webviewZoom, - - checkAppExists: async (appName: string) => { - return window.api.checkAppExists(appName) - }, - - async readClipboardImage() { - const image = await window.api.readClipboardImage().catch(() => null) - if (!image) return null - const blob = new Blob([image.buffer], { type: "image/png" }) - return new File([blob], `pasted-image-${Date.now()}.png`, { - type: "image/png", - }) - }, - } -} - -let menuTrigger = null as null | ((id: string) => void) -window.api.onMenuCommand((id) => { - menuTrigger?.(id) -}) -listenForDeepLinks() - -render(() => { - const platform = createPlatform() - const [windowConfig] = createResource(() => window.api.getWindowConfig().catch(() => ({ updaterEnabled: false }))) - const loadLocale = async () => { - const current = await platform.storage?.("opencode.global.dat").getItem("language") - const legacy = current ? undefined : await platform.storage?.().getItem("language.v1") - const raw = current ?? legacy - if (!raw) return - const locale = raw.match(/"locale"\s*:\s*"([^"]+)"/)?.[1] - if (!locale) return - const next = normalizeLocale(locale) - if (next !== "en") await loadLocaleDict(next) - return next satisfies Locale - } - - const [windowCount] = createResource(() => window.api.getWindowCount()) - - // Fetch sidecar credentials (available immediately, before health check) - const [sidecar] = createResource(() => window.api.awaitInitialization(() => undefined)) - - const [defaultServer] = createResource(() => - platform.getDefaultServer?.().then((url) => { - if (url) return ServerConnection.key({ type: "http", http: { url } }) - }), - ) - const [locale] = createResource(loadLocale) - - const servers = () => { - const data = sidecar() - if (!data) return [] - const server: ServerConnection.Sidecar = { - displayName: "Local Server", - type: "sidecar", - variant: "base", - http: { - url: data.url, - username: data.username ?? undefined, - password: data.password ?? undefined, - }, - } - return [server] as ServerConnection.Any[] - } - - function handleClick(e: MouseEvent) { - const link = (e.target as HTMLElement).closest("a.external-link") as HTMLAnchorElement | null - if (link?.href) { - e.preventDefault() - platform.openLink(link.href) - } - } - - function Inner() { - const cmd = useCommand() - menuTrigger = (id) => cmd.trigger(id) - - const theme = useTheme() - - createEffect(() => { - theme.themeId() - theme.mode() - const bg = getComputedStyle(document.documentElement).getPropertyValue("--background-base").trim() - if (bg) { - void window.api.setBackgroundColor(bg) - } - }) - - return null - } - - onMount(() => { - document.addEventListener("click", handleClick) - onCleanup(() => { - document.removeEventListener("click", handleClick) - }) - }) - - return ( - - - - {(_) => { - return ( - - - - ) - }} - - - - ) -}, root!) diff --git a/packages/desktop-electron/src/renderer/loading.html b/packages/desktop-electron/src/renderer/loading.html deleted file mode 100644 index ae3725af61..0000000000 --- a/packages/desktop-electron/src/renderer/loading.html +++ /dev/null @@ -1,22 +0,0 @@ - - - - - - OpenCode - - - - - - - - - - - - -
- - - diff --git a/packages/desktop-electron/src/renderer/loading.tsx b/packages/desktop-electron/src/renderer/loading.tsx deleted file mode 100644 index 000057e0a8..0000000000 --- a/packages/desktop-electron/src/renderer/loading.tsx +++ /dev/null @@ -1,83 +0,0 @@ -import { MetaProvider } from "@solidjs/meta" -import { render } from "solid-js/web" -import "@opencode-ai/app/index.css" -import { Font } from "@opencode-ai/ui/font" -import { Splash } from "@opencode-ai/ui/logo" -import { Progress } from "@opencode-ai/ui/progress" -import "./styles.css" -import { createEffect, createMemo, createSignal, onCleanup, onMount } from "solid-js" -import type { InitStep, SqliteMigrationProgress } from "../preload/types" - -const root = document.getElementById("root")! -const lines = ["Just a moment...", "Migrating your database", "This may take a couple of minutes"] -const delays = [3000, 9000] - -render(() => { - const [step, setStep] = createSignal(null) - const [line, setLine] = createSignal(0) - const [percent, setPercent] = createSignal(0) - - const phase = createMemo(() => step()?.phase) - - const value = createMemo(() => { - if (phase() === "done") return 100 - return Math.max(25, Math.min(100, percent())) - }) - - window.api.awaitInitialization((next) => setStep(next as InitStep)).catch(() => undefined) - - onMount(() => { - setLine(0) - setPercent(0) - - const timers = delays.map((ms, i) => setTimeout(() => setLine(i + 1), ms)) - - const listener = window.api.onSqliteMigrationProgress((progress: SqliteMigrationProgress) => { - if (progress.type === "InProgress") setPercent(Math.max(0, Math.min(100, progress.value))) - if (progress.type === "Done") { - setPercent(100) - setStep({ phase: "done" }) - } - }) - - onCleanup(() => { - listener() - timers.forEach(clearTimeout) - }) - }) - - createEffect(() => { - if (phase() !== "done") return - - const timer = setTimeout(() => window.api.loadingWindowComplete(), 1000) - onCleanup(() => clearTimeout(timer)) - }) - - const status = createMemo(() => { - if (phase() === "done") return "All done" - if (phase() === "sqlite_waiting") return lines[line()] - return "Just a moment..." - }) - - return ( - -
- -
- -
- - {status()} - - `${Math.round(value)}%`} - /> -
-
-
-
- ) -}, root) diff --git a/packages/desktop-electron/src/renderer/styles.css b/packages/desktop-electron/src/renderer/styles.css deleted file mode 100644 index e69de29bb2..0000000000 diff --git a/packages/desktop-electron/src/renderer/updater.ts b/packages/desktop-electron/src/renderer/updater.ts deleted file mode 100644 index c44ca26591..0000000000 --- a/packages/desktop-electron/src/renderer/updater.ts +++ /dev/null @@ -1,12 +0,0 @@ -import { initI18n, t } from "./i18n" - -export async function runUpdater({ alertOnFail }: { alertOnFail: boolean }) { - await initI18n() - try { - await window.api.runUpdater(alertOnFail) - } catch { - if (alertOnFail) { - window.alert(t("desktop.updater.checkFailed.message")) - } - } -} diff --git a/packages/desktop-electron/src/renderer/webview-zoom.ts b/packages/desktop-electron/src/renderer/webview-zoom.ts deleted file mode 100644 index 9c0a3a3a35..0000000000 --- a/packages/desktop-electron/src/renderer/webview-zoom.ts +++ /dev/null @@ -1,38 +0,0 @@ -// Copyright 2019-2024 Tauri Programme within The Commons Conservancy -// SPDX-License-Identifier: Apache-2.0 -// SPDX-License-Identifier: MIT - -import { createSignal } from "solid-js" - -const OS_NAME = (() => { - if (navigator.userAgent.includes("Mac")) return "macos" - if (navigator.userAgent.includes("Windows")) return "windows" - if (navigator.userAgent.includes("Linux")) return "linux" - return "unknown" -})() - -const [webviewZoom, setWebviewZoom] = createSignal(1) - -const MAX_ZOOM_LEVEL = 10 -const MIN_ZOOM_LEVEL = 0.2 - -const clamp = (value: number) => Math.min(Math.max(value, MIN_ZOOM_LEVEL), MAX_ZOOM_LEVEL) - -const applyZoom = (next: number) => { - setWebviewZoom(next) - void window.api.setZoomFactor(next) -} - -window.addEventListener("keydown", (event) => { - if (!(OS_NAME === "macos" ? event.metaKey : event.ctrlKey)) return - - let newZoom = webviewZoom() - - if (event.key === "-") newZoom -= 0.2 - if (event.key === "=" || event.key === "+") newZoom += 0.2 - if (event.key === "0") newZoom = 1 - - applyZoom(clamp(newZoom)) -}) - -export { webviewZoom } diff --git a/packages/desktop-electron/sst-env.d.ts b/packages/desktop-electron/sst-env.d.ts deleted file mode 100644 index f25b971455..0000000000 --- a/packages/desktop-electron/sst-env.d.ts +++ /dev/null @@ -1,10 +0,0 @@ -/* This file is auto-generated by SST. Do not edit. */ -/* tslint:disable */ -/* eslint-disable */ -/* deno-fmt-ignore-file */ -/* biome-ignore-all lint: auto-generated */ - -/// - -import "sst" -export {} diff --git a/packages/desktop-electron/tsconfig.json b/packages/desktop-electron/tsconfig.json deleted file mode 100644 index 9637fe03dd..0000000000 --- a/packages/desktop-electron/tsconfig.json +++ /dev/null @@ -1,23 +0,0 @@ -{ - "compilerOptions": { - "target": "ESNext", - "module": "ESNext", - "skipLibCheck": true, - "moduleResolution": "bundler", - "allowSyntheticDefaultImports": true, - "esModuleInterop": true, - "jsx": "preserve", - "jsxImportSource": "solid-js", - "allowJs": true, - "resolveJsonModule": true, - "strict": true, - "isolatedModules": true, - "noEmit": true, - "emitDeclarationOnly": false, - "outDir": "node_modules/.ts-dist", - "types": ["vite/client", "node", "electron"] - }, - "references": [{ "path": "../app" }], - "include": ["src", "package.json"], - "exclude": ["src/**/*.test.ts"] -} diff --git a/packages/desktop/.gitignore b/packages/desktop/.gitignore deleted file mode 100644 index a547bf36d8..0000000000 --- a/packages/desktop/.gitignore +++ /dev/null @@ -1,24 +0,0 @@ -# Logs -logs -*.log -npm-debug.log* -yarn-debug.log* -yarn-error.log* -pnpm-debug.log* -lerna-debug.log* - -node_modules -dist -dist-ssr -*.local - -# Editor directories and files -.vscode/* -!.vscode/extensions.json -.idea -.DS_Store -*.suo -*.ntvs* -*.njsproj -*.sln -*.sw? diff --git a/packages/desktop/README.md b/packages/desktop/README.md deleted file mode 100644 index 358b7d24d5..0000000000 --- a/packages/desktop/README.md +++ /dev/null @@ -1,32 +0,0 @@ -# OpenCode Desktop - -Native OpenCode desktop app, built with Tauri v2. - -## Prerequisites - -Building the desktop app requires additional Tauri dependencies (Rust toolchain, platform-specific libraries). See the [Tauri prerequisites](https://v2.tauri.app/start/prerequisites/) for setup instructions. - -## Development - -From the repo root: - -```bash -bun install -bun run --cwd packages/desktop tauri dev -``` - -## Build - -```bash -bun run --cwd packages/desktop tauri build -``` - -## Troubleshooting - -### Rust compiler not found - -If you see errors about Rust not being found, install it via [rustup](https://rustup.rs/): - -```bash -curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh -``` diff --git a/packages/desktop/index.html b/packages/desktop/index.html deleted file mode 100644 index ce2775a704..0000000000 --- a/packages/desktop/index.html +++ /dev/null @@ -1,24 +0,0 @@ - - - - - - OpenCode - - - - - - - - - - - - - -
-