Skip to content

Commit

Permalink
Add error handling to filters
Browse files Browse the repository at this point in the history
  • Loading branch information
acelaya committed Sep 2, 2024
1 parent e08bf70 commit 2edd699
Show file tree
Hide file tree
Showing 4 changed files with 254 additions and 123 deletions.
Original file line number Diff line number Diff line change
@@ -1,9 +1,13 @@
import {
Button,
CancelIcon,
IconButton,
MultiSelect,
RefreshIcon,
} from '@hypothesis/frontend-shared';
import { useMemo } from 'preact/hooks';
import type { ComponentChildren } from 'preact';
import type { MutableRef } from 'preact/hooks';
import { useMemo, useRef } from 'preact/hooks';
import { useParams } from 'wouter-preact';

import type {
Expand All @@ -15,6 +19,7 @@ import type {
StudentsResponse,
} from '../../api-types';
import { useConfig } from '../../config';
import type { PaginatedFetchResult } from '../../utils/api';
import { usePaginatedAPIFetch } from '../../utils/api';
import { formatDateTime } from '../../utils/date';

Expand Down Expand Up @@ -55,8 +60,6 @@ export type DashboardActivityFiltersProps = {
onClearSelection?: () => void;
};

type FiltersStudent = Student & { has_display_name: boolean };

/**
* Checks if provided element's scroll is at the bottom.
* @param offset - Return true if the difference between the element's current
Expand All @@ -69,12 +72,19 @@ function elementScrollIsAtBottom(element: HTMLElement, offset = 20): boolean {
return distanceToTop >= triggerPoint;
}

type PropsWithElementRef<T> = T & {
elementRef?: MutableRef<HTMLElement | null>;
};

/**
* Represents a `Select.Option` for a specific assignment
*/
function AssignmentOption({ assignment }: { assignment: Assignment }) {
function AssignmentOption({
assignment,
elementRef,
}: PropsWithElementRef<{ assignment: Assignment }>) {
return (
<MultiSelect.Option value={`${assignment.id}`}>
<MultiSelect.Option value={`${assignment.id}`} elementRef={elementRef}>
<div className="flex flex-col gap-0.5">
{assignment.title}
<div className="text-grey-6 text-xs">
Expand All @@ -85,14 +95,37 @@ function AssignmentOption({ assignment }: { assignment: Assignment }) {
);
}

/**
* Represents a `Select.Option` for a specific student
*/
function StudentOption({
student,
elementRef,
}: PropsWithElementRef<{ student: Student }>) {
const hasDisplayName = !!student.display_name;
const displayName =
student.display_name ??
`Student name unavailable (ID: ${student.lms_id.substring(0, 5)})`;

return (
<MultiSelect.Option value={student.h_userid} elementRef={elementRef}>
<span
className={hasDisplayName ? undefined : 'italic'}
title={hasDisplayName ? undefined : `User ID: ${student.lms_id}`}
data-testid="option-content-wrapper"
>
{displayName}
</span>
</MultiSelect.Option>
);
}

type FiltersEntity = 'courses' | 'assignments' | 'students';

/**
* Placeholder to indicate a loading is in progress in one of the dropdowns
*/
function LoadingOption({
entity,
}: {
entity: 'courses' | 'assignments' | 'students';
}) {
function LoadingOption({ entity }: { entity: FiltersEntity }) {
return (
<div
className="py-2 px-4 mb-1 text-grey-4 italic"
Expand All @@ -103,6 +136,108 @@ function LoadingOption({
);
}

type ErrorRetryOptionProps = {
entity: FiltersEntity;
retry: () => void;
};

/**
* Indicates an error occurred while loading filters, and presents a CTA to retry.
*/
function LoadingError({ entity, retry }: ErrorRetryOptionProps) {
return (
<div
className="flex gap-2 items-center py-2 px-4 mb-1"
data-testid={`${entity}-error`}
// Make this element "focusable" so that clicking on it does not cause
// the listbox containing it to be closed
tabIndex={-1}
>
<span className="italic text-red-error">Error loading {entity}</span>
<Button icon={RefreshIcon} onClick={retry} size="xs">
Retry
</Button>
</div>
);
}

type PaginatedMultiSelectProps<TResult, TSelect> = {
result: PaginatedFetchResult<NonNullable<TResult>[]>;
activeItem?: TResult;
renderOption: (
item: NonNullable<TResult>,
ref?: MutableRef<HTMLElement | null>,
) => ComponentChildren;
entity: FiltersEntity;
buttonContent?: ComponentChildren;
value: TSelect[];
onChange: (newValue: TSelect[]) => void;
};

function PaginatedMultiSelect<TResult, TSelect>({
result,
activeItem,
entity,
renderOption,
buttonContent,
value,
onChange,
}: PaginatedMultiSelectProps<TResult, TSelect>) {
const lastOptionRef = useRef<HTMLElement | null>(null);

return (
<MultiSelect
disabled={result.isLoadingFirstPage}
value={value}
onChange={onChange}
aria-label={`Select ${entity}`}
containerClasses="!w-auto min-w-[180px]"
buttonContent={buttonContent}
data-testid={`${entity}-select`}
onListboxScroll={e => {
if (elementScrollIsAtBottom(e.target as HTMLUListElement)) {
result.loadNextPage();
}
}}
>
<MultiSelect.Option
value={undefined}
elementRef={
!activeItem && (!result.data || result.data.length === 0)
? lastOptionRef
: undefined
}
>
All {entity}
</MultiSelect.Option>
{activeItem ? (
renderOption(activeItem, lastOptionRef)
) : (
<>
{result.data?.map((item, index, list) =>
renderOption(
item,
list.length - 1 === index ? lastOptionRef : undefined,
),
)}
{result.isLoading && <LoadingOption entity={entity} />}
{result.error && (
<LoadingError
entity={entity}
retry={() => {
// Focus last option before retrying, to avoid the listbox to
// be closed
lastOptionRef.current?.focus();
result.retry();
}}
/>
)}
</>
)}
</MultiSelect>
);
}

/**
* Renders drop-downs to select courses, assignments and/or students, used to
* filter dashboard activity metrics.
Expand Down Expand Up @@ -187,30 +322,18 @@ export default function DashboardActivityFilters({
Student[],
StudentsResponse
>('students', routes.students, studentsFilters);
const studentsWithFallbackName: FiltersStudent[] | undefined = useMemo(
() =>
studentsResult.data?.map(({ display_name, ...s }) => ({
...s,
display_name:
display_name ??
`Student name unavailable (ID: ${s.lms_id.substring(0, 5)})`,
has_display_name: !!display_name,
})),
[studentsResult.data],
);

return (
<div className="flex gap-2 flex-wrap">
<MultiSelect
disabled={coursesResult.isLoadingFirstPage}
<PaginatedMultiSelect
entity="courses"
result={coursesResult}
value={selectedCourseIds}
onChange={newCourseIds =>
'onChange' in courses
? courses.onChange(newCourseIds)
: newCourseIds.length === 0 && courses.onClear()
}
aria-label="Select courses"
containerClasses="!w-auto min-w-[180px]"
buttonContent={
activeCourse ? (
activeCourse.title
Expand All @@ -225,44 +348,26 @@ export default function DashboardActivityFilters({
<>{selectedCourseIds.length} courses</>
)
}
data-testid="courses-select"
onListboxScroll={e => {
if (elementScrollIsAtBottom(e.target as HTMLUListElement)) {
coursesResult.loadNextPage();
}
}}
>
<MultiSelect.Option value={undefined}>All courses</MultiSelect.Option>
{activeCourse ? (
activeItem={activeCourse}
renderOption={(course, elementRef) => (
<MultiSelect.Option
key={activeCourse.id}
value={`${activeCourse.id}`}
key={course.id}
value={`${course.id}`}
elementRef={elementRef}
>
{activeCourse.title}
{course.title}
</MultiSelect.Option>
) : (
<>
{coursesResult.data?.map(course => (
<MultiSelect.Option key={course.id} value={`${course.id}`}>
{course.title}
</MultiSelect.Option>
))}
{coursesResult.isLoading && !coursesResult.isLoadingFirstPage && (
<LoadingOption entity="courses" />
)}
</>
)}
</MultiSelect>
<MultiSelect
disabled={assignmentsResults.isLoadingFirstPage}
/>
<PaginatedMultiSelect
entity="assignments"
result={assignmentsResults}
value={selectedAssignmentIds}
onChange={newAssignmentIds =>
'onChange' in assignments
? assignments.onChange(newAssignmentIds)
: newAssignmentIds.length === 0 && assignments.onClear()
}
aria-label="Select assignments"
containerClasses="!w-auto min-w-[180px]"
buttonContent={
activeAssignment ? (
activeAssignment.title
Expand All @@ -278,76 +383,41 @@ export default function DashboardActivityFilters({
<>{selectedAssignmentIds.length} assignments</>
)
}
data-testid="assignments-select"
onListboxScroll={e => {
if (elementScrollIsAtBottom(e.target as HTMLUListElement)) {
assignmentsResults.loadNextPage();
}
}}
>
<MultiSelect.Option value={undefined}>
All assignments
</MultiSelect.Option>
{activeAssignment ? (
<AssignmentOption assignment={activeAssignment} />
) : (
<>
{assignmentsResults.data?.map(assignment => (
<AssignmentOption key={assignment.id} assignment={assignment} />
))}
{assignmentsResults.isLoading &&
!assignmentsResults.isLoadingFirstPage && (
<LoadingOption entity="assignments" />
)}
</>
activeItem={activeAssignment}
renderOption={(assignment, elementRef) => (
<AssignmentOption
key={assignment.id}
assignment={assignment}
elementRef={elementRef}
/>
)}
</MultiSelect>
<MultiSelect
disabled={studentsResult.isLoadingFirstPage}
/>
<PaginatedMultiSelect
entity="students"
result={studentsResult}
value={students.selectedIds}
onChange={newStudentIds => students.onChange(newStudentIds)}
aria-label="Select students"
containerClasses="!w-auto min-w-[180px]"
buttonContent={
studentsResult.isLoadingFirstPage ? (
<>...</>
) : students.selectedIds.length === 0 ? (
<>All students</>
) : students.selectedIds.length === 1 ? (
studentsWithFallbackName?.find(
studentsResult.data?.find(
s => s.h_userid === students.selectedIds[0],
)?.display_name ?? '1 student'
) : (
<>{students.selectedIds.length} students</>
)
}
data-testid="students-select"
onListboxScroll={e => {
if (elementScrollIsAtBottom(e.target as HTMLUListElement)) {
studentsResult.loadNextPage();
}
}}
>
<MultiSelect.Option value={undefined}>All students</MultiSelect.Option>
{studentsWithFallbackName?.map(student => (
<MultiSelect.Option key={student.lms_id} value={student.h_userid}>
<span
className={student.has_display_name ? undefined : 'italic'}
title={
student.has_display_name
? undefined
: `User ID: ${student.lms_id}`
}
data-testid="option-content-wrapper"
>
{student.display_name}
</span>
</MultiSelect.Option>
))}
{studentsResult.isLoading && !studentsResult.isLoadingFirstPage && (
<LoadingOption entity="students" />
renderOption={(student, elementRef) => (
<StudentOption
key={student.lms_id}
student={student}
elementRef={elementRef}
/>
)}
</MultiSelect>
/>
{hasSelection && onClearSelection && (
<IconButton
title="Clear filters"
Expand Down
Loading

0 comments on commit 2edd699

Please sign in to comment.