cognova 0.2.12 → 0.2.13

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 (188) hide show
  1. package/.output/nitro.json +1 -1
  2. package/.output/public/_nuxt/{DwY7rCd_.js → -HePz7lp.js} +2 -2
  3. package/.output/public/_nuxt/{CY-cjAwJ.js → 4ALIM-zZ.js} +1 -1
  4. package/.output/public/_nuxt/{DXdwpJ-I.js → 5sUbkh_6.js} +2 -2
  5. package/.output/public/_nuxt/{_cy8R3nk.js → 6bB8Ev7a.js} +1 -1
  6. package/.output/public/_nuxt/{1zUTf4AP.js → 7rT1DCe2.js} +1 -1
  7. package/.output/public/_nuxt/{vIOxcXKR.js → 9O1MXkck.js} +1 -1
  8. package/.output/public/_nuxt/{7oCGSglN.js → 9kAKMaPC.js} +1 -1
  9. package/.output/public/_nuxt/{CVqlefTY.js → B-DMcEU9.js} +1 -1
  10. package/.output/public/_nuxt/{BS2ZNXI1.js → B0nzD8Sk.js} +1 -1
  11. package/.output/public/_nuxt/B2PqR2Vu.js +1 -0
  12. package/.output/public/_nuxt/{Db4KMnt8.js → B2wI_pYg.js} +1 -1
  13. package/.output/public/_nuxt/{zq-a1TeT.js → B5FuOh7E.js} +1 -1
  14. package/.output/public/_nuxt/B6om4MW2.js +1 -0
  15. package/.output/public/_nuxt/{Dya5oK8u.js → B7ikW4eF.js} +1 -1
  16. package/.output/public/_nuxt/{BYHCP8x7.js → B8DFEjxA.js} +1 -1
  17. package/.output/public/_nuxt/{6boMs_nF.js → B9kWNcU7.js} +1 -1
  18. package/.output/public/_nuxt/BAZ8sewA.js +1 -0
  19. package/.output/public/_nuxt/{BXuWCWsJ.js → BCPwL4ma.js} +1 -1
  20. package/.output/public/_nuxt/{DY3uK7wM.js → BCrlmHh_.js} +1 -1
  21. package/.output/public/_nuxt/{BR8bKz8h.js → BD-AFJWW.js} +1 -1
  22. package/.output/public/_nuxt/{BtHQ1T0f.js → BGcQMReV.js} +1 -1
  23. package/.output/public/_nuxt/{BDyn4ApB.js → BL1mkqgd.js} +3 -3
  24. package/.output/public/_nuxt/BNJJSRKh.js +1 -0
  25. package/.output/public/_nuxt/{D8lwrAYS.js → BNeeERBt.js} +1 -1
  26. package/.output/public/_nuxt/{CVFwOzbl.js → BSJb5amZ.js} +1 -1
  27. package/.output/public/_nuxt/{CEnSeCqn.js → BSsaE9Db.js} +1 -1
  28. package/.output/public/_nuxt/{DzA58_Lm.js → BVO3XpHT.js} +1 -1
  29. package/.output/public/_nuxt/{CuxrHsu-.js → BWuL2CzO.js} +1 -1
  30. package/.output/public/_nuxt/{0yk-pS3R.js → B_9zJTqe.js} +1 -1
  31. package/.output/public/_nuxt/{DcJbYLTp.js → B_Lr1vlV.js} +1 -1
  32. package/.output/public/_nuxt/{CZoEPC_Q.js → Bc8bTnkd.js} +3 -3
  33. package/.output/public/_nuxt/{BKXg-alD.js → BghuwIGa.js} +1 -1
  34. package/.output/public/_nuxt/{PP_4ebzl.js → BjsKx597.js} +1 -1
  35. package/.output/public/_nuxt/{DuvzM-P1.js → Bl0qBhCG.js} +1 -1
  36. package/.output/public/_nuxt/{B_wilgcd.js → Bl43dkDq.js} +1 -1
  37. package/.output/public/_nuxt/{JbHa4oXq.js → BnMTFF5R.js} +1 -1
  38. package/.output/public/_nuxt/{CY-QVcA5.js → BrX-PnSK.js} +2 -2
  39. package/.output/public/_nuxt/{DYbZBZet.js → BtV7oNn3.js} +1 -1
  40. package/.output/public/_nuxt/{apYB9dr5.js → BvFYnRMR.js} +1 -1
  41. package/.output/public/_nuxt/{x6FRJ5ac.js → BvTUBUNN.js} +1 -1
  42. package/.output/public/_nuxt/{wO6z2ugJ.js → ByLsRRoQ.js} +1 -1
  43. package/.output/public/_nuxt/{DC4idAGt.js → C1LcEMwB.js} +1 -1
  44. package/.output/public/_nuxt/{YX8avsvq.js → C2IiXp9N.js} +2 -2
  45. package/.output/public/_nuxt/{fbyIeNkc.js → C6OSRpRd.js} +1 -1
  46. package/.output/public/_nuxt/{3jQMk_5H.js → C796kFT9.js} +1 -1
  47. package/.output/public/_nuxt/{DEd2xVbS.js → C7m9zGMG.js} +1 -1
  48. package/.output/public/_nuxt/C8AtdrMH.js +1 -0
  49. package/.output/public/_nuxt/CAWbHTgy.js +1 -0
  50. package/.output/public/_nuxt/{ixlNW2So.js → CFmvNggn.js} +1 -1
  51. package/.output/public/_nuxt/{gTrVszwd.js → CGplHfmR.js} +1 -1
  52. package/.output/public/_nuxt/CJPrV25_.js +1 -0
  53. package/.output/public/_nuxt/{U1MWjQMi.js → CKKEWapF.js} +1 -1
  54. package/.output/public/_nuxt/{C6qKcgOY.js → CLkkdThv.js} +22 -22
  55. package/.output/public/_nuxt/CNjv28J2.js +1 -0
  56. package/.output/public/_nuxt/{BLnYhy_t.js → CRRw0Thu.js} +1 -1
  57. package/.output/public/_nuxt/{10_wwHSz.js → CT8s4rqa.js} +1 -1
  58. package/.output/public/_nuxt/{C27WoGNZ.js → CZSOrxUy.js} +1 -1
  59. package/.output/public/_nuxt/{DNP5E1bC.js → CaNEL9JC.js} +5 -5
  60. package/.output/public/_nuxt/{DhuOKJda.js → CbTLg9oX.js} +1 -1
  61. package/.output/public/_nuxt/CfCXpVn5.js +1 -0
  62. package/.output/public/_nuxt/{C9WIgRRL.js → ChmNlvWR.js} +1 -1
  63. package/.output/public/_nuxt/{t8aDAkZ5.js → CiNxIhaq.js} +1 -1
  64. package/.output/public/_nuxt/{E3rXPwU8.js → Ck0QYdxe.js} +1 -1
  65. package/.output/public/_nuxt/{DUVzIl3o.js → Ckb0pu4-.js} +1 -1
  66. package/.output/public/_nuxt/{vScW1Zgm.js → CkwPRs1D.js} +1 -1
  67. package/.output/public/_nuxt/CmEJg8C6.js +2 -0
  68. package/.output/public/_nuxt/{CTCcEJU3.js → CmL_vMJD.js} +1 -1
  69. package/.output/public/_nuxt/{Bciqk4dX.js → Cq8zPxkT.js} +1 -1
  70. package/.output/public/_nuxt/{D8M722pn.js → CseYuM6E.js} +1 -1
  71. package/.output/public/_nuxt/{D0ifH682.js → CxRnB4NC.js} +1 -1
  72. package/.output/public/_nuxt/{CVdCqaby.js → D0zMyG8n.js} +1 -1
  73. package/.output/public/_nuxt/{DhI5cA_n.js → D12qPplu.js} +1 -1
  74. package/.output/public/_nuxt/{Beom-INj.js → D1XYB283.js} +1 -1
  75. package/.output/public/_nuxt/{BUghTae1.js → D2NxqyzW.js} +1 -1
  76. package/.output/public/_nuxt/{CZRnNmU8.js → D5xyrvFu.js} +1 -1
  77. package/.output/public/_nuxt/{2dNDtTiv.js → D6tVZcRs.js} +3 -3
  78. package/.output/public/_nuxt/{EgKnQnf-.js → D7N3EZ-I.js} +1 -1
  79. package/.output/public/_nuxt/{C5R5QaCA.js → D8iBB0Py.js} +1 -1
  80. package/.output/public/_nuxt/{XCjS70z4.js → D8oIH8m1.js} +1 -1
  81. package/.output/public/_nuxt/{BA_pRWwX.js → DB2zZev7.js} +1 -1
  82. package/.output/public/_nuxt/{Cl-LOSDV.js → DBrOAqkm.js} +1 -1
  83. package/.output/public/_nuxt/{DO9SFIh1.js → DDXDFweK.js} +1 -1
  84. package/.output/public/_nuxt/{B93pdGAW.js → DFZkCmLQ.js} +1 -1
  85. package/.output/public/_nuxt/{CLfF6dSn.js → DFpmIy6J.js} +1 -1
  86. package/.output/public/_nuxt/{BfcLZ_fC.js → DKZmgHNU.js} +1 -1
  87. package/.output/public/_nuxt/{CPutXj8l.js → DNcOC5zj.js} +1 -1
  88. package/.output/public/_nuxt/{BHtY0l0l.js → DQhm-UPG.js} +1 -1
  89. package/.output/public/_nuxt/{C2FxZbO_.js → DR8tGJhL.js} +1 -1
  90. package/.output/public/_nuxt/{D4UJwDQJ.js → DVAEVTjy.js} +1 -1
  91. package/.output/public/_nuxt/{2nOqGUPr.js → DVqrPkCJ.js} +1 -1
  92. package/.output/public/_nuxt/DWZ9ieYa.js +1 -0
  93. package/.output/public/_nuxt/{nnQqD5pb.js → DWj6VsZv.js} +1 -1
  94. package/.output/public/_nuxt/{C2vq6Te8.js → DXJ9PxCv.js} +1 -1
  95. package/.output/public/_nuxt/{CobqYwkp.js → D_ziPROx.js} +1 -1
  96. package/.output/public/_nuxt/{GtEM7xVU.js → Da2vwoYG.js} +1 -1
  97. package/.output/public/_nuxt/{Cc-2ziaZ.js → Ddz3HcBB.js} +1 -1
  98. package/.output/public/_nuxt/{Di-Nc75e.js → DgYei0bK.js} +1 -1
  99. package/.output/public/_nuxt/{DQh6I9z9.js → Di_3GV_R.js} +1 -1
  100. package/.output/public/_nuxt/{ByYh0uRg.js → Djog1f_e.js} +1 -1
  101. package/.output/public/_nuxt/{C4mwL7SE.js → Dkap2-Pj.js} +2 -2
  102. package/.output/public/_nuxt/{D5q8SnqE.js → DlJe_sHk.js} +1 -1
  103. package/.output/public/_nuxt/DlPOQpH_.js +1 -0
  104. package/.output/public/_nuxt/{DTGenhcA.js → DmYWMtVA.js} +1 -1
  105. package/.output/public/_nuxt/DmxKU7sH.js +1 -0
  106. package/.output/public/_nuxt/{DtjjnHnt.js → DowkFZ2V.js} +1 -1
  107. package/.output/public/_nuxt/{0NJ3PaRM.js → DsodHRrQ.js} +1 -1
  108. package/.output/public/_nuxt/{Bk6JUtIo.js → DuzpliIL.js} +1 -1
  109. package/.output/public/_nuxt/DvHS7_h2.js +1 -0
  110. package/.output/public/_nuxt/DwSY6_2U.js +1 -0
  111. package/.output/public/_nuxt/{Djs0Tlpa.js → DypJaSm-.js} +1 -1
  112. package/.output/public/_nuxt/{C71_a1IX.js → DzKaXH6P.js} +1 -1
  113. package/.output/public/_nuxt/{CBTkrk2M.js → FK-jSuOT.js} +1 -1
  114. package/.output/public/_nuxt/{ILEvizzp.js → Fc2JsQt4.js} +1 -1
  115. package/.output/public/_nuxt/{BIXrSYwR.js → IYCBGuRS.js} +1 -1
  116. package/.output/public/_nuxt/{JX1oqJI9.js → LRO7YHoH.js} +1 -1
  117. package/.output/public/_nuxt/{Kw0zy3FG.js → NrRzUJNY.js} +1 -1
  118. package/.output/public/_nuxt/{DJTCT0bl.js → OeVS0Rsb.js} +1 -1
  119. package/.output/public/_nuxt/{BmOtR1wX.js → OlF0QnKF.js} +1 -1
  120. package/.output/public/_nuxt/{DTAStixR.js → QuAoqY7t.js} +1 -1
  121. package/.output/public/_nuxt/{8q5NepGW.js → RlXHkmOd.js} +2 -2
  122. package/.output/public/_nuxt/{rfGRTJJW.js → TxMdabnL.js} +1 -1
  123. package/.output/public/_nuxt/{CZVzFlpI.js → WZnvL9Dh.js} +1 -1
  124. package/.output/public/_nuxt/ZvG7a1W3.js +1 -0
  125. package/.output/public/_nuxt/{N5XtbYVD.js → aPNZBVrG.js} +1 -1
  126. package/.output/public/_nuxt/builds/latest.json +1 -1
  127. package/.output/public/_nuxt/builds/meta/495aca73-bfa5-456e-b96a-f02e009c72d3.json +1 -0
  128. package/.output/public/_nuxt/entry.C_udkZt9.css +1 -0
  129. package/.output/public/_nuxt/{D_4LPm1U.js → evY81hM3.js} +1 -1
  130. package/.output/public/_nuxt/{DzGy77Vr.js → h_bXa-f2.js} +1 -1
  131. package/.output/public/_nuxt/{D9nmzBAw.js → i6QBzaEi.js} +1 -1
  132. package/.output/public/_nuxt/{BGgwYWjH.js → iTVKJ1kw.js} +1 -1
  133. package/.output/public/_nuxt/{Um1vPiAz.js → j92uR3uR.js} +1 -1
  134. package/.output/public/_nuxt/jdUkeAj_.js +1 -0
  135. package/.output/public/_nuxt/{BmVLxnDU.js → jeRvcMHU.js} +1 -1
  136. package/.output/public/_nuxt/{SrncdpaW.js → kuv0zpKq.js} +1 -1
  137. package/.output/public/_nuxt/{DfF81NlA.js → l3lKQs7b.js} +1 -1
  138. package/.output/public/_nuxt/{4K03TkGA.js → pdYvpiud.js} +3 -3
  139. package/.output/public/_nuxt/{C39dQ88F.js → qIgZ7OXa.js} +1 -1
  140. package/.output/public/_nuxt/{DS2wStH1.js → qcVwZv8e.js} +1 -1
  141. package/.output/public/_nuxt/{g5MjDvm5.js → ru8fqqWs.js} +1 -1
  142. package/.output/public/_nuxt/{CQqCBrXj.js → sBzyvWMJ.js} +1 -1
  143. package/.output/public/_nuxt/{D3AZaldL.js → sgmhgpCS.js} +1 -1
  144. package/.output/public/_nuxt/{ChcO9s3o.js → y2jv0da-.js} +1 -1
  145. package/.output/public/_nuxt/{Dn5a-guE.js → yyS4-kAX.js} +1 -1
  146. package/.output/public/_nuxt/zIoBLBHF.js +1 -0
  147. package/.output/public/_nuxt/zpazwPgx.js +1 -0
  148. package/.output/server/chunks/build/{chat-CR3JIVEq.mjs → chat-m4-n9vC6.mjs} +516 -14
  149. package/.output/server/chunks/build/chat-m4-n9vC6.mjs.map +1 -0
  150. package/.output/server/chunks/build/client.precomputed.mjs +1 -1
  151. package/.output/server/chunks/build/server.mjs +1 -1
  152. package/.output/server/chunks/build/styles.mjs +4 -4
  153. package/.output/server/chunks/nitro/nitro.mjs +875 -863
  154. package/.output/server/chunks/routes/_ws/chat.mjs +39 -5
  155. package/.output/server/chunks/routes/_ws/chat.mjs.map +1 -1
  156. package/Claude/skills/_lib/api.py +3 -2
  157. package/app/components/chat/ChatInput.vue +133 -5
  158. package/app/components/chat/MessageBubble.vue +188 -6
  159. package/app/composables/useAttachments.ts +196 -0
  160. package/app/composables/useChat.ts +14 -2
  161. package/app/pages/chat.vue +4 -2
  162. package/package.json +1 -1
  163. package/server/bridge/adapters/imessage.ts +9 -2
  164. package/server/routes/_ws/chat.ts +61 -7
  165. package/server/utils/chat-session-manager.ts +20 -2
  166. package/shared/types/index.ts +27 -1
  167. package/.output/public/_nuxt/BH0QDWEm.js +0 -1
  168. package/.output/public/_nuxt/BK8S2ony.js +0 -1
  169. package/.output/public/_nuxt/BZrSPX0S.js +0 -1
  170. package/.output/public/_nuxt/BaZwjF8o.js +0 -1
  171. package/.output/public/_nuxt/BkfI94R8.js +0 -1
  172. package/.output/public/_nuxt/C3NST0BU.js +0 -1
  173. package/.output/public/_nuxt/CKxSHyww.js +0 -1
  174. package/.output/public/_nuxt/CSKJ-Ahh.js +0 -1
  175. package/.output/public/_nuxt/CWyMCJQH.js +0 -1
  176. package/.output/public/_nuxt/CYPO5o_C.js +0 -1
  177. package/.output/public/_nuxt/CqNwHCpo.js +0 -1
  178. package/.output/public/_nuxt/CwPdIZ9J.js +0 -1
  179. package/.output/public/_nuxt/D5jZq8b3.js +0 -1
  180. package/.output/public/_nuxt/DH4YkDAi.js +0 -1
  181. package/.output/public/_nuxt/DPklr_tJ.js +0 -2
  182. package/.output/public/_nuxt/DQM4eBdt.js +0 -1
  183. package/.output/public/_nuxt/DUAAXJ6Q.js +0 -1
  184. package/.output/public/_nuxt/DmmdPt_5.js +0 -1
  185. package/.output/public/_nuxt/H6JbrRBU.js +0 -1
  186. package/.output/public/_nuxt/builds/meta/a892969f-e07d-4f44-8d7e-57a4b7f33d94.json +0 -1
  187. package/.output/public/_nuxt/entry.NKPfH2kE.css +0 -1
  188. package/.output/server/chunks/build/chat-CR3JIVEq.mjs.map +0 -1
