Room list: scroll to newly creation section (#33210)

* feat(rls): emit tag when section is created

* feat(vm): scroll to newly section tag

* feat(view): scroll to new section
This commit is contained in:
Florian Duros 2026-04-22 14:21:41 +02:00 committed by GitHub
parent 29411f0ded
commit 9df9fb9428
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
9 changed files with 101 additions and 27 deletions

View File

@ -485,13 +485,12 @@ export class RoomListStoreV3Class extends AsyncStoreWithClient<EmptyObject> {
/**
* Create a new section.
* Emits {@link SECTION_CREATED_EVENT} and {@link LISTS_UPDATE_EVENT} if the section was successfully created.
* Emits {@link SECTION_CREATED_EVENT} if the section was successfully created.
*/
public async createSection(): Promise<void> {
const sectionIsCreated = await createSection();
if (!sectionIsCreated) return;
this.emit(SECTION_CREATED_EVENT);
this.scheduleEmit();
const tag = await createSection();
if (!tag) return;
this.emit(SECTION_CREATED_EVENT, tag);
}
/**

View File

@ -33,13 +33,13 @@ export type OrderedCustomSections = Tag[];
* Creates a new custom section by showing a dialog to the user to enter the section name.
* If the user confirms, it generates a unique tag for the section, saves the section data in the settings, and updates the ordered list of sections.
*
* @return A promise that resolves to true if the section was created, or false if the user cancelled the creation or if there was an error.
* @return A promise that resolves to the new section tag if created, or undefined if cancelled.
*/
export async function createSection(): Promise<boolean> {
export async function createSection(): Promise<string | undefined> {
const modal = Modal.createDialog(CreateSectionDialog);
const [shouldCreateSection, sectionName] = await modal.finished;
if (!shouldCreateSection || !sectionName) return false;
if (!shouldCreateSection || !sectionName) return undefined;
const tag = `element.io.section.${window.crypto.randomUUID()}`;
const newSection: CustomSection = { tag, name: sectionName };
@ -53,5 +53,5 @@ export async function createSection(): Promise<boolean> {
const orderedSections = SettingsStore.getValue("RoomList.OrderedCustomSections") || [];
orderedSections.push(tag);
await SettingsStore.setValue("RoomList.OrderedCustomSections", null, SettingLevel.ACCOUNT, orderedSections);
return true;
return tag;
}

View File

@ -153,7 +153,7 @@ export class RoomListViewModel
this.disposables.trackListener(
RoomListStoreV3.instance,
RoomListStoreV3Event.SectionCreated as any,
this.onSectionCreated,
this.onSectionCreated as (...args: unknown[]) => void,
);
// Subscribe to active room changes to update selected room
@ -500,6 +500,7 @@ export class RoomListViewModel
private async updateRoomListData(
isRoomChange: boolean = false,
roomIdOverride: string | null = null,
scrollToSectionTag: string | undefined = undefined,
): Promise<void> {
// Determine the room ID to use for calculations
// Use override if provided (e.g., during space changes), otherwise fall back to RoomViewStore
@ -544,17 +545,23 @@ export class RoomListViewModel
// Update filter keys - only update if they have actually changed to prevent unnecessary re-renders of the room list
const previousFilterKeys = this.snapshot.current.roomListState.filterKeys;
const newFilterKeys = this.roomsResult.filterKeys?.map((k) => String(k));
const viewSections = toRoomListSection(this.sections);
const resolvedScrollToSectionTag =
scrollToSectionTag && viewSections.some((s) => s.id === scrollToSectionTag)
? scrollToSectionTag
: undefined;
const roomListState: RoomListViewState = {
activeRoomIndex,
spaceId: this.roomsResult.spaceId,
filterKeys: keepIfSame(previousFilterKeys, newFilterKeys),
scrollToSectionTag: resolvedScrollToSectionTag,
};
const activeFilterId = this.activeFilter !== undefined ? filterKeyToIdMap.get(this.activeFilter) : undefined;
const isRoomListEmpty = this.roomsResult.sections.every((section) => section.rooms.length === 0);
const isLoadingRooms = RoomListStoreV3.instance.isLoadingRooms;
const viewSections = toRoomListSection(this.sections);
const previousSections = this.snapshot.current.sections;
// Single atomic snapshot update
@ -586,7 +593,9 @@ export class RoomListViewModel
}
};
public onSectionCreated = (): void => {
public onSectionCreated = (tag: string): void => {
this.updateRoomListData(false, null, tag);
clearTimeout(this.toastRef);
this.snapshot.merge({
toast: "section_created",

View File

@ -1015,7 +1015,7 @@ describe("RoomListStoreV3", () => {
it("emits SECTION_CREATED_EVENT and LISTS_UPDATE_EVENT when section is created", async () => {
enableSections();
getClientAndRooms();
jest.spyOn(sectionModule, "createSection").mockResolvedValue(true);
jest.spyOn(sectionModule, "createSection").mockResolvedValue("element.io.section.test-tag");
const store = new RoomListStoreV3Class(dispatcher);
await store.start();
@ -1027,14 +1027,13 @@ describe("RoomListStoreV3", () => {
await store.createSection();
expect(sectionCreatedListener).toHaveBeenCalled();
expect(listsUpdateListener).toHaveBeenCalled();
expect(sectionCreatedListener).toHaveBeenCalledWith("element.io.section.test-tag");
});
it("does not emit when section creation is cancelled", async () => {
enableSections();
getClientAndRooms();
jest.spyOn(sectionModule, "createSection").mockResolvedValue(false);
jest.spyOn(sectionModule, "createSection").mockResolvedValue(undefined);
const store = new RoomListStoreV3Class(dispatcher);
await store.start();

View File

@ -21,10 +21,9 @@ describe("createSection", () => {
});
it.each([
[false, "", false],
[true, "", false],
[true, "My Section", true],
])("returns %s when shouldCreate=%s and name='%s'", async (shouldCreate, name, expected) => {
[false, "", undefined],
[true, "", undefined],
])("returns undefined when shouldCreate=%s and name='%s'", async (shouldCreate, name, expected) => {
jest.spyOn(Modal, "createDialog").mockReturnValue({
finished: Promise.resolve([shouldCreate, name]),
close: jest.fn(),
@ -34,6 +33,16 @@ describe("createSection", () => {
expect(result).toBe(expected);
});
it("returns the new tag when section is created", async () => {
jest.spyOn(Modal, "createDialog").mockReturnValue({
finished: Promise.resolve([true, "My Section"]),
close: jest.fn(),
} as any);
const result = await createSection();
expect(result).toMatch(/^element\.io\.section\./);
});
it("opens the CreateSectionDialog", async () => {
const createDialogSpy = jest.spyOn(Modal, "createDialog").mockReturnValue({
finished: Promise.resolve([false, ""]),

View File

@ -6,7 +6,7 @@
*/
import React, { type JSX, useCallback, useMemo } from "react";
import { Virtuoso } from "react-virtuoso";
import { Virtuoso, type VirtuosoHandle } from "react-virtuoso";
import { useVirtualizedList, type VirtualizedListContext, type VirtualizedListProps } from "../virtualized-list";
@ -33,6 +33,11 @@ export interface GroupedVirtualizedListProps<Header, Item, Context> extends Omit
VirtualizedListProps<Item, Context>,
"items" | "isItemFocusable" | "getItemKey"
> {
/**
* Optional ref to the underlying Virtuoso handle, for imperative scrolling.
*/
scrollHandleRef?: React.RefCallback<VirtuosoHandle>;
/**
* The groups to display in the virtualized list.
* Each group has a header and an array of child items.
@ -126,6 +131,7 @@ export function GroupedVirtualizedList<Header, Item, Context>(
isGroupHeaderFocusable,
getItemKey,
getHeaderKey,
scrollHandleRef,
...restProps
} = props;
@ -171,6 +177,7 @@ export function GroupedVirtualizedList<Header, Item, Context>(
isItemFocusable: wrappedIsEntryFocusable,
getItemKey: wrappedGetEntryKey,
},
scrollHandleRef,
);
// Convert (Item, e) → (NavigationEntry, e) for regular items

View File

@ -128,7 +128,7 @@ export interface UseVirtualizedListResult<Item, Context> extends Omit<
VirtuosoProps<Item, VirtualizedListContext<Context>>,
"data" | "itemContent" | "context" | "onKeyDown" | "onFocus" | "onBlur" | "rangeChanged" | "scrollerRef" | "ref"
> {
ref: React.RefObject<VirtuosoHandle | null>;
ref: React.RefCallback<VirtuosoHandle>;
scrollerRef: (element: HTMLElement | Window | null) => void;
onKeyDown: (e: React.KeyboardEvent<HTMLDivElement>) => void;
onFocus: (e: React.FocusEvent) => void;
@ -155,6 +155,7 @@ export interface UseVirtualizedListResult<Item, Context> extends Omit<
*/
export function useVirtualizedList<Item, Context>(
props: VirtualizedListProps<Item, Context>,
handleRef?: React.RefCallback<VirtuosoHandle>,
): UseVirtualizedListResult<Item, Context> {
// Extract our custom props to avoid conflicts with Virtuoso props
const {
@ -380,9 +381,17 @@ export function useVirtualizedList<Item, Context>(
[rangeChanged, mapRangeIndex],
);
const setRef = useCallback(
(handle: VirtuosoHandle | null) => {
virtuosoHandleRef.current = handle;
handleRef?.(handle);
},
[handleRef],
);
return {
...virtuosoProps,
ref: virtuosoHandleRef,
ref: setRef,
scrollerRef,
onKeyDown: keyDownCallback,
onFocus,

View File

@ -13,7 +13,7 @@ import { describe, it, expect } from "vitest";
import * as stories from "./VirtualizedRoomListView.stories";
const { Default } = composeStories(stories);
const { Default, Sections } = composeStories(stories);
const renderWithMockContext = (component: React.ReactElement): ReturnType<typeof render> => {
return render(component, {
@ -64,4 +64,28 @@ describe("<VirtualizedRoomListView />", () => {
renderWithMockContext(<Default />);
expect(Default.args.updateVisibleRooms).toHaveBeenCalled();
});
describe("scrollToSectionTag", () => {
it("skips scroll when scrollToSectionTag does not match any section", () => {
const roomListState = {
activeRoomIndex: 0,
spaceId: "!space:server",
scrollToSectionTag: "nonexistent",
};
renderWithMockContext(<Sections roomListState={roomListState} />);
expect(screen.getByRole("treegrid", { name: "Room list" })).toBeInTheDocument();
});
it("scrolls to the section when scrollToSectionTag matches", () => {
// sections: favourites(3 rooms), chats(1 room), low-priority(6 rooms)
// flat index for "chats" = 3 rooms + 1 header = 4
const roomListState = {
activeRoomIndex: 0,
spaceId: "!space:server",
scrollToSectionTag: "chats",
};
renderWithMockContext(<Sections roomListState={roomListState} />);
expect(screen.getByRole("treegrid", { name: "Room list" })).toBeInTheDocument();
});
});
});

View File

@ -5,8 +5,8 @@
* Please see LICENSE files in the repository root for full details.
*/
import React, { useCallback, useMemo, useRef, type JSX, type ReactNode } from "react";
import { type ScrollIntoViewLocation } from "react-virtuoso";
import React, { useCallback, useLayoutEffect, useMemo, useRef, type JSX, type ReactNode } from "react";
import { type ScrollIntoViewLocation, type VirtuosoHandle } from "react-virtuoso";
import { isEqual } from "lodash";
import { type Room } from "./RoomListItemAccessibilityWrapper/RoomListItemView";
@ -38,6 +38,8 @@ export interface RoomListViewState {
spaceId?: string;
/** Active filter keys for context tracking */
filterKeys?: FilterKey[];
/** Tag of a newly created section header to scroll into view */
scrollToSectionTag?: string;
}
/**
@ -110,8 +112,13 @@ export function VirtualizedRoomListView({ vm, renderAvatar, onKeyDown }: Virtual
const snapshot = useViewModel(vm);
const { roomListState, sections, isFlatList } = snapshot;
const activeRoomIndex = roomListState.activeRoomIndex;
const scrollToSectionTag = roomListState.scrollToSectionTag;
const lastSpaceId = useRef<string | undefined>(undefined);
const lastFilterKeys = useRef<FilterKey[] | undefined>(undefined);
const virtuosoHandleRef = useRef<VirtuosoHandle | null>(null);
const setVirtuosoHandle = useCallback((handle: VirtuosoHandle | null) => {
virtuosoHandleRef.current = handle;
}, []);
const roomIds = useMemo(() => sections.flatMap((section) => section.roomIds), [sections]);
const roomCount = roomIds.length;
const sectionCount = sections.length;
@ -328,6 +335,16 @@ export function VirtualizedRoomListView({ vm, renderAvatar, onKeyDown }: Virtual
[activeRoomIndex],
);
// Imperatively scroll to a newly created section header.
// scrollIntoView on virtuoso handle is more reliable in this case vs scrollIntoViewOnChange
useLayoutEffect(() => {
if (scrollToSectionTag === undefined) return;
const sectionIndex = sections.findIndex((s) => s.id === scrollToSectionTag);
if (sectionIndex === -1) return;
const flatIndex = sections.slice(0, sectionIndex).reduce((acc, s) => acc + s.roomIds.length + 1, 0);
virtuosoHandleRef.current?.scrollIntoView({ index: flatIndex, align: "start", behavior: "auto" });
}, [scrollToSectionTag, sections]);
const isItemFocusable = useCallback(() => true, []);
const isGroupHeaderFocusable = useCallback(() => true, []);
const increaseViewportBy = useMemo(
@ -369,6 +386,7 @@ export function VirtualizedRoomListView({ vm, renderAvatar, onKeyDown }: Virtual
<GroupedVirtualizedList<string, string, Context>
{...commonProps}
{...getContainerAccessibleProps("treegrid", totalCount)}
scrollHandleRef={setVirtuosoHandle}
groups={groups}
getHeaderKey={getHeaderKey}
getGroupHeaderComponent={getGroupHeaderComponent}