From 01895e76e2a42df9e662722c7f0f246373f9ceef Mon Sep 17 00:00:00 2001 From: darthmaim Date: Wed, 20 Dec 2023 14:40:37 +0100 Subject: [PATCH 1/4] Show correct item name/icon in tooltips for old revisions --- apps/web/app/[language]/item/[id]/component.tsx | 7 +++++-- apps/web/components/Item/ItemLinkTooltip.tsx | 15 +++++++++------ apps/web/components/Item/ItemTooltip.client.tsx | 15 +++++++++++---- apps/web/components/Item/ItemTooltip.module.css | 10 ++++++++++ apps/web/components/Item/ItemTooltip.tsx | 11 +++++++++-- 5 files changed, 44 insertions(+), 14 deletions(-) diff --git a/apps/web/app/[language]/item/[id]/component.tsx b/apps/web/app/[language]/item/[id]/component.tsx index c6018280a..11e10661e 100644 --- a/apps/web/app/[language]/item/[id]/component.tsx +++ b/apps/web/app/[language]/item/[id]/component.tsx @@ -40,6 +40,7 @@ import { ContentChanceColumn, ContentQuantityColumn, ItemContentQuantityColumn } import type { TODO } from '@/lib/todo'; import { pageView } from '@/lib/pageView'; import { GuildUpgradeLink } from '@/components/GuildUpgrade/GuildUpgradeLink'; +import { parseIcon } from '@/lib/parseIcon'; export interface ItemPageComponentProps { language: Language; @@ -74,10 +75,12 @@ export const ItemPageComponent: AsyncComponent = async ( const compareByName = compareLocalizedName(language); + const icon = parseIcon(data.icon); + return ( } @@ -94,7 +97,7 @@ export const ItemPageComponent: AsyncComponent = async ( )} Tooltip - + {item.unlocksSkinIds.length > 0 && ( <> diff --git a/apps/web/components/Item/ItemLinkTooltip.tsx b/apps/web/components/Item/ItemLinkTooltip.tsx index 2976517f2..31afbc048 100644 --- a/apps/web/components/Item/ItemLinkTooltip.tsx +++ b/apps/web/components/Item/ItemLinkTooltip.tsx @@ -30,13 +30,16 @@ export const ItemLinkTooltip: FC = ({ item, language, revi return (
-
- {item.icon && ()} - {localizedName(item, language)} -
- Error}> - {tooltip.loading &&

} + {tooltip.loading && ( + <> +
+ {item.icon && ()} + {localizedName(item, language)} +
+

+ + )} {!tooltip.loading && }
diff --git a/apps/web/components/Item/ItemTooltip.client.tsx b/apps/web/components/Item/ItemTooltip.client.tsx index 3e94a1fbd..99ff1f8fa 100644 --- a/apps/web/components/Item/ItemTooltip.client.tsx +++ b/apps/web/components/Item/ItemTooltip.client.tsx @@ -1,4 +1,4 @@ -import { type FC, Fragment } from 'react'; +import { type FC, Fragment, type ReactNode } from 'react'; import { FormatNumber } from '../Format/FormatNumber'; import { ItemTooltip } from './ItemTooltip'; import { Rarity } from './Rarity'; @@ -10,11 +10,11 @@ import { Icon } from '@gw2treasures/ui'; import { EntityIcon } from '@/components/Entity/EntityIcon'; import { DyeColor } from '../Color/DyeColor'; import { hexToRgb } from '../Color/hex-to-rgb'; -import { FlexRow } from '@gw2treasures/ui/components/Layout/FlexRow'; import { Tip } from '@gw2treasures/ui/components/Tip/Tip'; export interface ClientItemTooltipProps { tooltip: ItemTooltip; + hideTitle?: boolean; }; function renderAttributes(attributes: ItemTooltip['attributes']) { @@ -97,8 +97,8 @@ function renderConsumable(consumable: ItemTooltip['consumable']) { ); } -export const ClientItemTooltip: FC = ({ tooltip }) => { - const data = [ +export const ClientItemTooltip: FC = ({ tooltip, hideTitle = false }) => { + const data: ReactNode[] = [ tooltip.weaponStrength && (<>{tooltip.weaponStrength.label}: ), tooltip.defense && <>{tooltip.defense.label}: , renderAttributes(tooltip.attributes), @@ -158,6 +158,13 @@ export const ClientItemTooltip: FC = ({ tooltip }) => { return (
+ {!hideTitle && ( +
+ {tooltip.icon && ()} + {tooltip.name} +
+ )} + {data.filter(isTruthy).map((content, index) => { // eslint-disable-next-line react/no-array-index-key return
{content}
; diff --git a/apps/web/components/Item/ItemTooltip.module.css b/apps/web/components/Item/ItemTooltip.module.css index efdabf7d3..ad92c067d 100644 --- a/apps/web/components/Item/ItemTooltip.module.css +++ b/apps/web/components/Item/ItemTooltip.module.css @@ -1,3 +1,13 @@ +.title { + display: flex; + align-items: center; + gap: 8px; + margin-bottom: 8px; + + font-family: var(--font-bitter); + color: var(--color-rarity); +} + .row + .row { margin-top: 8px; } diff --git a/apps/web/components/Item/ItemTooltip.tsx b/apps/web/components/Item/ItemTooltip.tsx index 19488f431..d5f2f16d5 100644 --- a/apps/web/components/Item/ItemTooltip.tsx +++ b/apps/web/components/Item/ItemTooltip.tsx @@ -16,13 +16,14 @@ import type { RGB } from '../Color/types'; export interface ItemTooltipProps { item: Gw2Api.Item; language: Language; + hideTitle?: boolean; } -export const ItemTooltip: AsyncComponent = async ({ item, language }) => { +export const ItemTooltip: AsyncComponent = async ({ item, language, hideTitle }) => { const tooltip = await createTooltip(item, language); return ( - + ); }; @@ -65,8 +66,12 @@ export async function createTooltip(item: Gw2Api.Item, language: Language): Prom ? await db.color.findUnique({ where: { id: item.details.color_id }}) : null; + const icon = parseIcon(item.icon); + return { language, + name: item.name, + icon, weaponStrength: item.type === 'Weapon' ? { label: 'Strength', min: item.details?.min_power ?? 0, max: item.details?.max_power ?? 0 } : undefined, defense: item.type === 'Armor' ? { label: 'Defense', value: item.details?.defense ?? 0 } : undefined, attributes: item.details?.infix_upgrade?.attributes && item.details.infix_upgrade.attributes.length > 0 ? item.details.infix_upgrade.attributes.map((({ attribute, modifier }) => ({ label: t(`attribute.${attribute}`), value: modifier }))) : undefined, @@ -121,6 +126,8 @@ export type ItemWithAttributes = WithIcon & { export interface ItemTooltip { language: Language, + name: string, + icon?: { id: number, signature: string }, weaponStrength?: { label: string, min: number, max: number }, defense?: { label: string, value: number }, attributes?: { label: string, value: number }[], From 0b6a7f75b5000d50058211adb14026cf0013c647 Mon Sep 17 00:00:00 2001 From: darthmaim Date: Wed, 20 Dec 2023 19:51:22 +0100 Subject: [PATCH 2/4] Add item diff page --- .../app/[language]/item/[id]/component.tsx | 10 ++- .../app/[language]/item/diff/[a]/[b]/page.tsx | 78 +++++++++++++++++++ .../components/Item/ItemTooltip.client.tsx | 40 +++++----- 3 files changed, 109 insertions(+), 19 deletions(-) create mode 100644 apps/web/app/[language]/item/diff/[a]/[b]/page.tsx diff --git a/apps/web/app/[language]/item/[id]/component.tsx b/apps/web/app/[language]/item/[id]/component.tsx index 11e10661e..e3007eb00 100644 --- a/apps/web/app/[language]/item/[id]/component.tsx +++ b/apps/web/app/[language]/item/[id]/component.tsx @@ -40,6 +40,7 @@ import { ContentChanceColumn, ContentQuantityColumn, ItemContentQuantityColumn } import type { TODO } from '@/lib/todo'; import { pageView } from '@/lib/pageView'; import { GuildUpgradeLink } from '@/components/GuildUpgrade/GuildUpgradeLink'; +import { FlexRow } from '@gw2treasures/ui/components/Layout/FlexRow'; import { parseIcon } from '@/lib/parseIcon'; export interface ItemPageComponentProps { @@ -270,7 +271,14 @@ export const ItemPageComponent: AsyncComponent = async ( - {history.revisionId !== revision.id && View} + + {history.revisionId !== revision.id && ( + + View · + Compare + + )} + ))} diff --git a/apps/web/app/[language]/item/diff/[a]/[b]/page.tsx b/apps/web/app/[language]/item/diff/[a]/[b]/page.tsx new file mode 100644 index 000000000..c34fa1e3e --- /dev/null +++ b/apps/web/app/[language]/item/diff/[a]/[b]/page.tsx @@ -0,0 +1,78 @@ +import Link from 'next/link'; +import type { Gw2Api } from 'gw2-api-types'; +import { DiffLayout, DiffLayoutHeader, DiffLayoutRow } from '@/components/Layout/DiffLayout'; +import { EntityIcon } from '@/components/Entity/EntityIcon'; +import { parseIcon } from '@/lib/parseIcon'; +import { FormatDate } from '@/components/Format/FormatDate'; +import { Notice } from '@gw2treasures/ui/components/Notice/Notice'; +import { Separator } from '@gw2treasures/ui/components/Layout/Separator'; +import { Json } from '@/components/Format/Json'; +import { notFound } from 'next/navigation'; +import { Fragment } from 'react'; +import { db } from '@/lib/prisma'; +import { remember } from '@/lib/remember'; +import { ClientItemTooltip } from '@/components/Item/ItemTooltip.client'; +import { createTooltip } from '@/components/Item/ItemTooltip'; +import type { Language } from '@gw2treasures/database'; + +const getRevisions = remember(60, async function getRevisions(idA: string, idB: string) { + const [a, b] = await Promise.all([ + db?.revision.findUnique({ where: { id: idA, entity: 'Item' }}), + db?.revision.findUnique({ where: { id: idB, entity: 'Item' }}), + ]); + + if(!a || !b) { + notFound(); + } + + return { a, b }; +}); + +export default async function ItemDiffPage({ params }: { params: { a: string, b: string, language: Language }}) { + const idA = params.a.toString(); + const idB = params.b.toString(); + + const { a, b } = await getRevisions(idA, idB); + + const dataA: Gw2Api.Item = JSON.parse(a.data); + const dataB: Gw2Api.Item = JSON.parse(b.data); + + const iconA = parseIcon(dataA.icon); + const iconB = parseIcon(dataB.icon); + + const tooltipA = await createTooltip(dataA, params.language); + const tooltipB = await createTooltip(dataB, params.language); + + return ( + + , + iconB && , + ]} title={[ + dataA.name, + dataB.name, + ]} subtitle={[ + (Build {a.buildId}) ▪ View revision, + (Build {b.buildId}) ▪ View revision, + ]}/> + + {dataA.id !== dataB.id && ( +
+ You are comparing two different items +
+ )} + + {a.createdAt > b.createdAt && ( +
+ You are comparing an old version against a newer version. Switch around +
+ )} + + } right={}/> + + } right={}/> + } right={} changed/> +
+ ); +}; + diff --git a/apps/web/components/Item/ItemTooltip.client.tsx b/apps/web/components/Item/ItemTooltip.client.tsx index 99ff1f8fa..0866069a0 100644 --- a/apps/web/components/Item/ItemTooltip.client.tsx +++ b/apps/web/components/Item/ItemTooltip.client.tsx @@ -98,7 +98,27 @@ function renderConsumable(consumable: ItemTooltip['consumable']) { } export const ClientItemTooltip: FC = ({ tooltip, hideTitle = false }) => { - const data: ReactNode[] = [ + const data = renderItemTooltipRows(tooltip); + + return ( +
+ {!hideTitle && ( +
+ {tooltip.icon && ()} + {tooltip.name} +
+ )} + + {data.filter(isTruthy).map((content, index) => { + // eslint-disable-next-line react/no-array-index-key + return
{content}
; + })} +
+ ); +}; + +export function renderItemTooltipRows(tooltip: ItemTooltip): ReactNode[] { + return [ tooltip.weaponStrength && (<>{tooltip.weaponStrength.label}: ), tooltip.defense && <>{tooltip.defense.label}: , renderAttributes(tooltip.attributes), @@ -155,20 +175,4 @@ export const ClientItemTooltip: FC = ({ tooltip, hideTit ...tooltip.flags, tooltip.value && (), ]; - - return ( -
- {!hideTitle && ( -
- {tooltip.icon && ()} - {tooltip.name} -
- )} - - {data.filter(isTruthy).map((content, index) => { - // eslint-disable-next-line react/no-array-index-key - return
{content}
; - })} -
- ); -}; +} From 8d8ea57f95106e03e34cb6d3181a80928f7cfb65 Mon Sep 17 00:00:00 2001 From: darthmaim Date: Wed, 20 Dec 2023 20:15:07 +0100 Subject: [PATCH 3/4] Render weaponStrength and defense item diff --- .../app/[language]/item/diff/[a]/[b]/page.tsx | 5 +++- .../components/Item/ItemTooltip.client.tsx | 26 +++++++++++++++++-- apps/web/components/Layout/DiffLayout.tsx | 4 +++ 3 files changed, 32 insertions(+), 3 deletions(-) diff --git a/apps/web/app/[language]/item/diff/[a]/[b]/page.tsx b/apps/web/app/[language]/item/diff/[a]/[b]/page.tsx index c34fa1e3e..db9f0e971 100644 --- a/apps/web/app/[language]/item/diff/[a]/[b]/page.tsx +++ b/apps/web/app/[language]/item/diff/[a]/[b]/page.tsx @@ -11,7 +11,7 @@ import { notFound } from 'next/navigation'; import { Fragment } from 'react'; import { db } from '@/lib/prisma'; import { remember } from '@/lib/remember'; -import { ClientItemTooltip } from '@/components/Item/ItemTooltip.client'; +import { ClientItemTooltip, renderDefense, renderWeaponStrength } from '@/components/Item/ItemTooltip.client'; import { createTooltip } from '@/components/Item/ItemTooltip'; import type { Language } from '@gw2treasures/database'; @@ -68,6 +68,9 @@ export default async function ItemDiffPage({ params }: { params: { a: string, b:
)} + + + } right={}/> } right={}/> diff --git a/apps/web/components/Item/ItemTooltip.client.tsx b/apps/web/components/Item/ItemTooltip.client.tsx index 0866069a0..aade45b69 100644 --- a/apps/web/components/Item/ItemTooltip.client.tsx +++ b/apps/web/components/Item/ItemTooltip.client.tsx @@ -119,8 +119,8 @@ export const ClientItemTooltip: FC = ({ tooltip, hideTit export function renderItemTooltipRows(tooltip: ItemTooltip): ReactNode[] { return [ - tooltip.weaponStrength && (<>{tooltip.weaponStrength.label}: ), - tooltip.defense && <>{tooltip.defense.label}: , + renderWeaponStrength(tooltip), + renderDefense(tooltip), renderAttributes(tooltip.attributes), tooltip.buff && (

), renderConsumable(tooltip.consumable), @@ -176,3 +176,25 @@ export function renderItemTooltipRows(tooltip: ItemTooltip): ReactNode[] { tooltip.value && (), ]; } + +export function renderWeaponStrength({ weaponStrength }: ItemTooltip) { + if(!weaponStrength) { + return; + } + + return ( + <> + {weaponStrength.label}: + + ); +} + +export function renderDefense({ defense }: ItemTooltip) { + if(!defense) { + return; + } + + return ( + <>{defense.label}: + ); +} diff --git a/apps/web/components/Layout/DiffLayout.tsx b/apps/web/components/Layout/DiffLayout.tsx index 843d5b2d8..077e20595 100644 --- a/apps/web/components/Layout/DiffLayout.tsx +++ b/apps/web/components/Layout/DiffLayout.tsx @@ -45,6 +45,10 @@ interface DiffLayoutRowProps { }; export const DiffLayoutRow: FC = ({ left, right, changed = false }) => { + if(!left && !right) { + return null; + } + return (

{left}
From b80e891d4b9730d72417ac9733bb61f44d0e6f81 Mon Sep 17 00:00:00 2001 From: darthmaim Date: Tue, 26 Dec 2023 19:51:34 +0100 Subject: [PATCH 4/4] WIP itemdiff --- apps/web/app/[language]/item/diff/[a]/[b]/page.tsx | 13 +++++++++++-- apps/web/components/Item/ItemTooltip.client.tsx | 2 +- 2 files changed, 12 insertions(+), 3 deletions(-) diff --git a/apps/web/app/[language]/item/diff/[a]/[b]/page.tsx b/apps/web/app/[language]/item/diff/[a]/[b]/page.tsx index db9f0e971..86d4cddf3 100644 --- a/apps/web/app/[language]/item/diff/[a]/[b]/page.tsx +++ b/apps/web/app/[language]/item/diff/[a]/[b]/page.tsx @@ -11,7 +11,7 @@ import { notFound } from 'next/navigation'; import { Fragment } from 'react'; import { db } from '@/lib/prisma'; import { remember } from '@/lib/remember'; -import { ClientItemTooltip, renderDefense, renderWeaponStrength } from '@/components/Item/ItemTooltip.client'; +import { renderDefense, renderWeaponStrength } from '@/components/Item/ItemTooltip.client'; import { createTooltip } from '@/components/Item/ItemTooltip'; import type { Language } from '@gw2treasures/database'; @@ -71,7 +71,16 @@ export default async function ItemDiffPage({ params }: { params: { a: string, b: - } right={}/> + {(tooltipA.attributes || tooltipB.attributes) && ( + [...Array(Math.max(tooltipA.attributes?.length ?? 0, tooltipB.attributes?.length ?? 0)).keys()].map((_, i) => ( + + )) + )} + + {/* } right={}/> */} } right={}/> } right={} changed/> diff --git a/apps/web/components/Item/ItemTooltip.client.tsx b/apps/web/components/Item/ItemTooltip.client.tsx index aade45b69..adf1cd7f2 100644 --- a/apps/web/components/Item/ItemTooltip.client.tsx +++ b/apps/web/components/Item/ItemTooltip.client.tsx @@ -17,7 +17,7 @@ export interface ClientItemTooltipProps { hideTitle?: boolean; }; -function renderAttributes(attributes: ItemTooltip['attributes']) { +export function renderAttributes(attributes: ItemTooltip['attributes']) { if(!attributes) { return; }