@make-u-free/migi 0.3.8 → 0.3.9

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (2) hide show
  1. package/bin/migi.js +56 -17
  2. package/package.json +1 -1
package/bin/migi.js CHANGED
@@ -1,6 +1,6 @@
1
1
  #!/usr/bin/env node
2
2
  import '../src/tls.js' // 企業CA(Zscaler等)を起動直後に読み込む
3
- import readline from 'readline'
3
+ import readline, { emitKeypressEvents } from 'readline'
4
4
  import chalk from 'chalk'
5
5
  import dotenv from 'dotenv'
6
6
  import { MigiAgent } from '../src/agent.js'
@@ -78,24 +78,63 @@ function sepWithLabel(label) {
78
78
  return chalk.dim(left + right)
79
79
  }
80
80
 
81
- // ---- 複数行入力(空行で送信)----
82
- async function readMultiLine() {
83
- const lines = []
81
+ // ---- チャット入力(Enter送信 / Shift+Enter改行)----
82
+ async function readChatInput() {
84
83
  return new Promise((resolve) => {
85
- const onLine = (line) => {
86
- if (line === '' && lines.length > 0) {
87
- rl.removeListener('line', onLine)
88
- // 入力ボックス下辺(欄の閉じ)+ 欄外ガイド
89
- console.log(sep())
90
- console.log(chalk.dim(` ✦ ${model} · Enterで改行 / 空行で送信`))
91
- resolve(lines.join('\n'))
92
- } else {
93
- lines.push(line)
94
- process.stdout.write(chalk.cyan(' '))
84
+ const lines = ['']
85
+
86
+ emitKeypressEvents(process.stdin)
87
+ if (process.stdin.isTTY) process.stdin.setRawMode(true)
88
+ process.stdout.write(chalk.cyan(' '))
89
+
90
+ const onKey = (str, key) => {
91
+ if (!key) {
92
+ // IME確定などの複合文字
93
+ if (str) { lines[lines.length - 1] += str; process.stdout.write(str) }
94
+ return
95
+ }
96
+
97
+ // Ctrl+C
98
+ if (key.ctrl && key.name === 'c') {
99
+ console.log(chalk.cyan('\n\n お疲れ様でした!またね。\n'))
100
+ process.exit(0)
101
+ }
102
+
103
+ if (key.name === 'return') {
104
+ if (key.shift) {
105
+ // Shift+Enter → 改行
106
+ lines.push('')
107
+ process.stdout.write('\n ')
108
+ } else {
109
+ // Enter → 送信
110
+ const content = lines.join('\n').trim()
111
+ if (!content) return // 空は無視
112
+ process.stdin.removeListener('keypress', onKey)
113
+ if (process.stdin.isTTY) process.stdin.setRawMode(false)
114
+ process.stdout.write('\n')
115
+ console.log(sep())
116
+ console.log(chalk.dim(` ✦ ${model} · Shift+Enterで改行 / Enterで送信`))
117
+ resolve(content)
118
+ }
119
+ return
120
+ }
121
+
122
+ if (key.name === 'backspace') {
123
+ const cur = lines[lines.length - 1]
124
+ if (cur.length > 0) {
125
+ lines[lines.length - 1] = cur.slice(0, -1)
126
+ process.stdout.write('\b \b')
127
+ }
128
+ return
129
+ }
130
+
131
+ if (str && !key.ctrl && !key.meta) {
132
+ lines[lines.length - 1] += str
133
+ process.stdout.write(str)
95
134
  }
96
135
  }
97
- process.stdout.write(chalk.cyan(' '))
98
- rl.on('line', onLine)
136
+
137
+ process.stdin.on('keypress', onKey)
99
138
  })
100
139
  }
101
140
 
@@ -104,7 +143,7 @@ async function prompt() {
104
143
  // 入力ボックス上辺(ユーザー名をセパレーターに埋め込む)
105
144
  console.log('\n' + sepWithLabel(chalk.bold.cyan(userName || 'あなた')))
106
145
 
107
- const input = (await readMultiLine()).trim()
146
+ const input = (await readChatInput()).trim()
108
147
  if (!input) return prompt()
109
148
 
110
149
  // --- ビルトインコマンド ---
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@make-u-free/migi",
3
- "version": "0.3.8",
3
+ "version": "0.3.9",
4
4
  "description": "Your AI right-hand agent. Works anywhere, with any LLM API.",
5
5
  "type": "module",
6
6
  "bin": {