impress-2020/src/app/WardrobePage/SearchPanel.js

456 lines
14 KiB
JavaScript
Raw Normal View History

2020-04-24 21:17:03 -07:00
import React from "react";
import gql from "graphql-tag";
2020-12-25 09:08:33 -08:00
import { Box, Text, VisuallyHidden } from "@chakra-ui/react";
import { useQuery } from "@apollo/client";
2020-04-24 21:17:03 -07:00
2020-09-01 19:40:53 -07:00
import { Delay, useDebounce } from "../util";
import { emptySearchQuery } from "./SearchToolbar";
import Item, { ItemListContainer, ItemListSkeleton } from "./Item";
import { itemAppearanceFragment } from "../components/useOutfitAppearance";
2020-04-24 21:17:03 -07:00
2020-04-26 01:42:24 -07:00
/**
* SearchPanel shows item search results to the user, so they can preview them
* and add them to their outfit!
*
* It's tightly coordinated with SearchToolbar, using refs to control special
* keyboard and focus interactions.
*/
2020-04-25 01:55:48 -07:00
function SearchPanel({
query,
outfitState,
dispatchToOutfit,
scrollContainerRef,
searchQueryRef,
firstSearchResultRef,
2020-04-25 01:55:48 -07:00
}) {
// Whenever the search query changes, scroll back up to the top!
React.useEffect(() => {
if (scrollContainerRef.current) {
scrollContainerRef.current.scrollTop = 0;
}
}, [query, scrollContainerRef]);
2020-04-26 01:42:24 -07:00
// Sometimes we want to give focus back to the search field!
const onMoveFocusUpToQuery = (e) => {
if (searchQueryRef.current) {
searchQueryRef.current.focus();
e.preventDefault();
}
};
2020-04-24 21:17:03 -07:00
return (
2020-04-25 20:27:04 -07:00
<Box
onKeyDown={(e) => {
2020-04-26 01:42:24 -07:00
// This will catch any Escape presses when the user's focus is inside
// the SearchPanel.
2020-04-25 20:27:04 -07:00
if (e.key === "Escape") {
onMoveFocusUpToQuery(e);
}
}}
>
2020-04-24 21:17:03 -07:00
<SearchResults
// When the query changes, replace the SearchResults component with a
// new instance, to reset `itemIdsToReconsider`. That way, if you find
// an item you like in one search, then immediately do a second search
// and try a conflicting item, we'll restore the item you liked from
// your first search!
//
// NOTE: I wonder how this affects things like state. This component
// also tries to gracefully handle changes in the query, but tbh
// I wonder whether that's still necessary...
key={serializeQuery(query)}
2020-04-24 21:17:03 -07:00
query={query}
outfitState={outfitState}
dispatchToOutfit={dispatchToOutfit}
scrollContainerRef={scrollContainerRef}
firstSearchResultRef={firstSearchResultRef}
onMoveFocusUpToQuery={onMoveFocusUpToQuery}
2020-04-24 21:17:03 -07:00
/>
</Box>
);
}
2020-04-26 01:42:24 -07:00
/**
* SearchResults loads the search results from the user's query, renders them,
* and tracks the scroll container for infinite scrolling.
*
* For each item, we render a <label> with a visually-hidden checkbox and the
* Item component (which will visually reflect the radio's state). This makes
* the list screen-reader- and keyboard-accessible!
*/
function SearchResults({
query,
outfitState,
dispatchToOutfit,
scrollContainerRef,
firstSearchResultRef,
onMoveFocusUpToQuery,
}) {
2020-04-26 01:42:24 -07:00
const { loading, loadingMore, error, items, fetchMore } = useSearchResults(
query,
outfitState
);
useScrollTracker(scrollContainerRef, 300, fetchMore);
// This will save the `wornItemIds` when the SearchResults first mounts, and
// keep it saved even after the outfit changes. We use this to try to restore
// these items after the user makes changes, e.g., after they try on another
// Background we want to restore the previous one!
const [itemIdsToReconsider] = React.useState(outfitState.wornItemIds);
2020-04-26 01:42:24 -07:00
// When the checkbox changes, we should wear/unwear the item!
const onChange = (e) => {
const itemId = e.target.value;
const willBeWorn = e.target.checked;
if (willBeWorn) {
dispatchToOutfit({ type: "wearItem", itemId, itemIdsToReconsider });
2020-04-26 01:42:24 -07:00
} else {
dispatchToOutfit({ type: "unwearItem", itemId, itemIdsToReconsider });
2020-04-26 01:42:24 -07:00
}
};
// You can use UpArrow/DownArrow to navigate between items, and even back up
// to the search field!
const goToPrevItem = (e) => {
const prevLabel = e.target.closest("label").previousSibling;
if (prevLabel) {
prevLabel.querySelector("input[type=checkbox]").focus();
prevLabel.scrollIntoView({ block: "center" });
e.preventDefault();
} else {
// If we're at the top of the list, move back up to the search box!
onMoveFocusUpToQuery(e);
}
};
const goToNextItem = (e) => {
const nextLabel = e.target.closest("label").nextSibling;
if (nextLabel) {
nextLabel.querySelector("input[type=checkbox]").focus();
nextLabel.scrollIntoView({ block: "center" });
e.preventDefault();
}
};
// If the results aren't ready, we have some special case UI!
if (loading) {
return (
<Delay ms={500}>
<ItemListSkeleton count={8} />
</Delay>
);
} else if (error) {
return (
<Text>
2020-04-26 01:42:24 -07:00
We hit an error trying to load your search results{" "}
<span role="img" aria-label="(sweat emoji)">
😓
</span>{" "}
Try again?
</Text>
);
} else if (items.length === 0) {
return (
<Text>
2020-04-26 01:42:24 -07:00
We couldn't find any matching items{" "}
<span role="img" aria-label="(thinking emoji)">
🤔
</span>{" "}
Try again?
</Text>
);
}
// Finally, render the item list, with checkboxes and Item components!
// We also render some extra skeleton items at the bottom during infinite
// scroll loading.
return (
<>
<ItemListContainer>
{items.map((item, index) => (
<label key={item.id}>
<VisuallyHidden
as="input"
type="checkbox"
aria-label={`Wear "${item.name}"`}
value={item.id}
checked={outfitState.wornItemIds.includes(item.id)}
ref={index === 0 ? firstSearchResultRef : null}
onChange={onChange}
onKeyDown={(e) => {
if (e.key === "Enter") {
e.target.click();
} else if (e.key === "ArrowUp") {
goToPrevItem(e);
} else if (e.key === "ArrowDown") {
goToNextItem(e);
}
}}
/>
<Item
item={item}
isWorn={outfitState.wornItemIds.includes(item.id)}
isInOutfit={outfitState.allItemIds.includes(item.id)}
onRemove={() =>
dispatchToOutfit({
type: "removeItem",
itemId: item.id,
itemIdsToReconsider,
})
}
2020-04-26 01:42:24 -07:00
/>
</label>
))}
</ItemListContainer>
{loadingMore && <ItemListSkeleton count={8} />}
</>
);
}
/**
* useSearchResults manages the actual querying and state management of search!
* It's hefty, infinite-scroll pagination is a bit of a thing!
*/
function useSearchResults(query, outfitState) {
const { speciesId, colorId } = outfitState;
2020-04-26 01:42:24 -07:00
const [isEndOfResults, setIsEndOfResults] = React.useState(false);
2020-04-24 21:17:03 -07:00
2020-04-26 01:42:24 -07:00
// We debounce the search query, so that we don't resend a new query whenever
// the user types anything.
const debouncedQuery = useDebounce(query, 300, {
waitForFirstPause: true,
initialValue: emptySearchQuery,
});
2020-04-24 21:17:03 -07:00
2020-04-26 01:42:24 -07:00
// When the query changes, we should update our impression of whether we've
// reached the end!
2020-04-25 02:18:03 -07:00
React.useEffect(() => {
setIsEndOfResults(false);
}, [query]);
2020-09-01 20:06:54 -07:00
// NOTE: This query should always load ~instantly, from the client cache.
const { data: zoneData } = useQuery(gql`
query SearchPanelZones {
allZones {
id
label
}
}
`);
const allZones = zoneData?.allZones || [];
const filterToZones = query.filterToZoneLabel
? allZones.filter((z) => z.label === query.filterToZoneLabel)
: [];
const filterToZoneIds = filterToZones.map((z) => z.id);
2020-04-26 01:42:24 -07:00
// Here's the actual GQL query! At the bottom we have more config than usual!
const {
loading: loadingGQL,
error,
data,
fetchMore: fetchMoreGQL,
} = useQuery(
2020-04-24 21:17:03 -07:00
gql`
2020-05-19 14:48:54 -07:00
query SearchPanel(
$query: String!
2020-11-08 15:04:17 -08:00
$itemKind: ItemKindSearchFilter
2020-09-01 20:06:54 -07:00
$zoneIds: [ID!]!
2020-11-08 15:04:17 -08:00
$speciesId: ID!
2020-05-19 14:48:54 -07:00
$colorId: ID!
$offset: Int!
) {
itemSearchToFit(
query: $query
2020-11-08 15:04:17 -08:00
itemKind: $itemKind
2020-09-01 20:06:54 -07:00
zoneIds: $zoneIds
speciesId: $speciesId
colorId: $colorId
2020-04-25 01:55:48 -07:00
offset: $offset
limit: 50
) {
2020-04-25 01:55:48 -07:00
query
zones {
id
}
2020-04-25 01:55:48 -07:00
items {
# TODO: De-dupe this from useOutfitState?
id
name
thumbnailUrl
isNc
isPb
currentUserOwnsThis
currentUserWantsThis
2020-04-25 01:55:48 -07:00
appearanceOn(speciesId: $speciesId, colorId: $colorId) {
# This enables us to quickly show the item when the user clicks it!
...ItemAppearanceForOutfitPreview
2020-04-25 01:55:48 -07:00
# This is used to group items by zone, and to detect conflicts when
# wearing a new item.
layers {
zone {
id
label @client
2020-04-25 01:55:48 -07:00
}
2020-04-24 21:17:03 -07:00
}
restrictedZones {
id
label @client
isCommonlyUsedByItems @client
}
2020-04-24 21:17:03 -07:00
}
}
}
}
${itemAppearanceFragment}
`,
{
2020-09-01 20:06:54 -07:00
variables: {
query: debouncedQuery.value,
2020-11-08 15:04:17 -08:00
itemKind: debouncedQuery.filterToItemKind,
2020-09-01 20:06:54 -07:00
zoneIds: filterToZoneIds,
speciesId,
colorId,
offset: 0,
},
2020-11-08 15:04:17 -08:00
skip:
!debouncedQuery.value &&
!debouncedQuery.filterToItemKind &&
!debouncedQuery.filterToZoneLabel,
2020-04-25 01:55:48 -07:00
notifyOnNetworkStatusChange: true,
2020-04-25 02:18:03 -07:00
onCompleted: (d) => {
2020-04-26 01:42:24 -07:00
// This is called each time the query completes, including on
// `fetchMore`, with the extended results. But, on the first time, this
// logic can tell us whether we're at the end of the list, by counting
// whether there was <30. We also have to check in `fetchMore`!
2020-04-25 02:18:03 -07:00
const items = d && d.itemSearchToFit && d.itemSearchToFit.items;
if (items && items.length < 30) {
setIsEndOfResults(true);
}
},
onError: (e) => {
console.error("Error loading search results", e);
},
2020-04-24 21:17:03 -07:00
}
);
2020-04-26 01:42:24 -07:00
// Smooth over the data a bit, so that we can use key fields with confidence!
const result = data?.itemSearchToFit;
const resultValue = result?.query;
const zoneStr = filterToZoneIds.sort().join(",");
const resultZoneStr = (result?.zones || [])
.map((z) => z.id)
.sort()
.join(",");
const queriesMatch = resultValue === query.value && resultZoneStr === zoneStr;
const items = result?.items || [];
2020-04-25 01:55:48 -07:00
2020-04-26 01:42:24 -07:00
// Okay, what kind of loading state is this?
let loading;
let loadingMore;
if (loadingGQL && items.length > 0 && queriesMatch) {
// If we already have items for this query, but we're also loading GQL,
// then we're `loadingMore`.
2020-04-26 01:42:24 -07:00
loading = false;
loadingMore = true;
} else if (loadingGQL || query !== debouncedQuery) {
// Otherwise, if we're loading GQL or the user has changed the query, we're
// just `loading`.
loading = true;
loadingMore = false;
2020-04-26 01:42:24 -07:00
} else {
// Otherwise, we're not loading at all!
loading = false;
loadingMore = false;
}
// When SearchResults calls this, we'll resend the query, with the `offset`
// increased. We'll append the results to the original query!
const fetchMore = React.useCallback(() => {
if (!loadingGQL && !isEndOfResults) {
fetchMoreGQL({
2020-04-25 01:55:48 -07:00
variables: {
offset: items.length,
},
updateQuery: (prev, { fetchMoreResult }) => {
2020-04-25 02:18:03 -07:00
// Note: This is a bit awkward because, if the results count ends on
// a multiple of 30, the user will see a flash of loading before
// getting told it's actually the end. Ah well :/
//
// We could maybe make this more rigorous later with
// react-virtualized to have a better scrollbar anyway, and then
// we'd need to return the total result count... a bit annoying to
// potentially double the query runtime? We'd need to see how slow it
// actually makes things.
if (fetchMoreResult.itemSearchToFit.items.length < 30) {
setIsEndOfResults(true);
}
2020-04-25 01:55:48 -07:00
return {
...prev,
itemSearchToFit: {
...(prev?.itemSearchToFit || {}),
2020-04-25 01:55:48 -07:00
items: [
...(prev?.itemSearchToFit?.items || []),
...(fetchMoreResult?.itemSearchToFit?.items || []),
2020-04-25 01:55:48 -07:00
],
},
};
},
});
}
2020-04-26 01:42:24 -07:00
}, [loadingGQL, isEndOfResults, fetchMoreGQL, items.length]);
2020-04-25 01:55:48 -07:00
2020-04-26 01:42:24 -07:00
return { loading, loadingMore, error, items, fetchMore };
2020-04-25 01:55:48 -07:00
}
2020-04-26 01:42:24 -07:00
/**
* useScrollTracker watches for the given scroll container to scroll near the
* bottom, then fires a callback. We use this to fetch more search results!
*/
function useScrollTracker(scrollContainerRef, threshold, onScrolledToBottom) {
2020-04-25 01:55:48 -07:00
const onScroll = React.useCallback(
(e) => {
const topEdgeScrollPosition = e.target.scrollTop;
const bottomEdgeScrollPosition =
topEdgeScrollPosition + e.target.clientHeight;
const remainingScrollDistance =
e.target.scrollHeight - bottomEdgeScrollPosition;
if (remainingScrollDistance < threshold) {
onScrolledToBottom();
}
},
[onScrolledToBottom, threshold]
2020-04-24 21:17:03 -07:00
);
2020-04-25 01:55:48 -07:00
React.useLayoutEffect(() => {
const scrollContainer = scrollContainerRef.current;
if (!scrollContainer) {
2020-04-25 01:55:48 -07:00
return;
}
scrollContainer.addEventListener("scroll", onScroll);
2020-04-25 01:55:48 -07:00
return () => {
if (scrollContainer) {
scrollContainer.removeEventListener("scroll", onScroll);
2020-04-25 01:55:48 -07:00
}
};
}, [onScroll, scrollContainerRef]);
}
/**
* serializeQuery stably converts a search query object to a string, for easier
* JS comparison.
*/
function serializeQuery(query) {
2020-11-08 15:04:17 -08:00
return `${JSON.stringify([
query.value,
query.filterToItemKind,
query.filterToZoneLabel,
])}`;
}
2020-04-24 21:17:03 -07:00
export default SearchPanel;