mirror of
https://github.com/vercel/commerce.git
synced 2025-07-23 04:36:49 +00:00
Merge pull request #78 from KieIO/feature/m3-relevant-product
Relevant product in page product detail
This commit is contained in:
@@ -43,8 +43,10 @@ export type Product = {
|
|||||||
slug?: string
|
slug?: string
|
||||||
path?: string
|
path?: string
|
||||||
images: ProductImage[]
|
images: ProductImage[]
|
||||||
price: ProductPrice
|
price: number
|
||||||
|
currencyCode: CurrencyCode
|
||||||
options: ProductOption[]
|
options: ProductOption[]
|
||||||
|
facetValueIds?: string[]
|
||||||
}
|
}
|
||||||
|
|
||||||
export type ProductCard = {
|
export type ProductCard = {
|
||||||
|
@@ -16,10 +16,8 @@ export default function getProductOperation({
|
|||||||
variables: { slug: string }
|
variables: { slug: string }
|
||||||
config?: Partial<VendureConfig>
|
config?: Partial<VendureConfig>
|
||||||
preview?: boolean
|
preview?: boolean
|
||||||
}): Promise<Product | {} | any> {
|
}): Promise<Product | null> {
|
||||||
const config = commerce.getConfig(cfg)
|
const config = commerce.getConfig(cfg)
|
||||||
|
|
||||||
const locale = config.locale
|
|
||||||
const { data } = await config.fetch<GetProductQuery>(query, { variables })
|
const { data } = await config.fetch<GetProductQuery>(query, { variables })
|
||||||
const product = data.product
|
const product = data.product
|
||||||
|
|
||||||
@@ -28,7 +26,6 @@ export default function getProductOperation({
|
|||||||
return product.optionGroups.find((og) => og.id === id)!.name
|
return product.optionGroups.find((og) => og.id === id)!.name
|
||||||
}
|
}
|
||||||
return {
|
return {
|
||||||
product: {
|
|
||||||
id: product.id,
|
id: product.id,
|
||||||
name: product.name,
|
name: product.name,
|
||||||
description: product.description,
|
description: product.description,
|
||||||
@@ -49,20 +46,18 @@ export default function getProductOperation({
|
|||||||
values: [{ label: o.name }],
|
values: [{ label: o.name }],
|
||||||
})),
|
})),
|
||||||
})),
|
})),
|
||||||
price: {
|
price: product.variants[0].priceWithTax / 100,
|
||||||
value: product.variants[0].priceWithTax / 100,
|
currencyCode: product.variants[0].currencyCode,
|
||||||
currencyCode: product.variants[0].currencyCode,
|
|
||||||
},
|
|
||||||
options: product.optionGroups.map((og) => ({
|
options: product.optionGroups.map((og) => ({
|
||||||
id: og.id,
|
id: og.id,
|
||||||
displayName: og.name,
|
displayName: og.name,
|
||||||
values: og.options.map((o) => ({ label: o.name })),
|
values: og.options.map((o) => ({ label: o.name })),
|
||||||
})),
|
})),
|
||||||
} as Product,
|
facetValueIds: product.facetValues.map(item=> item.id)
|
||||||
}
|
} as Product
|
||||||
}
|
}
|
||||||
|
|
||||||
return {}
|
return null
|
||||||
}
|
}
|
||||||
|
|
||||||
return getProduct
|
return getProduct
|
||||||
|
12
framework/vendure/schema.d.ts
vendored
12
framework/vendure/schema.d.ts
vendored
@@ -3338,6 +3338,18 @@ export type GetProductQuery = { __typename?: 'Query' } & {
|
|||||||
>
|
>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
|
facetValues: Array<
|
||||||
|
{ __typename?: 'FacetValue' } & Pick<
|
||||||
|
FacetValue,
|
||||||
|
'id'
|
||||||
|
>
|
||||||
|
>
|
||||||
|
collections: Array<
|
||||||
|
{ __typename?: 'Collection' } & Pick<
|
||||||
|
Collection,
|
||||||
|
'id'
|
||||||
|
>
|
||||||
|
>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
}
|
}
|
||||||
|
@@ -36,6 +36,9 @@ export const getProductQuery = /* GraphQL */ `
|
|||||||
name
|
name
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
facetValues {
|
||||||
|
id
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
|
@@ -1,18 +1,114 @@
|
|||||||
|
|
||||||
|
import { Product } from '@framework/schema'
|
||||||
|
import commerce from '@lib/api/commerce'
|
||||||
|
import { GetStaticPathsContext, GetStaticPropsContext, InferGetStaticPropsType } from 'next'
|
||||||
import { Layout, RecipeDetail, RecommendedRecipes, RelevantBlogPosts } from 'src/components/common'
|
import { Layout, RecipeDetail, RecommendedRecipes, RelevantBlogPosts } from 'src/components/common'
|
||||||
import { ProductInfoDetail, ReleventProducts, ViewedProducts } from 'src/components/modules/product-detail'
|
import { ProductInfoDetail, ReleventProducts, ViewedProducts } from 'src/components/modules/product-detail'
|
||||||
|
import { MAX_PRODUCT_CAROUSEL, REVALIDATE_TIME } from 'src/utils/constanst.utils'
|
||||||
import { BLOGS_DATA_TEST, INGREDIENT_DATA_TEST, RECIPE_DATA_TEST } from 'src/utils/demo-data'
|
import { BLOGS_DATA_TEST, INGREDIENT_DATA_TEST, RECIPE_DATA_TEST } from 'src/utils/demo-data'
|
||||||
|
import { getAllPromies } from 'src/utils/funtion.utils'
|
||||||
|
import { PromiseWithKey } from 'src/utils/types.utils'
|
||||||
|
|
||||||
export default function Slug() {
|
export default function Slug({ product, relevantProducts, collections }: InferGetStaticPropsType<typeof getStaticProps>) {
|
||||||
|
|
||||||
return <>
|
return <>
|
||||||
<ProductInfoDetail />
|
<ProductInfoDetail />
|
||||||
<RecipeDetail ingredients={INGREDIENT_DATA_TEST} />
|
<RecipeDetail ingredients={INGREDIENT_DATA_TEST} />
|
||||||
<RecommendedRecipes data={RECIPE_DATA_TEST} />
|
<RecommendedRecipes data={RECIPE_DATA_TEST} />
|
||||||
<ReleventProducts />
|
<ReleventProducts data={relevantProducts} collections={collections}/>
|
||||||
<ViewedProducts />
|
<ViewedProducts />
|
||||||
<RelevantBlogPosts data={BLOGS_DATA_TEST} title="relevent blog posts"/>
|
<RelevantBlogPosts data={BLOGS_DATA_TEST} title="relevent blog posts" />
|
||||||
</>
|
</>
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export async function getStaticProps({
|
||||||
|
params,
|
||||||
|
locale,
|
||||||
|
locales,
|
||||||
|
preview,
|
||||||
|
}: GetStaticPropsContext<{ slug: string }>) {
|
||||||
|
const config = { locale, locales }
|
||||||
|
let promisesWithKey = [] as PromiseWithKey[]
|
||||||
|
let props = {} as any
|
||||||
|
|
||||||
|
const product = await commerce.getProduct({
|
||||||
|
variables: { slug: params!.slug },
|
||||||
|
config,
|
||||||
|
preview,
|
||||||
|
})
|
||||||
|
props.product = product
|
||||||
|
|
||||||
|
|
||||||
|
if (!product) {
|
||||||
|
throw new Error(`Product with slug '${params!.slug}' not found`)
|
||||||
|
}
|
||||||
|
|
||||||
|
// relevant product (filter by product detail's facetIds)
|
||||||
|
const relevantFacetIds = product.facetValueIds
|
||||||
|
if (relevantFacetIds && relevantFacetIds.length > 0) {
|
||||||
|
const relevantProductsPromise = commerce.getAllProducts({
|
||||||
|
variables: {
|
||||||
|
first: MAX_PRODUCT_CAROUSEL,
|
||||||
|
facetValueIds: relevantFacetIds,
|
||||||
|
},
|
||||||
|
config,
|
||||||
|
preview,
|
||||||
|
})
|
||||||
|
promisesWithKey.push({ key: 'relevantProducts', promise: relevantProductsPromise, keyResult: 'products' })
|
||||||
|
} else {
|
||||||
|
props.relevantProducts = []
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// collection
|
||||||
|
const collectionsPromise = commerce.getAllCollections({
|
||||||
|
variables: {},
|
||||||
|
config,
|
||||||
|
preview,
|
||||||
|
})
|
||||||
|
promisesWithKey.push({ key: 'collections', promise: collectionsPromise, keyResult: 'collections' })
|
||||||
|
|
||||||
|
|
||||||
|
try {
|
||||||
|
const promises = getAllPromies(promisesWithKey)
|
||||||
|
const rs = await Promise.all(promises)
|
||||||
|
|
||||||
|
promisesWithKey.map((item, index) => {
|
||||||
|
props[item.key] = item.keyResult ? rs[index][item.keyResult] : rs[index]
|
||||||
|
return null
|
||||||
|
})
|
||||||
|
|
||||||
|
if (props.relevantProducts.length > 0) {
|
||||||
|
const relevantProducts = props.relevantProducts.filter((item: Product) => item.id !== product.id)
|
||||||
|
props.relevantProducts = relevantProducts
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
props,
|
||||||
|
revalidate: REVALIDATE_TIME,
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
console.log('err: ', err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
export async function getStaticPaths({ locales }: GetStaticPathsContext) {
|
||||||
|
const { products } = await commerce.getAllProductPaths()
|
||||||
|
|
||||||
|
return {
|
||||||
|
paths: locales
|
||||||
|
? locales.reduce<string[]>((arr, locale) => {
|
||||||
|
// Add a product path for every locale
|
||||||
|
products.forEach((product: any) => {
|
||||||
|
arr.push(`/${locale}/product${product.path}`)
|
||||||
|
})
|
||||||
|
return arr
|
||||||
|
}, [])
|
||||||
|
: products.map((product: any) => `/product${product.path}`),
|
||||||
|
fallback: 'blocking',
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
Slug.Layout = Layout
|
Slug.Layout = Layout
|
||||||
|
@@ -5,7 +5,7 @@ import { GetStaticPropsContext } from 'next';
|
|||||||
import { Layout } from 'src/components/common';
|
import { Layout } from 'src/components/common';
|
||||||
import { ViewedProducts } from 'src/components/modules/product-detail';
|
import { ViewedProducts } from 'src/components/modules/product-detail';
|
||||||
import ProductListFilter from 'src/components/modules/product-list/ProductListFilter/ProductListFilter';
|
import ProductListFilter from 'src/components/modules/product-list/ProductListFilter/ProductListFilter';
|
||||||
import { CODE_FACET_BRAND, CODE_FACET_FEATURED, DEFAULT_PAGE_SIZE } from 'src/utils/constanst.utils';
|
import { CODE_FACET_BRAND, CODE_FACET_FEATURED, DEFAULT_PAGE_SIZE, REVALIDATE_TIME } from 'src/utils/constanst.utils';
|
||||||
import { getAllPromies } from 'src/utils/funtion.utils';
|
import { getAllPromies } from 'src/utils/funtion.utils';
|
||||||
import { PromiseWithKey, SortOrder } from 'src/utils/types.utils';
|
import { PromiseWithKey, SortOrder } from 'src/utils/types.utils';
|
||||||
import ProductListBanner from '../src/components/modules/product-list/ProductListBanner/ProductListBanner';
|
import ProductListBanner from '../src/components/modules/product-list/ProductListBanner/ProductListBanner';
|
||||||
@@ -87,7 +87,7 @@ export async function getStaticProps({
|
|||||||
|
|
||||||
return {
|
return {
|
||||||
props,
|
props,
|
||||||
revalidate: 60,
|
revalidate: REVALIDATE_TIME,
|
||||||
}
|
}
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
|
|
||||||
|
@@ -2,21 +2,13 @@ import { ProductCard } from '@commerce/types/product'
|
|||||||
import { Collection } from '@framework/schema'
|
import { Collection } from '@framework/schema'
|
||||||
import React, { useMemo } from 'react'
|
import React, { useMemo } from 'react'
|
||||||
import { OPTION_ALL, QUERY_KEY, ROUTE } from 'src/utils/constanst.utils'
|
import { OPTION_ALL, QUERY_KEY, ROUTE } from 'src/utils/constanst.utils'
|
||||||
|
import { getCategoryNameFromCollectionId } from 'src/utils/funtion.utils'
|
||||||
import { CollectionCarcousel } from '..'
|
import { CollectionCarcousel } from '..'
|
||||||
interface FreshProductsProps {
|
interface FreshProductsProps {
|
||||||
data: ProductCard[]
|
data: ProductCard[]
|
||||||
collections: Collection[]
|
collections: Collection[]
|
||||||
}
|
}
|
||||||
|
|
||||||
const getCategoryNameFromCollectionId = (colelctions: Collection[], collectionId?: string ) => {
|
|
||||||
if (!collectionId) {
|
|
||||||
return ''
|
|
||||||
}
|
|
||||||
|
|
||||||
const collection = colelctions.find(item => item.id === collectionId)
|
|
||||||
return collection?.name || ''
|
|
||||||
}
|
|
||||||
|
|
||||||
const FreshProducts = ({ data, collections }: FreshProductsProps) => {
|
const FreshProducts = ({ data, collections }: FreshProductsProps) => {
|
||||||
const dataWithCategory = useMemo(() => {
|
const dataWithCategory = useMemo(() => {
|
||||||
return data.map(item => {
|
return data.map(item => {
|
||||||
|
@@ -1,13 +1,34 @@
|
|||||||
import React from 'react';
|
import { ProductCard } from '@commerce/types/product';
|
||||||
|
import { Collection } from '@framework/schema';
|
||||||
|
import React, { useMemo } from 'react';
|
||||||
import ListProductWithInfo from 'src/components/common/ListProductWithInfo/ListProductWithInfo';
|
import ListProductWithInfo from 'src/components/common/ListProductWithInfo/ListProductWithInfo';
|
||||||
import { PRODUCT_DATA_TEST } from 'src/utils/demo-data';
|
import { getCategoryNameFromCollectionId } from 'src/utils/funtion.utils';
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
data: ProductCard[]
|
||||||
|
collections: Collection[]
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
const ReleventProducts = ({ data, collections }: Props) => {
|
||||||
|
const dataWithCategoryName = useMemo(() => {
|
||||||
|
return data.map(item => {
|
||||||
|
return {
|
||||||
|
...item,
|
||||||
|
collection: getCategoryNameFromCollectionId(collections, item.collectionIds ? item.collectionIds[0] : undefined)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}, [data, collections])
|
||||||
|
|
||||||
|
if (data.length === 0) {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
|
||||||
const ReleventProducts = () => {
|
|
||||||
return (
|
return (
|
||||||
<ListProductWithInfo
|
<ListProductWithInfo
|
||||||
title="Relevant Products"
|
title="Relevant Products"
|
||||||
subtitle="Last call! Shop deep deals on 100+ bulk picks while you can."
|
subtitle="Last call! Shop deep deals on 100+ bulk picks while you can."
|
||||||
data={PRODUCT_DATA_TEST}
|
data={dataWithCategoryName}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
@@ -1,5 +1,7 @@
|
|||||||
import DefaultImg from '../../public/assets/images/default_img.jpg'
|
import DefaultImg from '../../public/assets/images/default_img.jpg'
|
||||||
|
|
||||||
|
export const REVALIDATE_TIME = 60
|
||||||
|
export const MAX_PRODUCT_CAROUSEL = 20
|
||||||
export const BLUR_DATA_IMG = 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAQAAAC1HAwCAAAAC0lEQVR42mN8fBIAApUBruKYvzsAAAAASUVORK5CYII='
|
export const BLUR_DATA_IMG = 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAQAAAC1HAwCAAAAC0lEQVR42mN8fBIAApUBruKYvzsAAAAASUVORK5CYII='
|
||||||
export const DEFAULT_IMG = DefaultImg
|
export const DEFAULT_IMG = DefaultImg
|
||||||
|
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
import { Facet } from "@commerce/types/facet";
|
import { Facet } from "@commerce/types/facet";
|
||||||
import { FacetValue, SearchResultSortParameter } from './../../framework/vendure/schema.d';
|
import { Collection, FacetValue, SearchResultSortParameter } from './../../framework/vendure/schema.d';
|
||||||
import { CODE_FACET_DISCOUNT, CODE_FACET_FEATURED, CODE_FACET_FEATURED_VARIANT, PRODUCT_SORT_OPTION_VALUE } from "./constanst.utils";
|
import { CODE_FACET_DISCOUNT, CODE_FACET_FEATURED, CODE_FACET_FEATURED_VARIANT, PRODUCT_SORT_OPTION_VALUE } from "./constanst.utils";
|
||||||
import { PromiseWithKey, SortOrder } from "./types.utils";
|
import { PromiseWithKey, SortOrder } from "./types.utils";
|
||||||
|
|
||||||
@@ -116,6 +116,15 @@ export function getFacetIdsFromCodes(facets: FacetValue[], codes?: string[]): st
|
|||||||
return ids
|
return ids
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const getCategoryNameFromCollectionId = (colelctions: Collection[], collectionId?: string ) => {
|
||||||
|
if (!collectionId) {
|
||||||
|
return ''
|
||||||
|
}
|
||||||
|
|
||||||
|
const collection = colelctions.find(item => item.id === collectionId)
|
||||||
|
return collection?.name || ''
|
||||||
|
}
|
||||||
|
|
||||||
export function getAllPromies(promies: PromiseWithKey[]) {
|
export function getAllPromies(promies: PromiseWithKey[]) {
|
||||||
return promies.map(item => item.promise)
|
return promies.map(item => item.promise)
|
||||||
}
|
}
|
Reference in New Issue
Block a user