@ttt-productions/chat-core 0.2.9 → 0.3.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/firestore/queries.d.ts +3 -3
- package/dist/firestore/queries.d.ts.map +1 -1
- package/dist/firestore/queries.js +11 -6
- package/dist/firestore/queries.js.map +1 -1
- package/dist/hooks/useChatMessages.d.ts.map +1 -1
- package/dist/hooks/useChatMessages.js +15 -19
- package/dist/hooks/useChatMessages.js.map +1 -1
- package/dist/index.d.ts +0 -3
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +2 -3
- package/dist/index.js.map +1 -1
- package/dist/types.d.ts +34 -8
- package/dist/types.d.ts.map +1 -1
- package/dist/types.js +5 -1
- package/dist/types.js.map +1 -1
- package/dist/ui/ChatShell.d.ts +14 -6
- package/dist/ui/ChatShell.d.ts.map +1 -1
- package/dist/ui/ChatShell.js +3 -3
- package/dist/ui/ChatShell.js.map +1 -1
- package/dist/ui/Composer.d.ts +11 -3
- package/dist/ui/Composer.d.ts.map +1 -1
- package/dist/ui/Composer.js +104 -10
- package/dist/ui/Composer.js.map +1 -1
- package/dist/ui/MessageItemDefault.d.ts +5 -2
- package/dist/ui/MessageItemDefault.d.ts.map +1 -1
- package/dist/ui/MessageItemDefault.js +65 -2
- package/dist/ui/MessageItemDefault.js.map +1 -1
- package/dist/ui/MessageList.d.ts +1 -0
- package/dist/ui/MessageList.d.ts.map +1 -1
- package/dist/ui/MessageList.js +41 -5
- package/dist/ui/MessageList.js.map +1 -1
- package/package.json +13 -5
- package/src/styles/chat.css +137 -0
- package/dist/templates/AdminThreadChat.d.ts +0 -9
- package/dist/templates/AdminThreadChat.d.ts.map +0 -1
- package/dist/templates/AdminThreadChat.js +0 -8
- package/dist/templates/AdminThreadChat.js.map +0 -1
- package/dist/templates/ChannelChat.d.ts +0 -11
- package/dist/templates/ChannelChat.d.ts.map +0 -1
- package/dist/templates/ChannelChat.js +0 -8
- package/dist/templates/ChannelChat.js.map +0 -1
- package/dist/templates/InviteChat.d.ts +0 -9
- package/dist/templates/InviteChat.d.ts.map +0 -1
- package/dist/templates/InviteChat.js +0 -8
- package/dist/templates/InviteChat.js.map +0 -1
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import type { Firestore, Query, DocumentData } from "firebase/firestore";
|
|
2
|
-
export declare function threadDocPath(
|
|
3
|
-
export declare function messagesColPath(
|
|
4
|
-
export declare function newestWindowQuery(db: Firestore,
|
|
2
|
+
export declare function threadDocPath(chatCollectionPath: string | string[], threadId: string): string[];
|
|
3
|
+
export declare function messagesColPath(chatCollectionPath: string | string[], threadId: string, messagesSubcollection?: string): [string, string, ...string[]];
|
|
4
|
+
export declare function newestWindowQuery(db: Firestore, chatCollectionPath: string | string[], threadId: string, createdAtField: string, pageSize: number, messagesSubcollection?: string): Query<DocumentData>;
|
|
5
5
|
//# sourceMappingURL=queries.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"queries.d.ts","sourceRoot":"","sources":["../../src/firestore/queries.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,SAAS,EAAE,KAAK,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC;
|
|
1
|
+
{"version":3,"file":"queries.d.ts","sourceRoot":"","sources":["../../src/firestore/queries.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,SAAS,EAAE,KAAK,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC;AAOzE,wBAAgB,aAAa,CAAC,kBAAkB,EAAE,MAAM,GAAG,MAAM,EAAE,EAAE,QAAQ,EAAE,MAAM,GAAG,MAAM,EAAE,CAE/F;AAED,wBAAgB,eAAe,CAC7B,kBAAkB,EAAE,MAAM,GAAG,MAAM,EAAE,EACrC,QAAQ,EAAE,MAAM,EAChB,qBAAqB,SAAa,GACjC,CAAC,MAAM,EAAE,MAAM,EAAE,GAAG,MAAM,EAAE,CAAC,CAG/B;AAED,wBAAgB,iBAAiB,CAC/B,EAAE,EAAE,SAAS,EACb,kBAAkB,EAAE,MAAM,GAAG,MAAM,EAAE,EACrC,QAAQ,EAAE,MAAM,EAChB,cAAc,EAAE,MAAM,EACtB,QAAQ,EAAE,MAAM,EAChB,qBAAqB,SAAa,GACjC,KAAK,CAAC,YAAY,CAAC,CAOrB"}
|
|
@@ -1,11 +1,16 @@
|
|
|
1
1
|
import { collection, query, orderBy, limit } from "firebase/firestore";
|
|
2
|
-
|
|
3
|
-
return [
|
|
2
|
+
function normalizeSegments(path) {
|
|
3
|
+
return Array.isArray(path) ? path : [path];
|
|
4
4
|
}
|
|
5
|
-
export function
|
|
6
|
-
return [
|
|
5
|
+
export function threadDocPath(chatCollectionPath, threadId) {
|
|
6
|
+
return [...normalizeSegments(chatCollectionPath), threadId];
|
|
7
7
|
}
|
|
8
|
-
export function
|
|
9
|
-
|
|
8
|
+
export function messagesColPath(chatCollectionPath, threadId, messagesSubcollection = "messages") {
|
|
9
|
+
const segments = [...normalizeSegments(chatCollectionPath), threadId, messagesSubcollection];
|
|
10
|
+
return segments;
|
|
11
|
+
}
|
|
12
|
+
export function newestWindowQuery(db, chatCollectionPath, threadId, createdAtField, pageSize, messagesSubcollection = "messages") {
|
|
13
|
+
const path = messagesColPath(chatCollectionPath, threadId, messagesSubcollection);
|
|
14
|
+
return query(collection(db, path[0], ...path.slice(1)), orderBy(createdAtField, "desc"), limit(pageSize));
|
|
10
15
|
}
|
|
11
16
|
//# sourceMappingURL=queries.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"queries.js","sourceRoot":"","sources":["../../src/firestore/queries.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,UAAU,EAAE,KAAK,EAAE,OAAO,EAAE,KAAK,EAAE,MAAM,oBAAoB,CAAC;AAEvE,MAAM,UAAU,aAAa,CAAC,
|
|
1
|
+
{"version":3,"file":"queries.js","sourceRoot":"","sources":["../../src/firestore/queries.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,UAAU,EAAE,KAAK,EAAE,OAAO,EAAE,KAAK,EAAE,MAAM,oBAAoB,CAAC;AAEvE,SAAS,iBAAiB,CAAC,IAAuB;IAChD,OAAO,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;AAC7C,CAAC;AAED,MAAM,UAAU,aAAa,CAAC,kBAAqC,EAAE,QAAgB;IACnF,OAAO,CAAC,GAAG,iBAAiB,CAAC,kBAAkB,CAAC,EAAE,QAAQ,CAAC,CAAC;AAC9D,CAAC;AAED,MAAM,UAAU,eAAe,CAC7B,kBAAqC,EACrC,QAAgB,EAChB,qBAAqB,GAAG,UAAU;IAElC,MAAM,QAAQ,GAAG,CAAC,GAAG,iBAAiB,CAAC,kBAAkB,CAAC,EAAE,QAAQ,EAAE,qBAAqB,CAAC,CAAC;IAC7F,OAAO,QAAyC,CAAC;AACnD,CAAC;AAED,MAAM,UAAU,iBAAiB,CAC/B,EAAa,EACb,kBAAqC,EACrC,QAAgB,EAChB,cAAsB,EACtB,QAAgB,EAChB,qBAAqB,GAAG,UAAU;IAElC,MAAM,IAAI,GAAG,eAAe,CAAC,kBAAkB,EAAE,QAAQ,EAAE,qBAAqB,CAAC,CAAC;IAClF,OAAO,KAAK,CACV,UAAU,CAAC,EAAE,EAAE,IAAI,CAAC,CAAC,CAAC,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,EACzC,OAAO,CAAC,cAAc,EAAE,MAAM,CAAC,EAC/B,KAAK,CAAC,QAAQ,CAAC,CAChB,CAAC;AACJ,CAAC"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"useChatMessages.d.ts","sourceRoot":"","sources":["../../src/hooks/useChatMessages.ts"],"names":[],"mappings":"AAmBA,OAAO,KAAK,EAAE,cAAc,EAAE,aAAa,EAAE,MAAM,UAAU,CAAC;AAa9D,MAAM,MAAM,qBAAqB,GAAG;IAClC,OAAO,EAAE,OAAO,CAAC;IACjB,gBAAgB,EAAE,OAAO,CAAC;IAC1B,QAAQ,EAAE,aAAa,EAAE,CAAC;IAC1B,UAAU,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;IAChC,QAAQ,EAAE,OAAO,CAAC;IAClB,eAAe,EAAE,OAAO,CAAC;CAC1B,CAAC;
|
|
1
|
+
{"version":3,"file":"useChatMessages.d.ts","sourceRoot":"","sources":["../../src/hooks/useChatMessages.ts"],"names":[],"mappings":"AAmBA,OAAO,KAAK,EAAE,cAAc,EAAE,aAAa,EAAE,MAAM,UAAU,CAAC;AAa9D,MAAM,MAAM,qBAAqB,GAAG;IAClC,OAAO,EAAE,OAAO,CAAC;IACjB,gBAAgB,EAAE,OAAO,CAAC;IAC1B,QAAQ,EAAE,aAAa,EAAE,CAAC;IAC1B,UAAU,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;IAChC,QAAQ,EAAE,OAAO,CAAC;IAClB,eAAe,EAAE,OAAO,CAAC;CAC1B,CAAC;AA6CF,wBAAgB,eAAe,CAAC,MAAM,EAAE,cAAc,GAAG,qBAAqB,CAwH7E"}
|
|
@@ -9,9 +9,6 @@ import { messagesColPath, newestWindowQuery } from "../firestore/queries";
|
|
|
9
9
|
const MAX_PAGE_SIZE = 100;
|
|
10
10
|
function mapMsg(doc, threadId) {
|
|
11
11
|
const d = doc.data();
|
|
12
|
-
// CRITICAL FIX: Do not fallback to Date.now().
|
|
13
|
-
// If data is missing/corrupt, use 0 so it sorts to the beginning (oldest)
|
|
14
|
-
// rather than appearing as a brand new message.
|
|
15
12
|
const createdAt = toMillis(d.createdAt);
|
|
16
13
|
if (!createdAt) {
|
|
17
14
|
console.error('[useChatMessages] Message missing valid createdAt:', doc.id, d.createdAt);
|
|
@@ -22,8 +19,11 @@ function mapMsg(doc, threadId) {
|
|
|
22
19
|
createdAt: createdAt || 0,
|
|
23
20
|
senderId: d.senderId,
|
|
24
21
|
senderUsername: d.senderUsername,
|
|
25
|
-
text: d.
|
|
22
|
+
text: d.text ?? "",
|
|
26
23
|
type: d.type,
|
|
24
|
+
attachment: d.attachment ?? undefined,
|
|
25
|
+
replyTo: d.replyTo ?? undefined,
|
|
26
|
+
isSystemMessage: d.isSystemMessage ?? undefined,
|
|
27
27
|
meta: d.meta,
|
|
28
28
|
};
|
|
29
29
|
}
|
|
@@ -33,16 +33,17 @@ function dedupeAndSortAsc(all) {
|
|
|
33
33
|
m.set(x.messageId, x);
|
|
34
34
|
return Array.from(m.values()).sort((a, b) => a.createdAt - b.createdAt);
|
|
35
35
|
}
|
|
36
|
-
function messagesCol(db,
|
|
37
|
-
|
|
36
|
+
function messagesCol(db, chatCollectionPath, threadId, messagesSubcollection) {
|
|
37
|
+
const path = messagesColPath(chatCollectionPath, threadId, messagesSubcollection);
|
|
38
|
+
return collection(db, path[0], ...path.slice(1));
|
|
38
39
|
}
|
|
39
40
|
export function useChatMessages(config) {
|
|
40
|
-
const { db,
|
|
41
|
-
// SAFETY FIX: Clamp page size
|
|
41
|
+
const { db, chatCollectionPath, messagesSubcollection = "messages", threadId, pageSize: requestedPageSize = 20, currentUserId, isAdmin, threadAllowedUserIds, createdAtField = "createdAt", } = config;
|
|
42
42
|
const pageSize = Math.min(requestedPageSize, MAX_PAGE_SIZE);
|
|
43
43
|
if (requestedPageSize > MAX_PAGE_SIZE) {
|
|
44
44
|
console.warn(`[useChatMessages] Requested pageSize ${requestedPageSize} exceeds maximum ${MAX_PAGE_SIZE}. Using ${MAX_PAGE_SIZE}.`);
|
|
45
45
|
}
|
|
46
|
+
const pathKey = JSON.stringify(Array.isArray(chatCollectionPath) ? chatCollectionPath : [chatCollectionPath]);
|
|
46
47
|
const allowed = React.useMemo(() => canAccessThread({
|
|
47
48
|
isAdmin,
|
|
48
49
|
currentUserId,
|
|
@@ -53,14 +54,13 @@ export function useChatMessages(config) {
|
|
|
53
54
|
newestDesc: [],
|
|
54
55
|
oldestDocInWindow: null,
|
|
55
56
|
});
|
|
56
|
-
// realtime newest window only
|
|
57
57
|
React.useEffect(() => {
|
|
58
58
|
if (!allowed)
|
|
59
59
|
return;
|
|
60
|
-
const q = newestWindowQuery(db,
|
|
60
|
+
const q = newestWindowQuery(db, chatCollectionPath, threadId, createdAtField, pageSize, messagesSubcollection);
|
|
61
61
|
const unsub = onSnapshot(q, (snap) => {
|
|
62
62
|
const docs = snap.docs;
|
|
63
|
-
const mapped = docs.map((d) => mapMsg(d, threadId));
|
|
63
|
+
const mapped = docs.map((d) => mapMsg(d, threadId));
|
|
64
64
|
setNewest({
|
|
65
65
|
ready: true,
|
|
66
66
|
newestDesc: mapped,
|
|
@@ -68,14 +68,13 @@ export function useChatMessages(config) {
|
|
|
68
68
|
});
|
|
69
69
|
});
|
|
70
70
|
return () => unsub();
|
|
71
|
-
}, [allowed, db,
|
|
72
|
-
// older pagination (beyond newest window)
|
|
71
|
+
}, [allowed, db, pathKey, threadId, createdAtField, pageSize, messagesSubcollection]);
|
|
73
72
|
const older = useInfiniteQuery({
|
|
74
|
-
queryKey: ["chat-core", "older",
|
|
73
|
+
queryKey: ["chat-core", "older", pathKey, threadId, messagesSubcollection, pageSize],
|
|
75
74
|
enabled: allowed && newest.ready,
|
|
76
75
|
initialPageParam: undefined,
|
|
77
76
|
queryFn: async ({ pageParam }) => {
|
|
78
|
-
const baseCol = messagesCol(db,
|
|
77
|
+
const baseCol = messagesCol(db, chatCollectionPath, threadId, messagesSubcollection);
|
|
79
78
|
const cursor = pageParam ?? (newest.oldestDocInWindow ?? undefined);
|
|
80
79
|
if (!cursor) {
|
|
81
80
|
return {
|
|
@@ -86,7 +85,7 @@ export function useChatMessages(config) {
|
|
|
86
85
|
const q = query(baseCol, orderBy(createdAtField, "desc"), startAfter(cursor), limit(pageSize));
|
|
87
86
|
const snap = await getDocs(q);
|
|
88
87
|
const docs = snap.docs;
|
|
89
|
-
const itemsDesc = docs.map((d) => mapMsg(d, threadId));
|
|
88
|
+
const itemsDesc = docs.map((d) => mapMsg(d, threadId));
|
|
90
89
|
const nextCursor = docs.length === pageSize ? docs[docs.length - 1] : undefined;
|
|
91
90
|
return { itemsDesc, nextCursor };
|
|
92
91
|
},
|
|
@@ -99,9 +98,6 @@ export function useChatMessages(config) {
|
|
|
99
98
|
if (!allowed)
|
|
100
99
|
return [];
|
|
101
100
|
const olderDesc = (older.data?.pages ?? []).flatMap((p) => p.itemsDesc);
|
|
102
|
-
// Filter out any messages with 0 timestamp (invalid) if desired,
|
|
103
|
-
// or keep them (they will sort to top).
|
|
104
|
-
// For now we keep them to avoid data loss holes, but they won't look "new".
|
|
105
101
|
return dedupeAndSortAsc([...olderDesc, ...newest.newestDesc]);
|
|
106
102
|
}, [allowed, older.data, newest.newestDesc]);
|
|
107
103
|
return {
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"useChatMessages.js","sourceRoot":"","sources":["../../src/hooks/useChatMessages.ts"],"names":[],"mappings":"AAAA,YAAY,CAAC;AAEb,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAM/B,OAAO,EACL,UAAU,EACV,OAAO,EACP,UAAU,EACV,OAAO,EACP,KAAK,EACL,UAAU,EACV,KAAK,GACN,MAAM,oBAAoB,CAAC;AAC5B,OAAO,EAAE,gBAAgB,EAAE,MAAM,uBAAuB,CAAC;AACzD,OAAO,EAAE,QAAQ,EAAE,MAAM,mCAAmC,CAAC;AAE7D,OAAO,EAAE,eAAe,EAAE,MAAM,uBAAuB,CAAC;AACxD,OAAO,EAAE,eAAe,EAAE,iBAAiB,EAAE,MAAM,sBAAsB,CAAC;AAE1E,4CAA4C;AAC5C,MAAM,aAAa,GAAG,GAAG,CAAC;AAiB1B,SAAS,MAAM,CACb,GAAwC,EACxC,QAAgB;IAEhB,MAAM,CAAC,GAAG,GAAG,CAAC,IAAI,EAAS,CAAC;IAE5B
|
|
1
|
+
{"version":3,"file":"useChatMessages.js","sourceRoot":"","sources":["../../src/hooks/useChatMessages.ts"],"names":[],"mappings":"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"}
|
package/dist/index.d.ts
CHANGED
|
@@ -6,7 +6,4 @@ export * from "./ui/MessageList";
|
|
|
6
6
|
export * from "./ui/Composer";
|
|
7
7
|
export * from "./ui/MessageItemDefault";
|
|
8
8
|
export * from "./ui/menus";
|
|
9
|
-
export * from "./templates/ChannelChat";
|
|
10
|
-
export * from "./templates/AdminThreadChat";
|
|
11
|
-
export * from "./templates/InviteChat";
|
|
12
9
|
//# sourceMappingURL=index.d.ts.map
|
package/dist/index.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,cAAc,SAAS,CAAC;AAExB,cAAc,yBAAyB,CAAC;AACxC,cAAc,6BAA6B,CAAC;AAE5C,cAAc,gBAAgB,CAAC;AAC/B,cAAc,kBAAkB,CAAC;AACjC,cAAc,eAAe,CAAC;AAC9B,cAAc,yBAAyB,CAAC;AACxC,cAAc,YAAY,CAAC
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,cAAc,SAAS,CAAC;AAExB,cAAc,yBAAyB,CAAC;AACxC,cAAc,6BAA6B,CAAC;AAE5C,cAAc,gBAAgB,CAAC;AAC/B,cAAc,kBAAkB,CAAC;AACjC,cAAc,eAAe,CAAC;AAC9B,cAAc,yBAAyB,CAAC;AACxC,cAAc,YAAY,CAAC"}
|
package/dist/index.js
CHANGED
|
@@ -6,7 +6,6 @@ export * from "./ui/MessageList";
|
|
|
6
6
|
export * from "./ui/Composer";
|
|
7
7
|
export * from "./ui/MessageItemDefault";
|
|
8
8
|
export * from "./ui/menus";
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
export * from "./templates/InviteChat";
|
|
9
|
+
// Templates removed in v0.3.0 — use ChatShell directly with render slots.
|
|
10
|
+
// CSS: import "@ttt-productions/chat-core/styles" in your app layout.
|
|
12
11
|
//# sourceMappingURL=index.js.map
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,cAAc,SAAS,CAAC;AAExB,cAAc,yBAAyB,CAAC;AACxC,cAAc,6BAA6B,CAAC;AAE5C,cAAc,gBAAgB,CAAC;AAC/B,cAAc,kBAAkB,CAAC;AACjC,cAAc,eAAe,CAAC;AAC9B,cAAc,yBAAyB,CAAC;AACxC,cAAc,YAAY,CAAC;AAE3B,
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,cAAc,SAAS,CAAC;AAExB,cAAc,yBAAyB,CAAC;AACxC,cAAc,6BAA6B,CAAC;AAE5C,cAAc,gBAAgB,CAAC;AAC/B,cAAc,kBAAkB,CAAC;AACjC,cAAc,eAAe,CAAC;AAC9B,cAAc,yBAAyB,CAAC;AACxC,cAAc,YAAY,CAAC;AAE3B,0EAA0E;AAC1E,sEAAsE"}
|
package/dist/types.d.ts
CHANGED
|
@@ -1,4 +1,18 @@
|
|
|
1
1
|
import type { Firestore } from "firebase/firestore";
|
|
2
|
+
import type { FirebaseStorage } from "firebase/storage";
|
|
3
|
+
import type { MediaProcessingSpec } from "@ttt-productions/media-contracts";
|
|
4
|
+
export type ChatAttachmentStatus = "pending" | "processing" | "completed" | "failed" | "rejected";
|
|
5
|
+
export type ChatAttachment = {
|
|
6
|
+
id: string;
|
|
7
|
+
name: string;
|
|
8
|
+
type: "image" | "video" | "audio" | "text";
|
|
9
|
+
size: number;
|
|
10
|
+
status: ChatAttachmentStatus;
|
|
11
|
+
pendingStoragePath?: string;
|
|
12
|
+
url?: string;
|
|
13
|
+
storagePath?: string;
|
|
14
|
+
errorMessage?: string;
|
|
15
|
+
};
|
|
2
16
|
export type ChatId = string;
|
|
3
17
|
export type ChatThreadV1 = {
|
|
4
18
|
allowedUserIds: string[];
|
|
@@ -16,22 +30,32 @@ export type ChatMessageV1 = {
|
|
|
16
30
|
senderUsername?: string;
|
|
17
31
|
text?: string;
|
|
18
32
|
type?: string;
|
|
33
|
+
attachment?: ChatAttachment;
|
|
34
|
+
replyTo?: {
|
|
35
|
+
messageId: string;
|
|
36
|
+
senderUsername: string;
|
|
37
|
+
messagePreview: string;
|
|
38
|
+
};
|
|
39
|
+
isSystemMessage?: boolean;
|
|
19
40
|
meta?: Record<string, unknown>;
|
|
20
41
|
};
|
|
21
|
-
export type
|
|
22
|
-
/** top-level collection name, e.g. "projectChats" | "adminThreads" | "inviteChats" */
|
|
23
|
-
chatCollection: string;
|
|
24
|
-
};
|
|
25
|
-
export type ChatCoreConfig = ChatCollectionConfig & {
|
|
42
|
+
export type ChatCoreConfig = {
|
|
26
43
|
db: Firestore;
|
|
27
|
-
|
|
44
|
+
chatCollectionPath: string | string[];
|
|
45
|
+
messagesSubcollection?: string;
|
|
28
46
|
threadId: string;
|
|
29
47
|
currentUserId: string;
|
|
48
|
+
currentUserDisplayName?: string;
|
|
30
49
|
isAdmin: boolean;
|
|
31
|
-
/** If provided, used to gate access; otherwise only isAdmin applies */
|
|
32
50
|
threadAllowedUserIds?: string[];
|
|
33
|
-
/** Optional: tell chat-core which field to order by; must be indexed */
|
|
34
51
|
createdAtField?: string;
|
|
52
|
+
pageSize?: number;
|
|
53
|
+
};
|
|
54
|
+
export type ChatAttachmentConfig = {
|
|
55
|
+
attachmentSpec: MediaProcessingSpec;
|
|
56
|
+
storage: FirebaseStorage;
|
|
57
|
+
pendingStoragePath: string;
|
|
58
|
+
pendingMediaCollection?: string;
|
|
35
59
|
};
|
|
36
60
|
export type ModerationHandlers = {
|
|
37
61
|
onReportMessage?: (messageId: string, reason?: string) => void | Promise<void>;
|
|
@@ -41,4 +65,6 @@ export type ModerationHandlers = {
|
|
|
41
65
|
};
|
|
42
66
|
export type MessageRenderer = (m: ChatMessageV1) => React.ReactNode;
|
|
43
67
|
export type MessageRendererRegistry = Record<string, MessageRenderer>;
|
|
68
|
+
/** Max seconds between messages to be grouped as continuation */
|
|
69
|
+
export declare const GROUP_GAP_SEC = 120;
|
|
44
70
|
//# sourceMappingURL=types.d.ts.map
|
package/dist/types.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../src/types.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,oBAAoB,CAAC;
|
|
1
|
+
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../src/types.ts"],"names":[],"mappings":"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"}
|
package/dist/types.js
CHANGED
|
@@ -1,2 +1,6 @@
|
|
|
1
|
-
|
|
1
|
+
// ============================================
|
|
2
|
+
// MESSAGE GROUPING (internal, exported for tests)
|
|
3
|
+
// ============================================
|
|
4
|
+
/** Max seconds between messages to be grouped as continuation */
|
|
5
|
+
export const GROUP_GAP_SEC = 120;
|
|
2
6
|
//# sourceMappingURL=types.js.map
|
package/dist/types.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"types.js","sourceRoot":"","sources":["../src/types.ts"],"names":[],"mappings":""}
|
|
1
|
+
{"version":3,"file":"types.js","sourceRoot":"","sources":["../src/types.ts"],"names":[],"mappings":"AAsGA,+CAA+C;AAC/C,kDAAkD;AAClD,+CAA+C;AAE/C,iEAAiE;AACjE,MAAM,CAAC,MAAM,aAAa,GAAG,GAAG,CAAC"}
|
package/dist/ui/ChatShell.d.ts
CHANGED
|
@@ -1,13 +1,21 @@
|
|
|
1
1
|
import * as React from "react";
|
|
2
|
-
import type { ChatCoreConfig, MessageRendererRegistry, ModerationHandlers } from "../types";
|
|
3
|
-
export
|
|
2
|
+
import type { ChatCoreConfig, ChatAttachment, ChatAttachmentConfig, MessageRendererRegistry, ModerationHandlers } from "../types";
|
|
3
|
+
export type ChatShellProps = {
|
|
4
4
|
config: ChatCoreConfig;
|
|
5
5
|
header?: React.ReactNode;
|
|
6
|
-
|
|
7
|
-
onSend: (text: string) => void | Promise<void>;
|
|
6
|
+
onSend: (text: string, attachment?: ChatAttachment) => void | Promise<void>;
|
|
8
7
|
renderMessage?: (m: any) => React.ReactNode;
|
|
9
8
|
messageRenderers?: MessageRendererRegistry;
|
|
10
|
-
composerAutoFocus?: boolean;
|
|
11
9
|
handlers?: ModerationHandlers;
|
|
12
|
-
|
|
10
|
+
attachmentConfig?: ChatAttachmentConfig;
|
|
11
|
+
composerPlaceholder?: string;
|
|
12
|
+
autoFocus?: boolean;
|
|
13
|
+
renderAboveMessages?: () => React.ReactNode;
|
|
14
|
+
renderBelowMessages?: () => React.ReactNode;
|
|
15
|
+
/** If provided, replaces the Composer entirely */
|
|
16
|
+
renderFooter?: () => React.ReactNode;
|
|
17
|
+
onSenderClick?: (senderId: string, displayName: string) => void;
|
|
18
|
+
composerDisabled?: boolean;
|
|
19
|
+
};
|
|
20
|
+
export declare function ChatShell(props: ChatShellProps): import("react/jsx-runtime").JSX.Element;
|
|
13
21
|
//# sourceMappingURL=ChatShell.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"ChatShell.d.ts","sourceRoot":"","sources":["../../src/ui/ChatShell.tsx"],"names":[],"mappings":"AAEA,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAC/B,OAAO,KAAK,
|
|
1
|
+
{"version":3,"file":"ChatShell.d.ts","sourceRoot":"","sources":["../../src/ui/ChatShell.tsx"],"names":[],"mappings":"AAEA,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAC/B,OAAO,KAAK,EACV,cAAc,EACd,cAAc,EACd,oBAAoB,EACpB,uBAAuB,EACvB,kBAAkB,EACnB,MAAM,UAAU,CAAC;AAQlB,MAAM,MAAM,cAAc,GAAG;IAC3B,MAAM,EAAE,cAAc,CAAC;IAGvB,MAAM,CAAC,EAAE,KAAK,CAAC,SAAS,CAAC;IAGzB,MAAM,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,UAAU,CAAC,EAAE,cAAc,KAAK,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;IAG5E,aAAa,CAAC,EAAE,CAAC,CAAC,EAAE,GAAG,KAAK,KAAK,CAAC,SAAS,CAAC;IAC5C,gBAAgB,CAAC,EAAE,uBAAuB,CAAC;IAC3C,QAAQ,CAAC,EAAE,kBAAkB,CAAC;IAG9B,gBAAgB,CAAC,EAAE,oBAAoB,CAAC;IACxC,mBAAmB,CAAC,EAAE,MAAM,CAAC;IAC7B,SAAS,CAAC,EAAE,OAAO,CAAC;IAGpB,mBAAmB,CAAC,EAAE,MAAM,KAAK,CAAC,SAAS,CAAC;IAC5C,mBAAmB,CAAC,EAAE,MAAM,KAAK,CAAC,SAAS,CAAC;IAC5C,kDAAkD;IAClD,YAAY,CAAC,EAAE,MAAM,KAAK,CAAC,SAAS,CAAC;IAGrC,aAAa,CAAC,EAAE,CAAC,QAAQ,EAAE,MAAM,EAAE,WAAW,EAAE,MAAM,KAAK,IAAI,CAAC;IAGhE,gBAAgB,CAAC,EAAE,OAAO,CAAC;CAC5B,CAAC;AAEF,wBAAgB,SAAS,CAAC,KAAK,EAAE,cAAc,2CA0G9C"}
|
package/dist/ui/ChatShell.js
CHANGED
|
@@ -8,15 +8,15 @@ import { MessageList } from "./MessageList";
|
|
|
8
8
|
import { Composer } from "./Composer";
|
|
9
9
|
import { ThreadActions } from "./menus";
|
|
10
10
|
export function ChatShell(props) {
|
|
11
|
-
const { config, header,
|
|
11
|
+
const { config, header, onSend, renderMessage, messageRenderers, handlers, attachmentConfig, composerPlaceholder, autoFocus = false, renderAboveMessages, renderBelowMessages, renderFooter, onSenderClick, composerDisabled, } = props;
|
|
12
12
|
const { allowed, isInitialLoading, messages, fetchOlder, hasOlder, isFetchingOlder } = useChatMessages(config);
|
|
13
13
|
const [showScrollToBottom, setShowScrollToBottom] = React.useState(false);
|
|
14
14
|
if (!allowed) {
|
|
15
|
-
return _jsx("div", { className: "p-4 text-sm opacity-70", children: "You don
|
|
15
|
+
return _jsx("div", { className: "p-4 text-sm opacity-70", children: "You don't have access to this thread." });
|
|
16
16
|
}
|
|
17
17
|
if (isInitialLoading) {
|
|
18
18
|
return (_jsxs(Card, { className: "w-full", children: [_jsxs(CardHeader, { className: "space-y-2", children: [_jsx(Skeleton, { className: "h-5 w-40" }), _jsx(Skeleton, { className: "h-3 w-56" })] }), _jsx(CardContent, { children: _jsx(Skeleton, { className: "h-[240px] w-full" }) }), _jsx(CardFooter, { children: _jsx(Skeleton, { className: "h-10 w-full" }) })] }));
|
|
19
19
|
}
|
|
20
|
-
return (_jsxs(Card, { className: "w-full", children: [(header ||
|
|
20
|
+
return (_jsxs(Card, { className: "w-full", children: [(header || handlers) && (_jsxs(CardHeader, { className: "flex flex-row items-start justify-between gap-4 space-y-0", children: [_jsx("div", { children: header }), _jsx("div", { className: "flex flex-col items-end gap-2", children: _jsx(ThreadActions, { threadId: config.threadId, isAdmin: config.isAdmin, handlers: handlers }) })] })), renderAboveMessages && (_jsx("div", { className: "border-b", children: renderAboveMessages() })), _jsx(CardContent, { className: "p-0", children: _jsx(MessageList, { messages: messages, currentUserId: config.currentUserId, isAdmin: config.isAdmin, isFetchingOlder: isFetchingOlder, hasOlder: hasOlder, onLoadOlder: () => fetchOlder(), renderMessage: renderMessage, messageRenderers: messageRenderers, showScrollToBottom: showScrollToBottom, onScrollToBottom: () => setShowScrollToBottom(false), handlers: handlers, onSenderClick: onSenderClick }) }), renderBelowMessages && (_jsx("div", { className: "border-t", children: renderBelowMessages() })), renderFooter ? (_jsx(CardFooter, { className: "border-t", children: renderFooter() })) : (_jsx(CardFooter, { className: "border-t", children: _jsx(KeyboardAvoidingView, { padding: true, offset: 8, className: "w-full", children: _jsx(Composer, { onSend: onSend, attachmentConfig: attachmentConfig, db: config.db, currentUserId: config.currentUserId, disabled: composerDisabled, autoFocus: autoFocus, placeholder: composerPlaceholder }) }) }))] }));
|
|
21
21
|
}
|
|
22
22
|
//# sourceMappingURL=ChatShell.js.map
|
package/dist/ui/ChatShell.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"ChatShell.js","sourceRoot":"","sources":["../../src/ui/ChatShell.tsx"],"names":[],"mappings":"AAAA,YAAY,CAAC;;AAEb,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;
|
|
1
|
+
{"version":3,"file":"ChatShell.js","sourceRoot":"","sources":["../../src/ui/ChatShell.tsx"],"names":[],"mappings":"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"}
|
package/dist/ui/Composer.d.ts
CHANGED
|
@@ -1,7 +1,15 @@
|
|
|
1
|
-
|
|
1
|
+
import type { Firestore } from "firebase/firestore";
|
|
2
|
+
import type { ChatAttachment, ChatAttachmentConfig } from "../types";
|
|
3
|
+
export type ComposerProps = {
|
|
4
|
+
onSend: (text: string, attachment?: ChatAttachment) => void | Promise<void>;
|
|
5
|
+
attachmentConfig?: ChatAttachmentConfig;
|
|
6
|
+
/** Firestore instance for writing pendingMedia doc */
|
|
7
|
+
db?: Firestore;
|
|
8
|
+
/** Current user ID for the pending media upload path */
|
|
9
|
+
currentUserId?: string;
|
|
2
10
|
disabled?: boolean;
|
|
3
11
|
autoFocus?: boolean;
|
|
4
12
|
placeholder?: string;
|
|
5
|
-
|
|
6
|
-
|
|
13
|
+
};
|
|
14
|
+
export declare function Composer(props: ComposerProps): import("react/jsx-runtime").JSX.Element;
|
|
7
15
|
//# sourceMappingURL=Composer.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"Composer.d.ts","sourceRoot":"","sources":["../../src/ui/Composer.tsx"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"Composer.d.ts","sourceRoot":"","sources":["../../src/ui/Composer.tsx"],"names":[],"mappings":"AAOA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,oBAAoB,CAAC;AAGpD,OAAO,KAAK,EAAE,cAAc,EAAE,oBAAoB,EAAE,MAAM,UAAU,CAAC;AA2BrE,MAAM,MAAM,aAAa,GAAG;IAC1B,MAAM,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,UAAU,CAAC,EAAE,cAAc,KAAK,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;IAG5E,gBAAgB,CAAC,EAAE,oBAAoB,CAAC;IACxC,sDAAsD;IACtD,EAAE,CAAC,EAAE,SAAS,CAAC;IACf,wDAAwD;IACxD,aAAa,CAAC,EAAE,MAAM,CAAC;IAEvB,QAAQ,CAAC,EAAE,OAAO,CAAC;IACnB,SAAS,CAAC,EAAE,OAAO,CAAC;IACpB,WAAW,CAAC,EAAE,MAAM,CAAC;CACtB,CAAC;AAEF,wBAAgB,QAAQ,CAAC,KAAK,EAAE,aAAa,2CAgL5C"}
|
package/dist/ui/Composer.js
CHANGED
|
@@ -2,28 +2,122 @@
|
|
|
2
2
|
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
3
3
|
import * as React from "react";
|
|
4
4
|
import { Button, Textarea, cn } from "@ttt-productions/ui-core";
|
|
5
|
+
import { MediaInput } from "@ttt-productions/file-input";
|
|
6
|
+
import { uploadFileResumable } from "@ttt-productions/upload-core";
|
|
7
|
+
import { doc, setDoc, collection as firestoreCollection } from "firebase/firestore";
|
|
8
|
+
import { Loader2, X, FileText, ImageIcon, VideoIcon, MicIcon } from "lucide-react";
|
|
9
|
+
function genId() {
|
|
10
|
+
return `${Math.random().toString(36).slice(2)}${Date.now().toString(36)}`;
|
|
11
|
+
}
|
|
12
|
+
function getAttachmentType(file) {
|
|
13
|
+
if (file.type.startsWith("image/"))
|
|
14
|
+
return "image";
|
|
15
|
+
if (file.type.startsWith("video/"))
|
|
16
|
+
return "video";
|
|
17
|
+
if (file.type.startsWith("audio/"))
|
|
18
|
+
return "audio";
|
|
19
|
+
return "text";
|
|
20
|
+
}
|
|
21
|
+
function getFileExt(file) {
|
|
22
|
+
const parts = file.name.split(".");
|
|
23
|
+
return parts.length > 1 ? parts[parts.length - 1] : "bin";
|
|
24
|
+
}
|
|
25
|
+
function AttachmentTypeIcon({ type }) {
|
|
26
|
+
switch (type) {
|
|
27
|
+
case "image": return _jsx(ImageIcon, { className: "h-4 w-4 shrink-0" });
|
|
28
|
+
case "video": return _jsx(VideoIcon, { className: "h-4 w-4 shrink-0" });
|
|
29
|
+
case "audio": return _jsx(MicIcon, { className: "h-4 w-4 shrink-0" });
|
|
30
|
+
default: return _jsx(FileText, { className: "h-4 w-4 shrink-0" });
|
|
31
|
+
}
|
|
32
|
+
}
|
|
5
33
|
export function Composer(props) {
|
|
6
|
-
const { disabled, autoFocus = false, placeholder = "Type a message...",
|
|
34
|
+
const { onSend, attachmentConfig, db, currentUserId, disabled, autoFocus = false, placeholder = "Type a message...", } = props;
|
|
7
35
|
const [text, setText] = React.useState("");
|
|
36
|
+
const [pendingFile, setPendingFile] = React.useState(null);
|
|
37
|
+
const [isSending, setIsSending] = React.useState(false);
|
|
38
|
+
const [uploadProgress, setUploadProgress] = React.useState(null);
|
|
8
39
|
const ref = React.useRef(null);
|
|
40
|
+
const attachEnabled = Boolean(attachmentConfig && db && currentUserId);
|
|
9
41
|
// focus stability: never steal focus unless explicitly enabled
|
|
10
42
|
React.useEffect(() => {
|
|
11
43
|
if (!autoFocus)
|
|
12
44
|
return;
|
|
13
45
|
ref.current?.focus();
|
|
14
46
|
}, [autoFocus]);
|
|
47
|
+
const handleFileSelected = React.useCallback((payload) => {
|
|
48
|
+
if (payload.error || !payload.file) {
|
|
49
|
+
setPendingFile(null);
|
|
50
|
+
return;
|
|
51
|
+
}
|
|
52
|
+
setPendingFile(payload.file);
|
|
53
|
+
}, []);
|
|
15
54
|
const send = async () => {
|
|
16
55
|
const v = text.trim();
|
|
17
|
-
if (!v)
|
|
56
|
+
if (!v && !pendingFile)
|
|
18
57
|
return;
|
|
19
|
-
|
|
20
|
-
|
|
58
|
+
setIsSending(true);
|
|
59
|
+
try {
|
|
60
|
+
let attachment;
|
|
61
|
+
if (pendingFile && attachmentConfig && db && currentUserId) {
|
|
62
|
+
const uuid = genId();
|
|
63
|
+
const ext = getFileExt(pendingFile);
|
|
64
|
+
const storagePath = `${attachmentConfig.pendingStoragePath}/${uuid}.${ext}`;
|
|
65
|
+
const pendingCollection = attachmentConfig.pendingMediaCollection || "pendingMedia";
|
|
66
|
+
const attType = getAttachmentType(pendingFile);
|
|
67
|
+
// 1. Upload file to pending storage path
|
|
68
|
+
setUploadProgress(0);
|
|
69
|
+
await uploadFileResumable({
|
|
70
|
+
storage: attachmentConfig.storage,
|
|
71
|
+
path: storagePath,
|
|
72
|
+
file: pendingFile,
|
|
73
|
+
metadata: { contentType: pendingFile.type || "application/octet-stream" },
|
|
74
|
+
onProgress: ({ percent }) => setUploadProgress(percent),
|
|
75
|
+
});
|
|
76
|
+
setUploadProgress(null);
|
|
77
|
+
// Build the attachment object (status pending — will be sent with the message)
|
|
78
|
+
attachment = {
|
|
79
|
+
id: uuid,
|
|
80
|
+
name: pendingFile.name,
|
|
81
|
+
type: attType,
|
|
82
|
+
size: pendingFile.size,
|
|
83
|
+
status: "pending",
|
|
84
|
+
pendingStoragePath: storagePath,
|
|
85
|
+
};
|
|
86
|
+
// 2. Write the pendingMedia Firestore doc
|
|
87
|
+
// Note: targetDocPath will be set by the app's onSend handler since it knows the message doc path
|
|
88
|
+
const pendingDocRef = doc(firestoreCollection(db, pendingCollection), uuid);
|
|
89
|
+
await setDoc(pendingDocRef, {
|
|
90
|
+
id: uuid,
|
|
91
|
+
userId: currentUserId,
|
|
92
|
+
fileOrigin: "chat-attachment",
|
|
93
|
+
originalFileName: pendingFile.name,
|
|
94
|
+
pendingStoragePath: storagePath,
|
|
95
|
+
status: "pending",
|
|
96
|
+
createdAt: Date.now(),
|
|
97
|
+
});
|
|
98
|
+
}
|
|
99
|
+
await onSend(v, attachment);
|
|
100
|
+
setText("");
|
|
101
|
+
setPendingFile(null);
|
|
102
|
+
}
|
|
103
|
+
catch (err) {
|
|
104
|
+
console.error("[Composer] Send failed:", err);
|
|
105
|
+
// Remove the optimistic attachment preview on failure
|
|
106
|
+
setPendingFile(null);
|
|
107
|
+
setUploadProgress(null);
|
|
108
|
+
// Re-throw so the app can show an error toast
|
|
109
|
+
throw err;
|
|
110
|
+
}
|
|
111
|
+
finally {
|
|
112
|
+
setIsSending(false);
|
|
113
|
+
}
|
|
21
114
|
};
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
115
|
+
const isDisabled = disabled || isSending;
|
|
116
|
+
return (_jsxs("div", { className: "chat-composer", children: [pendingFile && !isSending && (_jsxs("div", { className: "chat-composer-file-preview", children: [_jsx(AttachmentTypeIcon, { type: getAttachmentType(pendingFile) }), _jsx("span", { className: "chat-composer-file-name", children: pendingFile.name }), _jsx(Button, { type: "button", variant: "ghost", size: "icon", className: "h-6 w-6 shrink-0", onClick: () => setPendingFile(null), children: _jsx(X, { className: "h-3 w-3" }) })] })), isSending && uploadProgress != null && (_jsxs("div", { className: "chat-composer-upload-status", children: [_jsx(Loader2, { className: "h-3 w-3 animate-spin" }), _jsxs("span", { className: "text-xs text-muted-foreground", children: ["Uploading ", Math.round(uploadProgress), "%..."] })] })), _jsxs("div", { className: "flex items-end gap-2", children: [_jsx(Textarea, { ref: ref, value: text, onChange: (e) => setText(e.target.value), placeholder: placeholder, disabled: isDisabled, rows: 1, className: cn("min-h-[40px] resize-none"), onKeyDown: (e) => {
|
|
117
|
+
if (e.key === "Enter" && !e.shiftKey) {
|
|
118
|
+
e.preventDefault();
|
|
119
|
+
send();
|
|
120
|
+
}
|
|
121
|
+
} }), _jsx(Button, { type: "button", variant: "default", disabled: isDisabled || (!text.trim() && !pendingFile), onClick: send, children: isSending ? _jsx(Loader2, { className: "h-4 w-4 animate-spin" }) : "Send" })] }), attachEnabled && attachmentConfig && (_jsx("div", { className: "mt-2", children: _jsx(MediaInput, { spec: attachmentConfig.attachmentSpec, onChange: handleFileSelected, selectedFile: pendingFile, isLoading: isSending && uploadProgress != null, uploadProgress: uploadProgress ?? undefined, disabled: isDisabled, buttonLabel: "Attach file", onClear: () => setPendingFile(null), className: "w-full" }) }))] }));
|
|
28
122
|
}
|
|
29
123
|
//# sourceMappingURL=Composer.js.map
|
package/dist/ui/Composer.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"Composer.js","sourceRoot":"","sources":["../../src/ui/Composer.tsx"],"names":[],"mappings":"AAAA,YAAY,CAAC;;AAEb,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAC/B,OAAO,EAAE,MAAM,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,0BAA0B,CAAC;
|
|
1
|
+
{"version":3,"file":"Composer.js","sourceRoot":"","sources":["../../src/ui/Composer.tsx"],"names":[],"mappings":"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"}
|
|
@@ -1,8 +1,11 @@
|
|
|
1
1
|
import type { ChatMessageV1, ModerationHandlers } from "../types";
|
|
2
|
-
export
|
|
2
|
+
export type MessageItemDefaultProps = {
|
|
3
3
|
m: ChatMessageV1;
|
|
4
4
|
currentUserId: string;
|
|
5
5
|
isAdmin: boolean;
|
|
6
6
|
handlers?: ModerationHandlers;
|
|
7
|
-
|
|
7
|
+
isContinuation?: boolean;
|
|
8
|
+
onSenderClick?: (senderId: string, displayName: string) => void;
|
|
9
|
+
};
|
|
10
|
+
export declare function MessageItemDefault(props: MessageItemDefaultProps): import("react/jsx-runtime").JSX.Element;
|
|
8
11
|
//# sourceMappingURL=MessageItemDefault.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"MessageItemDefault.d.ts","sourceRoot":"","sources":["../../src/ui/MessageItemDefault.tsx"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,aAAa,
|
|
1
|
+
{"version":3,"file":"MessageItemDefault.d.ts","sourceRoot":"","sources":["../../src/ui/MessageItemDefault.tsx"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,aAAa,EAAkB,kBAAkB,EAAE,MAAM,UAAU,CAAC;AA6GlF,MAAM,MAAM,uBAAuB,GAAG;IACpC,CAAC,EAAE,aAAa,CAAC;IACjB,aAAa,EAAE,MAAM,CAAC;IACtB,OAAO,EAAE,OAAO,CAAC;IACjB,QAAQ,CAAC,EAAE,kBAAkB,CAAC;IAG9B,cAAc,CAAC,EAAE,OAAO,CAAC;IAGzB,aAAa,CAAC,EAAE,CAAC,QAAQ,EAAE,MAAM,EAAE,WAAW,EAAE,MAAM,KAAK,IAAI,CAAC;CACjE,CAAC;AAEF,wBAAgB,kBAAkB,CAAC,KAAK,EAAE,uBAAuB,2CA4DhE"}
|
|
@@ -1,10 +1,73 @@
|
|
|
1
1
|
"use client";
|
|
2
2
|
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
3
3
|
import { cn } from "@ttt-productions/ui-core";
|
|
4
|
+
import { MediaViewer } from "@ttt-productions/media-viewer";
|
|
5
|
+
import { Loader2, FileText, AlertTriangle, ShieldAlert } from "lucide-react";
|
|
4
6
|
import { MessageActions } from "./menus";
|
|
7
|
+
// ============================================
|
|
8
|
+
// Attachment rendering
|
|
9
|
+
// ============================================
|
|
10
|
+
function AttachmentPending({ att }) {
|
|
11
|
+
return (_jsxs("div", { className: "chat-attachment-pending", children: [_jsx(Loader2, { className: "h-4 w-4 animate-spin shrink-0" }), _jsx("span", { className: "text-xs text-muted-foreground truncate", children: att.name }), _jsx("span", { className: "text-xs text-muted-foreground", children: "Processing\u2026" })] }));
|
|
12
|
+
}
|
|
13
|
+
function AttachmentCompleted({ att }) {
|
|
14
|
+
if (!att.url)
|
|
15
|
+
return null;
|
|
16
|
+
if (att.type === "image") {
|
|
17
|
+
return _jsx(MediaViewer, { type: "image", url: att.url, alt: att.name, className: "chat-attachment-media" });
|
|
18
|
+
}
|
|
19
|
+
if (att.type === "video") {
|
|
20
|
+
return _jsx(MediaViewer, { type: "video", url: att.url, controls: true, className: "chat-attachment-media" });
|
|
21
|
+
}
|
|
22
|
+
if (att.type === "audio") {
|
|
23
|
+
return _jsx(MediaViewer, { type: "audio", url: att.url, controls: true, className: "chat-attachment-media" });
|
|
24
|
+
}
|
|
25
|
+
// text/markdown — download link
|
|
26
|
+
return (_jsxs("a", { href: att.url, target: "_blank", rel: "noopener noreferrer", className: "chat-attachment-text-link", children: [_jsx(FileText, { className: "h-4 w-4 shrink-0" }), _jsx("span", { className: "truncate", children: att.name })] }));
|
|
27
|
+
}
|
|
28
|
+
function AttachmentFailed({ att }) {
|
|
29
|
+
return (_jsxs("div", { className: "chat-attachment-failed", children: [_jsx(AlertTriangle, { className: "h-4 w-4 shrink-0" }), _jsx("span", { className: "text-xs", children: att.errorMessage || "Attachment failed to process" })] }));
|
|
30
|
+
}
|
|
31
|
+
function AttachmentRejected({ att }) {
|
|
32
|
+
return (_jsxs("div", { className: "chat-attachment-rejected", children: [_jsx(ShieldAlert, { className: "h-4 w-4 shrink-0" }), _jsx("span", { className: "text-xs", children: att.errorMessage || "Attachment removed for policy violation" })] }));
|
|
33
|
+
}
|
|
34
|
+
function AttachmentRenderer({ attachment }) {
|
|
35
|
+
switch (attachment.status) {
|
|
36
|
+
case "pending":
|
|
37
|
+
case "processing":
|
|
38
|
+
return _jsx(AttachmentPending, { att: attachment });
|
|
39
|
+
case "completed":
|
|
40
|
+
return _jsx(AttachmentCompleted, { att: attachment });
|
|
41
|
+
case "failed":
|
|
42
|
+
return _jsx(AttachmentFailed, { att: attachment });
|
|
43
|
+
case "rejected":
|
|
44
|
+
return _jsx(AttachmentRejected, { att: attachment });
|
|
45
|
+
default:
|
|
46
|
+
return null;
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
// ============================================
|
|
50
|
+
// Reply-to quote
|
|
51
|
+
// ============================================
|
|
52
|
+
function ReplyQuote({ replyTo }) {
|
|
53
|
+
return (_jsxs("div", { className: "chat-reply-quote", children: [_jsx("span", { className: "chat-reply-quote-sender", children: replyTo.senderUsername }), _jsx("span", { className: "chat-reply-quote-preview", children: replyTo.messagePreview })] }));
|
|
54
|
+
}
|
|
55
|
+
// ============================================
|
|
56
|
+
// System message
|
|
57
|
+
// ============================================
|
|
58
|
+
function SystemMessage({ m }) {
|
|
59
|
+
return (_jsx("div", { className: "chat-system-message", children: m.text && _jsx("span", { children: m.text }) }));
|
|
60
|
+
}
|
|
5
61
|
export function MessageItemDefault(props) {
|
|
6
|
-
const { m, currentUserId, isAdmin, handlers } = props;
|
|
62
|
+
const { m, currentUserId, isAdmin, handlers, isContinuation, onSenderClick } = props;
|
|
63
|
+
// System messages render differently
|
|
64
|
+
if (m.isSystemMessage) {
|
|
65
|
+
return _jsx(SystemMessage, { m: m });
|
|
66
|
+
}
|
|
7
67
|
const mine = m.senderId === currentUserId;
|
|
8
|
-
return (_jsxs("div", { className: cn("flex flex-col w-fit max-w-[85%]", mine ? "ml-auto items-end" : "mr-auto items-start"), children: [_jsxs("div", { className: cn("
|
|
68
|
+
return (_jsxs("div", { className: cn("flex flex-col w-fit max-w-[85%]", mine ? "ml-auto items-end" : "mr-auto items-start", isContinuation ? "chat-continuation-gap" : "chat-group-gap"), children: [_jsxs("div", { className: cn("chat-bubble", mine ? "chat-bubble--mine" : "chat-bubble--theirs"), children: [!isContinuation && (_jsxs("div", { className: "flex items-center gap-2 text-xs opacity-80 mb-1", children: [_jsx("span", { className: cn("font-medium", onSenderClick && "cursor-pointer hover:underline"), onClick: onSenderClick ? () => onSenderClick(m.senderId, m.senderUsername ?? "User") : undefined, role: onSenderClick ? "button" : undefined, tabIndex: onSenderClick ? 0 : undefined, onKeyDown: onSenderClick ? (e) => {
|
|
69
|
+
if (e.key === "Enter" || e.key === " ")
|
|
70
|
+
onSenderClick(m.senderId, m.senderUsername ?? "User");
|
|
71
|
+
} : undefined, children: m.senderUsername ?? "User" }), _jsx("span", { children: "\u00B7" }), _jsx("span", { children: new Date(m.createdAt).toLocaleTimeString([], { hour: "numeric", minute: "2-digit" }) })] })), m.replyTo && _jsx(ReplyQuote, { replyTo: m.replyTo }), m.text && _jsx("p", { className: "text-sm whitespace-pre-wrap", children: m.text }), m.attachment && (_jsx("div", { className: "mt-2", children: _jsx(AttachmentRenderer, { attachment: m.attachment }) }))] }), !isContinuation && (_jsx("div", { className: "mt-1", children: _jsx(MessageActions, { messageId: m.messageId, isAdmin: isAdmin, handlers: handlers }) }))] }));
|
|
9
72
|
}
|
|
10
73
|
//# sourceMappingURL=MessageItemDefault.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"MessageItemDefault.js","sourceRoot":"","sources":["../../src/ui/MessageItemDefault.tsx"],"names":[],"mappings":"AAAA,YAAY,CAAC;;AAGb,OAAO,EAAE,EAAE,EAAE,MAAM,0BAA0B,CAAC;AAC9C,OAAO,EAAE,cAAc,EAAE,MAAM,SAAS,CAAC;AAEzC,MAAM,UAAU,kBAAkB,CAAC,
|
|
1
|
+
{"version":3,"file":"MessageItemDefault.js","sourceRoot":"","sources":["../../src/ui/MessageItemDefault.tsx"],"names":[],"mappings":"AAAA,YAAY,CAAC;;AAGb,OAAO,EAAE,EAAE,EAAE,MAAM,0BAA0B,CAAC;AAC9C,OAAO,EAAE,WAAW,EAAE,MAAM,+BAA+B,CAAC;AAC5D,OAAO,EAAE,OAAO,EAAE,QAAQ,EAAE,aAAa,EAAE,WAAW,EAAE,MAAM,cAAc,CAAC;AAC7E,OAAO,EAAE,cAAc,EAAE,MAAM,SAAS,CAAC;AAEzC,+CAA+C;AAC/C,uBAAuB;AACvB,+CAA+C;AAE/C,SAAS,iBAAiB,CAAC,EAAE,GAAG,EAA2B;IACzD,OAAO,CACL,eAAK,SAAS,EAAC,yBAAyB,aACtC,KAAC,OAAO,IAAC,SAAS,EAAC,+BAA+B,GAAG,EACrD,eAAM,SAAS,EAAC,wCAAwC,YAAE,GAAG,CAAC,IAAI,GAAQ,EAC1E,eAAM,SAAS,EAAC,+BAA+B,iCAAmB,IAC9D,CACP,CAAC;AACJ,CAAC;AAED,SAAS,mBAAmB,CAAC,EAAE,GAAG,EAA2B;IAC3D,IAAI,CAAC,GAAG,CAAC,GAAG;QAAE,OAAO,IAAI,CAAC;IAE1B,IAAI,GAAG,CAAC,IAAI,KAAK,OAAO,EAAE,CAAC;QACzB,OAAO,KAAC,WAAW,IAAC,IAAI,EAAC,OAAO,EAAC,GAAG,EAAE,GAAG,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,CAAC,IAAI,EAAE,SAAS,EAAC,uBAAuB,GAAG,CAAC;IACrG,CAAC;IACD,IAAI,GAAG,CAAC,IAAI,KAAK,OAAO,EAAE,CAAC;QACzB,OAAO,KAAC,WAAW,IAAC,IAAI,EAAC,OAAO,EAAC,GAAG,EAAE,GAAG,CAAC,GAAG,EAAE,QAAQ,QAAC,SAAS,EAAC,uBAAuB,GAAG,CAAC;IAC/F,CAAC;IACD,IAAI,GAAG,CAAC,IAAI,KAAK,OAAO,EAAE,CAAC;QACzB,OAAO,KAAC,WAAW,IAAC,IAAI,EAAC,OAAO,EAAC,GAAG,EAAE,GAAG,CAAC,GAAG,EAAE,QAAQ,QAAC,SAAS,EAAC,uBAAuB,GAAG,CAAC;IAC/F,CAAC;IACD,gCAAgC;IAChC,OAAO,CACL,aACE,IAAI,EAAE,GAAG,CAAC,GAAG,EACb,MAAM,EAAC,QAAQ,EACf,GAAG,EAAC,qBAAqB,EACzB,SAAS,EAAC,2BAA2B,aAErC,KAAC,QAAQ,IAAC,SAAS,EAAC,kBAAkB,GAAG,EACzC,eAAM,SAAS,EAAC,UAAU,YAAE,GAAG,CAAC,IAAI,GAAQ,IAC1C,CACL,CAAC;AACJ,CAAC;AAED,SAAS,gBAAgB,CAAC,EAAE,GAAG,EAA2B;IACxD,OAAO,CACL,eAAK,SAAS,EAAC,wBAAwB,aACrC,KAAC,aAAa,IAAC,SAAS,EAAC,kBAAkB,GAAG,EAC9C,eAAM,SAAS,EAAC,SAAS,YAAE,GAAG,CAAC,YAAY,IAAI,8BAA8B,GAAQ,IACjF,CACP,CAAC;AACJ,CAAC;AAED,SAAS,kBAAkB,CAAC,EAAE,GAAG,EAA2B;IAC1D,OAAO,CACL,eAAK,SAAS,EAAC,0BAA0B,aACvC,KAAC,WAAW,IAAC,SAAS,EAAC,kBAAkB,GAAG,EAC5C,eAAM,SAAS,EAAC,SAAS,YAAE,GAAG,CAAC,YAAY,IAAI,yCAAyC,GAAQ,IAC5F,CACP,CAAC;AACJ,CAAC;AAED,SAAS,kBAAkB,CAAC,EAAE,UAAU,EAAkC;IACxE,QAAQ,UAAU,CAAC,MAAM,EAAE,CAAC;QAC1B,KAAK,SAAS,CAAC;QACf,KAAK,YAAY;YACf,OAAO,KAAC,iBAAiB,IAAC,GAAG,EAAE,UAAU,GAAI,CAAC;QAChD,KAAK,WAAW;YACd,OAAO,KAAC,mBAAmB,IAAC,GAAG,EAAE,UAAU,GAAI,CAAC;QAClD,KAAK,QAAQ;YACX,OAAO,KAAC,gBAAgB,IAAC,GAAG,EAAE,UAAU,GAAI,CAAC;QAC/C,KAAK,UAAU;YACb,OAAO,KAAC,kBAAkB,IAAC,GAAG,EAAE,UAAU,GAAI,CAAC;QACjD;YACE,OAAO,IAAI,CAAC;IAChB,CAAC;AACH,CAAC;AAED,+CAA+C;AAC/C,iBAAiB;AACjB,+CAA+C;AAE/C,SAAS,UAAU,CAAC,EAAE,OAAO,EAAsD;IACjF,OAAO,CACL,eAAK,SAAS,EAAC,kBAAkB,aAC/B,eAAM,SAAS,EAAC,yBAAyB,YAAE,OAAO,CAAC,cAAc,GAAQ,EACzE,eAAM,SAAS,EAAC,0BAA0B,YAAE,OAAO,CAAC,cAAc,GAAQ,IACtE,CACP,CAAC;AACJ,CAAC;AAED,+CAA+C;AAC/C,iBAAiB;AACjB,+CAA+C;AAE/C,SAAS,aAAa,CAAC,EAAE,CAAC,EAAwB;IAChD,OAAO,CACL,cAAK,SAAS,EAAC,qBAAqB,YACjC,CAAC,CAAC,IAAI,IAAI,yBAAO,CAAC,CAAC,IAAI,GAAQ,GAC5B,CACP,CAAC;AACJ,CAAC;AAmBD,MAAM,UAAU,kBAAkB,CAAC,KAA8B;IAC/D,MAAM,EAAE,CAAC,EAAE,aAAa,EAAE,OAAO,EAAE,QAAQ,EAAE,cAAc,EAAE,aAAa,EAAE,GAAG,KAAK,CAAC;IAErF,qCAAqC;IACrC,IAAI,CAAC,CAAC,eAAe,EAAE,CAAC;QACtB,OAAO,KAAC,aAAa,IAAC,CAAC,EAAE,CAAC,GAAI,CAAC;IACjC,CAAC;IAED,MAAM,IAAI,GAAG,CAAC,CAAC,QAAQ,KAAK,aAAa,CAAC;IAE1C,OAAO,CACL,eACE,SAAS,EAAE,EAAE,CACX,iCAAiC,EACjC,IAAI,CAAC,CAAC,CAAC,mBAAmB,CAAC,CAAC,CAAC,qBAAqB,EAClD,cAAc,CAAC,CAAC,CAAC,uBAAuB,CAAC,CAAC,CAAC,gBAAgB,CAC5D,aAED,eAAK,SAAS,EAAE,EAAE,CAAC,aAAa,EAAE,IAAI,CAAC,CAAC,CAAC,mBAAmB,CAAC,CAAC,CAAC,qBAAqB,CAAC,aAElF,CAAC,cAAc,IAAI,CAClB,eAAK,SAAS,EAAC,iDAAiD,aAC9D,eACE,SAAS,EAAE,EAAE,CAAC,aAAa,EAAE,aAAa,IAAI,gCAAgC,CAAC,EAC/E,OAAO,EAAE,aAAa,CAAC,CAAC,CAAC,GAAG,EAAE,CAAC,aAAa,CAAC,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC,cAAc,IAAI,MAAM,CAAC,CAAC,CAAC,CAAC,SAAS,EAChG,IAAI,EAAE,aAAa,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,SAAS,EAC1C,QAAQ,EAAE,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,EACvC,SAAS,EAAE,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,EAAE;oCAC/B,IAAI,CAAC,CAAC,GAAG,KAAK,OAAO,IAAI,CAAC,CAAC,GAAG,KAAK,GAAG;wCAAE,aAAa,CAAC,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC,cAAc,IAAI,MAAM,CAAC,CAAC;gCAChG,CAAC,CAAC,CAAC,CAAC,SAAS,YAEZ,CAAC,CAAC,cAAc,IAAI,MAAM,GACtB,EACP,oCAAc,EACd,yBAAO,IAAI,IAAI,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,kBAAkB,CAAC,EAAE,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,MAAM,EAAE,SAAS,EAAE,CAAC,GAAQ,IAC/F,CACP,EAGA,CAAC,CAAC,OAAO,IAAI,KAAC,UAAU,IAAC,OAAO,EAAE,CAAC,CAAC,OAAO,GAAI,EAG/C,CAAC,CAAC,IAAI,IAAI,YAAG,SAAS,EAAC,6BAA6B,YAAE,CAAC,CAAC,IAAI,GAAK,EAGjE,CAAC,CAAC,UAAU,IAAI,CACf,cAAK,SAAS,EAAC,MAAM,YACnB,KAAC,kBAAkB,IAAC,UAAU,EAAE,CAAC,CAAC,UAAU,GAAI,GAC5C,CACP,IACG,EAGL,CAAC,cAAc,IAAI,CAClB,cAAK,SAAS,EAAC,MAAM,YACnB,KAAC,cAAc,IAAC,SAAS,EAAE,CAAC,CAAC,SAAS,EAAE,OAAO,EAAE,OAAO,EAAE,QAAQ,EAAE,QAAQ,GAAI,GAC5E,CACP,IACG,CACP,CAAC;AACJ,CAAC"}
|
package/dist/ui/MessageList.d.ts
CHANGED
|
@@ -12,5 +12,6 @@ export declare function MessageList(props: {
|
|
|
12
12
|
showScrollToBottom?: boolean;
|
|
13
13
|
onScrollToBottom?: () => void;
|
|
14
14
|
handlers?: ModerationHandlers;
|
|
15
|
+
onSenderClick?: (senderId: string, displayName: string) => void;
|
|
15
16
|
}): import("react/jsx-runtime").JSX.Element;
|
|
16
17
|
//# sourceMappingURL=MessageList.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"MessageList.d.ts","sourceRoot":"","sources":["../../src/ui/MessageList.tsx"],"names":[],"mappings":"AAEA,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAC/B,OAAO,KAAK,EAAE,aAAa,EAAE,uBAAuB,EAAE,kBAAkB,EAAE,MAAM,UAAU,CAAC;
|
|
1
|
+
{"version":3,"file":"MessageList.d.ts","sourceRoot":"","sources":["../../src/ui/MessageList.tsx"],"names":[],"mappings":"AAEA,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAC/B,OAAO,KAAK,EAAE,aAAa,EAAE,uBAAuB,EAAE,kBAAkB,EAAE,MAAM,UAAU,CAAC;AAgB3F,wBAAgB,WAAW,CAAC,KAAK,EAAE;IACjC,QAAQ,EAAE,aAAa,EAAE,CAAC;IAC1B,aAAa,EAAE,MAAM,CAAC;IACtB,OAAO,EAAE,OAAO,CAAC;IAEjB,eAAe,CAAC,EAAE,OAAO,CAAC;IAC1B,QAAQ,CAAC,EAAE,OAAO,CAAC;IACnB,WAAW,CAAC,EAAE,MAAM,IAAI,CAAC;IAEzB,aAAa,CAAC,EAAE,CAAC,CAAC,EAAE,aAAa,KAAK,KAAK,CAAC,SAAS,CAAC;IACtD,gBAAgB,CAAC,EAAE,uBAAuB,CAAC;IAE3C,kBAAkB,CAAC,EAAE,OAAO,CAAC;IAC7B,gBAAgB,CAAC,EAAE,MAAM,IAAI,CAAC;IAE9B,QAAQ,CAAC,EAAE,kBAAkB,CAAC;IAC9B,aAAa,CAAC,EAAE,CAAC,QAAQ,EAAE,MAAM,EAAE,WAAW,EAAE,MAAM,KAAK,IAAI,CAAC;CACjE,2CA+JA"}
|
package/dist/ui/MessageList.js
CHANGED
|
@@ -1,15 +1,30 @@
|
|
|
1
1
|
"use client";
|
|
2
2
|
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
3
3
|
import * as React from "react";
|
|
4
|
+
import { GROUP_GAP_SEC } from "../types";
|
|
4
5
|
import { Button } from "@ttt-productions/ui-core";
|
|
5
6
|
import { MessageItemDefault } from "./MessageItemDefault";
|
|
7
|
+
/**
|
|
8
|
+
* Determine if msg is a "continuation" of prev (same sender, within GROUP_GAP_SEC, no system break).
|
|
9
|
+
*/
|
|
10
|
+
function isContinuation(prev, msg) {
|
|
11
|
+
if (!prev)
|
|
12
|
+
return false;
|
|
13
|
+
if (msg.isSystemMessage || prev.isSystemMessage)
|
|
14
|
+
return false;
|
|
15
|
+
if (msg.senderId !== prev.senderId)
|
|
16
|
+
return false;
|
|
17
|
+
const gapMs = msg.createdAt - prev.createdAt;
|
|
18
|
+
return gapMs >= 0 && gapMs <= GROUP_GAP_SEC * 1000;
|
|
19
|
+
}
|
|
6
20
|
export function MessageList(props) {
|
|
7
|
-
const { messages, currentUserId, isAdmin, isFetchingOlder, hasOlder, onLoadOlder, renderMessage, messageRenderers, showScrollToBottom, onScrollToBottom, handlers } = props;
|
|
21
|
+
const { messages, currentUserId, isAdmin, isFetchingOlder, hasOlder, onLoadOlder, renderMessage, messageRenderers, showScrollToBottom, onScrollToBottom, handlers, onSenderClick, } = props;
|
|
8
22
|
const scrollRef = React.useRef(null);
|
|
9
23
|
const topSentinelRef = React.useRef(null);
|
|
10
24
|
const isAtBottomRef = React.useRef(true);
|
|
11
25
|
const prevScrollHeightRef = React.useRef(null);
|
|
12
26
|
const prevCountRef = React.useRef(0);
|
|
27
|
+
// IntersectionObserver for infinite scroll (older messages)
|
|
13
28
|
React.useEffect(() => {
|
|
14
29
|
const root = scrollRef.current;
|
|
15
30
|
const target = topSentinelRef.current;
|
|
@@ -29,6 +44,7 @@ export function MessageList(props) {
|
|
|
29
44
|
io.observe(target);
|
|
30
45
|
return () => io.disconnect();
|
|
31
46
|
}, [hasOlder, isFetchingOlder, onLoadOlder]);
|
|
47
|
+
// Scroll management — FIXES accordion scroll jump
|
|
32
48
|
React.useLayoutEffect(() => {
|
|
33
49
|
const el = scrollRef.current;
|
|
34
50
|
if (!el)
|
|
@@ -36,16 +52,33 @@ export function MessageList(props) {
|
|
|
36
52
|
const count = messages.length;
|
|
37
53
|
const prev = prevCountRef.current;
|
|
38
54
|
if (prev === 0 && count > 0) {
|
|
39
|
-
|
|
55
|
+
// Initial load — scroll to bottom WITHOUT triggering parent scroll
|
|
56
|
+
requestAnimationFrame(() => {
|
|
57
|
+
const savedScrollY = window.scrollY;
|
|
58
|
+
el.scrollTop = el.scrollHeight;
|
|
59
|
+
// Restore page scroll if the browser moved it
|
|
60
|
+
if (window.scrollY !== savedScrollY) {
|
|
61
|
+
window.scrollTo({ top: savedScrollY, behavior: "instant" });
|
|
62
|
+
}
|
|
63
|
+
});
|
|
40
64
|
isAtBottomRef.current = true;
|
|
41
65
|
}
|
|
42
66
|
else if (count > prev && prevScrollHeightRef.current != null) {
|
|
67
|
+
// Older messages prepended — maintain scroll position
|
|
43
68
|
const diff = el.scrollHeight - prevScrollHeightRef.current;
|
|
44
69
|
el.scrollTop += diff;
|
|
45
70
|
prevScrollHeightRef.current = null;
|
|
46
71
|
}
|
|
47
72
|
else if (count > prev && isAtBottomRef.current) {
|
|
48
|
-
|
|
73
|
+
// New message at bottom, user is at bottom — scroll down
|
|
74
|
+
// Use scrollTop assignment, NOT scrollTo with behavior:"smooth" (causes parent scroll jump)
|
|
75
|
+
requestAnimationFrame(() => {
|
|
76
|
+
const savedScrollY = window.scrollY;
|
|
77
|
+
el.scrollTop = el.scrollHeight;
|
|
78
|
+
if (window.scrollY !== savedScrollY) {
|
|
79
|
+
window.scrollTo({ top: savedScrollY, behavior: "instant" });
|
|
80
|
+
}
|
|
81
|
+
});
|
|
49
82
|
}
|
|
50
83
|
prevCountRef.current = count;
|
|
51
84
|
}, [messages]);
|
|
@@ -54,13 +87,16 @@ export function MessageList(props) {
|
|
|
54
87
|
isAtBottomRef.current = scrollHeight - (scrollTop + clientHeight) < 50;
|
|
55
88
|
};
|
|
56
89
|
let lastDay = null;
|
|
57
|
-
return (_jsxs("div", { className: "relative", children: [_jsxs("div", { ref: scrollRef, className: "h-[400px] overflow-y-auto p-4", onScroll: onScroll, children: [isFetchingOlder && _jsx("div", { className: "text-center text-xs opacity-70 mb-2", children: "Loading\u2026" }), _jsx("div", { ref: topSentinelRef }), messages.length === 0 ? (_jsx("div", { className: "h-full flex items-center justify-center text-sm opacity-70", children: "No messages yet" })) : (_jsx("div", { className: "flex flex-col
|
|
90
|
+
return (_jsxs("div", { className: "relative", children: [_jsxs("div", { ref: scrollRef, className: "h-[400px] overflow-y-auto p-4", onScroll: onScroll, children: [isFetchingOlder && _jsx("div", { className: "text-center text-xs opacity-70 mb-2", children: "Loading\u2026" }), _jsx("div", { ref: topSentinelRef }), messages.length === 0 ? (_jsx("div", { className: "h-full flex items-center justify-center text-sm opacity-70", children: "No messages yet" })) : (_jsx("div", { className: "flex flex-col", children: messages.map((m, idx) => {
|
|
58
91
|
const day = new Date(m.createdAt).toDateString();
|
|
59
92
|
const showDay = day !== lastDay;
|
|
60
93
|
lastDay = day;
|
|
94
|
+
const prevMsg = idx > 0 ? messages[idx - 1] : undefined;
|
|
95
|
+
// Date separators always break grouping
|
|
96
|
+
const continuation = showDay ? false : isContinuation(prevMsg, m);
|
|
61
97
|
const byType = m.type && messageRenderers?.[m.type] ? messageRenderers[m.type](m) : null;
|
|
62
98
|
const body = renderMessage?.(m) ??
|
|
63
|
-
byType ?? (_jsx(MessageItemDefault, { m: m, currentUserId: currentUserId, isAdmin: isAdmin, handlers: handlers }));
|
|
99
|
+
byType ?? (_jsx(MessageItemDefault, { m: m, currentUserId: currentUserId, isAdmin: isAdmin, handlers: handlers, isContinuation: continuation, onSenderClick: onSenderClick }));
|
|
64
100
|
return (_jsxs(React.Fragment, { children: [showDay && (_jsx("div", { className: "my-2 flex justify-center", children: _jsx("div", { className: "chat-date-separator", children: new Date(m.createdAt).toLocaleDateString() }) })), body] }, m.messageId));
|
|
65
101
|
}) }))] }), showScrollToBottom && (_jsx(Button, { type: "button", variant: "outline", size: "icon", className: "absolute bottom-4 left-1/2 -translate-x-1/2 rounded-full shadow-sm", onClick: onScrollToBottom, children: "\u2193" }))] }));
|
|
66
102
|
}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"MessageList.js","sourceRoot":"","sources":["../../src/ui/MessageList.tsx"],"names":[],"mappings":"AAAA,YAAY,CAAC;;AAEb,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAE/B,OAAO,EAAE,MAAM,EAAE,MAAM,0BAA0B,CAAC;AAClD,OAAO,EAAE,kBAAkB,EAAE,MAAM,sBAAsB,CAAC;AAE1D,MAAM,UAAU,WAAW,CAAC,
|
|
1
|
+
{"version":3,"file":"MessageList.js","sourceRoot":"","sources":["../../src/ui/MessageList.tsx"],"names":[],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@ttt-productions/chat-core",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.3.3",
|
|
4
4
|
"description": "Shared Firestore chat core (realtime newest window + infinite older pagination) for TTT Productions apps",
|
|
5
5
|
"repository": {
|
|
6
6
|
"type": "git",
|
|
@@ -11,15 +11,19 @@
|
|
|
11
11
|
"main": "dist/index.js",
|
|
12
12
|
"module": "dist/index.js",
|
|
13
13
|
"types": "dist/index.d.ts",
|
|
14
|
-
"sideEffects":
|
|
14
|
+
"sideEffects": [
|
|
15
|
+
"src/styles/*.css"
|
|
16
|
+
],
|
|
15
17
|
"files": [
|
|
16
|
-
"dist"
|
|
18
|
+
"dist",
|
|
19
|
+
"src/styles"
|
|
17
20
|
],
|
|
18
21
|
"exports": {
|
|
19
22
|
".": {
|
|
20
23
|
"types": "./dist/index.d.ts",
|
|
21
24
|
"default": "./dist/index.js"
|
|
22
|
-
}
|
|
25
|
+
},
|
|
26
|
+
"./styles": "./src/styles/chat.css"
|
|
23
27
|
},
|
|
24
28
|
"scripts": {
|
|
25
29
|
"build": "tsc",
|
|
@@ -36,7 +40,11 @@
|
|
|
36
40
|
"dependencies": {
|
|
37
41
|
"@ttt-productions/ui-core": "^0.2.3",
|
|
38
42
|
"@ttt-productions/firebase-helpers": "^0.2.14",
|
|
39
|
-
"@ttt-productions/mobile-core": "^0.2.5"
|
|
43
|
+
"@ttt-productions/mobile-core": "^0.2.5",
|
|
44
|
+
"@ttt-productions/file-input": "^0.0.5",
|
|
45
|
+
"@ttt-productions/upload-core": "^0.0.3",
|
|
46
|
+
"@ttt-productions/media-viewer": "^0.2.9",
|
|
47
|
+
"@ttt-productions/media-contracts": "^0.2.11"
|
|
40
48
|
},
|
|
41
49
|
"devDependencies": {
|
|
42
50
|
"@types/react": "^19.0.0",
|
|
@@ -0,0 +1,137 @@
|
|
|
1
|
+
/* ============================================
|
|
2
|
+
chat-core — default styles
|
|
3
|
+
Uses theme-core CSS custom property tokens.
|
|
4
|
+
Apps import: @ttt-productions/chat-core/styles
|
|
5
|
+
============================================ */
|
|
6
|
+
|
|
7
|
+
/* --- Message bubbles --- */
|
|
8
|
+
.chat-bubble {
|
|
9
|
+
padding: 0.75rem;
|
|
10
|
+
border-radius: 0.5rem;
|
|
11
|
+
}
|
|
12
|
+
.chat-bubble--mine {
|
|
13
|
+
background-color: hsl(var(--primary) / 0.1);
|
|
14
|
+
}
|
|
15
|
+
.chat-bubble--theirs {
|
|
16
|
+
background-color: hsl(var(--muted));
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
/* --- Message grouping --- */
|
|
20
|
+
.chat-group-gap {
|
|
21
|
+
margin-top: 0.75rem;
|
|
22
|
+
}
|
|
23
|
+
.chat-continuation-gap {
|
|
24
|
+
margin-top: 0.125rem;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
/* --- System messages --- */
|
|
28
|
+
.chat-system-message {
|
|
29
|
+
text-align: center;
|
|
30
|
+
font-size: 0.75rem;
|
|
31
|
+
color: hsl(var(--muted-foreground));
|
|
32
|
+
padding: 0.5rem 0;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
/* --- Reply-to quote --- */
|
|
36
|
+
.chat-reply-quote {
|
|
37
|
+
display: flex;
|
|
38
|
+
flex-direction: column;
|
|
39
|
+
gap: 0.125rem;
|
|
40
|
+
padding: 0.375rem 0.5rem;
|
|
41
|
+
margin-bottom: 0.375rem;
|
|
42
|
+
border-left: 2px solid hsl(var(--border));
|
|
43
|
+
border-radius: 0.125rem;
|
|
44
|
+
background-color: hsl(var(--muted) / 0.5);
|
|
45
|
+
font-size: 0.75rem;
|
|
46
|
+
}
|
|
47
|
+
.chat-reply-quote-sender {
|
|
48
|
+
font-weight: 600;
|
|
49
|
+
color: hsl(var(--foreground));
|
|
50
|
+
}
|
|
51
|
+
.chat-reply-quote-preview {
|
|
52
|
+
color: hsl(var(--muted-foreground));
|
|
53
|
+
overflow: hidden;
|
|
54
|
+
text-overflow: ellipsis;
|
|
55
|
+
white-space: nowrap;
|
|
56
|
+
max-width: 280px;
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
/* --- Attachment states --- */
|
|
60
|
+
.chat-attachment-pending {
|
|
61
|
+
display: flex;
|
|
62
|
+
align-items: center;
|
|
63
|
+
gap: 0.5rem;
|
|
64
|
+
padding: 0.5rem 0.75rem;
|
|
65
|
+
border-radius: 0.375rem;
|
|
66
|
+
background-color: hsl(var(--muted));
|
|
67
|
+
font-size: 0.75rem;
|
|
68
|
+
}
|
|
69
|
+
.chat-attachment-media {
|
|
70
|
+
max-width: 320px;
|
|
71
|
+
border-radius: 0.375rem;
|
|
72
|
+
overflow: hidden;
|
|
73
|
+
}
|
|
74
|
+
.chat-attachment-text-link {
|
|
75
|
+
display: inline-flex;
|
|
76
|
+
align-items: center;
|
|
77
|
+
gap: 0.375rem;
|
|
78
|
+
padding: 0.375rem 0.625rem;
|
|
79
|
+
border-radius: 0.375rem;
|
|
80
|
+
background-color: hsl(var(--muted));
|
|
81
|
+
color: hsl(var(--primary));
|
|
82
|
+
font-size: 0.8125rem;
|
|
83
|
+
text-decoration: none;
|
|
84
|
+
transition: opacity 0.15s;
|
|
85
|
+
}
|
|
86
|
+
.chat-attachment-text-link:hover {
|
|
87
|
+
opacity: 0.8;
|
|
88
|
+
}
|
|
89
|
+
.chat-attachment-failed {
|
|
90
|
+
display: flex;
|
|
91
|
+
align-items: center;
|
|
92
|
+
gap: 0.375rem;
|
|
93
|
+
padding: 0.5rem 0.75rem;
|
|
94
|
+
border-radius: 0.375rem;
|
|
95
|
+
background-color: hsl(var(--destructive) / 0.1);
|
|
96
|
+
color: hsl(var(--destructive));
|
|
97
|
+
font-size: 0.75rem;
|
|
98
|
+
}
|
|
99
|
+
.chat-attachment-rejected {
|
|
100
|
+
display: flex;
|
|
101
|
+
align-items: center;
|
|
102
|
+
gap: 0.375rem;
|
|
103
|
+
padding: 0.5rem 0.75rem;
|
|
104
|
+
border-radius: 0.375rem;
|
|
105
|
+
background-color: hsl(var(--warning, var(--muted)) / 0.15);
|
|
106
|
+
color: hsl(var(--warning-foreground, var(--muted-foreground)));
|
|
107
|
+
font-size: 0.75rem;
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
/* --- Composer --- */
|
|
111
|
+
.chat-composer {
|
|
112
|
+
display: flex;
|
|
113
|
+
flex-direction: column;
|
|
114
|
+
gap: 0.5rem;
|
|
115
|
+
width: 100%;
|
|
116
|
+
}
|
|
117
|
+
.chat-composer-file-preview {
|
|
118
|
+
display: flex;
|
|
119
|
+
align-items: center;
|
|
120
|
+
gap: 0.5rem;
|
|
121
|
+
padding: 0.375rem 0.5rem;
|
|
122
|
+
border-radius: 0.375rem;
|
|
123
|
+
background-color: hsl(var(--muted));
|
|
124
|
+
font-size: 0.8125rem;
|
|
125
|
+
}
|
|
126
|
+
.chat-composer-file-name {
|
|
127
|
+
flex: 1;
|
|
128
|
+
min-width: 0;
|
|
129
|
+
overflow: hidden;
|
|
130
|
+
text-overflow: ellipsis;
|
|
131
|
+
white-space: nowrap;
|
|
132
|
+
}
|
|
133
|
+
.chat-composer-upload-status {
|
|
134
|
+
display: flex;
|
|
135
|
+
align-items: center;
|
|
136
|
+
gap: 0.375rem;
|
|
137
|
+
}
|
|
@@ -1,9 +0,0 @@
|
|
|
1
|
-
import * as React from "react";
|
|
2
|
-
import type { ChatCoreConfig, ModerationHandlers } from "../types";
|
|
3
|
-
export declare function AdminThreadChat(props: {
|
|
4
|
-
config: ChatCoreConfig;
|
|
5
|
-
onSend: (text: string) => void | Promise<void>;
|
|
6
|
-
renderThreadActions?: () => React.ReactNode;
|
|
7
|
-
handlers?: ModerationHandlers;
|
|
8
|
-
}): import("react/jsx-runtime").JSX.Element;
|
|
9
|
-
//# sourceMappingURL=AdminThreadChat.d.ts.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"AdminThreadChat.d.ts","sourceRoot":"","sources":["../../src/templates/AdminThreadChat.tsx"],"names":[],"mappings":"AAEA,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAC/B,OAAO,KAAK,EAAE,cAAc,EAAE,kBAAkB,EAAE,MAAM,UAAU,CAAC;AAGnE,wBAAgB,eAAe,CAAC,KAAK,EAAE;IACrC,MAAM,EAAE,cAAc,CAAC;IACvB,MAAM,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;IAC/C,mBAAmB,CAAC,EAAE,MAAM,KAAK,CAAC,SAAS,CAAC;IAC5C,QAAQ,CAAC,EAAE,kBAAkB,CAAC;CAC/B,2CAWA"}
|
|
@@ -1,8 +0,0 @@
|
|
|
1
|
-
"use client";
|
|
2
|
-
import { jsx as _jsx } from "react/jsx-runtime";
|
|
3
|
-
import { ChatShell } from "../ui/ChatShell";
|
|
4
|
-
export function AdminThreadChat(props) {
|
|
5
|
-
const { config, onSend, renderThreadActions, handlers } = props;
|
|
6
|
-
return (_jsx(ChatShell, { config: config, header: _jsx("div", { className: "font-medium", children: "Support" }), renderThreadActions: renderThreadActions, onSend: onSend, handlers: handlers }));
|
|
7
|
-
}
|
|
8
|
-
//# sourceMappingURL=AdminThreadChat.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"AdminThreadChat.js","sourceRoot":"","sources":["../../src/templates/AdminThreadChat.tsx"],"names":[],"mappings":"AAAA,YAAY,CAAC;;AAIb,OAAO,EAAE,SAAS,EAAE,MAAM,iBAAiB,CAAC;AAE5C,MAAM,UAAU,eAAe,CAAC,KAK/B;IACC,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,mBAAmB,EAAE,QAAQ,EAAE,GAAG,KAAK,CAAC;IAChE,OAAO,CACL,KAAC,SAAS,IACR,MAAM,EAAE,MAAM,EACd,MAAM,EAAE,cAAK,SAAS,EAAC,aAAa,wBAAc,EAClD,mBAAmB,EAAE,mBAAmB,EACxC,MAAM,EAAE,MAAM,EACd,QAAQ,EAAE,QAAQ,GAClB,CACH,CAAC;AACJ,CAAC"}
|
|
@@ -1,11 +0,0 @@
|
|
|
1
|
-
import * as React from "react";
|
|
2
|
-
import type { ChatCoreConfig, ModerationHandlers, MessageRendererRegistry } from "../types";
|
|
3
|
-
export declare function ChannelChat(props: {
|
|
4
|
-
config: ChatCoreConfig;
|
|
5
|
-
title?: string;
|
|
6
|
-
onSend: (text: string) => void | Promise<void>;
|
|
7
|
-
renderMessage?: (m: any) => React.ReactNode;
|
|
8
|
-
messageRenderers?: MessageRendererRegistry;
|
|
9
|
-
handlers?: ModerationHandlers;
|
|
10
|
-
}): import("react/jsx-runtime").JSX.Element;
|
|
11
|
-
//# sourceMappingURL=ChannelChat.d.ts.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"ChannelChat.d.ts","sourceRoot":"","sources":["../../src/templates/ChannelChat.tsx"],"names":[],"mappings":"AAEA,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAC/B,OAAO,KAAK,EAAE,cAAc,EAAE,kBAAkB,EAAE,uBAAuB,EAAE,MAAM,UAAU,CAAC;AAG5F,wBAAgB,WAAW,CAAC,KAAK,EAAE;IACjC,MAAM,EAAE,cAAc,CAAC;IACvB,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,MAAM,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;IAC/C,aAAa,CAAC,EAAE,CAAC,CAAC,EAAE,GAAG,KAAK,KAAK,CAAC,SAAS,CAAC;IAC5C,gBAAgB,CAAC,EAAE,uBAAuB,CAAC;IAC3C,QAAQ,CAAC,EAAE,kBAAkB,CAAC;CAC/B,2CAYA"}
|
|
@@ -1,8 +0,0 @@
|
|
|
1
|
-
"use client";
|
|
2
|
-
import { jsx as _jsx } from "react/jsx-runtime";
|
|
3
|
-
import { ChatShell } from "../ui/ChatShell";
|
|
4
|
-
export function ChannelChat(props) {
|
|
5
|
-
const { config, title, onSend, renderMessage, messageRenderers, handlers } = props;
|
|
6
|
-
return (_jsx(ChatShell, { config: config, header: _jsx("div", { className: "font-medium", children: title ?? "Channel" }), onSend: onSend, renderMessage: renderMessage, messageRenderers: messageRenderers, handlers: handlers }));
|
|
7
|
-
}
|
|
8
|
-
//# sourceMappingURL=ChannelChat.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"ChannelChat.js","sourceRoot":"","sources":["../../src/templates/ChannelChat.tsx"],"names":[],"mappings":"AAAA,YAAY,CAAC;;AAIb,OAAO,EAAE,SAAS,EAAE,MAAM,iBAAiB,CAAC;AAE5C,MAAM,UAAU,WAAW,CAAC,KAO3B;IACC,MAAM,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,aAAa,EAAE,gBAAgB,EAAE,QAAQ,EAAE,GAAG,KAAK,CAAC;IACnF,OAAO,CACL,KAAC,SAAS,IACR,MAAM,EAAE,MAAM,EACd,MAAM,EAAE,cAAK,SAAS,EAAC,aAAa,YAAE,KAAK,IAAI,SAAS,GAAO,EAC/D,MAAM,EAAE,MAAM,EACd,aAAa,EAAE,aAAa,EAC5B,gBAAgB,EAAE,gBAAgB,EAClC,QAAQ,EAAE,QAAQ,GAClB,CACH,CAAC;AACJ,CAAC"}
|
|
@@ -1,9 +0,0 @@
|
|
|
1
|
-
import * as React from "react";
|
|
2
|
-
import type { ChatCoreConfig, ModerationHandlers } from "../types";
|
|
3
|
-
export declare function InviteChat(props: {
|
|
4
|
-
config: ChatCoreConfig;
|
|
5
|
-
onSend: (text: string) => void | Promise<void>;
|
|
6
|
-
renderThreadActions?: () => React.ReactNode;
|
|
7
|
-
handlers?: ModerationHandlers;
|
|
8
|
-
}): import("react/jsx-runtime").JSX.Element;
|
|
9
|
-
//# sourceMappingURL=InviteChat.d.ts.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"InviteChat.d.ts","sourceRoot":"","sources":["../../src/templates/InviteChat.tsx"],"names":[],"mappings":"AAEA,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAC/B,OAAO,KAAK,EAAE,cAAc,EAAE,kBAAkB,EAAE,MAAM,UAAU,CAAC;AAGnE,wBAAgB,UAAU,CAAC,KAAK,EAAE;IAChC,MAAM,EAAE,cAAc,CAAC;IACvB,MAAM,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;IAC/C,mBAAmB,CAAC,EAAE,MAAM,KAAK,CAAC,SAAS,CAAC;IAC5C,QAAQ,CAAC,EAAE,kBAAkB,CAAC;CAC/B,2CAWA"}
|
|
@@ -1,8 +0,0 @@
|
|
|
1
|
-
"use client";
|
|
2
|
-
import { jsx as _jsx } from "react/jsx-runtime";
|
|
3
|
-
import { ChatShell } from "../ui/ChatShell";
|
|
4
|
-
export function InviteChat(props) {
|
|
5
|
-
const { config, onSend, renderThreadActions, handlers } = props;
|
|
6
|
-
return (_jsx(ChatShell, { config: config, header: _jsx("div", { className: "font-medium", children: "Invite" }), renderThreadActions: renderThreadActions, onSend: onSend, handlers: handlers }));
|
|
7
|
-
}
|
|
8
|
-
//# sourceMappingURL=InviteChat.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"InviteChat.js","sourceRoot":"","sources":["../../src/templates/InviteChat.tsx"],"names":[],"mappings":"AAAA,YAAY,CAAC;;AAIb,OAAO,EAAE,SAAS,EAAE,MAAM,iBAAiB,CAAC;AAE5C,MAAM,UAAU,UAAU,CAAC,KAK1B;IACC,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,mBAAmB,EAAE,QAAQ,EAAE,GAAG,KAAK,CAAC;IAChE,OAAO,CACL,KAAC,SAAS,IACR,MAAM,EAAE,MAAM,EACd,MAAM,EAAE,cAAK,SAAS,EAAC,aAAa,uBAAa,EACjD,mBAAmB,EAAE,mBAAmB,EACxC,MAAM,EAAE,MAAM,EACd,QAAQ,EAAE,QAAQ,GAClB,CACH,CAAC;AACJ,CAAC"}
|