summaryrefslogtreecommitdiffhomepage
path: root/src/features/tabs/tabs.ts
blob: a4db6f3f9c846ee3a40a9ae7f01247584ca438c5 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
export interface Tab {
	readonly conversationId: string;
	readonly model: string;
	readonly title: string;
}

export interface TabsState {
	readonly tabs: readonly Tab[];
	readonly activeConversationId: string | null;
}

const DEFAULT_TITLE = "New chat";
const DEFAULT_MAX_TITLE_LENGTH = 40;

export function initialState(persisted?: TabsState): TabsState {
	if (persisted !== undefined) return persisted;
	return { tabs: [], activeConversationId: null };
}

export function newDraft(state: TabsState): TabsState {
	return { ...state, activeConversationId: null };
}

export function createTab(state: TabsState, tab: Tab): TabsState {
	const exists = state.tabs.some((t) => t.conversationId === tab.conversationId);
	const tabs = exists ? state.tabs : [...state.tabs, tab];
	return { tabs, activeConversationId: tab.conversationId };
}

export function selectTab(state: TabsState, conversationId: string): TabsState {
	return { ...state, activeConversationId: conversationId };
}

export function closeTab(state: TabsState, conversationId: string): TabsState {
	const idx = state.tabs.findIndex((t) => t.conversationId === conversationId);
	if (idx === -1) return state;

	const tabs = state.tabs.filter((t) => t.conversationId !== conversationId);

	if (state.activeConversationId !== conversationId) {
		return { tabs, activeConversationId: state.activeConversationId };
	}

	if (tabs.length === 0) {
		return { tabs, activeConversationId: null };
	}

	// prefer previous tab, else next
	const neighborIdx = idx > 0 ? idx - 1 : 0;
	const neighbor = tabs[neighborIdx];
	return { tabs, activeConversationId: neighbor?.conversationId ?? null };
}

export function setModel(state: TabsState, conversationId: string, model: string): TabsState {
	const tabs = state.tabs.map((t) => (t.conversationId === conversationId ? { ...t, model } : t));
	return { tabs, activeConversationId: state.activeConversationId };
}

export function setTitle(state: TabsState, conversationId: string, title: string): TabsState {
	const tabs = state.tabs.map((t) => (t.conversationId === conversationId ? { ...t, title } : t));
	return { tabs, activeConversationId: state.activeConversationId };
}

export function activeTab(state: TabsState): Tab | null {
	if (state.activeConversationId === null) return null;
	return state.tabs.find((t) => t.conversationId === state.activeConversationId) ?? null;
}

export interface ScrollMetrics {
	readonly scrollLeft: number;
	readonly clientWidth: number;
	readonly scrollWidth: number;
}

const STUCK_EPSILON = 1;

/**
 * True when a right-pinned sticky element is floating over scrolled content — the
 * strip overflows horizontally AND is not scrolled fully to the right. When it is
 * at rest (no overflow, or scrolled to the end so it sits at its natural position)
 * this returns false. Pure: layout measurements in, boolean out.
 */
export function isStuckToEnd(m: ScrollMetrics): boolean {
	const overflows = m.scrollWidth > m.clientWidth + STUCK_EPSILON;
	const notAtEnd = m.scrollLeft + m.clientWidth < m.scrollWidth - STUCK_EPSILON;
	return overflows && notAtEnd;
}

export function deriveTitle(message: string, max: number = DEFAULT_MAX_TITLE_LENGTH): string {
	const trimmed = message.trim().replace(/\s+/g, " ");
	if (trimmed.length === 0) return DEFAULT_TITLE;
	if (trimmed.length <= max) return trimmed;
	return `${trimmed.slice(0, max)}\u2026`;
}