@rencar-dev/feature-modules-public 1.2.2 → 1.2.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.
@@ -62,18 +62,29 @@ function usePresence(options) {
62
62
  setUsers([]);
63
63
  }
64
64
  }, [enabled]);
65
+ const currentUserKey = JSON.stringify(currentUser);
65
66
  useEffect(() => {
66
67
  if (!enabled) {
67
68
  return;
68
69
  }
70
+ if (!isPresenceSocketInitialized()) {
71
+ console.warn(
72
+ "[presence] Socket not initialized. Presence features disabled."
73
+ );
74
+ return;
75
+ }
69
76
  const socket = getPresenceSocket();
77
+ let hasJoined = false;
70
78
  const joinRoom = () => {
79
+ if (hasJoined) return;
80
+ hasJoined = true;
71
81
  socket.emit("presence:join", { roomId, user: currentUser });
72
82
  };
73
83
  if (socket.connected) {
74
84
  joinRoom();
85
+ } else {
86
+ socket.once("connect", joinRoom);
75
87
  }
76
- socket.on("connect", joinRoom);
77
88
  const heartbeat = () => {
78
89
  socket.emit("presence:heartbeat", { roomId });
79
90
  };
@@ -99,8 +110,7 @@ function usePresence(options) {
99
110
  };
100
111
  }, [
101
112
  roomId,
102
- currentUser.userId,
103
- currentUser.name,
113
+ currentUserKey,
104
114
  enabled,
105
115
  heartbeatInterval,
106
116
  normalizeUsers2
@@ -109,7 +119,7 @@ function usePresence(options) {
109
119
  }
110
120
 
111
121
  // src/presence/hooks/usePresenceWatch.ts
112
- import { useEffect as useEffect2, useMemo, useRef, useState as useState2 } from "react";
122
+ import { useEffect as useEffect2, useMemo, useRef as useRef2, useState as useState2 } from "react";
113
123
  function normalizeUsers(value) {
114
124
  if (Array.isArray(value)) return value;
115
125
  if (value && typeof value === "object" && Array.isArray(value.users)) {
@@ -122,17 +132,25 @@ function usePresenceWatch(options) {
122
132
  const [roomsUsers, setRoomsUsers] = useState2(
123
133
  {}
124
134
  );
135
+ const roomIdsKey = JSON.stringify(roomIds ?? []);
125
136
  const normalizedRoomIds = useMemo(() => {
126
- const uniq = Array.from(new Set((roomIds ?? []).filter(Boolean)));
137
+ const parsed = JSON.parse(roomIdsKey);
138
+ const uniq = Array.from(new Set(parsed.filter(Boolean)));
127
139
  return uniq;
128
- }, [roomIds]);
129
- const watchedRef = useRef(/* @__PURE__ */ new Set());
140
+ }, [roomIdsKey]);
141
+ const watchedRef = useRef2(/* @__PURE__ */ new Set());
130
142
  useEffect2(() => {
131
143
  if (!enabled) {
132
144
  setRoomsUsers({});
133
145
  watchedRef.current = /* @__PURE__ */ new Set();
134
146
  return;
135
147
  }
148
+ if (!isPresenceSocketInitialized()) {
149
+ console.warn(
150
+ "[presence-watch] Socket not initialized. Watch features disabled."
151
+ );
152
+ return;
153
+ }
136
154
  const socket = getPresenceSocket();
137
155
  const toWatch = normalizedRoomIds.filter((r) => !watchedRef.current.has(r));
138
156
  const toUnwatch = Array.from(watchedRef.current).filter(
@@ -151,10 +169,6 @@ function usePresenceWatch(options) {
151
169
  return next;
152
170
  });
153
171
  }
154
- }, [enabled, normalizedRoomIds]);
155
- useEffect2(() => {
156
- if (!enabled) return;
157
- const socket = getPresenceSocket();
158
172
  const handleSnapshot = (payload) => {
159
173
  const rooms = payload?.rooms ?? {};
160
174
  setRoomsUsers((prev) => {
@@ -172,33 +186,38 @@ function usePresenceWatch(options) {
172
186
  if (!watchedRef.current.has(roomId)) return;
173
187
  setRoomsUsers((prev) => ({ ...prev, [roomId]: normalizeUsers(users) }));
174
188
  };
175
- socket.on("presence:snapshot", handleSnapshot);
176
- socket.on("presence:update", handleUpdate);
177
189
  const handleConnect = () => {
178
190
  const ids = Array.from(watchedRef.current);
179
191
  if (ids.length > 0) socket.emit("presence:watch", { roomIds: ids });
180
192
  };
193
+ socket.on("presence:snapshot", handleSnapshot);
194
+ socket.on("presence:update", handleUpdate);
181
195
  socket.on("connect", handleConnect);
182
196
  return () => {
183
197
  socket.off("presence:snapshot", handleSnapshot);
184
198
  socket.off("presence:update", handleUpdate);
185
199
  socket.off("connect", handleConnect);
186
- const ids = Array.from(watchedRef.current);
187
- if (ids.length > 0) socket.emit("presence:unwatch", { roomIds: ids });
200
+ if (normalizedRoomIds.length > 0) {
201
+ socket.emit("presence:unwatch", { roomIds: normalizedRoomIds });
202
+ }
188
203
  watchedRef.current = /* @__PURE__ */ new Set();
189
204
  };
190
- }, [enabled]);
205
+ }, [enabled, normalizedRoomIds]);
191
206
  return roomsUsers;
192
207
  }
193
208
 
194
209
  // src/presence/headless/usePresenceAvatarsState.ts
195
210
  import { useMemo as useMemo2, useState as useState3, useCallback as useCallback3 } from "react";
196
- function getInitialFromUser(user) {
211
+
212
+ // src/presence/core/utils.ts
213
+ function getInitial(user) {
197
214
  const base = (user?.name || user?.userId || "").trim();
198
215
  if (!base) return "?";
199
216
  const ch = base.charAt(0);
200
217
  return /[a-z]/i.test(ch) ? ch.toUpperCase() : ch;
201
218
  }
219
+
220
+ // src/presence/headless/usePresenceAvatarsState.ts
202
221
  function usePresenceAvatarsState(options) {
203
222
  const { users, maxVisible = 3 } = options;
204
223
  const safeUsers = Array.isArray(users) ? users : [];
@@ -210,9 +229,6 @@ function usePresenceAvatarsState(options) {
210
229
  const [hoveredIndex, setHoveredIndex] = useState3(null);
211
230
  const hoveredUser = hoveredIndex != null ? visibleUsers[hoveredIndex] : null;
212
231
  const baseTopZ = visibleUsers.length + 1;
213
- const getInitial2 = useCallback3((user) => {
214
- return getInitialFromUser(user);
215
- }, []);
216
232
  const getZIndex = useCallback3(
217
233
  (index, isHovered) => {
218
234
  return isHovered ? 100 : baseTopZ - index;
@@ -225,16 +241,16 @@ function usePresenceAvatarsState(options) {
225
241
  hoveredUser,
226
242
  hoveredIndex,
227
243
  setHoveredIndex,
228
- getInitial: getInitial2,
244
+ getInitial,
229
245
  getZIndex
230
246
  };
231
247
  }
232
248
 
233
249
  // src/presence/headless/usePresenceFloatingState.ts
234
- import { useCallback as useCallback4, useEffect as useEffect3, useMemo as useMemo3, useRef as useRef2, useState as useState4 } from "react";
250
+ import { useCallback as useCallback4, useEffect as useEffect3, useMemo as useMemo3, useRef as useRef3, useState as useState4 } from "react";
235
251
  function usePresenceFloatingState(options) {
236
252
  const { users, maxVisible = 8, initialPosition = { top: 166 } } = options;
237
- const containerRef = useRef2(null);
253
+ const containerRef = useRef3(null);
238
254
  const [dragging, setDragging] = useState4(false);
239
255
  const [hasDragged, setHasDragged] = useState4(false);
240
256
  const [dragOffset, setDragOffset] = useState4({
@@ -377,22 +393,19 @@ function usePresenceFloatingState(options) {
377
393
  };
378
394
  }
379
395
  }, [position.top, position.left, hasDragged]);
380
- const onAvatarEnter = useCallback4(
381
- (e, user) => {
382
- const container = containerRef.current;
383
- if (!container) {
384
- setHoveredUser(user);
385
- setTooltipTop(0);
386
- return;
387
- }
388
- const containerRect = container.getBoundingClientRect();
389
- const targetRect = e.currentTarget.getBoundingClientRect();
390
- const topWithinContainer = targetRect.top - containerRect.top;
396
+ const onAvatarEnter = useCallback4((e, user) => {
397
+ const container = containerRef.current;
398
+ if (!container) {
391
399
  setHoveredUser(user);
392
- setTooltipTop(topWithinContainer);
393
- },
394
- []
395
- );
400
+ setTooltipTop(0);
401
+ return;
402
+ }
403
+ const containerRect = container.getBoundingClientRect();
404
+ const targetRect = e.currentTarget.getBoundingClientRect();
405
+ const topWithinContainer = targetRect.top - containerRect.top;
406
+ setHoveredUser(user);
407
+ setTooltipTop(topWithinContainer);
408
+ }, []);
396
409
  const onAvatarLeave = useCallback4(() => {
397
410
  setHoveredUser(null);
398
411
  }, []);
@@ -521,12 +534,6 @@ function PresenceAvatars({
521
534
  // src/presence/components/PresenceFloating.tsx
522
535
  import { useCallback as useCallback6 } from "react";
523
536
  import { Fragment as Fragment2, jsx as jsx2, jsxs as jsxs2 } from "react/jsx-runtime";
524
- function getInitial(user) {
525
- const base = (user?.name || user?.userId || "").trim();
526
- if (!base) return "?";
527
- const ch = base.charAt(0);
528
- return /[a-z]/i.test(ch) ? ch.toUpperCase() : ch;
529
- }
530
537
  function PresenceFloating({
531
538
  users,
532
539
  maxVisible = 8,
@@ -636,9 +643,73 @@ function PresenceFloating({
636
643
  }
637
644
  );
638
645
  }
646
+
647
+ // src/presence/components/PresenceProvider.tsx
648
+ import { createContext, useContext, useEffect as useEffect4, useState as useState6 } from "react";
649
+ import { jsx as jsx3 } from "react/jsx-runtime";
650
+ var PresenceContext = createContext(null);
651
+ function PresenceProvider({
652
+ url,
653
+ transports,
654
+ debug = false,
655
+ children
656
+ }) {
657
+ const [status, setStatus] = useState6("disconnected");
658
+ const [isInitialized, setIsInitialized] = useState6(false);
659
+ useEffect4(() => {
660
+ initPresenceSocket({ url, transports });
661
+ setIsInitialized(true);
662
+ if (debug) {
663
+ console.log("[presence] Socket initialized with URL:", url);
664
+ }
665
+ const socket = getPresenceSocket();
666
+ const handleConnect = () => {
667
+ setStatus("connected");
668
+ if (debug) {
669
+ console.log("[presence] Connected");
670
+ }
671
+ };
672
+ const handleDisconnect = () => {
673
+ setStatus("disconnected");
674
+ if (debug) {
675
+ console.log("[presence] Disconnected");
676
+ }
677
+ };
678
+ const handleConnecting = () => {
679
+ setStatus("connecting");
680
+ };
681
+ if (socket.connected) {
682
+ setStatus("connected");
683
+ } else {
684
+ setStatus("connecting");
685
+ }
686
+ socket.on("connect", handleConnect);
687
+ socket.on("disconnect", handleDisconnect);
688
+ socket.io.on("reconnect_attempt", handleConnecting);
689
+ return () => {
690
+ socket.off("connect", handleConnect);
691
+ socket.off("disconnect", handleDisconnect);
692
+ socket.io.off("reconnect_attempt", handleConnecting);
693
+ disconnectPresenceSocket();
694
+ setIsInitialized(false);
695
+ if (debug) {
696
+ console.log("[presence] Socket disconnected and cleaned up");
697
+ }
698
+ };
699
+ }, [url, transports, debug]);
700
+ const contextValue = {
701
+ isInitialized,
702
+ status
703
+ };
704
+ return /* @__PURE__ */ jsx3(PresenceContext.Provider, { value: contextValue, children });
705
+ }
706
+ function usePresenceStatus() {
707
+ return useContext(PresenceContext);
708
+ }
639
709
  export {
640
710
  PresenceAvatars,
641
711
  PresenceFloating,
712
+ PresenceProvider,
642
713
  disconnectPresenceSocket,
643
714
  getPresenceSocket,
644
715
  initPresenceSocket,
@@ -646,6 +717,7 @@ export {
646
717
  usePresence,
647
718
  usePresenceAvatarsState,
648
719
  usePresenceFloatingState,
720
+ usePresenceStatus,
649
721
  usePresenceWatch
650
722
  };
651
723
  //# sourceMappingURL=index.js.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/presence/core/socket.ts","../../src/presence/hooks/usePresence.ts","../../src/presence/hooks/usePresenceWatch.ts","../../src/presence/headless/usePresenceAvatarsState.ts","../../src/presence/headless/usePresenceFloatingState.ts","../../src/presence/components/PresenceAvatars.tsx","../../src/presence/components/PresenceFloating.tsx"],"sourcesContent":["import { io, Socket } from 'socket.io-client';\nimport type { PresenceConfig } from './types';\n\nlet config: PresenceConfig | null = null;\nlet sharedSocket: Socket | null = null;\n\n/**\n * Initialize the presence socket with configuration\n * Must be called before using any presence hooks\n */\nexport function initPresenceSocket(presenceConfig: PresenceConfig): void {\n config = presenceConfig;\n\n // If socket already exists, disconnect and recreate\n if (sharedSocket) {\n sharedSocket.disconnect();\n sharedSocket = null;\n }\n}\n\n/**\n * Get the shared socket instance\n * @throws Error if socket is not initialized\n */\nexport function getPresenceSocket(): Socket {\n if (!config) {\n throw new Error(\n '[presence-module] Socket not initialized. Call initPresenceSocket() first.',\n );\n }\n\n if (sharedSocket && sharedSocket.connected) {\n return sharedSocket;\n }\n\n if (!sharedSocket) {\n sharedSocket = io(config.url, {\n transports: config.transports ?? ['websocket'],\n });\n } else if (!sharedSocket.connected) {\n sharedSocket.connect();\n }\n\n return sharedSocket;\n}\n\n/**\n * Disconnect and cleanup the socket connection\n */\nexport function disconnectPresenceSocket(): void {\n if (sharedSocket) {\n sharedSocket.disconnect();\n sharedSocket = null;\n }\n}\n\n/**\n * Check if socket is initialized\n */\nexport function isPresenceSocketInitialized(): boolean {\n return config !== null;\n}\n","import { useEffect, useState, useCallback } from 'react';\nimport { getPresenceSocket } from '../core/socket';\nimport type { PresenceUser, UsePresenceOptions } from '../core/types';\n\nconst DEFAULT_HEARTBEAT_INTERVAL = 10 * 60 * 1000; // 10 minutes\n\n/**\n * Hook to join a presence room and track users in that room\n *\n * @param options - Configuration options\n * @returns Array of users currently in the room\n *\n * @example\n * ```tsx\n * const users = usePresence({\n * roomId: 'my-app:page-123',\n * currentUser: { userId: '1', name: 'John' },\n * });\n * ```\n */\nexport function usePresence(options: UsePresenceOptions): PresenceUser[] {\n const {\n roomId,\n currentUser,\n enabled = true,\n heartbeatInterval = DEFAULT_HEARTBEAT_INTERVAL,\n } = options;\n\n const [users, setUsers] = useState<PresenceUser[]>([]);\n\n // Normalize users from server response\n const normalizeUsers = useCallback((incoming: unknown): PresenceUser[] => {\n if (Array.isArray(incoming)) {\n return incoming as PresenceUser[];\n }\n if (\n incoming &&\n typeof incoming === 'object' &&\n Array.isArray((incoming as { users?: unknown }).users)\n ) {\n return (incoming as { users: unknown[] }).users as PresenceUser[];\n }\n return [];\n }, []);\n\n // Clear users when disabled\n useEffect(() => {\n if (!enabled) {\n setUsers([]);\n }\n }, [enabled]);\n\n useEffect(() => {\n if (!enabled) {\n return;\n }\n\n const socket = getPresenceSocket();\n\n const joinRoom = () => {\n socket.emit('presence:join', { roomId, user: currentUser });\n };\n\n // Join immediately if connected, otherwise wait for connect event\n if (socket.connected) {\n joinRoom();\n }\n\n socket.on('connect', joinRoom);\n\n // Heartbeat to extend TTL\n const heartbeat = () => {\n socket.emit('presence:heartbeat', { roomId });\n };\n heartbeat();\n const heartbeatIntervalId = window.setInterval(\n heartbeat,\n heartbeatInterval,\n );\n\n // Handle presence updates from server\n const handlePresenceUpdate = (incoming: unknown) => {\n setUsers(normalizeUsers(incoming));\n };\n socket.on('presence:update', handlePresenceUpdate);\n\n // Notify server when tab/window closes\n const handleBeforeUnload = () => {\n socket.emit('presence:leave', { roomId, userId: currentUser.userId });\n };\n window.addEventListener('beforeunload', handleBeforeUnload);\n\n return () => {\n socket.off('connect', joinRoom);\n socket.off('presence:update', handlePresenceUpdate);\n window.removeEventListener('beforeunload', handleBeforeUnload);\n window.clearInterval(heartbeatIntervalId);\n socket.emit('presence:leave', { roomId, userId: currentUser.userId });\n };\n }, [\n roomId,\n currentUser.userId,\n currentUser.name,\n enabled,\n heartbeatInterval,\n normalizeUsers,\n ]);\n\n return users;\n}\n","import { useEffect, useMemo, useRef, useState, useCallback } from 'react';\nimport { getPresenceSocket } from '../core/socket';\nimport type { PresenceUser, UsePresenceWatchOptions } from '../core/types';\n\ntype SnapshotPayload = {\n rooms?: Record<string, unknown>;\n};\n\ntype UpdatePayload = {\n roomId?: string;\n users?: unknown;\n};\n\nfunction normalizeUsers(value: unknown): PresenceUser[] {\n if (Array.isArray(value)) return value as PresenceUser[];\n if (\n value &&\n typeof value === 'object' &&\n Array.isArray((value as { users?: unknown }).users)\n ) {\n return (value as { users: unknown[] }).users as PresenceUser[];\n }\n return [];\n}\n\n/**\n * Hook to watch multiple rooms without joining them\n * (read-only observation, current user is not added to the rooms)\n *\n * @param options - Configuration options\n * @returns Map of roomId to array of users in that room\n *\n * @example\n * ```tsx\n * const roomsUsers = usePresenceWatch({\n * roomIds: ['room-1', 'room-2', 'room-3'],\n * });\n * // roomsUsers = { 'room-1': [...], 'room-2': [...], ... }\n * ```\n */\nexport function usePresenceWatch(\n options: UsePresenceWatchOptions,\n): Record<string, PresenceUser[]> {\n const { roomIds, enabled = true } = options;\n\n const [roomsUsers, setRoomsUsers] = useState<Record<string, PresenceUser[]>>(\n {},\n );\n\n const normalizedRoomIds = useMemo(() => {\n const uniq = Array.from(new Set((roomIds ?? []).filter(Boolean)));\n return uniq;\n }, [roomIds]);\n\n const watchedRef = useRef<Set<string>>(new Set());\n\n // Handle watch/unwatch based on roomIds changes\n useEffect(() => {\n if (!enabled) {\n setRoomsUsers({});\n watchedRef.current = new Set();\n return;\n }\n\n const socket = getPresenceSocket();\n\n const toWatch = normalizedRoomIds.filter((r) => !watchedRef.current.has(r));\n const toUnwatch = Array.from(watchedRef.current).filter(\n (r) => !normalizedRoomIds.includes(r),\n );\n\n if (toWatch.length > 0) {\n socket.emit('presence:watch', { roomIds: toWatch });\n for (const r of toWatch) watchedRef.current.add(r);\n }\n\n if (toUnwatch.length > 0) {\n socket.emit('presence:unwatch', { roomIds: toUnwatch });\n for (const r of toUnwatch) watchedRef.current.delete(r);\n setRoomsUsers((prev) => {\n const next = { ...prev };\n for (const r of toUnwatch) delete next[r];\n return next;\n });\n }\n }, [enabled, normalizedRoomIds]);\n\n // Handle socket events\n useEffect(() => {\n if (!enabled) return;\n\n const socket = getPresenceSocket();\n\n const handleSnapshot = (payload: SnapshotPayload) => {\n const rooms = payload?.rooms ?? {};\n setRoomsUsers((prev) => {\n const next = { ...prev };\n for (const [roomId, users] of Object.entries(rooms)) {\n next[roomId] = normalizeUsers(users);\n }\n return next;\n });\n };\n\n const handleUpdate = (payload: unknown) => {\n if (!payload || typeof payload !== 'object') return;\n const { roomId, users } = payload as UpdatePayload;\n if (!roomId) return;\n if (!watchedRef.current.has(roomId)) return;\n setRoomsUsers((prev) => ({ ...prev, [roomId]: normalizeUsers(users) }));\n };\n\n socket.on('presence:snapshot', handleSnapshot);\n socket.on('presence:update', handleUpdate);\n\n // Re-watch on reconnect\n const handleConnect = () => {\n const ids = Array.from(watchedRef.current);\n if (ids.length > 0) socket.emit('presence:watch', { roomIds: ids });\n };\n socket.on('connect', handleConnect);\n\n return () => {\n socket.off('presence:snapshot', handleSnapshot);\n socket.off('presence:update', handleUpdate);\n socket.off('connect', handleConnect);\n\n const ids = Array.from(watchedRef.current);\n if (ids.length > 0) socket.emit('presence:unwatch', { roomIds: ids });\n watchedRef.current = new Set();\n };\n }, [enabled]);\n\n return roomsUsers;\n}\n","import { useMemo, useState, useCallback } from 'react';\nimport type {\n PresenceUser,\n UsePresenceAvatarsStateOptions,\n UsePresenceAvatarsStateReturn,\n} from '../core/types';\n\n/**\n * Extract initial character from user name/id\n */\nfunction getInitialFromUser(user: PresenceUser): string {\n const base = (user?.name || user?.userId || '').trim();\n if (!base) return '?';\n const ch = base.charAt(0);\n return /[a-z]/i.test(ch) ? ch.toUpperCase() : ch;\n}\n\n/**\n * Headless hook for presence avatars UI state management\n *\n * This hook provides all the state and logic needed to build\n * a presence avatars component without any styling.\n *\n * @param options - Configuration options\n * @returns State and handlers for building presence avatars UI\n *\n * @example\n * ```tsx\n * const {\n * visibleUsers,\n * moreCount,\n * hoveredUser,\n * setHoveredIndex,\n * getInitial,\n * getZIndex,\n * } = usePresenceAvatarsState({ users, maxVisible: 3 });\n *\n * return (\n * <div>\n * {visibleUsers.map((user, idx) => (\n * <span\n * key={user.userId}\n * style={{ zIndex: getZIndex(idx, hoveredUser?.userId === user.userId) }}\n * onMouseEnter={() => setHoveredIndex(idx)}\n * onMouseLeave={() => setHoveredIndex(null)}\n * >\n * {getInitial(user)}\n * </span>\n * ))}\n * {moreCount > 0 && <span>+{moreCount}</span>}\n * </div>\n * );\n * ```\n */\nexport function usePresenceAvatarsState(\n options: UsePresenceAvatarsStateOptions,\n): UsePresenceAvatarsStateReturn {\n const { users, maxVisible = 3 } = options;\n\n const safeUsers = Array.isArray(users) ? users : [];\n\n const visibleUsers = useMemo(\n () => safeUsers.slice(0, maxVisible),\n [safeUsers, maxVisible],\n );\n\n const moreCount = Math.max(0, safeUsers.length - visibleUsers.length);\n\n const [hoveredIndex, setHoveredIndex] = useState<number | null>(null);\n\n const hoveredUser = hoveredIndex != null ? visibleUsers[hoveredIndex] : null;\n\n const baseTopZ = visibleUsers.length + 1;\n\n const getInitial = useCallback((user: PresenceUser): string => {\n return getInitialFromUser(user);\n }, []);\n\n const getZIndex = useCallback(\n (index: number, isHovered: boolean): number => {\n return isHovered ? 100 : baseTopZ - index;\n },\n [baseTopZ],\n );\n\n return {\n visibleUsers,\n moreCount,\n hoveredUser,\n hoveredIndex,\n setHoveredIndex,\n getInitial,\n getZIndex,\n };\n}\n","import { useCallback, useEffect, useMemo, useRef, useState } from \"react\";\nimport type {\n PresenceUser,\n UsePresenceFloatingStateOptions,\n UsePresenceFloatingStateReturn,\n} from \"../core/types\";\n\n/**\n * Headless hook for presence floating UI state management\n *\n * This hook provides all the state and logic needed to build\n * a draggable floating presence panel without any styling.\n *\n * @param options - Configuration options\n * @returns State and handlers for building floating presence UI\n *\n * @example\n * ```tsx\n * const {\n * containerRef,\n * inlineStyle,\n * isDragging,\n * visibleUsers,\n * moreCount,\n * hoveredUser,\n * tooltipTop,\n * onMouseDownHeader,\n * onAvatarEnter,\n * onAvatarLeave,\n * } = usePresenceFloatingState({ users, maxVisible: 8 });\n *\n * return (\n * <div ref={containerRef} style={inlineStyle} onMouseDown={onMouseDownHeader}>\n * <div className=\"header\">열람중 {users.length}</div>\n * <div className=\"body\">\n * {visibleUsers.map((user) => (\n * <span\n * key={user.userId}\n * onMouseEnter={(e) => onAvatarEnter(e, user)}\n * onMouseLeave={onAvatarLeave}\n * >\n * {user.name}\n * </span>\n * ))}\n * </div>\n * </div>\n * );\n * ```\n */\nexport function usePresenceFloatingState(\n options: UsePresenceFloatingStateOptions\n): UsePresenceFloatingStateReturn {\n const { users, maxVisible = 8, initialPosition = { top: 166 } } = options;\n\n const containerRef = useRef<HTMLDivElement>(null);\n const [dragging, setDragging] = useState(false);\n const [hasDragged, setHasDragged] = useState(false);\n const [dragOffset, setDragOffset] = useState<{ x: number; y: number }>({\n x: 0,\n y: 0,\n });\n const [position, setPosition] = useState<{ top: number; left: number }>({\n top: initialPosition.top,\n left: 0,\n });\n const [hoveredUser, setHoveredUser] = useState<PresenceUser | null>(null);\n const [tooltipTop, setTooltipTop] = useState<number>(0);\n\n const safeUsers = Array.isArray(users) ? users : [];\n const visibleUsers = safeUsers.slice(0, maxVisible);\n const moreCount = Math.max(0, safeUsers.length - visibleUsers.length);\n\n // Position adjustment after drag (resize handling)\n useEffect(() => {\n if (!hasDragged) return;\n\n const el = containerRef.current;\n if (!el) return;\n\n const adjustPosition = () => {\n const rect = el.getBoundingClientRect();\n const maxLeft = window.innerWidth - rect.width;\n const maxTop = window.innerHeight - rect.height - 8;\n setPosition((prev) => ({\n left: Math.min(Math.max(0, prev.left), Math.max(0, maxLeft)),\n top: Math.min(Math.max(8, prev.top), Math.max(8, maxTop)),\n }));\n };\n\n const resizeObserver = new ResizeObserver(() => {\n adjustPosition();\n });\n resizeObserver.observe(el);\n\n window.addEventListener(\"resize\", adjustPosition);\n\n return () => {\n resizeObserver.disconnect();\n window.removeEventListener(\"resize\", adjustPosition);\n };\n }, [hasDragged]);\n\n // Helper to get coordinates from mouse or touch event\n const getEventCoordinates = useCallback(\n (e: MouseEvent | TouchEvent | React.MouseEvent | React.TouchEvent) => {\n if (\"touches\" in e && e.touches.length > 0) {\n return { clientX: e.touches[0].clientX, clientY: e.touches[0].clientY };\n }\n if (\"changedTouches\" in e && e.changedTouches.length > 0) {\n return {\n clientX: e.changedTouches[0].clientX,\n clientY: e.changedTouches[0].clientY,\n };\n }\n if (\"clientX\" in e) {\n return { clientX: e.clientX, clientY: e.clientY };\n }\n return { clientX: 0, clientY: 0 };\n },\n []\n );\n\n const onMouseDownHeader = useCallback(\n (e: React.MouseEvent) => {\n const el = containerRef.current;\n if (!el) return;\n const rect = el.getBoundingClientRect();\n\n // On first drag, switch to left-based positioning\n if (!hasDragged) {\n setHasDragged(true);\n setPosition({\n top: rect.top,\n left: rect.left,\n });\n }\n\n setDragOffset({ x: e.clientX - rect.left, y: e.clientY - rect.top });\n setDragging(true);\n e.preventDefault();\n },\n [hasDragged]\n );\n\n const onTouchStartHeader = useCallback(\n (e: React.TouchEvent) => {\n const el = containerRef.current;\n if (!el) return;\n const rect = el.getBoundingClientRect();\n const { clientX, clientY } = getEventCoordinates(e);\n\n // On first drag, switch to left-based positioning\n if (!hasDragged) {\n setHasDragged(true);\n setPosition({\n top: rect.top,\n left: rect.left,\n });\n }\n\n setDragOffset({ x: clientX - rect.left, y: clientY - rect.top });\n setDragging(true);\n // Note: Don't prevent default to allow scrolling detection\n },\n [hasDragged, getEventCoordinates]\n );\n\n // Handle drag movement (mouse and touch)\n useEffect(() => {\n if (!dragging) return;\n\n const el = containerRef.current;\n\n const onMove = (e: MouseEvent | TouchEvent) => {\n if (!el) return;\n const rect = el.getBoundingClientRect();\n const { clientX, clientY } = getEventCoordinates(e);\n const newLeft = clientX - dragOffset.x;\n const newTop = clientY - dragOffset.y;\n const maxLeft = window.innerWidth - rect.width;\n const maxTop = window.innerHeight - rect.height - 8;\n setPosition({\n left: Math.min(Math.max(0, newLeft), Math.max(0, maxLeft)),\n top: Math.min(Math.max(8, newTop), Math.max(8, maxTop)),\n });\n\n // Prevent scrolling while dragging on touch\n if (\"touches\" in e) {\n e.preventDefault();\n }\n };\n\n const onUp = () => setDragging(false);\n\n // Mouse events\n window.addEventListener(\"mousemove\", onMove);\n window.addEventListener(\"mouseup\", onUp, { once: true });\n\n // Touch events\n window.addEventListener(\"touchmove\", onMove, { passive: false });\n window.addEventListener(\"touchend\", onUp, { once: true });\n window.addEventListener(\"touchcancel\", onUp, { once: true });\n\n return () => {\n window.removeEventListener(\"mousemove\", onMove);\n window.removeEventListener(\"mouseup\", onUp);\n window.removeEventListener(\"touchmove\", onMove);\n window.removeEventListener(\"touchend\", onUp);\n window.removeEventListener(\"touchcancel\", onUp);\n };\n }, [dragging, dragOffset.x, dragOffset.y, getEventCoordinates]);\n\n const inlineStyle = useMemo<React.CSSProperties>(() => {\n if (hasDragged) {\n return {\n position: \"fixed\",\n top: position.top,\n left: position.left,\n right: \"auto\",\n };\n } else {\n return {\n position: \"fixed\",\n top: position.top,\n right: 0,\n left: \"auto\",\n };\n }\n }, [position.top, position.left, hasDragged]);\n\n const onAvatarEnter = useCallback(\n (e: React.MouseEvent, user: PresenceUser) => {\n const container = containerRef.current;\n if (!container) {\n setHoveredUser(user);\n setTooltipTop(0);\n return;\n }\n const containerRect = container.getBoundingClientRect();\n const targetRect = (\n e.currentTarget as HTMLElement\n ).getBoundingClientRect();\n const topWithinContainer = targetRect.top - containerRect.top;\n setHoveredUser(user);\n setTooltipTop(topWithinContainer);\n },\n []\n );\n\n const onAvatarLeave = useCallback(() => {\n setHoveredUser(null);\n }, []);\n\n return {\n containerRef,\n position,\n isDragging: dragging,\n hasDragged,\n inlineStyle,\n visibleUsers,\n moreCount,\n hoveredUser,\n tooltipTop,\n onMouseDownHeader,\n onTouchStartHeader,\n onAvatarEnter,\n onAvatarLeave,\n };\n}\n","import React, { useState, useCallback } from \"react\";\nimport { usePresenceAvatarsState } from \"../headless/usePresenceAvatarsState\";\nimport type {\n PresenceUser,\n PresenceAvatarsProps,\n PresenceAvatarRenderProps,\n PresenceAvatarsTooltipRenderProps,\n} from \"../core/types\";\n\n/**\n * PresenceAvatars - Displays a stack of user avatars with hover tooltips\n *\n * @example\n * ```tsx\n * // Basic usage\n * <PresenceAvatars users={users} />\n *\n * // Custom avatar\n * <PresenceAvatars\n * users={users}\n * renderAvatar={({ user, initial }) => (\n * <img src={user.avatarUrl} alt={user.name} />\n * )}\n * />\n *\n * // Custom tooltip\n * <PresenceAvatars\n * users={users}\n * renderTooltip={({ user, position }) => (\n * <div style={{ top: position.top, left: position.left }}>\n * {user.name}\n * </div>\n * )}\n * />\n * ```\n */\nexport function PresenceAvatars({\n users,\n maxVisible = 3,\n className,\n renderAvatar,\n renderTooltip,\n renderMore,\n}: PresenceAvatarsProps) {\n const {\n visibleUsers,\n moreCount,\n hoveredUser,\n hoveredIndex,\n setHoveredIndex,\n getInitial,\n getZIndex,\n } = usePresenceAvatarsState({ users, maxVisible });\n\n const [tooltipPos, setTooltipPos] = useState<{\n top: number;\n left: number;\n } | null>(null);\n\n const updateTooltipPos = useCallback((el: HTMLElement) => {\n const rect = el.getBoundingClientRect();\n const tooltipWidth = 200;\n const half = tooltipWidth / 2;\n const centerX = rect.left + rect.width / 2;\n const left = Math.min(\n Math.max(centerX, half + 8),\n window.innerWidth - half - 8\n );\n const top = Math.min(rect.bottom + 8, window.innerHeight - 8);\n setTooltipPos({ top, left });\n }, []);\n\n const handleMouseEnter = useCallback(\n (e: React.MouseEvent, idx: number) => {\n setHoveredIndex(idx);\n updateTooltipPos(e.currentTarget as HTMLElement);\n },\n [setHoveredIndex, updateTooltipPos]\n );\n\n const handleMouseLeave = useCallback(() => {\n setHoveredIndex(null);\n }, [setHoveredIndex]);\n\n // Default tooltip renderer\n const defaultTooltipRenderer = useCallback(\n ({ user, position }: PresenceAvatarsTooltipRenderProps) => (\n <div\n className=\"presence-avatars__tooltip\"\n style={{ top: position.top, left: position.left }}\n >\n <div className=\"presence-avatars__tooltip-row\">\n <span className=\"presence-avatars__tooltip-key\">Name</span>\n <span className=\"presence-avatars__tooltip-val\">\n {user.name || \"-\"}\n </span>\n </div>\n <div className=\"presence-avatars__tooltip-row\">\n <span className=\"presence-avatars__tooltip-key\">ID</span>\n <span className=\"presence-avatars__tooltip-val\">\n {user.userId || \"-\"}\n </span>\n </div>\n </div>\n ),\n []\n );\n\n // Default avatar renderer\n const defaultAvatarRenderer = useCallback(\n ({ initial, isHovered }: PresenceAvatarRenderProps) => <>{initial}</>,\n []\n );\n\n return (\n <div className={`presence-avatars ${className ?? \"\"}`}>\n <div className=\"presence-avatars__stack\">\n {visibleUsers.map((user, idx) => {\n const isHovered = hoveredIndex === idx;\n const initial = getInitial(user);\n const zIndex = getZIndex(idx, isHovered);\n\n const renderProps: PresenceAvatarRenderProps = {\n user,\n index: idx,\n initial,\n isHovered,\n zIndex,\n };\n\n return (\n <span\n key={`${user.userId}-${idx}`}\n className={`presence-avatars__avatar ${\n isHovered ? \"is-hovered\" : \"\"\n }`}\n style={{ zIndex }}\n onMouseEnter={(e) => handleMouseEnter(e, idx)}\n onMouseLeave={handleMouseLeave}\n aria-label={user.name || user.userId || \"\"}\n >\n {renderAvatar\n ? renderAvatar(renderProps)\n : defaultAvatarRenderer(renderProps)}\n </span>\n );\n })}\n {moreCount > 0 &&\n (renderMore ? (\n renderMore(moreCount)\n ) : (\n <span className=\"presence-avatars__more\">+{moreCount}</span>\n ))}\n </div>\n\n {hoveredUser &&\n tooltipPos &&\n (renderTooltip\n ? renderTooltip({ user: hoveredUser, position: tooltipPos })\n : defaultTooltipRenderer({\n user: hoveredUser,\n position: tooltipPos,\n }))}\n </div>\n );\n}\n\nexport default PresenceAvatars;\n","import React, { useCallback } from \"react\";\nimport { usePresenceFloatingState } from \"../headless/usePresenceFloatingState\";\nimport type {\n PresenceUser,\n PresenceFloatingProps,\n PresenceFloatingAvatarRenderProps,\n PresenceFloatingTooltipRenderProps,\n} from \"../core/types\";\n\n/**\n * Extract initial character from user name/id\n */\nfunction getInitial(user: PresenceUser): string {\n const base = (user?.name || user?.userId || \"\").trim();\n if (!base) return \"?\";\n const ch = base.charAt(0);\n return /[a-z]/i.test(ch) ? ch.toUpperCase() : ch;\n}\n\n/**\n * PresenceFloating - Draggable floating panel showing presence users\n *\n * @example\n * ```tsx\n * // Basic usage\n * <PresenceFloating users={users} />\n *\n * // Custom text\n * <PresenceFloating\n * users={users}\n * title=\"Viewing\"\n * emptyText=\"No one\"\n * moreText={(n) => `+${n} more`}\n * />\n *\n * // Custom avatar\n * <PresenceFloating\n * users={users}\n * renderAvatar={({ user, initial }) => (\n * <img src={user.avatarUrl} alt={user.name} />\n * )}\n * />\n * ```\n */\nexport function PresenceFloating({\n users,\n maxVisible = 8,\n className,\n style,\n initialPosition,\n title = \"열람중\",\n emptyText = \"없음\",\n moreText = (n) => `+${n}명`,\n renderAvatar,\n renderTooltip,\n}: PresenceFloatingProps) {\n const {\n containerRef,\n inlineStyle,\n isDragging,\n visibleUsers,\n moreCount,\n hoveredUser,\n tooltipTop,\n onMouseDownHeader,\n onTouchStartHeader,\n onAvatarEnter,\n onAvatarLeave,\n } = usePresenceFloatingState({ users, maxVisible, initialPosition });\n\n // Default tooltip renderer\n const defaultTooltipRenderer = useCallback(\n ({ user, top }: PresenceFloatingTooltipRenderProps) => (\n <div\n className=\"presence-floating__tooltip\"\n style={{ top: Math.max(0, top - 6) }}\n onMouseLeave={onAvatarLeave}\n >\n <div className=\"presence-floating__tooltip-row\">\n <span className=\"presence-floating__tooltip-key\">Name</span>\n <span className=\"presence-floating__tooltip-val\">\n {user.name || \"-\"}\n </span>\n </div>\n <div className=\"presence-floating__tooltip-row\">\n <span className=\"presence-floating__tooltip-key\">ID</span>\n <span className=\"presence-floating__tooltip-val\">\n {user.userId || \"-\"}\n </span>\n </div>\n </div>\n ),\n [onAvatarLeave]\n );\n\n // Default avatar renderer\n const defaultAvatarRenderer = useCallback(\n ({ initial }: PresenceFloatingAvatarRenderProps) => <>{initial}</>,\n []\n );\n\n const combinedStyle: React.CSSProperties = {\n ...inlineStyle,\n ...style,\n };\n\n return (\n <div\n ref={containerRef}\n className={`presence-floating ${isDragging ? \"is-dragging\" : \"\"} ${\n className ?? \"\"\n }`}\n style={combinedStyle}\n >\n <div\n className=\"presence-floating__header\"\n onMouseDown={onMouseDownHeader}\n onTouchStart={onTouchStartHeader}\n >\n <span\n className=\"presence-floating__title\"\n title={title}\n aria-label={title}\n >\n {title}\n </span>\n <span className=\"presence-floating__count\">{users.length}</span>\n </div>\n\n {hoveredUser &&\n (renderTooltip\n ? renderTooltip({ user: hoveredUser, top: tooltipTop })\n : defaultTooltipRenderer({ user: hoveredUser, top: tooltipTop }))}\n\n <div className=\"presence-floating__body\">\n {visibleUsers.length === 0 ? (\n <span className=\"presence-floating__empty\">{emptyText}</span>\n ) : (\n visibleUsers.map((user, idx) => {\n const initial = getInitial(user);\n const label = user.name || user.userId || \"\";\n\n const renderProps: PresenceFloatingAvatarRenderProps = {\n user,\n initial,\n };\n\n return (\n <span\n key={`${user.userId}-${idx}`}\n className=\"presence-floating__avatar\"\n title={label}\n aria-label={label}\n onMouseEnter={(e) => onAvatarEnter(e, user)}\n onMouseLeave={onAvatarLeave}\n >\n {renderAvatar\n ? renderAvatar(renderProps)\n : defaultAvatarRenderer(renderProps)}\n </span>\n );\n })\n )}\n {moreCount > 0 && (\n <span className=\"presence-floating__more\">{moreText(moreCount)}</span>\n )}\n </div>\n </div>\n );\n}\n\nexport default PresenceFloating;\n"],"mappings":"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","names":["normalizeUsers","useEffect","useState","useState","useEffect","useMemo","useState","useCallback","getInitial","useCallback","useEffect","useMemo","useRef","useState","useState","useCallback","getInitial","useState","useCallback","useCallback","Fragment","jsx","jsxs","useCallback"]}
1
+ {"version":3,"sources":["../../src/presence/core/socket.ts","../../src/presence/hooks/usePresence.ts","../../src/presence/hooks/usePresenceWatch.ts","../../src/presence/headless/usePresenceAvatarsState.ts","../../src/presence/core/utils.ts","../../src/presence/headless/usePresenceFloatingState.ts","../../src/presence/components/PresenceAvatars.tsx","../../src/presence/components/PresenceFloating.tsx","../../src/presence/components/PresenceProvider.tsx"],"sourcesContent":["import { io, Socket } from 'socket.io-client';\nimport type { PresenceConfig } from './types';\n\nlet config: PresenceConfig | null = null;\nlet sharedSocket: Socket | null = null;\n\n/**\n * Initialize the presence socket with configuration\n * Must be called before using any presence hooks\n */\nexport function initPresenceSocket(presenceConfig: PresenceConfig): void {\n config = presenceConfig;\n\n // If socket already exists, disconnect and recreate\n if (sharedSocket) {\n sharedSocket.disconnect();\n sharedSocket = null;\n }\n}\n\n/**\n * Get the shared socket instance\n * @throws Error if socket is not initialized\n */\nexport function getPresenceSocket(): Socket {\n if (!config) {\n throw new Error(\n '[presence-module] Socket not initialized. Call initPresenceSocket() first.',\n );\n }\n\n if (sharedSocket && sharedSocket.connected) {\n return sharedSocket;\n }\n\n if (!sharedSocket) {\n sharedSocket = io(config.url, {\n transports: config.transports ?? ['websocket'],\n });\n } else if (!sharedSocket.connected) {\n sharedSocket.connect();\n }\n\n return sharedSocket;\n}\n\n/**\n * Disconnect and cleanup the socket connection\n */\nexport function disconnectPresenceSocket(): void {\n if (sharedSocket) {\n sharedSocket.disconnect();\n sharedSocket = null;\n }\n}\n\n/**\n * Check if socket is initialized\n */\nexport function isPresenceSocketInitialized(): boolean {\n return config !== null;\n}\n","import { useEffect, useState, useCallback, useRef } from \"react\";\nimport { getPresenceSocket, isPresenceSocketInitialized } from \"../core/socket\";\nimport type { PresenceUser, UsePresenceOptions } from \"../core/types\";\n\nconst DEFAULT_HEARTBEAT_INTERVAL = 10 * 60 * 1000; // 10 minutes\n\n/**\n * Hook to join a presence room and track users in that room\n *\n * @param options - Configuration options\n * @returns Array of users currently in the room\n *\n * @example\n * ```tsx\n * const users = usePresence({\n * roomId: 'my-app:page-123',\n * currentUser: { userId: '1', name: 'John' },\n * });\n * ```\n */\nexport function usePresence(options: UsePresenceOptions): PresenceUser[] {\n const {\n roomId,\n currentUser,\n enabled = true,\n heartbeatInterval = DEFAULT_HEARTBEAT_INTERVAL,\n } = options;\n\n const [users, setUsers] = useState<PresenceUser[]>([]);\n\n // Normalize users from server response\n const normalizeUsers = useCallback((incoming: unknown): PresenceUser[] => {\n if (Array.isArray(incoming)) {\n return incoming as PresenceUser[];\n }\n if (\n incoming &&\n typeof incoming === \"object\" &&\n Array.isArray((incoming as { users?: unknown }).users)\n ) {\n return (incoming as { users: unknown[] }).users as PresenceUser[];\n }\n return [];\n }, []);\n\n // Clear users when disabled\n useEffect(() => {\n if (!enabled) {\n setUsers([]);\n }\n }, [enabled]);\n\n // Serialize currentUser to detect deep changes without infinite loops\n const currentUserKey = JSON.stringify(currentUser);\n\n useEffect(() => {\n if (!enabled) {\n return;\n }\n\n if (!isPresenceSocketInitialized()) {\n console.warn(\n \"[presence] Socket not initialized. Presence features disabled.\",\n );\n return;\n }\n\n const socket = getPresenceSocket();\n let hasJoined = false;\n\n const joinRoom = () => {\n if (hasJoined) return; // 중복 join 방지\n hasJoined = true;\n socket.emit(\"presence:join\", { roomId, user: currentUser });\n };\n\n // Join immediately if connected, otherwise wait for connect event (once)\n if (socket.connected) {\n joinRoom();\n } else {\n socket.once(\"connect\", joinRoom); // on → once로 변경하여 중복 방지\n }\n\n // Heartbeat to extend TTL\n const heartbeat = () => {\n socket.emit(\"presence:heartbeat\", { roomId });\n };\n heartbeat();\n const heartbeatIntervalId = window.setInterval(\n heartbeat,\n heartbeatInterval,\n );\n\n // Handle presence updates from server\n const handlePresenceUpdate = (incoming: unknown) => {\n setUsers(normalizeUsers(incoming));\n };\n socket.on(\"presence:update\", handlePresenceUpdate);\n\n // Notify server when tab/window closes\n const handleBeforeUnload = () => {\n socket.emit(\"presence:leave\", { roomId, userId: currentUser.userId });\n };\n window.addEventListener(\"beforeunload\", handleBeforeUnload);\n\n return () => {\n socket.off(\"connect\", joinRoom);\n socket.off(\"presence:update\", handlePresenceUpdate);\n window.removeEventListener(\"beforeunload\", handleBeforeUnload);\n window.clearInterval(heartbeatIntervalId);\n socket.emit(\"presence:leave\", { roomId, userId: currentUser.userId });\n };\n }, [\n roomId,\n currentUserKey,\n enabled,\n heartbeatInterval,\n normalizeUsers,\n ]);\n\n return users;\n}\n","import { useEffect, useMemo, useRef, useState, useCallback } from \"react\";\nimport { getPresenceSocket, isPresenceSocketInitialized } from \"../core/socket\";\nimport type { PresenceUser, UsePresenceWatchOptions } from \"../core/types\";\n\ntype SnapshotPayload = {\n rooms?: Record<string, unknown>;\n};\n\ntype UpdatePayload = {\n roomId?: string;\n users?: unknown;\n};\n\nfunction normalizeUsers(value: unknown): PresenceUser[] {\n if (Array.isArray(value)) return value as PresenceUser[];\n if (\n value &&\n typeof value === \"object\" &&\n Array.isArray((value as { users?: unknown }).users)\n ) {\n return (value as { users: unknown[] }).users as PresenceUser[];\n }\n return [];\n}\n\n/**\n * Hook to watch multiple rooms without joining them\n * (read-only observation, current user is not added to the rooms)\n *\n * @param options - Configuration options\n * @returns Map of roomId to array of users in that room\n *\n * @example\n * ```tsx\n * const roomsUsers = usePresenceWatch({\n * roomIds: ['room-1', 'room-2', 'room-3'],\n * });\n * // roomsUsers = { 'room-1': [...], 'room-2': [...], ... }\n * ```\n */\nexport function usePresenceWatch(\n options: UsePresenceWatchOptions,\n): Record<string, PresenceUser[]> {\n const { roomIds, enabled = true } = options;\n\n const [roomsUsers, setRoomsUsers] = useState<Record<string, PresenceUser[]>>(\n {},\n );\n\n // Serialize roomIds to prevent infinite effect loops when array reference changes\n const roomIdsKey = JSON.stringify(roomIds ?? []);\n const normalizedRoomIds = useMemo(() => {\n const parsed = JSON.parse(roomIdsKey) as string[];\n const uniq = Array.from(new Set(parsed.filter(Boolean)));\n return uniq;\n }, [roomIdsKey]);\n\n const watchedRef = useRef<Set<string>>(new Set());\n\n // 통합된 useEffect: watch/unwatch + socket 이벤트 처리\n useEffect(() => {\n if (!enabled) {\n setRoomsUsers({});\n watchedRef.current = new Set();\n return;\n }\n\n if (!isPresenceSocketInitialized()) {\n console.warn(\n \"[presence-watch] Socket not initialized. Watch features disabled.\",\n );\n return;\n }\n\n const socket = getPresenceSocket();\n\n // Watch/unwatch 로직\n const toWatch = normalizedRoomIds.filter((r) => !watchedRef.current.has(r));\n const toUnwatch = Array.from(watchedRef.current).filter(\n (r) => !normalizedRoomIds.includes(r),\n );\n\n if (toWatch.length > 0) {\n socket.emit(\"presence:watch\", { roomIds: toWatch });\n for (const r of toWatch) watchedRef.current.add(r);\n }\n\n if (toUnwatch.length > 0) {\n socket.emit(\"presence:unwatch\", { roomIds: toUnwatch });\n for (const r of toUnwatch) watchedRef.current.delete(r);\n setRoomsUsers((prev) => {\n const next = { ...prev };\n for (const r of toUnwatch) delete next[r];\n return next;\n });\n }\n\n // 이벤트 핸들러\n const handleSnapshot = (payload: SnapshotPayload) => {\n const rooms = payload?.rooms ?? {};\n setRoomsUsers((prev) => {\n const next = { ...prev };\n for (const [roomId, users] of Object.entries(rooms)) {\n next[roomId] = normalizeUsers(users);\n }\n return next;\n });\n };\n\n const handleUpdate = (payload: unknown) => {\n if (!payload || typeof payload !== \"object\") return;\n const { roomId, users } = payload as UpdatePayload;\n if (!roomId) return;\n if (!watchedRef.current.has(roomId)) return;\n setRoomsUsers((prev) => ({ ...prev, [roomId]: normalizeUsers(users) }));\n };\n\n // Re-watch on reconnect\n const handleConnect = () => {\n const ids = Array.from(watchedRef.current);\n if (ids.length > 0) socket.emit(\"presence:watch\", { roomIds: ids });\n };\n\n socket.on(\"presence:snapshot\", handleSnapshot);\n socket.on(\"presence:update\", handleUpdate);\n socket.on(\"connect\", handleConnect);\n\n return () => {\n socket.off(\"presence:snapshot\", handleSnapshot);\n socket.off(\"presence:update\", handleUpdate);\n socket.off(\"connect\", handleConnect);\n\n // cleanup 시 현재 normalizedRoomIds 기준으로 unwatch\n if (normalizedRoomIds.length > 0) {\n socket.emit(\"presence:unwatch\", { roomIds: normalizedRoomIds });\n }\n watchedRef.current = new Set();\n };\n }, [enabled, normalizedRoomIds]);\n\n return roomsUsers;\n}\n","import { useMemo, useState, useCallback } from \"react\";\nimport type {\n PresenceUser,\n UsePresenceAvatarsStateOptions,\n UsePresenceAvatarsStateReturn,\n} from \"../core/types\";\nimport { getInitial } from \"../core/utils\";\n\n/**\n * Headless hook for presence avatars UI state management\n *\n * This hook provides all the state and logic needed to build\n * a presence avatars component without any styling.\n *\n * @param options - Configuration options\n * @returns State and handlers for building presence avatars UI\n *\n * @example\n * ```tsx\n * const {\n * visibleUsers,\n * moreCount,\n * hoveredUser,\n * setHoveredIndex,\n * getInitial,\n * getZIndex,\n * } = usePresenceAvatarsState({ users, maxVisible: 3 });\n *\n * return (\n * <div>\n * {visibleUsers.map((user, idx) => (\n * <span\n * key={user.userId}\n * style={{ zIndex: getZIndex(idx, hoveredUser?.userId === user.userId) }}\n * onMouseEnter={() => setHoveredIndex(idx)}\n * onMouseLeave={() => setHoveredIndex(null)}\n * >\n * {getInitial(user)}\n * </span>\n * ))}\n * {moreCount > 0 && <span>+{moreCount}</span>}\n * </div>\n * );\n * ```\n */\nexport function usePresenceAvatarsState<T extends PresenceUser = PresenceUser>(\n options: UsePresenceAvatarsStateOptions<T>,\n): UsePresenceAvatarsStateReturn<T> {\n const { users, maxVisible = 3 } = options;\n\n const safeUsers = Array.isArray(users) ? users : [];\n\n const visibleUsers = useMemo(\n () => safeUsers.slice(0, maxVisible),\n [safeUsers, maxVisible],\n );\n\n const moreCount = Math.max(0, safeUsers.length - visibleUsers.length);\n\n const [hoveredIndex, setHoveredIndex] = useState<number | null>(null);\n\n const hoveredUser = hoveredIndex != null ? visibleUsers[hoveredIndex] : null;\n\n const baseTopZ = visibleUsers.length + 1;\n\n const getZIndex = useCallback(\n (index: number, isHovered: boolean): number => {\n return isHovered ? 100 : baseTopZ - index;\n },\n [baseTopZ],\n );\n\n return {\n visibleUsers,\n moreCount,\n hoveredUser,\n hoveredIndex,\n setHoveredIndex,\n getInitial,\n getZIndex,\n };\n}\n","import type { PresenceUser } from \"./types\";\n\n/**\n * Extract initial character from user name/id\n * @param user - PresenceUser object\n * @returns Single character initial (uppercase for letters)\n */\nexport function getInitial(user: PresenceUser): string {\n const base = (user?.name || user?.userId || \"\").trim();\n if (!base) return \"?\";\n const ch = base.charAt(0);\n return /[a-z]/i.test(ch) ? ch.toUpperCase() : ch;\n}\n","import { useCallback, useEffect, useMemo, useRef, useState } from \"react\";\nimport type {\n PresenceUser,\n UsePresenceFloatingStateOptions,\n UsePresenceFloatingStateReturn,\n} from \"../core/types\";\n\n/**\n * Headless hook for presence floating UI state management\n *\n * This hook provides all the state and logic needed to build\n * a draggable floating presence panel without any styling.\n *\n * @param options - Configuration options\n * @returns State and handlers for building floating presence UI\n *\n * @example\n * ```tsx\n * const {\n * containerRef,\n * inlineStyle,\n * isDragging,\n * visibleUsers,\n * moreCount,\n * hoveredUser,\n * tooltipTop,\n * onMouseDownHeader,\n * onAvatarEnter,\n * onAvatarLeave,\n * } = usePresenceFloatingState({ users, maxVisible: 8 });\n *\n * return (\n * <div ref={containerRef} style={inlineStyle} onMouseDown={onMouseDownHeader}>\n * <div className=\"header\">열람중 {users.length}</div>\n * <div className=\"body\">\n * {visibleUsers.map((user) => (\n * <span\n * key={user.userId}\n * onMouseEnter={(e) => onAvatarEnter(e, user)}\n * onMouseLeave={onAvatarLeave}\n * >\n * {user.name}\n * </span>\n * ))}\n * </div>\n * </div>\n * );\n * ```\n */\nexport function usePresenceFloatingState<T extends PresenceUser = PresenceUser>(\n options: UsePresenceFloatingStateOptions<T>,\n): UsePresenceFloatingStateReturn<T> {\n const { users, maxVisible = 8, initialPosition = { top: 166 } } = options;\n\n const containerRef = useRef<HTMLDivElement>(null);\n const [dragging, setDragging] = useState(false);\n const [hasDragged, setHasDragged] = useState(false);\n const [dragOffset, setDragOffset] = useState<{ x: number; y: number }>({\n x: 0,\n y: 0,\n });\n const [position, setPosition] = useState<{ top: number; left: number }>({\n top: initialPosition.top,\n left: 0,\n });\n const [hoveredUser, setHoveredUser] = useState<T | null>(null);\n const [tooltipTop, setTooltipTop] = useState<number>(0);\n\n const safeUsers = Array.isArray(users) ? users : [];\n const visibleUsers = safeUsers.slice(0, maxVisible);\n const moreCount = Math.max(0, safeUsers.length - visibleUsers.length);\n\n // Position adjustment after drag (resize handling)\n useEffect(() => {\n if (!hasDragged) return;\n\n const el = containerRef.current;\n if (!el) return;\n\n const adjustPosition = () => {\n const rect = el.getBoundingClientRect();\n const maxLeft = window.innerWidth - rect.width;\n const maxTop = window.innerHeight - rect.height - 8;\n setPosition((prev) => ({\n left: Math.min(Math.max(0, prev.left), Math.max(0, maxLeft)),\n top: Math.min(Math.max(8, prev.top), Math.max(8, maxTop)),\n }));\n };\n\n const resizeObserver = new ResizeObserver(() => {\n adjustPosition();\n });\n resizeObserver.observe(el);\n\n window.addEventListener(\"resize\", adjustPosition);\n\n return () => {\n resizeObserver.disconnect();\n window.removeEventListener(\"resize\", adjustPosition);\n };\n }, [hasDragged]);\n\n // Helper to get coordinates from mouse or touch event\n const getEventCoordinates = useCallback(\n (e: MouseEvent | TouchEvent | React.MouseEvent | React.TouchEvent) => {\n if (\"touches\" in e && e.touches.length > 0) {\n return { clientX: e.touches[0].clientX, clientY: e.touches[0].clientY };\n }\n if (\"changedTouches\" in e && e.changedTouches.length > 0) {\n return {\n clientX: e.changedTouches[0].clientX,\n clientY: e.changedTouches[0].clientY,\n };\n }\n if (\"clientX\" in e) {\n return { clientX: e.clientX, clientY: e.clientY };\n }\n return { clientX: 0, clientY: 0 };\n },\n [],\n );\n\n const onMouseDownHeader = useCallback(\n (e: React.MouseEvent) => {\n const el = containerRef.current;\n if (!el) return;\n const rect = el.getBoundingClientRect();\n\n // On first drag, switch to left-based positioning\n if (!hasDragged) {\n setHasDragged(true);\n setPosition({\n top: rect.top,\n left: rect.left,\n });\n }\n\n setDragOffset({ x: e.clientX - rect.left, y: e.clientY - rect.top });\n setDragging(true);\n e.preventDefault();\n },\n [hasDragged],\n );\n\n const onTouchStartHeader = useCallback(\n (e: React.TouchEvent) => {\n const el = containerRef.current;\n if (!el) return;\n const rect = el.getBoundingClientRect();\n const { clientX, clientY } = getEventCoordinates(e);\n\n // On first drag, switch to left-based positioning\n if (!hasDragged) {\n setHasDragged(true);\n setPosition({\n top: rect.top,\n left: rect.left,\n });\n }\n\n setDragOffset({ x: clientX - rect.left, y: clientY - rect.top });\n setDragging(true);\n // Note: Don't prevent default to allow scrolling detection\n },\n [hasDragged, getEventCoordinates],\n );\n\n // Handle drag movement (mouse and touch)\n useEffect(() => {\n if (!dragging) return;\n\n const el = containerRef.current;\n\n const onMove = (e: MouseEvent | TouchEvent) => {\n if (!el) return;\n const rect = el.getBoundingClientRect();\n const { clientX, clientY } = getEventCoordinates(e);\n const newLeft = clientX - dragOffset.x;\n const newTop = clientY - dragOffset.y;\n const maxLeft = window.innerWidth - rect.width;\n const maxTop = window.innerHeight - rect.height - 8;\n setPosition({\n left: Math.min(Math.max(0, newLeft), Math.max(0, maxLeft)),\n top: Math.min(Math.max(8, newTop), Math.max(8, maxTop)),\n });\n\n // Prevent scrolling while dragging on touch\n if (\"touches\" in e) {\n e.preventDefault();\n }\n };\n\n const onUp = () => setDragging(false);\n\n // Mouse events\n window.addEventListener(\"mousemove\", onMove);\n window.addEventListener(\"mouseup\", onUp, { once: true });\n\n // Touch events\n window.addEventListener(\"touchmove\", onMove, { passive: false });\n window.addEventListener(\"touchend\", onUp, { once: true });\n window.addEventListener(\"touchcancel\", onUp, { once: true });\n\n return () => {\n window.removeEventListener(\"mousemove\", onMove);\n window.removeEventListener(\"mouseup\", onUp);\n window.removeEventListener(\"touchmove\", onMove);\n window.removeEventListener(\"touchend\", onUp);\n window.removeEventListener(\"touchcancel\", onUp);\n };\n }, [dragging, dragOffset.x, dragOffset.y, getEventCoordinates]);\n\n const inlineStyle = useMemo<React.CSSProperties>(() => {\n if (hasDragged) {\n return {\n position: \"fixed\",\n top: position.top,\n left: position.left,\n right: \"auto\",\n };\n } else {\n return {\n position: \"fixed\",\n top: position.top,\n right: 0,\n left: \"auto\",\n };\n }\n }, [position.top, position.left, hasDragged]);\n\n const onAvatarEnter = useCallback((e: React.MouseEvent, user: T) => {\n const container = containerRef.current;\n if (!container) {\n setHoveredUser(user);\n setTooltipTop(0);\n return;\n }\n const containerRect = container.getBoundingClientRect();\n const targetRect = (e.currentTarget as HTMLElement).getBoundingClientRect();\n const topWithinContainer = targetRect.top - containerRect.top;\n setHoveredUser(user);\n setTooltipTop(topWithinContainer);\n }, []);\n\n const onAvatarLeave = useCallback(() => {\n setHoveredUser(null);\n }, []);\n\n return {\n containerRef,\n position,\n isDragging: dragging,\n hasDragged,\n inlineStyle,\n visibleUsers,\n moreCount,\n hoveredUser,\n tooltipTop,\n onMouseDownHeader,\n onTouchStartHeader,\n onAvatarEnter,\n onAvatarLeave,\n };\n}\n","import React, { useState, useCallback } from \"react\";\nimport { usePresenceAvatarsState } from \"../headless/usePresenceAvatarsState\";\nimport type {\n PresenceUser,\n PresenceAvatarsProps,\n PresenceAvatarRenderProps,\n PresenceAvatarsTooltipRenderProps,\n} from \"../core/types\";\n\n/**\n * PresenceAvatars - Displays a stack of user avatars with hover tooltips\n *\n * @example\n * ```tsx\n * // Basic usage\n * <PresenceAvatars users={users} />\n *\n * // Custom avatar\n * <PresenceAvatars\n * users={users}\n * renderAvatar={({ user, initial }) => (\n * <img src={user.avatarUrl} alt={user.name} />\n * )}\n * />\n *\n * // Custom tooltip\n * <PresenceAvatars\n * users={users}\n * renderTooltip={({ user, position }) => (\n * <div style={{ top: position.top, left: position.left }}>\n * {user.name}\n * </div>\n * )}\n * />\n * ```\n */\nexport function PresenceAvatars<T extends PresenceUser = PresenceUser>({\n users,\n maxVisible = 3,\n className,\n renderAvatar,\n renderTooltip,\n renderMore,\n}: PresenceAvatarsProps<T>) {\n const {\n visibleUsers,\n moreCount,\n hoveredUser,\n hoveredIndex,\n setHoveredIndex,\n getInitial,\n getZIndex,\n } = usePresenceAvatarsState<T>({ users, maxVisible });\n\n const [tooltipPos, setTooltipPos] = useState<{\n top: number;\n left: number;\n } | null>(null);\n\n const updateTooltipPos = useCallback((el: HTMLElement) => {\n const rect = el.getBoundingClientRect();\n const tooltipWidth = 200;\n const half = tooltipWidth / 2;\n const centerX = rect.left + rect.width / 2;\n const left = Math.min(\n Math.max(centerX, half + 8),\n window.innerWidth - half - 8,\n );\n const top = Math.min(rect.bottom + 8, window.innerHeight - 8);\n setTooltipPos({ top, left });\n }, []);\n\n const handleMouseEnter = useCallback(\n (e: React.MouseEvent, idx: number) => {\n setHoveredIndex(idx);\n updateTooltipPos(e.currentTarget as HTMLElement);\n },\n [setHoveredIndex, updateTooltipPos],\n );\n\n const handleMouseLeave = useCallback(() => {\n setHoveredIndex(null);\n }, [setHoveredIndex]);\n\n // Default tooltip renderer\n const defaultTooltipRenderer = useCallback(\n ({ user, position }: PresenceAvatarsTooltipRenderProps) => (\n <div\n className=\"presence-avatars__tooltip\"\n style={{ top: position.top, left: position.left }}\n >\n <div className=\"presence-avatars__tooltip-row\">\n <span className=\"presence-avatars__tooltip-key\">Name</span>\n <span className=\"presence-avatars__tooltip-val\">\n {user.name || \"-\"}\n </span>\n </div>\n <div className=\"presence-avatars__tooltip-row\">\n <span className=\"presence-avatars__tooltip-key\">ID</span>\n <span className=\"presence-avatars__tooltip-val\">\n {user.userId || \"-\"}\n </span>\n </div>\n </div>\n ),\n [],\n );\n\n // Default avatar renderer\n const defaultAvatarRenderer = useCallback(\n ({ initial, isHovered }: PresenceAvatarRenderProps) => <>{initial}</>,\n [],\n );\n\n return (\n <div className={`presence-avatars ${className ?? \"\"}`}>\n <div className=\"presence-avatars__stack\">\n {visibleUsers.map((user, idx) => {\n const isHovered = hoveredIndex === idx;\n const initial = getInitial(user);\n const zIndex = getZIndex(idx, isHovered);\n\n const renderProps: PresenceAvatarRenderProps<T> = {\n user,\n index: idx,\n initial,\n isHovered,\n zIndex,\n };\n\n return (\n <span\n key={`${user.userId}-${idx}`}\n className={`presence-avatars__avatar ${\n isHovered ? \"is-hovered\" : \"\"\n }`}\n style={{ zIndex }}\n onMouseEnter={(e) => handleMouseEnter(e, idx)}\n onMouseLeave={handleMouseLeave}\n aria-label={user.name || user.userId || \"\"}\n >\n {renderAvatar\n ? renderAvatar(renderProps)\n : defaultAvatarRenderer(renderProps)}\n </span>\n );\n })}\n {moreCount > 0 &&\n (renderMore ? (\n renderMore(moreCount)\n ) : (\n <span className=\"presence-avatars__more\">+{moreCount}</span>\n ))}\n </div>\n\n {hoveredUser &&\n tooltipPos &&\n (renderTooltip\n ? renderTooltip({ user: hoveredUser, position: tooltipPos })\n : defaultTooltipRenderer({\n user: hoveredUser,\n position: tooltipPos,\n }))}\n </div>\n );\n}\n\nexport default PresenceAvatars;\n","import React, { useCallback } from \"react\";\nimport { usePresenceFloatingState } from \"../headless/usePresenceFloatingState\";\nimport type {\n PresenceUser,\n PresenceFloatingProps,\n PresenceFloatingAvatarRenderProps,\n PresenceFloatingTooltipRenderProps,\n} from \"../core/types\";\nimport { getInitial } from \"../core/utils\";\n\n/**\n * PresenceFloating - Draggable floating panel showing presence users\n *\n * @example\n * ```tsx\n * // Basic usage\n * <PresenceFloating users={users} />\n *\n * // Custom text\n * <PresenceFloating\n * users={users}\n * title=\"Viewing\"\n * emptyText=\"No one\"\n * moreText={(n) => `+${n} more`}\n * />\n *\n * // Custom avatar\n * <PresenceFloating\n * users={users}\n * renderAvatar={({ user, initial }) => (\n * <img src={user.avatarUrl} alt={user.name} />\n * )}\n * />\n * ```\n */\nexport function PresenceFloating<T extends PresenceUser = PresenceUser>({\n users,\n maxVisible = 8,\n className,\n style,\n initialPosition,\n title = \"열람중\",\n emptyText = \"없음\",\n moreText = (n) => `+${n}명`,\n renderAvatar,\n renderTooltip,\n}: PresenceFloatingProps<T>) {\n const {\n containerRef,\n inlineStyle,\n isDragging,\n visibleUsers,\n moreCount,\n hoveredUser,\n tooltipTop,\n onMouseDownHeader,\n onTouchStartHeader,\n onAvatarEnter,\n onAvatarLeave,\n } = usePresenceFloatingState<T>({ users, maxVisible, initialPosition });\n\n // Default tooltip renderer\n const defaultTooltipRenderer = useCallback(\n ({ user, top }: PresenceFloatingTooltipRenderProps) => (\n <div\n className=\"presence-floating__tooltip\"\n style={{ top: Math.max(0, top - 6) }}\n onMouseLeave={onAvatarLeave}\n >\n <div className=\"presence-floating__tooltip-row\">\n <span className=\"presence-floating__tooltip-key\">Name</span>\n <span className=\"presence-floating__tooltip-val\">\n {user.name || \"-\"}\n </span>\n </div>\n <div className=\"presence-floating__tooltip-row\">\n <span className=\"presence-floating__tooltip-key\">ID</span>\n <span className=\"presence-floating__tooltip-val\">\n {user.userId || \"-\"}\n </span>\n </div>\n </div>\n ),\n [onAvatarLeave],\n );\n\n // Default avatar renderer\n const defaultAvatarRenderer = useCallback(\n ({ initial }: PresenceFloatingAvatarRenderProps) => <>{initial}</>,\n [],\n );\n\n const combinedStyle: React.CSSProperties = {\n ...inlineStyle,\n ...style,\n };\n\n return (\n <div\n ref={containerRef}\n className={`presence-floating ${isDragging ? \"is-dragging\" : \"\"} ${\n className ?? \"\"\n }`}\n style={combinedStyle}\n >\n <div\n className=\"presence-floating__header\"\n onMouseDown={onMouseDownHeader}\n onTouchStart={onTouchStartHeader}\n >\n <span\n className=\"presence-floating__title\"\n title={title}\n aria-label={title}\n >\n {title}\n </span>\n <span className=\"presence-floating__count\">{users.length}</span>\n </div>\n\n {hoveredUser &&\n (renderTooltip\n ? renderTooltip({ user: hoveredUser, top: tooltipTop })\n : defaultTooltipRenderer({ user: hoveredUser, top: tooltipTop }))}\n\n <div className=\"presence-floating__body\">\n {visibleUsers.length === 0 ? (\n <span className=\"presence-floating__empty\">{emptyText}</span>\n ) : (\n visibleUsers.map((user, idx) => {\n const initial = getInitial(user);\n const label = user.name || user.userId || \"\";\n\n const renderProps: PresenceFloatingAvatarRenderProps<T> = {\n user,\n initial,\n };\n\n return (\n <span\n key={`${user.userId}-${idx}`}\n className=\"presence-floating__avatar\"\n title={label}\n aria-label={label}\n onMouseEnter={(e) => onAvatarEnter(e, user)}\n onMouseLeave={onAvatarLeave}\n >\n {renderAvatar\n ? renderAvatar(renderProps)\n : defaultAvatarRenderer(renderProps)}\n </span>\n );\n })\n )}\n {moreCount > 0 && (\n <span className=\"presence-floating__more\">{moreText(moreCount)}</span>\n )}\n </div>\n </div>\n );\n}\n\nexport default PresenceFloating;\n","import React, { createContext, useContext, useEffect, useState } from \"react\";\nimport {\n initPresenceSocket,\n disconnectPresenceSocket,\n isPresenceSocketInitialized,\n getPresenceSocket,\n} from \"../core/socket\";\nimport type { PresenceProviderProps } from \"../core/types\";\n\n/**\n * Presence Context value type\n */\ntype PresenceContextValue = {\n /** Whether the socket is initialized */\n isInitialized: boolean;\n /** Current connection status */\n status: \"connecting\" | \"connected\" | \"disconnected\";\n};\n\nconst PresenceContext = createContext<PresenceContextValue | null>(null);\n\n/**\n * PresenceProvider - Manages socket initialization lifecycle\n *\n * Wrap your app with this provider to automatically initialize\n * and cleanup the presence socket connection.\n *\n * @example\n * ```tsx\n * // In your app entry point\n * import { PresenceProvider } from '@rencar-dev/feature-modules-public';\n *\n * function App() {\n * return (\n * <PresenceProvider url=\"https://socket.example.com\">\n * <MyApp />\n * </PresenceProvider>\n * );\n * }\n *\n * // In child components, hooks work automatically\n * function MyComponent() {\n * const users = usePresence({ roomId: 'room-1', currentUser });\n * return <PresenceFloating users={users} />;\n * }\n * ```\n */\nexport function PresenceProvider({\n url,\n transports,\n debug = false,\n children,\n}: PresenceProviderProps) {\n const [status, setStatus] =\n useState<PresenceContextValue[\"status\"]>(\"disconnected\");\n const [isInitialized, setIsInitialized] = useState(false);\n\n useEffect(() => {\n // Initialize socket\n initPresenceSocket({ url, transports });\n setIsInitialized(true);\n\n if (debug) {\n console.log(\"[presence] Socket initialized with URL:\", url);\n }\n\n // Get socket and setup connection listeners\n const socket = getPresenceSocket();\n\n const handleConnect = () => {\n setStatus(\"connected\");\n if (debug) {\n console.log(\"[presence] Connected\");\n }\n };\n\n const handleDisconnect = () => {\n setStatus(\"disconnected\");\n if (debug) {\n console.log(\"[presence] Disconnected\");\n }\n };\n\n const handleConnecting = () => {\n setStatus(\"connecting\");\n };\n\n // Set initial status\n if (socket.connected) {\n setStatus(\"connected\");\n } else {\n setStatus(\"connecting\");\n }\n\n socket.on(\"connect\", handleConnect);\n socket.on(\"disconnect\", handleDisconnect);\n socket.io.on(\"reconnect_attempt\", handleConnecting);\n\n return () => {\n socket.off(\"connect\", handleConnect);\n socket.off(\"disconnect\", handleDisconnect);\n socket.io.off(\"reconnect_attempt\", handleConnecting);\n\n disconnectPresenceSocket();\n setIsInitialized(false);\n\n if (debug) {\n console.log(\"[presence] Socket disconnected and cleaned up\");\n }\n };\n }, [url, transports, debug]);\n\n const contextValue: PresenceContextValue = {\n isInitialized,\n status,\n };\n\n return (\n <PresenceContext.Provider value={contextValue}>\n {children}\n </PresenceContext.Provider>\n );\n}\n\n/**\n * Hook to access presence connection status\n *\n * @returns Connection status object or null if not inside PresenceProvider\n *\n * @example\n * ```tsx\n * function StatusIndicator() {\n * const presence = usePresenceStatus();\n *\n * if (!presence) {\n * return <span>Not initialized</span>;\n * }\n *\n * return (\n * <span>\n * Status: {presence.status}\n * {presence.status === 'connecting' && <Spinner />}\n * </span>\n * );\n * }\n * ```\n */\nexport function usePresenceStatus(): PresenceContextValue | null {\n return useContext(PresenceContext);\n}\n\nexport default PresenceProvider;\n"],"mappings":"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","names":["normalizeUsers","useEffect","useRef","useState","useState","useRef","useEffect","useMemo","useState","useCallback","useMemo","useState","useCallback","useCallback","useEffect","useMemo","useRef","useState","useState","useCallback","getInitial","useState","useCallback","useCallback","Fragment","jsx","jsxs","useCallback","useEffect","useState","jsx","useState","useEffect"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@rencar-dev/feature-modules-public",
3
- "version": "1.2.2",
3
+ "version": "1.2.3",
4
4
  "description": "Rencar feature modules - hooks, utils, and UI components",
5
5
  "type": "module",
6
6
  "exports": {
@@ -36,7 +36,7 @@
36
36
  },
37
37
  "repository": {
38
38
  "type": "git",
39
- "url": "git+https://github.com/Rencar-dev/feature-modules-public.git"
39
+ "url": "git+https://github.com/Rencar-dev/fe-toolkit.git"
40
40
  },
41
41
  "publishConfig": {
42
42
  "registry": "https://registry.npmjs.org",