Skip to content

Commit c1401d4

Browse files
committed
Load more rooms if needed
1 parent 1d34114 commit c1401d4

File tree

5 files changed

+210
-64
lines changed

5 files changed

+210
-64
lines changed

packages/host/app/components/ai-assistant/past-session-item.gts

+6-1
Original file line numberDiff line numberDiff line change
@@ -209,7 +209,12 @@ export default class PastSessionItem extends Component<Signature> {
209209
if (!this.args.session.lastMessage) {
210210
return false;
211211
}
212-
return !this.args.session.lastMessage.isStreamingFinished;
212+
213+
return (
214+
this.args.session.lastMessage.author.userId !==
215+
this.matrixService.userId &&
216+
!this.args.session.lastMessage.isStreamingFinished
217+
);
213218
}
214219

215220
get hasUnseenMessage() {
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,18 @@
1-
import type { TemplateOnlyComponent } from '@ember/component/template-only';
21
import { on } from '@ember/modifier';
2+
import { service } from '@ember/service';
3+
import Component from '@glimmer/component';
34

4-
import { IconButton } from '@cardstack/boxel-ui/components';
5+
import { modifier } from 'ember-modifier';
6+
7+
import { IconButton, LoadingIndicator } from '@cardstack/boxel-ui/components';
58
import { DropdownArrowFilled } from '@cardstack/boxel-ui/icons';
69

710
import { SessionRoomData } from './panel';
811
import AiAssistantPanelPopover from './panel-popover';
912
import PastSessionItem, { type RoomActions } from './past-session-item';
1013

14+
import type MatrixService from '../../services/matrix-service';
15+
1116
interface Signature {
1217
Args: {
1318
sessions: SessionRoomData[];
@@ -17,51 +22,93 @@ interface Signature {
1722
Element: HTMLElement;
1823
}
1924

20-
const AiAssistantPastSessionsList: TemplateOnlyComponent<Signature> = <template>
21-
<AiAssistantPanelPopover
22-
@onClose={{@onClose}}
23-
data-test-past-sessions
24-
...attributes
25-
>
26-
<:header>
27-
All Sessions
28-
<IconButton
29-
@icon={{DropdownArrowFilled}}
30-
@width='12px'
31-
@height='12px'
32-
{{on 'click' @onClose}}
33-
aria-label='Close Past Sessions'
34-
data-test-close-past-sessions
35-
/>
36-
</:header>
37-
<:body>
38-
{{#if @sessions}}
39-
<ul class='past-sessions'>
40-
{{#each @sessions key='roomId' as |session|}}
41-
<PastSessionItem @session={{session}} @actions={{@roomActions}} />
42-
{{/each}}
43-
</ul>
44-
{{else}}
45-
<div class='empty-collection'>
46-
No past sessions to show.
47-
</div>
48-
{{/if}}
49-
</:body>
50-
</AiAssistantPanelPopover>
25+
export default class AiAssistantPastSessionsList extends Component<Signature> {
26+
@service declare matrixService: MatrixService;
27+
28+
checkScroll = modifier((element: HTMLElement) => {
29+
let checkScrollPosition = () => {
30+
let { scrollHeight, scrollTop, clientHeight } = element;
31+
let isAtBottom = Math.abs(scrollHeight - scrollTop - clientHeight) < 1;
32+
let hasNoScroll = scrollHeight <= clientHeight;
33+
34+
if (isAtBottom || hasNoScroll) {
35+
this.matrixService.loadMoreAIRooms();
36+
}
37+
};
5138

52-
<style scoped>
53-
.past-sessions {
54-
list-style-type: none;
55-
padding: 0;
56-
margin: 0;
57-
margin-bottom: var(--boxel-sp-xs);
58-
}
59-
.empty-collection {
60-
padding: var(--boxel-sp-sm);
61-
text-align: center;
62-
color: var(--boxel-450);
63-
}
64-
</style>
65-
</template>;
39+
checkScrollPosition();
6640

67-
export default AiAssistantPastSessionsList;
41+
element.addEventListener('scroll', checkScrollPosition);
42+
43+
return () => {
44+
element.removeEventListener('scroll', checkScrollPosition);
45+
};
46+
});
47+
48+
<template>
49+
<AiAssistantPanelPopover
50+
@onClose={{@onClose}}
51+
data-test-past-sessions
52+
...attributes
53+
>
54+
<:header>
55+
All Sessions
56+
<IconButton
57+
@icon={{DropdownArrowFilled}}
58+
@width='12px'
59+
@height='12px'
60+
{{on 'click' @onClose}}
61+
aria-label='Close Past Sessions'
62+
data-test-close-past-sessions
63+
/>
64+
</:header>
65+
<:body>
66+
{{#if @sessions}}
67+
<ul class='past-sessions' {{this.checkScroll}}>
68+
{{#each @sessions key='roomId' as |session|}}
69+
<PastSessionItem @session={{session}} @actions={{@roomActions}} />
70+
{{/each}}
71+
{{#if this.matrixService.loadingMoreAIRooms}}
72+
<li class='loading-indicator-container'>
73+
<LoadingIndicator
74+
@color='var(--boxel-dark)'
75+
class='loading-indicator'
76+
/>
77+
</li>
78+
{{/if}}
79+
</ul>
80+
{{else}}
81+
<div class='empty-collection'>
82+
No past sessions to show.
83+
</div>
84+
{{/if}}
85+
</:body>
86+
</AiAssistantPanelPopover>
87+
88+
<style scoped>
89+
.past-sessions {
90+
list-style-type: none;
91+
padding: 0;
92+
margin: 0;
93+
margin-bottom: var(--boxel-sp-xs);
94+
max-height: 400px;
95+
overflow-y: auto;
96+
}
97+
.empty-collection {
98+
padding: var(--boxel-sp-sm);
99+
text-align: center;
100+
color: var(--boxel-450);
101+
}
102+
.loading-indicator-container {
103+
padding: var(--boxel-sp-sm);
104+
text-align: center;
105+
display: flex;
106+
justify-content: center;
107+
align-items: center;
108+
}
109+
.loading-indicator {
110+
margin: 0 auto;
111+
}
112+
</style>
113+
</template>
114+
}

packages/host/app/services/matrix-service.ts

+103-9
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,13 @@ import {
1515
type EmittedEvents,
1616
type ISendEventResponse,
1717
} from 'matrix-js-sdk';
18-
import { SlidingSync, MSC3575List } from 'matrix-js-sdk/lib/sliding-sync';
18+
import {
19+
SlidingSync,
20+
MSC3575List,
21+
SlidingSyncEvent,
22+
SlidingSyncState,
23+
MSC3575SlidingSyncResponse,
24+
} from 'matrix-js-sdk/lib/sliding-sync';
1925
import stringify from 'safe-stable-stringify';
2026
import { md5 } from 'super-fast-md5';
2127
import { TrackedMap } from 'tracked-built-ins';
@@ -114,7 +120,7 @@ import type ResetService from './reset';
114120

115121
import type * as MatrixSDK from 'matrix-js-sdk';
116122

117-
const { matrixURL } = ENV;
123+
const { matrixURL, environment } = ENV;
118124
const MAX_CARD_SIZE_KB = 60;
119125
const STATE_EVENTS_OF_INTEREST = ['m.room.create', 'm.room.name'];
120126
const SLIDING_SYNC_AI_ROOM_LIST_NAME = 'ai-room';
@@ -173,7 +179,10 @@ export default class MatrixService extends Service {
173179
new TrackedMap();
174180
private cardHashes: Map<string, string> = new Map(); // hashes <> event id
175181
private skillCardHashes: Map<string, string> = new Map(); // hashes <> event id
182+
176183
private slidingSync: SlidingSync | undefined;
184+
private aiRoomIds: Set<string> = new Set();
185+
@tracked private isLoadingMoreAIRooms = false;
177186

178187
constructor(owner: Owner) {
179188
super(owner);
@@ -191,7 +200,7 @@ export default class MatrixService extends Service {
191200
set currentRoomId(value: string | undefined) {
192201
this._currentRoomId = value;
193202
if (value) {
194-
this.loadAllTimelineEvents.perform(value);
203+
this.loadAllTimelineEvents(value);
195204
window.localStorage.setItem(CurrentRoomIdPersistenceKey, value);
196205
} else {
197206
window.localStorage.removeItem(CurrentRoomIdPersistenceKey);
@@ -243,6 +252,7 @@ export default class MatrixService extends Service {
243252
e.event.content.realms,
244253
);
245254
await this.loginToRealms();
255+
await this.maybeLoadMoreAuthRooms(e.event.content.realms);
246256
}
247257
},
248258
],
@@ -486,7 +496,7 @@ export default class MatrixService extends Service {
486496
this.bindEventListeners();
487497

488498
try {
489-
this.initSlidingSync();
499+
await this.initSlidingSync();
490500
await this.client.startClient({ slidingSync: this.slidingSync });
491501
let accountDataContent = await this.client.getAccountDataFromServer<{
492502
realms: string[];
@@ -510,7 +520,11 @@ export default class MatrixService extends Service {
510520
}
511521
}
512522

513-
private initSlidingSync() {
523+
private async initSlidingSync() {
524+
let accountData = await this.client.getAccountDataFromServer<{
525+
realms: string[];
526+
}>(APP_BOXEL_REALMS_EVENT_TYPE);
527+
514528
let lists: Map<string, MSC3575List> = new Map();
515529
lists.set(SLIDING_SYNC_AI_ROOM_LIST_NAME, {
516530
ranges: [[0, SLIDING_SYNC_LIST_RANGE_END]],
@@ -521,7 +535,14 @@ export default class MatrixService extends Service {
521535
required_state: [['*', '*']],
522536
});
523537
lists.set(SLIDING_SYNC_AUTH_ROOM_LIST_NAME, {
524-
ranges: [[0, SLIDING_SYNC_LIST_RANGE_END]],
538+
ranges: [
539+
[
540+
0,
541+
accountData
542+
? accountData?.realms.length
543+
: SLIDING_SYNC_LIST_RANGE_END,
544+
],
545+
],
525546
filters: {
526547
is_dm: true,
527548
},
@@ -535,7 +556,25 @@ export default class MatrixService extends Service {
535556
timeline_limit: SLIDING_SYNC_LIST_TIMELINE_LIMIT,
536557
},
537558
this.client as any,
538-
3000,
559+
environment === 'test' ? 200 : 30000,
560+
);
561+
this.slidingSync.on(
562+
SlidingSyncEvent.Lifecycle,
563+
(
564+
state: SlidingSyncState | null,
565+
resp: MSC3575SlidingSyncResponse | null,
566+
) => {
567+
if (
568+
state === SlidingSyncState.Complete &&
569+
resp &&
570+
resp.lists[SLIDING_SYNC_AI_ROOM_LIST_NAME].ops?.[0]?.op === 'SYNC'
571+
) {
572+
for (let roomId of resp.lists[SLIDING_SYNC_AI_ROOM_LIST_NAME].ops[0]
573+
.room_ids) {
574+
this.aiRoomIds.add(roomId);
575+
}
576+
}
577+
},
539578
);
540579

541580
return this.slidingSync;
@@ -1174,7 +1213,7 @@ export default class MatrixService extends Service {
11741213
return await this.client.isUsernameAvailable(username);
11751214
}
11761215

1177-
private loadAllTimelineEvents = restartableTask(async (roomId: string) => {
1216+
private async loadAllTimelineEvents(roomId: string) {
11781217
let roomData = this.ensureRoomData(roomId);
11791218
let room = this.client.getRoom(roomId);
11801219

@@ -1214,7 +1253,7 @@ export default class MatrixService extends Service {
12141253
this.timelineLoadingState.set(roomId, false);
12151254
waiter.endAsync(token);
12161255
}
1217-
});
1256+
}
12181257

12191258
get isLoadingTimeline() {
12201259
if (!this.currentRoomId) {
@@ -1597,6 +1636,61 @@ export default class MatrixService extends Service {
15971636
await roomResource.loading;
15981637
roomResource.activateLLM(model);
15991638
}
1639+
1640+
loadMoreAIRooms() {
1641+
this.loadMoreAIRoomsTask.perform();
1642+
}
1643+
1644+
private loadMoreAIRoomsTask = restartableTask(async () => {
1645+
if (!this.slidingSync) return;
1646+
1647+
let currentList = this.slidingSync.getListParams(
1648+
SLIDING_SYNC_AI_ROOM_LIST_NAME,
1649+
);
1650+
if (!currentList) return;
1651+
1652+
let currentRange = currentList.ranges[0];
1653+
if (!currentRange) return;
1654+
1655+
if (this.aiRoomIds.size < currentRange[1] - 1) {
1656+
return;
1657+
}
1658+
1659+
let newEndRange = currentRange[1] + 10;
1660+
1661+
this.isLoadingMoreAIRooms = true;
1662+
try {
1663+
await this.slidingSync.setListRanges(SLIDING_SYNC_AI_ROOM_LIST_NAME, [
1664+
[0, newEndRange],
1665+
]);
1666+
} finally {
1667+
this.isLoadingMoreAIRooms = false;
1668+
}
1669+
});
1670+
1671+
get loadingMoreAIRooms() {
1672+
return this.isLoadingMoreAIRooms;
1673+
}
1674+
1675+
async maybeLoadMoreAuthRooms(realms: string[]) {
1676+
if (!this.slidingSync) return;
1677+
1678+
let currentList = this.slidingSync.getListParams(
1679+
SLIDING_SYNC_AUTH_ROOM_LIST_NAME,
1680+
);
1681+
if (!currentList) return;
1682+
1683+
let currentRange = currentList.ranges[0];
1684+
if (!currentRange) return;
1685+
if (realms.length - 1 <= currentRange[1]) {
1686+
return;
1687+
}
1688+
1689+
let newEndRange = realms.length - 1;
1690+
await this.slidingSync.setListRanges(SLIDING_SYNC_AUTH_ROOM_LIST_NAME, [
1691+
[0, newEndRange],
1692+
]);
1693+
}
16001694
}
16011695

16021696
function saveAuth(auth: LoginResponse) {

patches/matrix-js-sdk@31.0.0.patch

2.08 KB
Binary file not shown.

0 commit comments

Comments
 (0)