@@ -46,6 +46,33 @@ function send(peer, data) {
46
46
  console.error("[chat] send failed:", err);
47
47
  }
48
48
  }
49
+ function buildSdkContent(message, attachments, documents) {
50
+ if (!(attachments == null ? void 0 : attachments.length) && !(documents == null ? void 0 : documents.length)) return message;
51
+ const blocks = [];
52
+ if (attachments == null ? void 0 : attachments.length) {
53
+ for (const img of attachments) {
54
+ blocks.push({
55
+ type: "image",
56
+ source: {
57
+ type: "base64",
58
+ media_type: img.source.media_type,
59
+ data: img.source.data
60
+ }
61
+ });
62
+ }
63
+ }
64
+ if (documents == null ? void 0 : documents.length) {
65
+ for (const doc of documents) {
66
+ blocks.push({
67
+ type: "document",
68
+ source: doc.source,
69
+ title: doc.title
70
+ });
71
+ }
72
+ }
73
+ if (message) blocks.push({ type: "text", text: message });
74
+ return blocks;
75
+ }
49
76
  const chat = defineWebSocketHandler({
50
77
  open(peer) {
51
78
  console.log(`[chat] WebSocket opened: ${peer.id}`);
@@ -56,7 +83,7 @@ const chat = defineWebSocketHandler({
56
83
  const msg = JSON.parse(rawMessage.text());
57
84
  switch (msg.type) {
58
85
  case "chat:send":
59
- handleSend(peer, msg.message, msg.conversationId);
86
+ handleSend(peer, msg.message, msg.conversationId, msg.attachments, msg.documents);
60
87
  break;
61
88
  case "chat:interrupt":
62
89
  handleInterrupt(peer, msg.conversationId);
@@ -74,14 +101,14 @@ const chat = defineWebSocketHandler({
74
101
  console.error(`[chat] WebSocket error for ${peer.id}:`, error);
75
102
  }
76
103
  });
77
- async function handleSend(peer, message, conversationId) {
104
+ async function handleSend(peer, message, conversationId, attachments, documents) {
78
105
  const db = getDb();
79
106
  let convId = conversationId;
80
107
  let resumeSessionId;
81
108
  if (!convId) {
82
109
  const [conv] = await db.insert(conversations).values({
83
110
  sessionId: randomUUID(),
84
- title: message.slice(0, 100),
111
+ title: message.slice(0, 100) || "File attachment",
85
112
  status: "streaming",
86
113
  messageCount: 0,
87
114
  totalCostUsd: 0
@@ -101,13 +128,20 @@ async function handleSend(peer, message, conversationId) {
101
128
  }
102
129
  await db.update(conversations).set({ status: "streaming" }).where(eq(conversations.id, convId));
103
130
  }
131
+ const userContent = [];
132
+ if (attachments == null ? void 0 : attachments.length)
133
+ for (const img of attachments) userContent.push(img);
134
+ if (documents == null ? void 0 : documents.length)
135
+ for (const doc of documents) userContent.push(doc);
136
+ if (message) userContent.push({ type: "text", text: message });
104
137
  await db.insert(conversationMessages).values({
105
138
  conversationId: convId,
106
139
  role: "user",
107
- content: JSON.stringify([{ type: "text", text: message }]),
140
+ content: JSON.stringify(userContent),
108
141
  source: "web"
109
142
  });
110
- const session = chatSessionManager.startSession(convId, message, resumeSessionId);
143
+ const prompt = buildSdkContent(message, attachments, documents);
144
+ const session = chatSessionManager.startSession(convId, prompt, resumeSessionId);
111
145
  send(peer, { type: "chat:stream_start", conversationId: convId });
112
146
  streamSDKResponse(peer, session, convId).catch((err) => {
113
147
  console.error("[chat] Unhandled stream error:", err);
@@ -1 +1 @@
1
- {"version":3,"file":"chat.mjs","sources":["../../../../../server/routes/_ws/chat.ts"],"names":["schema.conversations","schema.conversationMessages"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AASA,SAAA,IAAA,CAAA,MAAA,IAAA,EAAA;AACA,EAAA,IAAA;AACA,IAAA,MAAA,IAAA,GAAA,IAAA,CAAA,SAAA,CAAA,IAAA,CAAA;AACA,IAAA,IAAA,CAAA,KAAA,IAAA,CAAA;AAAA,EACA,SAAA,GAAA,EAAA;AACA,IAAA,OAAA,CAAA,KAAA,CAAA,uBAAA,GAAA,CAAA;AAAA,EACA;AACA;AAEA,aAAA,sBAAA,CAAA;AAAA,EACA,KAAA,IAAA,EAAA;AACA,IAAA,OAAA,CAAA,GAAA,CAAA,CAAA,yBAAA,EAAA,IAAA,CAAA,EAAA,CAAA,CAAA,CAAA;AACA,IAAA,IAAA,CAAA,IAAA,EAAA,EAAA,IAAA,EAAA,gBAAA,EAAA,CAAA;AAAA,EACA,CAAA;AAAA,EAEA,OAAA,CAAA,MAAA,UAAA,EAAA;AACA,IAAA,IAAA;AACA,MAAA,MAAA,GAAA,GAAA,IAAA,CAAA,KAAA,CAAA,UAAA,CAAA,MAAA,CAAA;AAEA,MAAA,QAAA,IAAA,IAAA;AAAA,QACA,KAAA,WAAA;AACA,UAAA,UAAA,CAAA,IAAA,EAAA,GAAA,CAAA,OAAA,EAAA,GAAA,CAAA,cAAA,CAAA;AACA,UAAA;AAAA,QACA,KAAA,gBAAA;AACA,UAAA,eAAA,CAAA,IAAA,EAAA,IAAA,cAAA,CAAA;AACA,UAAA;AAAA;AACA,IACA,SAAA,CAAA,EAAA;AACA,MAAA,OAAA,CAAA,KAAA,CAAA,yBAAA,CAAA,CAAA;AACA,MAAA,IAAA,CAAA,MAAA,EAAA,IAAA,EAAA,YAAA,EAAA,OAAA,EAAA,0BAAA,CAAA;AAAA,IACA;AAAA,EACA,CAAA;AAAA,EAEA,MAAA,IAAA,EAAA;AACA,IAAA,OAAA,CAAA,GAAA,CAAA,CAAA,yBAAA,EAAA,IAAA,CAAA,EAAA,CAAA,CAAA,CAAA;AAAA,EACA,CAAA;AAAA,EAEA,KAAA,CAAA,MAAA,KAAA,EAAA;AACA,IAAA,OAAA,CAAA,KAAA,CAAA,CAAA,2BAAA,EAAA,IAAA,CAAA,EAAA,KAAA,KAAA,CAAA;AAAA,EACA;AACA,CAAA,CAAA;AAGA,eAAA,UAAA,CAAA,IAAA,EAAA,OAAA,EAAA,cAAA,EAAA;AACA,EAAA,MAAA,KAAA,KAAA,EAAA;AAEA,EAAA,IAAA,MAAA,GAAA,cAAA;AACA,EAAA,IAAA,eAAA;AAEA,EAAA,IAAA,CAAA,MAAA,EAAA;AAEA,IAAA,MAAA,CAAA,IAAA,CAAA,GAAA,MAAA,GAAA,MAAA,CAAAA,aAAA,CAAA,CACA,MAAA,CAAA;AAAA,MACA,WAAA,UAAA,EAAA;AAAA,MACA,KAAA,EAAA,OAAA,CAAA,KAAA,CAAA,CAAA,EAAA,GAAA,CAAA;AAAA,MACA,MAAA,EAAA,WAAA;AAAA,MACA,YAAA,EAAA,CAAA;AAAA,MACA,YAAA,EAAA;AAAA,KACA,EACA,SAAA,EAAA;AAEA,IAAA,MAAA,GAAA,IAAA,CAAA,EAAA;AACA,IAAA,IAAA,CAAA,MAAA,EAAA,IAAA,EAAA,sBAAA,EAAA,cAAA,EAAA,QAAA,CAAA;AAAA,EACA,CAAA,MAAA;AAEA,IAAA,MAAA,QAAA,GAAA,kBAAA,CAAA,UAAA,CAAA,MAAA,CAAA;AACA,IAAA,IAAA,QAAA,oBAAA,QAAA,CAAA,YAAA;AAEA,IAAA,IAAA,CAAA,eAAA,EAAA;AACA,MAAA,MAAA,CAAA,IAAA,CAAA,GAAA,MAAA,GAAA,MAAA,EAAA,CACA,KAAAA,aAAA,EACA,KAAA,CAAA,EAAA,CAAAA,aAAA,CAAA,EAAA,EAAA,MAAA,CAAA,CAAA,CACA,MAAA,CAAA,CAAA;AAEA,MAAA,IAAA,CAAA,IAAA,EAAA;AACA,QAAA,IAAA,CAAA,MAAA,EAAA,IAAA,EAAA,YAAA,EAAA,OAAA,EAAA,0BAAA,CAAA;AACA,QAAA;AAAA,MACA;AACA,MAAA,IAAA,IAAA,CAAA,YAAA,EAAA,eAAA,GAAA,IAAA,CAAA,YAAA;AAAA,IACA;AAEA,IAAA,MAAA,GAAA,MAAA,CAAAA,aAAA,CAAA,CACA,IAAA,EAAA,MAAA,EAAA,WAAA,EAAA,EACA,KAAA,CAAA,EAAA,CAAAA,aAAA,CAAA,EAAA,EAAA,MAAA,CAAA,CAAA;AAAA,EACA;AAGA,EAAA,MAAA,EAAA,CAAA,MAAA,CAAAC,oBAAA,EAAA,MAAA,CAAA;AAAA,IACA,cAAA,EAAA,MAAA;AAAA,IACA,IAAA,EAAA,MAAA;AAAA,IACA,OAAA,EAAA,IAAA,CAAA,SAAA,CAAA,CAAA,EAAA,MAAA,MAAA,EAAA,IAAA,EAAA,OAAA,EAAA,CAAA,CAAA;AAAA,IACA,MAAA,EAAA;AAAA,GACA,CAAA;AAGA,EAAA,MAAA,OAAA,GAAA,kBAAA,CAAA,YAAA,CAAA,MAAA,EAAA,SAAA,eAAA,CAAA;AACA,EAAA,IAAA,CAAA,MAAA,EAAA,IAAA,EAAA,mBAAA,EAAA,cAAA,EAAA,QAAA,CAAA;AAEA,EAAA,iBAAA,CAAA,MAAA,OAAA,EAAA,MAAA,CAAA,CAAA,KAAA,CAAA,CAAA,GAAA,KAAA;AACA,IAAA,OAAA,CAAA,KAAA,CAAA,kCAAA,GAAA,CAAA;AAAA,EACA,CAAA,CAAA;AACA;AAGA,SAAA,eAAA,CAAA,MAAA,cAAA,EAAA;AACA,EAAA,MAAA,WAAA,GAAA,kBAAA,CAAA,SAAA,CAAA,cAAA,CAAA;AACA,EAAA,IAAA,CAAA,WAAA;AACA,IAAA,IAAA,CAAA,MAAA,EAAA,IAAA,EAAA,cAAA,cAAA,EAAA,OAAA,EAAA,kCAAA,CAAA;AACA;AAGA,eAAA,iBAAA,CAAA,IAAA,EAAA,OAAA,EAAA,cAAA,EAAA;;AACA,EAAA,MAAA,KAAA,KAAA,EAAA;AACA,EAAA,MAAA,gBAAA,EAAA;AACA,EAAA,IAAA,WAAA,GAAA,EAAA;AAEA,EAAA,IAAA;AACA,IAAA,WAAA,MAAA,OAAA,IAAA,QAAA,YAAA,EAAA;AACA,MAAA,IAAA,QAAA,WAAA,EAAA;AACA,QAAA,IAAA,CAAA,IAAA,EAAA,EAAA,IAAA,EAAA,kBAAA,EAAA,gBAAA,CAAA;AACA,QAAA,MAAA,GAAA,MAAA,CAAAD,aAAA,CAAA,CACA,IAAA,EAAA,MAAA,EAAA,aAAA,EAAA,EACA,KAAA,CAAA,EAAA,CAAAA,aAAA,CAAA,EAAA,EAAA,cAAA,CAAA,CAAA;AACA,QAAA;AAAA,MACA;AAEA,MAAA,IAAA,OAAA,CAAA,IAAA,KAAA,QAAA,IAAA,OAAA,CAAA,YAAA,MAAA,EAAA;AACA,QAAA,OAAA,CAAA,eAAA,OAAA,CAAA,UAAA;AACA,QAAA,MAAA,GAAA,MAAA,CAAAA,aAAA,CAAA,CACA,GAAA,CAAA,EAAA,YAAA,EAAA,OAAA,CAAA,UAAA,EAAA,EACA,KAAA,CAAA,EAAA,CAAAA,aAAA,CAAA,EAAA,EAAA,cAAA,CAAA,CAAA;AAAA,MACA,CAAA,MAAA,IAAA,OAAA,CAAA,IAAA,KAAA,cAAA,EAAA;AAEA,QAAA,MAAA,QAAA,OAAA,CAAA,KAAA;AACA,QAAA,IAAA,KAAA,CAAA,SAAA,qBAAA,EAAA;AACA,UAAA,MAAA,SAAA,GAAA,CAAA,EAAA,GAAA,KAAA,CAAA,KAAA,KAAA,IAAA,GAAA,KAAA,CAAA,GAAA,EAAA,CAAA,IAAA;AACA,UAAA,IAAA,SAAA,KAAA,YAAA,IAAA,KAAA,CAAA,KAAA,CAAA,IAAA,EAAA;AACA,YAAA,WAAA,IAAA,MAAA,KAAA,CAAA,IAAA;AACA,YAAA,IAAA,CAAA,IAAA,EAAA,EAAA,IAAA,EAAA,iBAAA,EAAA,gBAAA,KAAA,EAAA,KAAA,CAAA,KAAA,CAAA,IAAA,EAAA,CAAA;AAAA,UACA;AAAA,QACA,CAAA,MAAA,IAAA,KAAA,CAAA,IAAA,KAAA,qBAAA,EAAA;AACA,UAAA,IAAA,CAAA,CAAA,EAAA,GAAA,KAAA,CAAA,aAAA,KAAA,IAAA,GAAA,KAAA,CAAA,GAAA,EAAA,CAAA,IAAA,MAAA,UAAA,EAAA;AACA,YAAA,IAAA,CAAA,IAAA,EAAA;AAAA,cACA,IAAA,EAAA,iBAAA;AAAA,cACA,cAAA;AAAA,cACA,SAAA,EAAA,MAAA,aAAA,CAAA,EAAA;AAAA,cACA,QAAA,EAAA,MAAA,aAAA,CAAA;AAAA,aACA,CAAA;AAAA,UACA;AAAA,QACA;AAAA,MACA,CAAA,MAAA,IAAA,OAAA,CAAA,IAAA,KAAA,WAAA,EAAA;AAEA,QAAA,KAAA,MAAA,WAAA,EAAA,GAAA,OAAA,CAAA,OAAA,KAAA,IAAA,GAAA,KAAA,CAAA,GAAA,EAAA,CAAA,OAAA,KAAA,EAAA,EAAA;AACA,UAAA,IAAA,KAAA,CAAA,SAAA,MAAA,EAAA;AACA,YAAA,aAAA,CAAA,KAAA,EAAA,IAAA,EAAA,QAAA,IAAA,EAAA,KAAA,CAAA,MAAA,CAAA;AAEA,YAAA,IAAA,CAAA,WAAA;AACA,cAAA,IAAA,CAAA,IAAA,EAAA,EAAA,IAAA,EAAA,iBAAA,EAAA,gBAAA,KAAA,EAAA,KAAA,CAAA,MAAA,CAAA;AAAA,UACA,CAAA,MAAA,IAAA,KAAA,CAAA,IAAA,KAAA,UAAA,EAAA;AACA,YAAA,aAAA,CAAA,IAAA,CAAA;AAAA,cACA,IAAA,EAAA,UAAA;AAAA,cACA,IAAA,KAAA,CAAA,EAAA;AAAA,cACA,MAAA,KAAA,CAAA,IAAA;AAAA,cACA,OAAA,KAAA,CAAA;AAAA,aACA,CAAA;AAAA,UACA;AAAA,QACA;AACA,QAAA,WAAA,GAAA,EAAA;AAAA,MACA,CAAA,MAAA,IAAA,OAAA,CAAA,IAAA,KAAA,MAAA,EAAA;AAEA,QAAA,MAAA,OAAA,GAAA,CAAA,EAAA,GAAA,OAAA,CAAA,OAAA,KAAA,IAAA,GAAA,KAAA,CAAA,GAAA,EAAA,CAAA,OAAA;AACA,QAAA,IAAA,KAAA,CAAA,OAAA,CAAA,OAAA,CAAA,EAAA;AACA,UAAA,KAAA,MAAA,SAAA,OAAA,EAAA;AACA,YAAA,IAAA,KAAA,CAAA,SAAA,aAAA,EAAA;AACA,cAAA,MAAA,UAAA,GAAA,OAAA,KAAA,CAAA,OAAA,KAAA,QAAA,GACA,KAAA,CAAA,OAAA,GACA,KAAA,CAAA,OAAA,CAAA,KAAA,CAAA,OAAA,CAAA,GACA,KAAA,CAAA,OAAA,CAAA,GAAA,CAAA,CAAA,CAAA,KAAA,CAAA,CAAA,IAAA,IAAA,EAAA,CAAA,CAAA,IAAA,CAAA,EAAA,CAAA,GACA,IAAA,CAAA,SAAA,CAAA,CAAA,EAAA,GAAA,KAAA,CAAA,OAAA,KAAA,YAAA,EAAA,CAAA;AACA,cAAA,aAAA,CAAA,IAAA,CAAA;AAAA,gBACA,IAAA,EAAA,aAAA;AAAA,gBACA,aAAA,KAAA,CAAA,WAAA;AAAA,gBACA,OAAA,EAAA,UAAA;AAAA,gBACA,QAAA,EAAA,CAAA,CAAA,KAAA,CAAA;AAAA,eACA,CAAA;AACA,cAAA,IAAA,CAAA,IAAA,EAAA;AAAA,gBACA,IAAA,EAAA,eAAA;AAAA,gBACA,cAAA;AAAA,gBACA,WAAA,KAAA,CAAA,WAAA;AAAA,gBACA,MAAA,EAAA,UAAA,CAAA,KAAA,CAAA,CAAA,EAAA,GAAA,CAAA;AAAA,gBACA,OAAA,EAAA,CAAA,CAAA,KAAA,CAAA;AAAA,eACA,CAAA;AAAA,YACA;AAAA,UACA;AAAA,QACA;AAAA,MACA,CAAA,MAAA,IAAA,OAAA,CAAA,IAAA,KAAA,QAAA,EAAA;AAEA,QAAA,MAAA,GAAA,GAAA,OAAA;AAMA,QAAA,MAAA,OAAA,GAAA,IAAA,cAAA,IAAA,CAAA;AACA,QAAA,MAAA,UAAA,GAAA,IAAA,WAAA,IAAA,CAAA;AACA,QAAA,MAAA,WAAA,GAAA,CAAA,CAAA,EAAA,GAAA,GAAA,CAAA,KAAA,KAAA,IAAA,GAAA,KAAA,CAAA,GAAA,EAAA,CAAA,YAAA,KAAA,CAAA;AACA,QAAA,MAAA,YAAA,GAAA,CAAA,CAAA,EAAA,GAAA,GAAA,CAAA,KAAA,KAAA,IAAA,GAAA,KAAA,CAAA,GAAA,EAAA,CAAA,aAAA,KAAA,CAAA;AAGA,QAAA,IAAA,aAAA,CAAA,SAAA,CAAA,EAAA;AACA,UAAA,MAAA,EAAA,CAAA,MAAA,CAAAC,oBAAA,EAAA,MAAA,CAAA;AAAA,YACA,cAAA;AAAA,YACA,IAAA,EAAA,WAAA;AAAA,YACA,OAAA,EAAA,IAAA,CAAA,SAAA,CAAA,aAAA,CAAA;AAAA,YACA,OAAA;AAAA,YACA;AAAA,WACA,CAAA;AAAA,QACA;AAGA,QAAA,MAAA,IAAA,GAAA,MAAA,EAAA,CAAA,MAAA,GACA,IAAA,CAAAA,oBAAA,CAAA,CACA,MAAA,EAAA,CAAAA,oBAAA,CAAA,cAAA,EAAA,cAAA,CAAA,CAAA;AAGA,QAAA,MAAA,EAAA,CAAA,MAAA,CAAAD,aAAA,EACA,GAAA,CAAA;AAAA,UACA,MAAA,EAAA,MAAA;AAAA,UACA,cAAA,IAAA,CAAA,MAAA;AAAA,UACA,YAAA,EAAA,OAAA;AAAA,UACA,OAAA,sBAAA,IAAA;AAAA,SACA,EACA,KAAA,CAAA,EAAA,CAAAA,aAAA,CAAA,EAAA,EAAA,cAAA,CAAA,CAAA;AAGA,QAAA,MAAA,CAAA,IAAA,CAAA,GAAA,MAAA,EAAA,CAAA,MAAA,CAAA,EAAA,KAAA,EAAAA,aAAA,CAAA,KAAA,EAAA,CAAA,CACA,KAAAA,aAAA,CAAA,CACA,KAAA,CAAA,EAAA,CAAAA,aAAA,CAAA,EAAA,EAAA,cAAA,CAAA,CAAA,CACA,KAAA,CAAA,CAAA,CAAA;AAEA,QAAA,aAAA,CAAA;AAAA,UACA,MAAA,EAAA,MAAA;AAAA,UACA,QAAA,EAAA,cAAA;AAAA,UACA,UAAA,EAAA,CAAA,6BAAA,KAAA,KAAA,MAAA;AAAA,UACA,WAAA;AAAA,UACA,YAAA;AAAA,UACA,OAAA;AAAA,UACA,UAAA;AAAA,UACA,QAAA,EAAA,IAAA,SAAA,IAAA;AAAA,SACA,CAAA;AAEA,QAAA,IAAA,CAAA,IAAA,EAAA;AAAA,UACA,IAAA,EAAA,iBAAA;AAAA,UACA,cAAA;AAAA,UACA,OAAA;AAAA,UACA;AAAA,SACA,CAAA;AAAA,MACA;AAAA,IACA;AAAA,EACA,SAAA,KAAA,EAAA;AACA,IAAA,MAAA,WAAA,KAAA,YAAA,KAAA,GAAA,KAAA,CAAA,OAAA,GAAA,OAAA,KAAA,CAAA;AACA,IAAA,OAAA,CAAA,KAAA,CAAA,wBAAA,QAAA,CAAA;AACA,IAAA,IAAA,CAAA,MAAA,EAAA,IAAA,EAAA,cAAA,cAAA,EAAA,OAAA,EAAA,UAAA,CAAA;AAEA,IAAA,MAAA,GAAA,MAAA,CAAAA,aAAA,CAAA,CACA,IAAA,EAAA,MAAA,EAAA,OAAA,EAAA,EACA,KAAA,CAAA,EAAA,CAAAA,aAAA,CAAA,EAAA,EAAA,cAAA,CAAA,CAAA;AAAA,EACA,CAAA,SAAA;AACA,IAAA,kBAAA,CAAA,cAAA,cAAA,CAAA;AAAA,EACA;AACA;;;;"}
1
+ {"version":3,"file":"chat.mjs","sources":["../../../../../server/routes/_ws/chat.ts"],"names":["schema.conversations","schema.conversationMessages"],"mappings":"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"}
@@ -84,14 +84,15 @@ def api_request(
84
84
  cmd.extend(["-H", f"X-API-Token: {API_TOKEN}"])
85
85
 
86
86
  if params:
87
+ from urllib.parse import quote
87
88
  query_parts = []
88
89
  for k, v in params.items():
89
90
  if v is not None:
90
91
  if isinstance(v, list):
91
92
  for item in v:
92
- query_parts.append(f"{k}={item}")
93
+ query_parts.append(f"{k}={quote(str(item))}")
93
94
  else:
94
- query_parts.append(f"{k}={v}")
95
+ query_parts.append(f"{k}={quote(str(v))}")
95
96
  if query_parts:
96
97
  url = f"{url}?{'&'.join(query_parts)}"
97
98
 
@@ -1,5 +1,5 @@
1
1
  <script setup lang="ts">
2
- import type { ChatSessionStatus, ChatConnectionStatus } from '~~/shared/types'
2
+ import type { ChatSessionStatus, ChatConnectionStatus, ChatImageBlock, ChatDocumentBlock } from '~~/shared/types'
3
3
 
4
4
  const props = defineProps<{
5
5
  sessionStatus: ChatSessionStatus
@@ -7,16 +7,24 @@ const props = defineProps<{
7
7
  }>()
8
8
 
9
9
  const emit = defineEmits<{
10
- send: [message: string]
10
+ send: [message: string, attachments?: ChatImageBlock[], documents?: ChatDocumentBlock[]]
11
11
  interrupt: []
12
12
  }>()
13
13
 
14
+ const { attachments, addFiles, removeAttachment, clearAttachments, toImageBlocks, toDocumentBlocks } = useAttachments()
15
+
14
16
  const inputText = ref('')
15
17
  const textareaRef = ref<HTMLTextAreaElement | null>(null)
18
+ const fileInputRef = ref<HTMLInputElement | null>(null)
19
+ const isDragging = ref(false)
16
20
 
17
21
  const isStreaming = computed(() => props.sessionStatus === 'streaming')
18
22
  const isConnected = computed(() => props.connectionStatus === 'connected')
19
- const canSend = computed(() => isConnected.value && !isStreaming.value && inputText.value.trim().length > 0)
23
+ const canSend = computed(() =>
24
+ isConnected.value
25
+ && !isStreaming.value
26
+ && (inputText.value.trim().length > 0 || attachments.value.length > 0)
27
+ )
20
28
 
21
29
  function handleKeydown(e: KeyboardEvent) {
22
30
  if (e.key === 'Enter' && !e.shiftKey) {
@@ -27,8 +35,16 @@ function handleKeydown(e: KeyboardEvent) {
27
35
 
28
36
  function handleSend() {
29
37
  if (!canSend.value) return
30
- emit('send', inputText.value.trim())
38
+ const images = toImageBlocks()
39
+ const docs = toDocumentBlocks()
40
+ emit(
41
+ 'send',
42
+ inputText.value.trim(),
43
+ images.length > 0 ? images : undefined,
44
+ docs.length > 0 ? docs : undefined
45
+ )
31
46
  inputText.value = ''
47
+ clearAttachments()
32
48
  nextTick(() => {
33
49
  if (textareaRef.value) textareaRef.value.style.height = 'auto'
34
50
  })
@@ -39,10 +55,64 @@ function autoResize(e: Event) {
39
55
  target.style.height = 'auto'
40
56
  target.style.height = Math.min(target.scrollHeight, 200) + 'px'
41
57
  }
58
+
59
+ function handlePaste(e: ClipboardEvent) {
60
+ const files = Array.from(e.clipboardData?.files || [])
61
+ if (files.length > 0) {
62
+ e.preventDefault()
63
+ addFiles(files)
64
+ }
65
+ }
66
+
67
+ function handleDragOver(e: DragEvent) {
68
+ e.preventDefault()
69
+ isDragging.value = true
70
+ }
71
+
72
+ function handleDragLeave() {
73
+ isDragging.value = false
74
+ }
75
+
76
+ function handleDrop(e: DragEvent) {
77
+ e.preventDefault()
78
+ isDragging.value = false
79
+ const files = Array.from(e.dataTransfer?.files || [])
80
+ if (files.length > 0) addFiles(files)
81
+ }
82
+
83
+ function handleFileSelect(e: Event) {
84
+ const input = e.target as HTMLInputElement
85
+ if (input.files?.length) addFiles(input.files)
86
+ input.value = ''
87
+ }
88
+
89
+ const FILE_ACCEPT = [
90
+ 'image/jpeg', 'image/png', 'image/gif', 'image/webp',
91
+ 'application/pdf',
92
+ '.txt', '.md', '.js', '.ts', '.jsx', '.tsx', '.py', '.rb', '.rs', '.go',
93
+ '.java', '.c', '.cpp', '.h', '.cs', '.swift', '.kt',
94
+ '.json', '.yaml', '.yml', '.toml', '.ini',
95
+ '.xml', '.html', '.css', '.scss',
96
+ '.sh', '.sql', '.graphql', '.csv', '.log',
97
+ '.vue', '.svelte', '.prisma', '.lua', '.dart'
98
+ ].join(',')
42
99
  </script>
43
100
 
44
101
  <template>
45
- <div class="border-t border-default p-4">
102
+ <div
103
+ class="border-t border-default p-4 relative"
104
+ @dragover="handleDragOver"
105
+ @dragleave="handleDragLeave"
106
+ @drop="handleDrop"
107
+ >
108
+ <!-- Drag overlay -->
109
+ <div
110
+ v-if="isDragging"
111
+ class="absolute inset-0 bg-primary/10 border-2 border-dashed border-primary rounded-lg flex items-center justify-center z-10"
112
+ >
113
+ <span class="text-sm text-primary font-medium">Drop files here</span>
114
+ </div>
115
+
46
116
  <!-- Connection status -->
47
117
  <div
48
118
  v-if="!isConnected"
@@ -52,6 +122,45 @@ function autoResize(e: Event) {
52
122
  <span>{{ connectionStatus === 'connecting' ? 'Connecting...' : 'Disconnected' }}</span>
53
123
  </div>
54
124
 
125
+ <!-- Attachment preview strip -->
126
+ <div
127
+ v-if="attachments.length"
128
+ class="flex gap-2 mb-2 flex-wrap"
129
+ >
130
+ <div
131
+ v-for="att in attachments"
132
+ :key="att.id"
133
+ class="relative group rounded-lg overflow-hidden border border-default"
134
+ :class="att.kind === 'image' ? 'size-16' : 'h-10 px-3 flex items-center gap-1.5 bg-elevated/50'"
135
+ >
136
+ <!-- Image thumbnail -->
137
+ <img
138
+ v-if="att.kind === 'image'"
139
+ :src="att.previewUrl"
140
+ :alt="att.name"
141
+ class="size-full object-cover"
142
+ >
143
+ <!-- Document chip -->
144
+ <template v-else>
145
+ <UIcon
146
+ :name="att.name.endsWith('.pdf') ? 'i-lucide-file-text' : 'i-lucide-file-code'"
147
+ class="size-4 text-dimmed shrink-0"
148
+ />
149
+ <span class="text-xs truncate max-w-24">{{ att.name }}</span>
150
+ </template>
151
+
152
+ <button
153
+ class="absolute top-0 right-0 p-0.5 bg-error/80 rounded-bl-lg opacity-0 group-hover:opacity-100 transition-opacity"
154
+ @click="removeAttachment(att.id)"
155
+ >
156
+ <UIcon
157
+ name="i-lucide-x"
158
+ class="size-3 text-white"
159
+ />
160
+ </button>
161
+ </div>
162
+ </div>
163
+
55
164
  <div class="flex items-start gap-2">
56
165
  <textarea
57
166
  ref="textareaRef"
@@ -62,6 +171,25 @@ function autoResize(e: Event) {
62
171
  class="flex-1 resize-none bg-elevated/50 border border-default rounded-lg px-3 py-2 text-sm focus:outline-none focus:ring-2 focus:ring-primary/50 disabled:opacity-50"
63
172
  @keydown="handleKeydown"
64
173
  @input="autoResize"
174
+ @paste="handlePaste"
175
+ />
176
+
177
+ <input
178
+ ref="fileInputRef"
179
+ type="file"
180
+ :accept="FILE_ACCEPT"
181
+ multiple
182
+ class="hidden"
183
+ @change="handleFileSelect"
184
+ >
185
+
186
+ <UButton
187
+ icon="i-lucide-paperclip"
188
+ variant="ghost"
189
+ color="neutral"
190
+ size="md"
191
+ :disabled="!isConnected || isStreaming"
192
+ @click="fileInputRef?.click()"
65
193
  />
66
194
 
67
195
  <UButton
@@ -1,5 +1,5 @@
1
1
  <script setup lang="ts">
2
- import type { ChatMessage, ChatContentBlock, MessageSource } from '~~/shared/types'
2
+ import type { ChatMessage, ChatContentBlock, ChatImageBlock, ChatDocumentBlock, MessageSource, CodeLanguage } from '~~/shared/types'
3
3
 
4
4
  defineProps<{
5
5
  message: ChatMessage
@@ -12,6 +12,14 @@ function getTextContent(blocks: ChatContentBlock[]): string {
12
12
  .join('\n')
13
13
  }
14
14
 
15
+ function getImageBlocks(blocks: ChatContentBlock[]): ChatImageBlock[] {
16
+ return blocks.filter((b): b is ChatImageBlock => b.type === 'image')
17
+ }
18
+
19
+ function getDocumentBlocks(blocks: ChatContentBlock[]): ChatDocumentBlock[] {
20
+ return blocks.filter((b): b is ChatDocumentBlock => b.type === 'document')
21
+ }
22
+
15
23
  function getToolPairs(blocks: ChatContentBlock[]) {
16
24
  const tools: { name: string, id: string, result?: string, isError?: boolean }[] = []
17
25
  for (const block of blocks) {
@@ -28,6 +36,99 @@ function getToolPairs(blocks: ChatContentBlock[]) {
28
36
  return tools
29
37
  }
30
38
 
39
+ function getDocIcon(doc: ChatDocumentBlock): string {
40
+ if (doc.source.media_type === 'application/pdf') return 'i-lucide-file-text'
41
+ return 'i-lucide-file-code'
42
+ }
43
+
44
+ // Image preview
45
+ const previewImage = ref<ChatImageBlock | null>(null)
46
+ const previewOpen = computed({
47
+ get: () => previewImage.value !== null,
48
+ set: (v) => { if (!v) previewImage.value = null }
49
+ })
50
+
51
+ // Code/document preview
52
+ const previewDoc = ref<ChatDocumentBlock | null>(null)
53
+ const codePreviewOpen = computed({
54
+ get: () => previewDoc.value !== null,
55
+ set: (v) => { if (!v) previewDoc.value = null }
56
+ })
57
+
58
+ const codePreviewContent = computed(() => previewDoc.value?.source.data || '')
59
+
60
+ const extToLang: Record<string, CodeLanguage> = {
61
+ md: 'markdown',
62
+ js: 'javascript',
63
+ mjs: 'javascript',
64
+ cjs: 'javascript',
65
+ jsx: 'javascript',
66
+ ts: 'typescript',
67
+ tsx: 'typescript',
68
+ mts: 'typescript',
69
+ json: 'json',
70
+ html: 'html',
71
+ htm: 'html',
72
+ css: 'css',
73
+ scss: 'css',
74
+ sass: 'css',
75
+ less: 'css',
76
+ vue: 'vue',
77
+ svelte: 'html',
78
+ py: 'python',
79
+ sql: 'sql',
80
+ yaml: 'yaml',
81
+ yml: 'yaml',
82
+ sh: 'bash',
83
+ bash: 'bash',
84
+ zsh: 'bash',
85
+ fish: 'bash',
86
+ go: 'go',
87
+ rs: 'rust',
88
+ dockerfile: 'dockerfile',
89
+ java: 'java',
90
+ kt: 'java',
91
+ c: 'cpp',
92
+ cpp: 'cpp',
93
+ h: 'cpp',
94
+ hpp: 'cpp',
95
+ cs: 'cpp',
96
+ xml: 'xml',
97
+ graphql: 'plaintext',
98
+ gql: 'plaintext',
99
+ toml: 'plaintext',
100
+ ini: 'plaintext',
101
+ csv: 'plaintext',
102
+ txt: 'plaintext',
103
+ log: 'plaintext',
104
+ env: 'plaintext',
105
+ prisma: 'plaintext',
106
+ proto: 'plaintext'
107
+ }
108
+
109
+ const codePreviewLang = computed<CodeLanguage>(() => {
110
+ if (!previewDoc.value?.title) return 'plaintext'
111
+ const ext = previewDoc.value.title.split('.').pop()?.toLowerCase() || ''
112
+ return extToLang[ext] || 'plaintext'
113
+ })
114
+
115
+ function handleDocClick(doc: ChatDocumentBlock) {
116
+ if (doc.source.media_type === 'application/pdf') {
117
+ // Open PDF in new tab
118
+ const byteChars = atob(doc.source.data)
119
+ const bytes = new Uint8Array(byteChars.length)
120
+ for (let i = 0; i < byteChars.length; i++)
121
+ bytes[i] = byteChars.charCodeAt(i)
122
+ const blob = new Blob([bytes], { type: 'application/pdf' })
123
+ const url = URL.createObjectURL(blob)
124
+ window.open(url, '_blank')
125
+ setTimeout(() => URL.revokeObjectURL(url), 10000)
126
+ } else {
127
+ // Open text file in code viewer modal
128
+ previewDoc.value = doc
129
+ }
130
+ }
131
+
31
132
  const sourceIconMap: Record<string, string> = {
32
133
  telegram: 'i-simple-icons-telegram',
33
134
  discord: 'i-simple-icons-discord',
@@ -76,11 +177,52 @@ function formatTime(date: Date | string | undefined): string {
76
177
  : 'bg-muted'"
77
178
  >
78
179
  <!-- User message -->
79
- <div
80
- v-if="message.role === 'user'"
81
- class="text-sm whitespace-pre-wrap"
82
- >
83
- {{ getTextContent(message.content) }}
180
+ <div v-if="message.role === 'user'">
181
+ <!-- Image attachments -->
182
+ <div
183
+ v-if="getImageBlocks(message.content).length"
184
+ class="flex flex-wrap gap-2 mb-2"
185
+ >
186
+ <img
187
+ v-for="(img, i) in getImageBlocks(message.content)"
188
+ :key="i"
189
+ :src="`data:${img.source.media_type};base64,${img.source.data}`"
190
+ class="max-w-48 max-h-48 rounded-lg object-contain cursor-pointer hover:opacity-80 transition-opacity"
191
+ @click="previewImage = img"
192
+ >
193
+ </div>
194
+
195
+ <!-- Document attachments -->
196
+ <div
197
+ v-if="getDocumentBlocks(message.content).length"
198
+ class="flex flex-wrap gap-2 mb-2"
199
+ >
200
+ <button
201
+ v-for="(doc, i) in getDocumentBlocks(message.content)"
202
+ :key="i"
203
+ class="flex items-center gap-1.5 px-2.5 py-1.5 rounded-lg bg-elevated/50 border border-default text-xs hover:bg-elevated transition-colors cursor-pointer"
204
+ @click="handleDocClick(doc)"
205
+ >
206
+ <UIcon
207
+ :name="getDocIcon(doc)"
208
+ class="size-3.5 text-dimmed shrink-0"
209
+ />
210
+ <span class="truncate max-w-32">{{ doc.title || 'Document' }}</span>
211
+ <UIcon
212
+ v-if="doc.source.media_type === 'application/pdf'"
213
+ name="i-lucide-external-link"
214
+ class="size-3 text-dimmed shrink-0"
215
+ />
216
+ </button>
217
+ </div>
218
+
219
+ <!-- Text content -->
220
+ <div
221
+ v-if="getTextContent(message.content)"
222
+ class="text-sm whitespace-pre-wrap"
223
+ >
224
+ {{ getTextContent(message.content) }}
225
+ </div>
84
226
  </div>
85
227
 
86
228
  <!-- Assistant message -->
@@ -121,5 +263,45 @@ function formatTime(date: Date | string | undefined): string {
121
263
  {{ formatTime(message.createdAt) }}
122
264
  </div>
123
265
  </div>
266
+
267
+ <!-- Image preview modal -->
268
+ <UModal
269
+ v-model:open="previewOpen"
270
+ >
271
+ <template #content>
272
+ <div class="flex items-center justify-center">
273
+ <img
274
+ v-if="previewImage"
275
+ :src="`data:${previewImage.source.media_type};base64,${previewImage.source.data}`"
276
+ class="max-w-full max-h-[80vh] object-contain rounded-lg"
277
+ >
278
+ </div>
279
+ </template>
280
+ </UModal>
281
+
282
+ <!-- Code preview modal -->
283
+ <UModal
284
+ v-model:open="codePreviewOpen"
285
+ >
286
+ <template #header>
287
+ <div class="flex items-center gap-2">
288
+ <UIcon
289
+ name="i-lucide-file-code"
290
+ class="size-4 text-dimmed"
291
+ />
292
+ <span class="text-sm font-medium truncate">{{ previewDoc?.title || 'File' }}</span>
293
+ </div>
294
+ </template>
295
+ <template #body>
296
+ <div class="h-[60vh] overflow-hidden rounded-lg border border-default">
297
+ <EditorCodeEditor
298
+ v-if="previewDoc"
299
+ :model-value="codePreviewContent"
300
+ :language="codePreviewLang"
301
+ read-only
302
+ />
303
+ </div>
304
+ </template>
305
+ </UModal>
124
306
  </div>
125
307
  </template>