Skip to content

Commit fc3372d

Browse files
authored
Refactor category metadata handling and enhance product banner styles (#1531)
- Introduced a new utility function `getCategoryMetadata` to streamline metadata retrieval for categories. - Removed outdated metadata checks in `generateMetadata` function, improving clarity and efficiency. - Updated styles in the `ProductBanner` component for improved responsiveness and visual appeal, including hover effects and spacing adjustments.
2 parents b3484c3 + ef8d155 commit fc3372d

File tree

3 files changed

+52
-16
lines changed

3 files changed

+52
-16
lines changed

frontend/src/app/apps/category/[category]/page.tsx

+2-8
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ import {
1212
generateCollectionPageSchema,
1313
generateBreadcrumbSchema,
1414
generateAppListSchema,
15+
getCategoryMetadata,
1516
} from '../../utils/metadata';
1617
import { ProductBanner } from '@/src/app/components/product-banner';
1718
import { getAppsByCategory } from '@/src/lib/api/apps';
@@ -41,14 +42,7 @@ async function getCategoryData(category: string) {
4142
export async function generateMetadata({ params }: CategoryPageProps): Promise<Metadata> {
4243
const { category } = params;
4344
const { categoryPlugins } = await getCategoryData(category);
44-
const metadata = categoryMetadata[category];
45-
46-
if (!metadata) {
47-
return {
48-
title: 'Category Not Found - OMI Apps',
49-
description: 'The requested category could not be found.',
50-
};
51-
}
45+
const metadata = getCategoryMetadata(category);
5246

5347
const title = `${metadata.title} - OMI Apps Marketplace`;
5448
const description = `${metadata.description} Browse ${categoryPlugins.length}+ ${category} apps for your OMI Necklace.`;

frontend/src/app/apps/utils/metadata.ts

+42
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
import { Metadata } from 'next';
22
import envConfig from '@/src/constants/envConfig';
3+
import { getCategoryMetadata as getUICategoryMetadata } from './category';
34

45
export interface CategoryMetadata {
56
title: string;
@@ -71,6 +72,30 @@ export const categoryMetadata: Record<string, CategoryMetadata> = {
7172
'service integration',
7273
],
7374
},
75+
utility: {
76+
title: 'Utility Apps for OMI Necklace',
77+
description:
78+
'Essential utility apps for your OMI Necklace. Access practical tools and helpful functions through voice commands and AI assistance.',
79+
keywords: [
80+
'utility apps',
81+
'tools',
82+
'practical apps',
83+
'voice commands',
84+
'AI assistance',
85+
],
86+
},
87+
lifestyle: {
88+
title: 'Lifestyle Apps for OMI Necklace',
89+
description:
90+
'Enhance your daily life with OMI lifestyle apps. From personal organization to habit tracking, make everyday tasks more intuitive with AI assistance.',
91+
keywords: [
92+
'lifestyle apps',
93+
'personal organization',
94+
'habit tracking',
95+
'daily assistance',
96+
'AI companion',
97+
],
98+
},
7499
};
75100

76101
const productInfo = {
@@ -207,3 +232,20 @@ export function getBaseMetadata(title: string, description: string): Metadata {
207232
},
208233
};
209234
}
235+
236+
export function getCategoryMetadata(category: string): CategoryMetadata {
237+
const uiMetadata = getUICategoryMetadata(category);
238+
239+
return {
240+
title: `${uiMetadata.displayName} Apps for OMI Necklace`,
241+
description: `${uiMetadata.description}. Browse and discover apps designed for voice control and ambient computing.`,
242+
keywords: [
243+
`${category.toLowerCase()} apps`,
244+
'OMI apps',
245+
'voice control',
246+
'AI assistant',
247+
'wearable apps',
248+
...uiMetadata.displayName.toLowerCase().split(' '),
249+
],
250+
};
251+
}

frontend/src/app/components/product-banner/index.tsx

+8-8
Original file line numberDiff line numberDiff line change
@@ -144,12 +144,12 @@ export function ProductBanner({
144144
href={PRODUCT_INFO.url}
145145
target="_blank"
146146
rel="noopener noreferrer"
147-
className="group relative inline-flex items-center justify-center overflow-hidden rounded-xl bg-gradient-to-r from-indigo-500 via-purple-500 to-pink-500 p-0.5 transition-all duration-300 ease-out hover:bg-gradient-to-br hover:shadow-lg hover:shadow-indigo-500/25"
147+
className="group relative inline-flex w-full items-center justify-center overflow-hidden rounded-xl bg-gradient-to-r from-indigo-500 via-purple-500 to-pink-500 transition-all duration-300 ease-out hover:scale-[1.02] hover:shadow-lg hover:shadow-indigo-500/25 active:scale-[0.98] sm:w-auto"
148148
>
149-
<span className="relative inline-flex items-center gap-2 rounded-[0.625rem] bg-[#1A1F2E] px-6 py-3 text-base font-medium text-white transition-all duration-300 group-hover:bg-opacity-90 sm:text-lg">
149+
<span className="relative inline-flex items-center gap-2 px-6 py-3.5 text-base font-semibold text-white transition-all duration-300 sm:px-8 sm:py-4 sm:text-lg">
150150
<span>Order Now</span>
151151
<svg
152-
className="h-5 w-5 transition-transform duration-300 group-hover:translate-x-1"
152+
className="h-4 w-4 transition-transform duration-300 group-hover:translate-x-1 sm:h-5 sm:w-5"
153153
fill="none"
154154
stroke="currentColor"
155155
viewBox="0 0 24 24"
@@ -243,12 +243,12 @@ export function ProductBanner({
243243
href={PRODUCT_INFO.url}
244244
target="_blank"
245245
rel="noopener noreferrer"
246-
className="relative mt-3 inline-flex w-full items-center justify-center overflow-hidden rounded-xl bg-gradient-to-r from-indigo-500 via-purple-500 to-pink-500 p-0.5 transition-all duration-300 ease-out hover:bg-gradient-to-br hover:shadow-lg hover:shadow-indigo-500/25"
246+
className="group relative mt-4 inline-flex w-full items-center justify-center overflow-hidden rounded-xl bg-gradient-to-r from-indigo-500 via-purple-500 to-pink-500 transition-all duration-300 ease-out hover:scale-[1.02] hover:shadow-lg hover:shadow-indigo-500/25 active:scale-[0.98]"
247247
>
248-
<span className="relative inline-flex w-full items-center justify-center gap-2 rounded-[0.625rem] bg-[#1A1F2E] px-4 py-2 text-sm font-medium text-white transition-all duration-300 group-hover:bg-opacity-90">
248+
<span className="relative inline-flex items-center gap-2 px-3 py-2 text-xs font-semibold text-white transition-all duration-300 sm:gap-2 sm:px-4 sm:py-2.5 sm:text-sm md:px-6 md:py-3 md:text-base">
249249
<span>Order Now</span>
250250
<svg
251-
className="h-4 w-4 transition-transform duration-300 group-hover:translate-x-1"
251+
className="h-3 w-3 transition-transform duration-300 group-hover:translate-x-1 sm:h-4 sm:w-4 md:h-5 md:w-5"
252252
fill="none"
253253
stroke="currentColor"
254254
viewBox="0 0 24 24"
@@ -307,9 +307,9 @@ export function ProductBanner({
307307
href={PRODUCT_INFO.url}
308308
target="_blank"
309309
rel="noopener noreferrer"
310-
className="group relative inline-flex items-center justify-center overflow-hidden rounded-lg bg-gradient-to-r from-indigo-500 via-purple-500 to-pink-500 p-0.5 transition-all duration-300 ease-out hover:bg-gradient-to-br hover:shadow-lg hover:shadow-indigo-500/25"
310+
className="group relative inline-flex items-center justify-center overflow-hidden rounded-xl bg-gradient-to-r from-indigo-500 via-purple-500 to-pink-500 transition-all duration-300 ease-out hover:scale-[1.02] hover:shadow-lg hover:shadow-indigo-500/25 active:scale-[0.98]"
311311
>
312-
<span className="relative inline-flex items-center gap-1 rounded-[0.375rem] bg-[#1A1F2E] px-2 py-1 text-xs font-medium text-white transition-all duration-300 group-hover:bg-opacity-90 sm:gap-2 sm:px-4 sm:py-2 sm:text-sm md:px-6 md:py-3">
312+
<span className="relative inline-flex items-center gap-2 px-3 py-2 text-xs font-semibold text-white transition-all duration-300 sm:gap-2 sm:px-4 sm:py-2.5 sm:text-sm md:px-6 md:py-3 md:text-base">
313313
<span>Order Now</span>
314314
<svg
315315
className="h-3 w-3 transition-transform duration-300 group-hover:translate-x-1 sm:h-4 sm:w-4 md:h-5 md:w-5"

0 commit comments

Comments
 (0)