mirror of
https://github.com/getcompanion-ai/co-mono.git
synced 2026-04-15 09:01:14 +00:00
Merge syntax-highlighting branch: add syntax highlighting and intra-line diff
- Syntax highlighting for markdown code blocks, read/write tool output - Intra-line diff highlighting for edit tool with word-level inverse - VS Code-style syntax colors in themes - Fix Google provider FinishReason handling - Upgrade @google/genai to 1.34.0
This commit is contained in:
commit
039b3a0845
10 changed files with 594 additions and 47 deletions
|
|
@ -15,6 +15,8 @@ read README.md, then ask which module(s) to work on. Based on the answer, read t
|
||||||
- No `any` types unless absolutely necessary
|
- No `any` types unless absolutely necessary
|
||||||
- Check node_modules for external API type definitions instead of guessing
|
- Check node_modules for external API type definitions instead of guessing
|
||||||
- No inline imports like `await import("./foo.js")`
|
- No inline imports like `await import("./foo.js")`
|
||||||
|
- NEVER remove or downgrade code to fix type errors from outdated dependencies; upgrade the dependency instead
|
||||||
|
- Always ask before removing functionality or code that appears to be intentional
|
||||||
|
|
||||||
## Commands
|
## Commands
|
||||||
- After code changes: `npm run check` (get full output, no tail)
|
- After code changes: `npm run check` (get full output, no tail)
|
||||||
|
|
|
||||||
236
package-lock.json
generated
236
package-lock.json
generated
|
|
@ -2583,6 +2583,12 @@
|
||||||
"url": "https://github.com/chalk/ansi-styles?sponsor=1"
|
"url": "https://github.com/chalk/ansi-styles?sponsor=1"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/any-promise": {
|
||||||
|
"version": "1.3.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/any-promise/-/any-promise-1.3.0.tgz",
|
||||||
|
"integrity": "sha512-7UvmKalWRt1wgjL1RrGxoSJW/0QZFIegpeGvZG9kjp8vrRu55XTHbwnqq2GpXm9uLbcuhxm3IqX9OB4MZR1b2A==",
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
"node_modules/assertion-error": {
|
"node_modules/assertion-error": {
|
||||||
"version": "2.0.1",
|
"version": "2.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/assertion-error/-/assertion-error-2.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/assertion-error/-/assertion-error-2.0.1.tgz",
|
||||||
|
|
@ -2806,6 +2812,169 @@
|
||||||
"url": "https://polar.sh/cva"
|
"url": "https://polar.sh/cva"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/cli-highlight": {
|
||||||
|
"version": "2.1.11",
|
||||||
|
"resolved": "https://registry.npmjs.org/cli-highlight/-/cli-highlight-2.1.11.tgz",
|
||||||
|
"integrity": "sha512-9KDcoEVwyUXrjcJNvHD0NFc/hiwe/WPVYIleQh2O1N2Zro5gWJZ/K+3DGn8w8P/F6FxOgzyC5bxDyHIgCSPhGg==",
|
||||||
|
"license": "ISC",
|
||||||
|
"dependencies": {
|
||||||
|
"chalk": "^4.0.0",
|
||||||
|
"highlight.js": "^10.7.1",
|
||||||
|
"mz": "^2.4.0",
|
||||||
|
"parse5": "^5.1.1",
|
||||||
|
"parse5-htmlparser2-tree-adapter": "^6.0.0",
|
||||||
|
"yargs": "^16.0.0"
|
||||||
|
},
|
||||||
|
"bin": {
|
||||||
|
"highlight": "bin/highlight"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=8.0.0",
|
||||||
|
"npm": ">=5.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/cli-highlight/node_modules/ansi-regex": {
|
||||||
|
"version": "5.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz",
|
||||||
|
"integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=8"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/cli-highlight/node_modules/ansi-styles": {
|
||||||
|
"version": "4.3.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz",
|
||||||
|
"integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"color-convert": "^2.0.1"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=8"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/chalk/ansi-styles?sponsor=1"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/cli-highlight/node_modules/chalk": {
|
||||||
|
"version": "4.1.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz",
|
||||||
|
"integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"ansi-styles": "^4.1.0",
|
||||||
|
"supports-color": "^7.1.0"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=10"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/chalk/chalk?sponsor=1"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/cli-highlight/node_modules/cliui": {
|
||||||
|
"version": "7.0.4",
|
||||||
|
"resolved": "https://registry.npmjs.org/cliui/-/cliui-7.0.4.tgz",
|
||||||
|
"integrity": "sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==",
|
||||||
|
"license": "ISC",
|
||||||
|
"dependencies": {
|
||||||
|
"string-width": "^4.2.0",
|
||||||
|
"strip-ansi": "^6.0.0",
|
||||||
|
"wrap-ansi": "^7.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/cli-highlight/node_modules/highlight.js": {
|
||||||
|
"version": "10.7.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/highlight.js/-/highlight.js-10.7.3.tgz",
|
||||||
|
"integrity": "sha512-tzcUFauisWKNHaRkN4Wjl/ZA07gENAjFl3J/c480dprkGTg5EQstgaNFqBfUqCq54kZRIEcreTsAgF/m2quD7A==",
|
||||||
|
"license": "BSD-3-Clause",
|
||||||
|
"engines": {
|
||||||
|
"node": "*"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/cli-highlight/node_modules/string-width": {
|
||||||
|
"version": "4.2.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz",
|
||||||
|
"integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"emoji-regex": "^8.0.0",
|
||||||
|
"is-fullwidth-code-point": "^3.0.0",
|
||||||
|
"strip-ansi": "^6.0.1"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=8"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/cli-highlight/node_modules/strip-ansi": {
|
||||||
|
"version": "6.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz",
|
||||||
|
"integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"ansi-regex": "^5.0.1"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=8"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/cli-highlight/node_modules/supports-color": {
|
||||||
|
"version": "7.2.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz",
|
||||||
|
"integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"has-flag": "^4.0.0"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=8"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/cli-highlight/node_modules/wrap-ansi": {
|
||||||
|
"version": "7.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz",
|
||||||
|
"integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"ansi-styles": "^4.0.0",
|
||||||
|
"string-width": "^4.1.0",
|
||||||
|
"strip-ansi": "^6.0.0"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=10"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/chalk/wrap-ansi?sponsor=1"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/cli-highlight/node_modules/yargs": {
|
||||||
|
"version": "16.2.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/yargs/-/yargs-16.2.0.tgz",
|
||||||
|
"integrity": "sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"cliui": "^7.0.2",
|
||||||
|
"escalade": "^3.1.1",
|
||||||
|
"get-caller-file": "^2.0.5",
|
||||||
|
"require-directory": "^2.1.1",
|
||||||
|
"string-width": "^4.2.0",
|
||||||
|
"y18n": "^5.0.5",
|
||||||
|
"yargs-parser": "^20.2.2"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=10"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/cli-highlight/node_modules/yargs-parser": {
|
||||||
|
"version": "20.2.9",
|
||||||
|
"resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.9.tgz",
|
||||||
|
"integrity": "sha512-y11nGElTIV+CT3Zv9t7VKl+Q3hTQoT9a1Qzezhhl6Rp21gJ/IVTW7Z3y9EWXhuUBC2Shnf+DX0antecpAwSP8w==",
|
||||||
|
"license": "ISC",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=10"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/cliui": {
|
"node_modules/cliui": {
|
||||||
"version": "8.0.1",
|
"version": "8.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/cliui/-/cliui-8.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/cliui/-/cliui-8.0.1.tgz",
|
||||||
|
|
@ -3294,7 +3463,6 @@
|
||||||
"version": "3.2.0",
|
"version": "3.2.0",
|
||||||
"resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz",
|
"resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz",
|
||||||
"integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==",
|
"integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==",
|
||||||
"dev": true,
|
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=6"
|
"node": ">=6"
|
||||||
|
|
@ -3562,7 +3730,6 @@
|
||||||
"version": "2.0.5",
|
"version": "2.0.5",
|
||||||
"resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz",
|
"resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz",
|
||||||
"integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==",
|
"integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==",
|
||||||
"dev": true,
|
|
||||||
"license": "ISC",
|
"license": "ISC",
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": "6.* || 8.* || >= 10.*"
|
"node": "6.* || 8.* || >= 10.*"
|
||||||
|
|
@ -4551,6 +4718,17 @@
|
||||||
"integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==",
|
"integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==",
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
|
"node_modules/mz": {
|
||||||
|
"version": "2.7.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/mz/-/mz-2.7.0.tgz",
|
||||||
|
"integrity": "sha512-z81GNO7nnYMEhrGh9LeymoE4+Yr0Wn5McHIZMK5cfQCl+NDX08sCZgUc9/6MHni9IWuFLm1Z3HTCXu2z9fN62Q==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"any-promise": "^1.0.0",
|
||||||
|
"object-assign": "^4.0.1",
|
||||||
|
"thenify-all": "^1.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/nanoid": {
|
"node_modules/nanoid": {
|
||||||
"version": "3.3.11",
|
"version": "3.3.11",
|
||||||
"resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.11.tgz",
|
"resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.11.tgz",
|
||||||
|
|
@ -4634,6 +4812,15 @@
|
||||||
"url": "https://opencollective.com/node-fetch"
|
"url": "https://opencollective.com/node-fetch"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/object-assign": {
|
||||||
|
"version": "4.1.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
|
||||||
|
"integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=0.10.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/ollama": {
|
"node_modules/ollama": {
|
||||||
"version": "0.6.3",
|
"version": "0.6.3",
|
||||||
"resolved": "https://registry.npmjs.org/ollama/-/ollama-0.6.3.tgz",
|
"resolved": "https://registry.npmjs.org/ollama/-/ollama-0.6.3.tgz",
|
||||||
|
|
@ -4742,6 +4929,27 @@
|
||||||
"integrity": "sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw==",
|
"integrity": "sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw==",
|
||||||
"license": "(MIT AND Zlib)"
|
"license": "(MIT AND Zlib)"
|
||||||
},
|
},
|
||||||
|
"node_modules/parse5": {
|
||||||
|
"version": "5.1.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/parse5/-/parse5-5.1.1.tgz",
|
||||||
|
"integrity": "sha512-ugq4DFI0Ptb+WWjAdOK16+u/nHfiIrcE+sh8kZMaM0WllQKLI9rOUq6c2b7cwPkXdzfQESqvoqK6ug7U/Yyzug==",
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
|
"node_modules/parse5-htmlparser2-tree-adapter": {
|
||||||
|
"version": "6.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/parse5-htmlparser2-tree-adapter/-/parse5-htmlparser2-tree-adapter-6.0.1.tgz",
|
||||||
|
"integrity": "sha512-qPuWvbLgvDGilKc5BoicRovlT4MtYT6JfJyBOMDsKoiT+GiuP5qyrPCnR9HcPECIJJmZh5jRndyNThnhhb/vlA==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"parse5": "^6.0.1"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/parse5-htmlparser2-tree-adapter/node_modules/parse5": {
|
||||||
|
"version": "6.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/parse5/-/parse5-6.0.1.tgz",
|
||||||
|
"integrity": "sha512-Ofn/CTFzRGTTxwpNEs9PP93gXShHcTq255nzRYSKe8AkVpZY7e1fpmTfOyoIvjP5HG7Z2ZM7VS9PPhQGW2pOpw==",
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
"node_modules/partial-json": {
|
"node_modules/partial-json": {
|
||||||
"version": "0.1.7",
|
"version": "0.1.7",
|
||||||
"resolved": "https://registry.npmjs.org/partial-json/-/partial-json-0.1.7.tgz",
|
"resolved": "https://registry.npmjs.org/partial-json/-/partial-json-0.1.7.tgz",
|
||||||
|
|
@ -4937,7 +5145,6 @@
|
||||||
"version": "2.1.1",
|
"version": "2.1.1",
|
||||||
"resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz",
|
||||||
"integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==",
|
"integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==",
|
||||||
"dev": true,
|
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=0.10.0"
|
"node": ">=0.10.0"
|
||||||
|
|
@ -5502,6 +5709,27 @@
|
||||||
"node": ">=6"
|
"node": ">=6"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/thenify": {
|
||||||
|
"version": "3.3.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/thenify/-/thenify-3.3.1.tgz",
|
||||||
|
"integrity": "sha512-RVZSIV5IG10Hk3enotrhvz0T9em6cyHBLkH/YAZuKqd8hRkKhSfCGIcP2KUY0EPxndzANBmNllzWPwak+bheSw==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"any-promise": "^1.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/thenify-all": {
|
||||||
|
"version": "1.6.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/thenify-all/-/thenify-all-1.6.0.tgz",
|
||||||
|
"integrity": "sha512-RNxQH/qI8/t3thXJDwcstUO4zeqo64+Uy/+sNVRBx4Xn2OX+OZ9oP+iJnNFqplFra2ZUVeKCSa2oVWi3T4uVmA==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"thenify": ">= 3.1.0 < 4"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=0.8"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/tinybench": {
|
"node_modules/tinybench": {
|
||||||
"version": "2.9.0",
|
"version": "2.9.0",
|
||||||
"resolved": "https://registry.npmjs.org/tinybench/-/tinybench-2.9.0.tgz",
|
"resolved": "https://registry.npmjs.org/tinybench/-/tinybench-2.9.0.tgz",
|
||||||
|
|
@ -6065,7 +6293,6 @@
|
||||||
"version": "5.0.8",
|
"version": "5.0.8",
|
||||||
"resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz",
|
"resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz",
|
||||||
"integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==",
|
"integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==",
|
||||||
"dev": true,
|
|
||||||
"license": "ISC",
|
"license": "ISC",
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=10"
|
"node": ">=10"
|
||||||
|
|
@ -6242,6 +6469,7 @@
|
||||||
"@mariozechner/pi-ai": "^0.23.3",
|
"@mariozechner/pi-ai": "^0.23.3",
|
||||||
"@mariozechner/pi-tui": "^0.23.3",
|
"@mariozechner/pi-tui": "^0.23.3",
|
||||||
"chalk": "^5.5.0",
|
"chalk": "^5.5.0",
|
||||||
|
"cli-highlight": "^2.1.11",
|
||||||
"diff": "^8.0.2",
|
"diff": "^8.0.2",
|
||||||
"file-type": "^21.1.1",
|
"file-type": "^21.1.1",
|
||||||
"glob": "^11.0.3",
|
"glob": "^11.0.3",
|
||||||
|
|
|
||||||
|
|
@ -2,12 +2,20 @@
|
||||||
|
|
||||||
## [Unreleased]
|
## [Unreleased]
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- **Syntax highlighting**: Added syntax highlighting for markdown code blocks, read tool output, and write tool content. Uses cli-highlight with theme-aware color mapping and VS Code-style syntax colors. ([#214](https://github.com/badlogic/pi-mono/pull/214) by [@svkozak](https://github.com/svkozak))
|
||||||
|
|
||||||
|
- **Intra-line diff highlighting**: Edit tool now shows word-level changes with inverse highlighting when a single line is modified. Multi-line changes show all removed lines first, then all added lines.
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
||||||
- **Gemini tool result format**: Fixed tool result format for Gemini 3 Flash Preview which strictly requires `{ output: value }` for success and `{ error: value }` for errors. Previous format using `{ result, isError }` was rejected by newer Gemini models. ([#213](https://github.com/badlogic/pi-mono/issues/213), [#220](https://github.com/badlogic/pi-mono/pull/220))
|
- **Gemini tool result format**: Fixed tool result format for Gemini 3 Flash Preview which strictly requires `{ output: value }` for success and `{ error: value }` for errors. Previous format using `{ result, isError }` was rejected by newer Gemini models. ([#213](https://github.com/badlogic/pi-mono/issues/213), [#220](https://github.com/badlogic/pi-mono/pull/220))
|
||||||
|
|
||||||
- **Google baseUrl configuration**: Google provider now respects `baseUrl` configuration for custom endpoints or API proxies. ([#216](https://github.com/badlogic/pi-mono/issues/216), [#221](https://github.com/badlogic/pi-mono/pull/221) by [@theBucky](https://github.com/theBucky))
|
- **Google baseUrl configuration**: Google provider now respects `baseUrl` configuration for custom endpoints or API proxies. ([#216](https://github.com/badlogic/pi-mono/issues/216), [#221](https://github.com/badlogic/pi-mono/pull/221) by [@theBucky](https://github.com/theBucky))
|
||||||
|
|
||||||
|
- **Google provider FinishReason**: Added handling for new `IMAGE_RECITATION` and `IMAGE_OTHER` finish reasons. Upgraded @google/genai to 1.34.0.
|
||||||
|
|
||||||
## [0.23.3] - 2025-12-17
|
## [0.23.3] - 2025-12-17
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
|
||||||
|
|
@ -43,6 +43,7 @@
|
||||||
"@mariozechner/pi-ai": "^0.23.3",
|
"@mariozechner/pi-ai": "^0.23.3",
|
||||||
"@mariozechner/pi-tui": "^0.23.3",
|
"@mariozechner/pi-tui": "^0.23.3",
|
||||||
"chalk": "^5.5.0",
|
"chalk": "^5.5.0",
|
||||||
|
"cli-highlight": "^2.1.11",
|
||||||
"diff": "^8.0.2",
|
"diff": "^8.0.2",
|
||||||
"file-type": "^21.1.1",
|
"file-type": "^21.1.1",
|
||||||
"glob": "^11.0.3",
|
"glob": "^11.0.3",
|
||||||
|
|
|
||||||
147
packages/coding-agent/src/modes/interactive/components/diff.ts
Normal file
147
packages/coding-agent/src/modes/interactive/components/diff.ts
Normal file
|
|
@ -0,0 +1,147 @@
|
||||||
|
import * as Diff from "diff";
|
||||||
|
import { theme } from "../theme/theme.js";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parse diff line to extract prefix, line number, and content.
|
||||||
|
* Format: "+123 content" or "-123 content" or " 123 content" or " ..."
|
||||||
|
*/
|
||||||
|
function parseDiffLine(line: string): { prefix: string; lineNum: string; content: string } | null {
|
||||||
|
const match = line.match(/^([+-\s])(\s*\d*)\s(.*)$/);
|
||||||
|
if (!match) return null;
|
||||||
|
return { prefix: match[1], lineNum: match[2], content: match[3] };
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Replace tabs with spaces for consistent rendering.
|
||||||
|
*/
|
||||||
|
function replaceTabs(text: string): string {
|
||||||
|
return text.replace(/\t/g, " ");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Compute word-level diff and render with inverse on changed parts.
|
||||||
|
* Uses diffWords which groups whitespace with adjacent words for cleaner highlighting.
|
||||||
|
* Strips leading whitespace from inverse to avoid highlighting indentation.
|
||||||
|
*/
|
||||||
|
function renderIntraLineDiff(oldContent: string, newContent: string): { removedLine: string; addedLine: string } {
|
||||||
|
const wordDiff = Diff.diffWords(oldContent, newContent);
|
||||||
|
|
||||||
|
let removedLine = "";
|
||||||
|
let addedLine = "";
|
||||||
|
let isFirstRemoved = true;
|
||||||
|
let isFirstAdded = true;
|
||||||
|
|
||||||
|
for (const part of wordDiff) {
|
||||||
|
if (part.removed) {
|
||||||
|
let value = part.value;
|
||||||
|
// Strip leading whitespace from the first removed part
|
||||||
|
if (isFirstRemoved) {
|
||||||
|
const leadingWs = value.match(/^(\s*)/)?.[1] || "";
|
||||||
|
value = value.slice(leadingWs.length);
|
||||||
|
removedLine += leadingWs;
|
||||||
|
isFirstRemoved = false;
|
||||||
|
}
|
||||||
|
if (value) {
|
||||||
|
removedLine += theme.inverse(value);
|
||||||
|
}
|
||||||
|
} else if (part.added) {
|
||||||
|
let value = part.value;
|
||||||
|
// Strip leading whitespace from the first added part
|
||||||
|
if (isFirstAdded) {
|
||||||
|
const leadingWs = value.match(/^(\s*)/)?.[1] || "";
|
||||||
|
value = value.slice(leadingWs.length);
|
||||||
|
addedLine += leadingWs;
|
||||||
|
isFirstAdded = false;
|
||||||
|
}
|
||||||
|
if (value) {
|
||||||
|
addedLine += theme.inverse(value);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
removedLine += part.value;
|
||||||
|
addedLine += part.value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return { removedLine, addedLine };
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface RenderDiffOptions {
|
||||||
|
/** File path (unused, kept for API compatibility) */
|
||||||
|
filePath?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Render a diff string with colored lines and intra-line change highlighting.
|
||||||
|
* - Context lines: dim/gray
|
||||||
|
* - Removed lines: red, with inverse on changed tokens
|
||||||
|
* - Added lines: green, with inverse on changed tokens
|
||||||
|
*/
|
||||||
|
export function renderDiff(diffText: string, _options: RenderDiffOptions = {}): string {
|
||||||
|
const lines = diffText.split("\n");
|
||||||
|
const result: string[] = [];
|
||||||
|
|
||||||
|
let i = 0;
|
||||||
|
while (i < lines.length) {
|
||||||
|
const line = lines[i];
|
||||||
|
const parsed = parseDiffLine(line);
|
||||||
|
|
||||||
|
if (!parsed) {
|
||||||
|
result.push(theme.fg("toolDiffContext", line));
|
||||||
|
i++;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (parsed.prefix === "-") {
|
||||||
|
// Collect consecutive removed lines
|
||||||
|
const removedLines: { lineNum: string; content: string }[] = [];
|
||||||
|
while (i < lines.length) {
|
||||||
|
const p = parseDiffLine(lines[i]);
|
||||||
|
if (!p || p.prefix !== "-") break;
|
||||||
|
removedLines.push({ lineNum: p.lineNum, content: p.content });
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Collect consecutive added lines
|
||||||
|
const addedLines: { lineNum: string; content: string }[] = [];
|
||||||
|
while (i < lines.length) {
|
||||||
|
const p = parseDiffLine(lines[i]);
|
||||||
|
if (!p || p.prefix !== "+") break;
|
||||||
|
addedLines.push({ lineNum: p.lineNum, content: p.content });
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Only do intra-line diffing when there's exactly one removed and one added line
|
||||||
|
// (indicating a single line modification). Otherwise, show lines as-is.
|
||||||
|
if (removedLines.length === 1 && addedLines.length === 1) {
|
||||||
|
const removed = removedLines[0];
|
||||||
|
const added = addedLines[0];
|
||||||
|
|
||||||
|
const { removedLine, addedLine } = renderIntraLineDiff(
|
||||||
|
replaceTabs(removed.content),
|
||||||
|
replaceTabs(added.content),
|
||||||
|
);
|
||||||
|
|
||||||
|
result.push(theme.fg("toolDiffRemoved", `-${removed.lineNum} ${removedLine}`));
|
||||||
|
result.push(theme.fg("toolDiffAdded", `+${added.lineNum} ${addedLine}`));
|
||||||
|
} else {
|
||||||
|
// Show all removed lines first, then all added lines
|
||||||
|
for (const removed of removedLines) {
|
||||||
|
result.push(theme.fg("toolDiffRemoved", `-${removed.lineNum} ${replaceTabs(removed.content)}`));
|
||||||
|
}
|
||||||
|
for (const added of addedLines) {
|
||||||
|
result.push(theme.fg("toolDiffAdded", `+${added.lineNum} ${replaceTabs(added.content)}`));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (parsed.prefix === "+") {
|
||||||
|
// Standalone added line
|
||||||
|
result.push(theme.fg("toolDiffAdded", `+${parsed.lineNum} ${replaceTabs(parsed.content)}`));
|
||||||
|
i++;
|
||||||
|
} else {
|
||||||
|
// Context line
|
||||||
|
result.push(theme.fg("toolDiffContext", ` ${parsed.lineNum} ${replaceTabs(parsed.content)}`));
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result.join("\n");
|
||||||
|
}
|
||||||
|
|
@ -12,7 +12,8 @@ import {
|
||||||
import stripAnsi from "strip-ansi";
|
import stripAnsi from "strip-ansi";
|
||||||
import type { CustomAgentTool } from "../../../core/custom-tools/types.js";
|
import type { CustomAgentTool } from "../../../core/custom-tools/types.js";
|
||||||
import { DEFAULT_MAX_BYTES, DEFAULT_MAX_LINES, formatSize } from "../../../core/tools/truncate.js";
|
import { DEFAULT_MAX_BYTES, DEFAULT_MAX_LINES, formatSize } from "../../../core/tools/truncate.js";
|
||||||
import { theme } from "../theme/theme.js";
|
import { getLanguageFromPath, highlightCode, theme } from "../theme/theme.js";
|
||||||
|
import { renderDiff } from "./diff.js";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Convert absolute path to tilde notation if it's in home directory
|
* Convert absolute path to tilde notation if it's in home directory
|
||||||
|
|
@ -280,13 +281,19 @@ export class ToolExecutionComponent extends Container {
|
||||||
|
|
||||||
if (this.result) {
|
if (this.result) {
|
||||||
const output = this.getTextOutput();
|
const output = this.getTextOutput();
|
||||||
const lines = output.split("\n");
|
const rawPath = this.args?.file_path || this.args?.path || "";
|
||||||
|
const lang = getLanguageFromPath(rawPath);
|
||||||
|
const lines = lang ? highlightCode(replaceTabs(output), lang) : output.split("\n");
|
||||||
|
|
||||||
const maxLines = this.expanded ? lines.length : 10;
|
const maxLines = this.expanded ? lines.length : 10;
|
||||||
const displayLines = lines.slice(0, maxLines);
|
const displayLines = lines.slice(0, maxLines);
|
||||||
const remaining = lines.length - maxLines;
|
const remaining = lines.length - maxLines;
|
||||||
|
|
||||||
text += "\n\n" + displayLines.map((line: string) => theme.fg("toolOutput", replaceTabs(line))).join("\n");
|
text +=
|
||||||
|
"\n\n" +
|
||||||
|
displayLines
|
||||||
|
.map((line: string) => (lang ? replaceTabs(line) : theme.fg("toolOutput", replaceTabs(line))))
|
||||||
|
.join("\n");
|
||||||
if (remaining > 0) {
|
if (remaining > 0) {
|
||||||
text += theme.fg("toolOutput", `\n... (${remaining} more lines)`);
|
text += theme.fg("toolOutput", `\n... (${remaining} more lines)`);
|
||||||
}
|
}
|
||||||
|
|
@ -318,9 +325,15 @@ export class ToolExecutionComponent extends Container {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (this.toolName === "write") {
|
} else if (this.toolName === "write") {
|
||||||
const path = shortenPath(this.args?.file_path || this.args?.path || "");
|
const rawPath = this.args?.file_path || this.args?.path || "";
|
||||||
|
const path = shortenPath(rawPath);
|
||||||
const fileContent = this.args?.content || "";
|
const fileContent = this.args?.content || "";
|
||||||
const lines = fileContent ? fileContent.split("\n") : [];
|
const lang = getLanguageFromPath(rawPath);
|
||||||
|
const lines = fileContent
|
||||||
|
? lang
|
||||||
|
? highlightCode(replaceTabs(fileContent), lang)
|
||||||
|
: fileContent.split("\n")
|
||||||
|
: [];
|
||||||
const totalLines = lines.length;
|
const totalLines = lines.length;
|
||||||
|
|
||||||
text =
|
text =
|
||||||
|
|
@ -336,13 +349,18 @@ export class ToolExecutionComponent extends Container {
|
||||||
const displayLines = lines.slice(0, maxLines);
|
const displayLines = lines.slice(0, maxLines);
|
||||||
const remaining = lines.length - maxLines;
|
const remaining = lines.length - maxLines;
|
||||||
|
|
||||||
text += "\n\n" + displayLines.map((line: string) => theme.fg("toolOutput", replaceTabs(line))).join("\n");
|
text +=
|
||||||
|
"\n\n" +
|
||||||
|
displayLines
|
||||||
|
.map((line: string) => (lang ? replaceTabs(line) : theme.fg("toolOutput", replaceTabs(line))))
|
||||||
|
.join("\n");
|
||||||
if (remaining > 0) {
|
if (remaining > 0) {
|
||||||
text += theme.fg("toolOutput", `\n... (${remaining} more lines)`);
|
text += theme.fg("toolOutput", `\n... (${remaining} more lines)`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (this.toolName === "edit") {
|
} else if (this.toolName === "edit") {
|
||||||
const path = shortenPath(this.args?.file_path || this.args?.path || "");
|
const rawPath = this.args?.file_path || this.args?.path || "";
|
||||||
|
const path = shortenPath(rawPath);
|
||||||
text =
|
text =
|
||||||
theme.fg("toolTitle", theme.bold("edit")) +
|
theme.fg("toolTitle", theme.bold("edit")) +
|
||||||
" " +
|
" " +
|
||||||
|
|
@ -355,17 +373,7 @@ export class ToolExecutionComponent extends Container {
|
||||||
text += "\n\n" + theme.fg("error", errorText);
|
text += "\n\n" + theme.fg("error", errorText);
|
||||||
}
|
}
|
||||||
} else if (this.result.details?.diff) {
|
} else if (this.result.details?.diff) {
|
||||||
const diffLines = this.result.details.diff.split("\n");
|
text += "\n\n" + renderDiff(this.result.details.diff, { filePath: rawPath });
|
||||||
const coloredLines = diffLines.map((line: string) => {
|
|
||||||
if (line.startsWith("+")) {
|
|
||||||
return theme.fg("toolDiffAdded", line);
|
|
||||||
} else if (line.startsWith("-")) {
|
|
||||||
return theme.fg("toolDiffRemoved", line);
|
|
||||||
} else {
|
|
||||||
return theme.fg("toolDiffContext", line);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
text += "\n\n" + coloredLines.join("\n");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (this.toolName === "ls") {
|
} else if (this.toolName === "ls") {
|
||||||
|
|
|
||||||
|
|
@ -51,15 +51,15 @@
|
||||||
"toolDiffRemoved": "red",
|
"toolDiffRemoved": "red",
|
||||||
"toolDiffContext": "gray",
|
"toolDiffContext": "gray",
|
||||||
|
|
||||||
"syntaxComment": "gray",
|
"syntaxComment": "#6A9955",
|
||||||
"syntaxKeyword": "cyan",
|
"syntaxKeyword": "#569CD6",
|
||||||
"syntaxFunction": "blue",
|
"syntaxFunction": "#DCDCAA",
|
||||||
"syntaxVariable": "",
|
"syntaxVariable": "#9CDCFE",
|
||||||
"syntaxString": "green",
|
"syntaxString": "#CE9178",
|
||||||
"syntaxNumber": "yellow",
|
"syntaxNumber": "#B5CEA8",
|
||||||
"syntaxType": "cyan",
|
"syntaxType": "#4EC9B0",
|
||||||
"syntaxOperator": "",
|
"syntaxOperator": "#D4D4D4",
|
||||||
"syntaxPunctuation": "gray",
|
"syntaxPunctuation": "#D4D4D4",
|
||||||
|
|
||||||
"thinkingOff": "darkGray",
|
"thinkingOff": "darkGray",
|
||||||
"thinkingMinimal": "#6e6e6e",
|
"thinkingMinimal": "#6e6e6e",
|
||||||
|
|
|
||||||
|
|
@ -50,15 +50,15 @@
|
||||||
"toolDiffRemoved": "red",
|
"toolDiffRemoved": "red",
|
||||||
"toolDiffContext": "mediumGray",
|
"toolDiffContext": "mediumGray",
|
||||||
|
|
||||||
"syntaxComment": "mediumGray",
|
"syntaxComment": "#008000",
|
||||||
"syntaxKeyword": "teal",
|
"syntaxKeyword": "#0000FF",
|
||||||
"syntaxFunction": "blue",
|
"syntaxFunction": "#795E26",
|
||||||
"syntaxVariable": "",
|
"syntaxVariable": "#001080",
|
||||||
"syntaxString": "green",
|
"syntaxString": "#A31515",
|
||||||
"syntaxNumber": "yellow",
|
"syntaxNumber": "#098658",
|
||||||
"syntaxType": "teal",
|
"syntaxType": "#267F99",
|
||||||
"syntaxOperator": "",
|
"syntaxOperator": "#000000",
|
||||||
"syntaxPunctuation": "mediumGray",
|
"syntaxPunctuation": "#000000",
|
||||||
|
|
||||||
"thinkingOff": "lightGray",
|
"thinkingOff": "lightGray",
|
||||||
"thinkingMinimal": "#9e9e9e",
|
"thinkingMinimal": "#9e9e9e",
|
||||||
|
|
|
||||||
|
|
@ -4,6 +4,7 @@ import type { EditorTheme, MarkdownTheme, SelectListTheme } from "@mariozechner/
|
||||||
import { type Static, Type } from "@sinclair/typebox";
|
import { type Static, Type } from "@sinclair/typebox";
|
||||||
import { TypeCompiler } from "@sinclair/typebox/compiler";
|
import { TypeCompiler } from "@sinclair/typebox/compiler";
|
||||||
import chalk from "chalk";
|
import chalk from "chalk";
|
||||||
|
import { highlight } from "cli-highlight";
|
||||||
import { getCustomThemesDir, getThemesDir } from "../../../config.js";
|
import { getCustomThemesDir, getThemesDir } from "../../../config.js";
|
||||||
|
|
||||||
// ============================================================================
|
// ============================================================================
|
||||||
|
|
@ -350,6 +351,10 @@ export class Theme {
|
||||||
return chalk.underline(text);
|
return chalk.underline(text);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
inverse(text: string): string {
|
||||||
|
return chalk.inverse(text);
|
||||||
|
}
|
||||||
|
|
||||||
getFgAnsi(color: ThemeColor): string {
|
getFgAnsi(color: ThemeColor): string {
|
||||||
const ansi = this.fgColors.get(color);
|
const ansi = this.fgColors.get(color);
|
||||||
if (!ansi) throw new Error(`Unknown theme color: ${color}`);
|
if (!ansi) throw new Error(`Unknown theme color: ${color}`);
|
||||||
|
|
@ -630,6 +635,127 @@ export function stopThemeWatcher(): void {
|
||||||
// TUI Helpers
|
// TUI Helpers
|
||||||
// ============================================================================
|
// ============================================================================
|
||||||
|
|
||||||
|
type CliHighlightTheme = Record<string, (s: string) => string>;
|
||||||
|
|
||||||
|
let cachedHighlightThemeFor: Theme | undefined;
|
||||||
|
let cachedCliHighlightTheme: CliHighlightTheme | undefined;
|
||||||
|
|
||||||
|
function buildCliHighlightTheme(t: Theme): CliHighlightTheme {
|
||||||
|
return {
|
||||||
|
keyword: (s: string) => t.fg("syntaxKeyword", s),
|
||||||
|
built_in: (s: string) => t.fg("syntaxType", s),
|
||||||
|
literal: (s: string) => t.fg("syntaxNumber", s),
|
||||||
|
number: (s: string) => t.fg("syntaxNumber", s),
|
||||||
|
string: (s: string) => t.fg("syntaxString", s),
|
||||||
|
comment: (s: string) => t.fg("syntaxComment", s),
|
||||||
|
function: (s: string) => t.fg("syntaxFunction", s),
|
||||||
|
title: (s: string) => t.fg("syntaxFunction", s),
|
||||||
|
class: (s: string) => t.fg("syntaxType", s),
|
||||||
|
type: (s: string) => t.fg("syntaxType", s),
|
||||||
|
attr: (s: string) => t.fg("syntaxVariable", s),
|
||||||
|
variable: (s: string) => t.fg("syntaxVariable", s),
|
||||||
|
params: (s: string) => t.fg("syntaxVariable", s),
|
||||||
|
operator: (s: string) => t.fg("syntaxOperator", s),
|
||||||
|
punctuation: (s: string) => t.fg("syntaxPunctuation", s),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function getCliHighlightTheme(t: Theme): CliHighlightTheme {
|
||||||
|
if (cachedHighlightThemeFor !== t || !cachedCliHighlightTheme) {
|
||||||
|
cachedHighlightThemeFor = t;
|
||||||
|
cachedCliHighlightTheme = buildCliHighlightTheme(t);
|
||||||
|
}
|
||||||
|
return cachedCliHighlightTheme;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Highlight code with syntax coloring based on file extension or language.
|
||||||
|
* Returns array of highlighted lines.
|
||||||
|
*/
|
||||||
|
export function highlightCode(code: string, lang?: string): string[] {
|
||||||
|
const opts = {
|
||||||
|
language: lang,
|
||||||
|
ignoreIllegals: true,
|
||||||
|
theme: getCliHighlightTheme(theme),
|
||||||
|
};
|
||||||
|
try {
|
||||||
|
return highlight(code, opts).split("\n");
|
||||||
|
} catch {
|
||||||
|
return code.split("\n");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get language identifier from file path extension.
|
||||||
|
*/
|
||||||
|
export function getLanguageFromPath(filePath: string): string | undefined {
|
||||||
|
const ext = filePath.split(".").pop()?.toLowerCase();
|
||||||
|
if (!ext) return undefined;
|
||||||
|
|
||||||
|
const extToLang: Record<string, string> = {
|
||||||
|
ts: "typescript",
|
||||||
|
tsx: "typescript",
|
||||||
|
js: "javascript",
|
||||||
|
jsx: "javascript",
|
||||||
|
mjs: "javascript",
|
||||||
|
cjs: "javascript",
|
||||||
|
py: "python",
|
||||||
|
rb: "ruby",
|
||||||
|
rs: "rust",
|
||||||
|
go: "go",
|
||||||
|
java: "java",
|
||||||
|
kt: "kotlin",
|
||||||
|
swift: "swift",
|
||||||
|
c: "c",
|
||||||
|
h: "c",
|
||||||
|
cpp: "cpp",
|
||||||
|
cc: "cpp",
|
||||||
|
cxx: "cpp",
|
||||||
|
hpp: "cpp",
|
||||||
|
cs: "csharp",
|
||||||
|
php: "php",
|
||||||
|
sh: "bash",
|
||||||
|
bash: "bash",
|
||||||
|
zsh: "bash",
|
||||||
|
fish: "fish",
|
||||||
|
ps1: "powershell",
|
||||||
|
sql: "sql",
|
||||||
|
html: "html",
|
||||||
|
htm: "html",
|
||||||
|
css: "css",
|
||||||
|
scss: "scss",
|
||||||
|
sass: "sass",
|
||||||
|
less: "less",
|
||||||
|
json: "json",
|
||||||
|
yaml: "yaml",
|
||||||
|
yml: "yaml",
|
||||||
|
toml: "toml",
|
||||||
|
xml: "xml",
|
||||||
|
md: "markdown",
|
||||||
|
markdown: "markdown",
|
||||||
|
dockerfile: "dockerfile",
|
||||||
|
makefile: "makefile",
|
||||||
|
cmake: "cmake",
|
||||||
|
lua: "lua",
|
||||||
|
perl: "perl",
|
||||||
|
r: "r",
|
||||||
|
scala: "scala",
|
||||||
|
clj: "clojure",
|
||||||
|
ex: "elixir",
|
||||||
|
exs: "elixir",
|
||||||
|
erl: "erlang",
|
||||||
|
hs: "haskell",
|
||||||
|
ml: "ocaml",
|
||||||
|
vim: "vim",
|
||||||
|
graphql: "graphql",
|
||||||
|
proto: "protobuf",
|
||||||
|
tf: "hcl",
|
||||||
|
hcl: "hcl",
|
||||||
|
};
|
||||||
|
|
||||||
|
return extToLang[ext];
|
||||||
|
}
|
||||||
|
|
||||||
export function getMarkdownTheme(): MarkdownTheme {
|
export function getMarkdownTheme(): MarkdownTheme {
|
||||||
return {
|
return {
|
||||||
heading: (text: string) => theme.fg("mdHeading", text),
|
heading: (text: string) => theme.fg("mdHeading", text),
|
||||||
|
|
@ -646,6 +772,18 @@ export function getMarkdownTheme(): MarkdownTheme {
|
||||||
italic: (text: string) => theme.italic(text),
|
italic: (text: string) => theme.italic(text),
|
||||||
underline: (text: string) => theme.underline(text),
|
underline: (text: string) => theme.underline(text),
|
||||||
strikethrough: (text: string) => chalk.strikethrough(text),
|
strikethrough: (text: string) => chalk.strikethrough(text),
|
||||||
|
highlightCode: (code: string, lang?: string): string[] => {
|
||||||
|
const opts = {
|
||||||
|
language: lang,
|
||||||
|
ignoreIllegals: true,
|
||||||
|
theme: getCliHighlightTheme(theme),
|
||||||
|
};
|
||||||
|
try {
|
||||||
|
return highlight(code, opts).split("\n");
|
||||||
|
} catch {
|
||||||
|
return code.split("\n").map((line) => theme.fg("mdCodeBlock", line));
|
||||||
|
}
|
||||||
|
},
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -40,6 +40,7 @@ export interface MarkdownTheme {
|
||||||
italic: (text: string) => string;
|
italic: (text: string) => string;
|
||||||
strikethrough: (text: string) => string;
|
strikethrough: (text: string) => string;
|
||||||
underline: (text: string) => string;
|
underline: (text: string) => string;
|
||||||
|
highlightCode?: (code: string, lang?: string) => string[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export class Markdown implements Component {
|
export class Markdown implements Component {
|
||||||
|
|
@ -263,10 +264,17 @@ export class Markdown implements Component {
|
||||||
|
|
||||||
case "code": {
|
case "code": {
|
||||||
lines.push(this.theme.codeBlockBorder("```" + (token.lang || "")));
|
lines.push(this.theme.codeBlockBorder("```" + (token.lang || "")));
|
||||||
// Split code by newlines and style each line
|
if (this.theme.highlightCode) {
|
||||||
const codeLines = token.text.split("\n");
|
const highlightedLines = this.theme.highlightCode(token.text, token.lang);
|
||||||
for (const codeLine of codeLines) {
|
for (const hlLine of highlightedLines) {
|
||||||
lines.push(" " + this.theme.codeBlock(codeLine));
|
lines.push(" " + hlLine);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Split code by newlines and style each line
|
||||||
|
const codeLines = token.text.split("\n");
|
||||||
|
for (const codeLine of codeLines) {
|
||||||
|
lines.push(" " + this.theme.codeBlock(codeLine));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
lines.push(this.theme.codeBlockBorder("```"));
|
lines.push(this.theme.codeBlockBorder("```"));
|
||||||
if (nextTokenType !== "space") {
|
if (nextTokenType !== "space") {
|
||||||
|
|
@ -471,9 +479,16 @@ export class Markdown implements Component {
|
||||||
} else if (token.type === "code") {
|
} else if (token.type === "code") {
|
||||||
// Code block in list item
|
// Code block in list item
|
||||||
lines.push(this.theme.codeBlockBorder("```" + (token.lang || "")));
|
lines.push(this.theme.codeBlockBorder("```" + (token.lang || "")));
|
||||||
const codeLines = token.text.split("\n");
|
if (this.theme.highlightCode) {
|
||||||
for (const codeLine of codeLines) {
|
const highlightedLines = this.theme.highlightCode(token.text, token.lang);
|
||||||
lines.push(" " + this.theme.codeBlock(codeLine));
|
for (const hlLine of highlightedLines) {
|
||||||
|
lines.push(" " + hlLine);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
const codeLines = token.text.split("\n");
|
||||||
|
for (const codeLine of codeLines) {
|
||||||
|
lines.push(" " + this.theme.codeBlock(codeLine));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
lines.push(this.theme.codeBlockBorder("```"));
|
lines.push(this.theme.codeBlockBorder("```"));
|
||||||
} else {
|
} else {
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue