fix: sort button visual and filter by variant metafield

Signed-off-by: Chloe <pinkcloudvnn@gmail.com>
This commit is contained in:
Chloe 2024-05-07 13:30:26 +07:00
parent 305fe3d458
commit 145eb3eaed
No known key found for this signature in database
GPG Key ID: CFD53CE570D42DF5
9 changed files with 109 additions and 28 deletions

View File

@ -8,7 +8,14 @@ import Grid from 'components/grid';
import ProductGridItems from 'components/layout/product-grid-items'; import ProductGridItems from 'components/layout/product-grid-items';
import Filters from 'components/layout/search/filters'; import Filters from 'components/layout/search/filters';
import SortingMenu from 'components/layout/search/sorting-menu'; import SortingMenu from 'components/layout/search/sorting-menu';
import { AVAILABILITY_FILTER_ID, PRICE_FILTER_ID, defaultSort, sorting } from 'lib/constants'; import {
AVAILABILITY_FILTER_ID,
PRICE_FILTER_ID,
PRODUCT_METAFIELD_PREFIX,
VARIANT_METAFIELD_PREFIX,
defaultSort,
sorting
} from 'lib/constants';
import { Suspense } from 'react'; import { Suspense } from 'react';
export const runtime = 'edge'; export const runtime = 'edge';
@ -34,12 +41,18 @@ const constructFilterInput = (filters: {
}): Array<object> => { }): Array<object> => {
const results = [] as Array<object>; const results = [] as Array<object>;
Object.entries(filters) Object.entries(filters)
.filter(([key]) => ![AVAILABILITY_FILTER_ID, PRICE_FILTER_ID].includes(key)) .filter(([key]) => key !== PRICE_FILTER_ID)
.forEach(([key, value]) => { .forEach(([key, value]) => {
const [namespace, metafieldKey] = key.split('.').slice(-2); const [namespace, metafieldKey] = key.split('.').slice(-2);
if (Array.isArray(value)) { const values = Array.isArray(value) ? value : [value];
if (key === AVAILABILITY_FILTER_ID) {
results.push({
available: value === 'true'
});
} else if (key.startsWith(PRODUCT_METAFIELD_PREFIX)) {
results.push( results.push(
...value.map((v) => ({ ...values.map((v) => ({
productMetafield: { productMetafield: {
namespace, namespace,
key: metafieldKey, key: metafieldKey,
@ -47,14 +60,16 @@ const constructFilterInput = (filters: {
} }
})) }))
); );
} else { } else if (key.startsWith(VARIANT_METAFIELD_PREFIX)) {
results.push({ results.push(
productMetafield: { ...values.map((v) => ({
namespace, variantMetafield: {
key: metafieldKey, namespace,
value key: metafieldKey,
} value: v
}); }
}))
);
} }
}); });

View File

@ -8,10 +8,16 @@ const Filters = ({ filters }: { filters: Filter[] }) => {
const router = useRouter(); const router = useRouter();
const pathname = usePathname(); const pathname = usePathname();
const searchParams = useSearchParams(); const searchParams = useSearchParams();
const { q, sort, collection } = Object.fromEntries(searchParams);
const initialFilters = {
...(q && { q }),
...(sort && { sort }),
...(collection && { collection })
};
const handleChange = (e: React.FormEvent<HTMLFormElement>) => { const handleChange = (e: React.FormEvent<HTMLFormElement>) => {
const formData = new FormData(e.currentTarget); const formData = new FormData(e.currentTarget);
const newSearchParams = new URLSearchParams(searchParams); const newSearchParams = new URLSearchParams(initialFilters);
Array.from(formData.keys()).forEach((key) => { Array.from(formData.keys()).forEach((key) => {
const values = formData.getAll(key); const values = formData.getAll(key);

View File

@ -2,18 +2,27 @@
import { Menu, Transition } from '@headlessui/react'; import { Menu, Transition } from '@headlessui/react';
import { ChevronDownIcon } from '@heroicons/react/20/solid'; import { ChevronDownIcon } from '@heroicons/react/20/solid';
import { sorting } from 'lib/constants'; import { defaultSort, sorting } from 'lib/constants';
import { useSearchParams } from 'next/navigation';
import { Fragment } from 'react'; import { Fragment } from 'react';
import SortingItem from './item'; import SortingItem from './item';
const SortingMenu = () => { const SortingMenu = () => {
const searchParams = useSearchParams();
const sort = searchParams.get('sort');
return ( return (
<Menu as="div" className="relative inline-block text-left"> <Menu as="div" className="relative inline-block text-left">
<div> <div>
<Menu.Button className="group inline-flex justify-center text-sm font-medium text-gray-700 hover:text-gray-900"> <Menu.Button className="group inline-flex justify-center rounded border border-gray-300 px-3 py-1 text-sm text-gray-700 hover:bg-gray-100">
Sort <div className="flex items-center gap-2">
Sort by:{' '}
<span>
{sorting.find((option) => option.slug === sort)?.title || defaultSort.title}
</span>
</div>
<ChevronDownIcon <ChevronDownIcon
className="-mr-1 ml-1 h-5 w-5 flex-shrink-0 text-gray-400 group-hover:text-gray-500" className="-mr-1 ml-1.5 h-5 w-5 flex-shrink-0 text-gray-400 group-hover:text-gray-500"
aria-hidden="true" aria-hidden="true"
/> />
</Menu.Button> </Menu.Button>
@ -28,7 +37,7 @@ const SortingMenu = () => {
leaveFrom="transform opacity-100 scale-100" leaveFrom="transform opacity-100 scale-100"
leaveTo="transform opacity-0 scale-95" leaveTo="transform opacity-0 scale-95"
> >
<Menu.Items className="absolute right-0 z-10 mt-2 w-40 origin-top-right rounded-md bg-white shadow-lg ring-1 ring-black ring-opacity-5 focus:outline-none"> <Menu.Items className="absolute right-0 z-10 mt-2 w-full origin-top-right rounded-md bg-white shadow-lg ring-1 ring-black ring-opacity-5 focus:outline-none">
<div className="py-1"> <div className="py-1">
{sorting.map((option) => ( {sorting.map((option) => (
<Menu.Item key={option.title}> <Menu.Item key={option.title}>

View File

@ -35,3 +35,5 @@ export const CORE_VARIANT_ID_KEY = 'coreVariantId';
export const AVAILABILITY_FILTER_ID = 'filter.v.availability'; export const AVAILABILITY_FILTER_ID = 'filter.v.availability';
export const PRICE_FILTER_ID = 'filter.v.price'; export const PRICE_FILTER_ID = 'filter.v.price';
export const PRODUCT_METAFIELD_PREFIX = 'filter.p.m';
export const VARIANT_METAFIELD_PREFIX = 'filter.v.m';

View File

@ -2,8 +2,10 @@ import {
AVAILABILITY_FILTER_ID, AVAILABILITY_FILTER_ID,
HIDDEN_PRODUCT_TAG, HIDDEN_PRODUCT_TAG,
PRICE_FILTER_ID, PRICE_FILTER_ID,
PRODUCT_METAFIELD_PREFIX,
SHOPIFY_GRAPHQL_API_ENDPOINT, SHOPIFY_GRAPHQL_API_ENDPOINT,
TAGS TAGS,
VARIANT_METAFIELD_PREFIX
} from 'lib/constants'; } from 'lib/constants';
import { isShopifyError } from 'lib/type-guards'; import { isShopifyError } from 'lib/type-guards';
import { ensureStartsWith, normalizeUrl, parseMetaFieldValue } from 'lib/utils'; import { ensureStartsWith, normalizeUrl, parseMetaFieldValue } from 'lib/utils';
@ -38,6 +40,7 @@ import {
Menu, Menu,
Money, Money,
Page, Page,
PageInfo,
Product, Product,
ProductVariant, ProductVariant,
ShopifyAddToCartOperation, ShopifyAddToCartOperation,
@ -182,16 +185,35 @@ const reshapeFilters = (filters: ShopifyFilter[]): Filter[] => {
for (const filter of filters) { for (const filter of filters) {
const values = filter.values const values = filter.values
.map((valueItem) => { .map((valueItem) => {
try { if (filter.id === AVAILABILITY_FILTER_ID) {
return { return {
...valueItem, ...valueItem,
...(![AVAILABILITY_FILTER_ID, PRICE_FILTER_ID].includes(filter.id) value: JSON.parse(valueItem.input).available
? { value: JSON.parse(valueItem.input).productMetafield.value }
: { value: JSON.parse(valueItem.input) })
}; };
} catch (error) {
return null;
} }
if (filter.id === PRICE_FILTER_ID) {
return {
...valueItem,
value: JSON.parse(valueItem.input)
};
}
if (filter.id.startsWith(PRODUCT_METAFIELD_PREFIX)) {
return {
...valueItem,
value: JSON.parse(valueItem.input).productMetafield.value
};
}
if (filter.id.startsWith(VARIANT_METAFIELD_PREFIX)) {
return {
...valueItem,
value: JSON.parse(valueItem.input).variantMetafield.value
};
}
return null;
}) })
.filter(Boolean) as Filter['values']; .filter(Boolean) as Filter['values'];
@ -345,7 +367,7 @@ export async function getCollectionProducts({
reverse?: boolean; reverse?: boolean;
sortKey?: string; sortKey?: string;
filters?: Array<object>; filters?: Array<object>;
}): Promise<{ products: Product[]; filters: Filter[] }> { }): Promise<{ products: Product[]; filters: Filter[]; pageInfo: PageInfo }> {
const res = await shopifyFetch<ShopifyCollectionProductsOperation>({ const res = await shopifyFetch<ShopifyCollectionProductsOperation>({
query: getCollectionProductsQuery, query: getCollectionProductsQuery,
tags: [TAGS.collections, TAGS.products], tags: [TAGS.collections, TAGS.products],
@ -359,12 +381,18 @@ export async function getCollectionProducts({
if (!res.body.data.collection) { if (!res.body.data.collection) {
console.log(`No collection found for \`${collection}\``); console.log(`No collection found for \`${collection}\``);
return { products: [], filters: [] }; return {
products: [],
filters: [],
pageInfo: { startCursor: '', hasNextPage: false, endCursor: '' }
};
} }
const pageInfo = res.body.data.collection.products.pageInfo;
return { return {
products: reshapeProducts(removeEdgesAndNodes(res.body.data.collection.products)), products: reshapeProducts(removeEdgesAndNodes(res.body.data.collection.products)),
filters: reshapeFilters(res.body.data.collection.products.filters) filters: reshapeFilters(res.body.data.collection.products.filters),
pageInfo
}; };
} }

View File

@ -49,6 +49,7 @@ export const getCollectionProductsQuery = /* GraphQL */ `
node { node {
...product ...product
} }
cursor
} }
filters { filters {
id id
@ -61,6 +62,11 @@ export const getCollectionProductsQuery = /* GraphQL */ `
label label
} }
} }
pageInfo {
endCursor
startCursor
hasNextPage
}
} }
} }
} }

View File

@ -220,11 +220,18 @@ export type ShopifyCollectionOperation = {
}; };
}; };
export type PageInfo = {
startCursor: string;
hasNextPage: boolean;
endCursor: string;
};
export type ShopifyCollectionProductsOperation = { export type ShopifyCollectionProductsOperation = {
data: { data: {
collection: { collection: {
products: Connection<ShopifyProduct> & { products: Connection<ShopifyProduct> & {
filters: ShopifyFilter[]; filters: ShopifyFilter[];
pageInfo: PageInfo;
}; };
}; };
}; };

View File

@ -27,6 +27,7 @@
"@radix-ui/react-checkbox": "^1.0.4", "@radix-ui/react-checkbox": "^1.0.4",
"clsx": "^2.1.0", "clsx": "^2.1.0",
"geist": "^1.3.0", "geist": "^1.3.0",
"lodash.get": "^4.4.2",
"next": "14.1.4", "next": "14.1.4",
"react": "18.2.0", "react": "18.2.0",
"react-dom": "18.2.0", "react-dom": "18.2.0",

7
pnpm-lock.yaml generated
View File

@ -20,6 +20,9 @@ dependencies:
geist: geist:
specifier: ^1.3.0 specifier: ^1.3.0
version: 1.3.0(next@14.1.4) version: 1.3.0(next@14.1.4)
lodash.get:
specifier: ^4.4.2
version: 4.4.2
next: next:
specifier: 14.1.4 specifier: 14.1.4
version: 14.1.4(react-dom@18.2.0)(react@18.2.0) version: 14.1.4(react-dom@18.2.0)(react@18.2.0)
@ -2502,6 +2505,10 @@ packages:
resolution: {integrity: sha512-aVx8ztPv7/2ULbArGJ2Y42bG1mEQ5mGjpdvrbJcJFU3TbYybe+QlLS4pst9zV52ymy2in1KpFPiZnAOATxD4+Q==} resolution: {integrity: sha512-aVx8ztPv7/2ULbArGJ2Y42bG1mEQ5mGjpdvrbJcJFU3TbYybe+QlLS4pst9zV52ymy2in1KpFPiZnAOATxD4+Q==}
dev: true dev: true
/lodash.get@4.4.2:
resolution: {integrity: sha512-z+Uw/vLuy6gQe8cfaFWD7p0wVv8fJl3mbzXh33RS+0oW2wvUqiRXiQ69gLWSLpgB5/6sU+r6BlQR0MBILadqTQ==}
dev: false
/lodash.isplainobject@4.0.6: /lodash.isplainobject@4.0.6:
resolution: {integrity: sha512-oSXzaWypCMHkPC3NvBEaPHf0KsA5mvPrOPgQWDsbg8n7orZ290M0BmC/jgRZ4vcJ6DTAhjrsSYgdsW/F+MFOBA==} resolution: {integrity: sha512-oSXzaWypCMHkPC3NvBEaPHf0KsA5mvPrOPgQWDsbg8n7orZ290M0BmC/jgRZ4vcJ6DTAhjrsSYgdsW/F+MFOBA==}
dev: true dev: true