From c38764432274d5fa0aca3e42a837b3d6ff031a52 Mon Sep 17 00:00:00 2001 From: Mikael Brevik Date: Thu, 12 Dec 2024 15:59:15 +0100 Subject: [PATCH 1/3] refactor: reuse contact box in case pages (#1035) --- .../contactInformation/ContactInformation.tsx | 84 ++++--------------- .../contactInformation/ContactSelector.tsx | 74 ---------------- .../customerCase/contactInformation/api.ts | 65 ++++++++++++++ .../contactInformation.module.css | 74 ---------------- .../sections/contact-box/ContactBox.tsx | 45 ++-------- .../contact-box/ContactBoxContent.tsx | 61 ++++++++++++++ .../contact-box/contact-box.module.css | 3 +- src/components/sections/contact-box/types.ts | 2 +- 8 files changed, 152 insertions(+), 256 deletions(-) delete mode 100644 src/components/customerCases/customerCase/contactInformation/ContactSelector.tsx create mode 100644 src/components/customerCases/customerCase/contactInformation/api.ts delete mode 100644 src/components/customerCases/customerCase/contactInformation/contactInformation.module.css create mode 100644 src/components/sections/contact-box/ContactBoxContent.tsx diff --git a/src/components/customerCases/customerCase/contactInformation/ContactInformation.tsx b/src/components/customerCases/customerCase/contactInformation/ContactInformation.tsx index 471a2828d..3afa9760d 100644 --- a/src/components/customerCases/customerCase/contactInformation/ContactInformation.tsx +++ b/src/components/customerCases/customerCase/contactInformation/ContactInformation.tsx @@ -1,83 +1,33 @@ import { getTranslations } from "next-intl/server"; import { Suspense } from "react"; -import { EmployeeCardSkeleton } from "src/components/employeeCard/EmployeeCard"; -import Text from "src/components/text/Text"; -import { fetchEmployeesByEmails } from "src/utils/employees"; -import { CompanyLocation } from "studio/lib/interfaces/companyDetails"; -import { COMPANY_LOCATIONS_QUERY } from "studio/lib/queries/admin"; -import { EMPLOYEE_PAGE_SLUG_QUERY } from "studio/lib/queries/siteSettings"; -import { loadStudioQuery } from "studio/lib/store"; +import ContactBoxContent from "src/components/sections/contact-box/ContactBoxContent"; -import styles from "./contactInformation.module.css"; -import ContactSelector, { ContactByLocationMap } from "./ContactSelector"; +import { fetchLocationsWithContact, getEmployeePageSlug } from "./api"; interface ContactInformationProps { language: string; } -async function fetchContactByLocationMap( - companyLocations: CompanyLocation[], -): Promise { - const contactByLocation: ContactByLocationMap = {}; - for (const location of companyLocations) { - if ( - location.contactPerson === undefined || - location.contactPerson.length === 0 - ) { - continue; - } - const contactRes = await fetchEmployeesByEmails([location.contactPerson]); - if (!contactRes.ok || contactRes.value.length === 0) { - continue; - } - contactByLocation[location._id] = contactRes.value[0]; - } - return contactByLocation; -} - export default async function ContactInformation({ language, }: ContactInformationProps) { - const employeePageSlug = ( - await loadStudioQuery<{ slug: string } | null>(EMPLOYEE_PAGE_SLUG_QUERY, { - language, - }) - ).data?.slug; - const companyLocations = ( - await loadStudioQuery(COMPANY_LOCATIONS_QUERY, {}) - ).data; - const contactByLocation = await fetchContactByLocationMap(companyLocations); - const locationIdsWithContact = Object.keys(contactByLocation); - const locationsWithContact = companyLocations.filter((location) => - locationIdsWithContact.includes(location._id), - ); + const employeePageSlug = await getEmployeePageSlug(language); + const contactPoints = fetchLocationsWithContact(); + const t = await getTranslations("contact_information"); return ( - <> -
-
-
- - {t("help")} - - {t("contact_sale")} -
-
- }> - - -
-
-
- + Loading...}> + + ); } diff --git a/src/components/customerCases/customerCase/contactInformation/ContactSelector.tsx b/src/components/customerCases/customerCase/contactInformation/ContactSelector.tsx deleted file mode 100644 index 507db68d1..000000000 --- a/src/components/customerCases/customerCase/contactInformation/ContactSelector.tsx +++ /dev/null @@ -1,74 +0,0 @@ -"use client"; - -import { useState } from "react"; - -import EmployeeCard from "src/components/employeeCard/EmployeeCard"; -import { Tag } from "src/components/tag"; -import { ChewbaccaEmployee } from "src/types/employees"; -import { CompanyLocation } from "studio/lib/interfaces/companyDetails"; - -import styles from "./contactInformation.module.css"; - -export type ContactByLocationMap = { - [locationId: string]: ChewbaccaEmployee; -}; - -export interface ContactSelectorProps { - language: string; - locations: CompanyLocation[]; - contactByLocation: ContactByLocationMap; - employeePageSlug?: string; - background: "dark" | "light"; -} - -export default function ContactSelector({ - language, - locations, - contactByLocation, - employeePageSlug, - background = "light", -}: ContactSelectorProps) { - const locationIds = Object.keys(contactByLocation); - - const [selectedLocationId, setSelectedLocationId] = useState( - locationIds[0], - ); - - if (locationIds.length === 0) { - return; - } - - const selectedOrDefaultLocationId = selectedLocationId ?? locationIds[0]; - - return ( - <> -
-
- {locations.map((location) => ( - setSelectedLocationId(location._id)} - text={location.companyLocationName} - /> - ))} -
-
-
- -
-
-
- - ); -} diff --git a/src/components/customerCases/customerCase/contactInformation/api.ts b/src/components/customerCases/customerCase/contactInformation/api.ts new file mode 100644 index 000000000..9d8264c20 --- /dev/null +++ b/src/components/customerCases/customerCase/contactInformation/api.ts @@ -0,0 +1,65 @@ +import { EmployeeAndMetadata } from "src/components/sections/contact-box/types"; +import { fetchEmployeesByEmails } from "src/utils/employees"; +import { CompanyLocation } from "studio/lib/interfaces/companyDetails"; +import { COMPANY_LOCATIONS_QUERY } from "studio/lib/queries/admin"; +import { EMPLOYEE_PAGE_SLUG_QUERY } from "studio/lib/queries/siteSettings"; +import { loadStudioQuery } from "studio/lib/store"; + +export async function getEmployeePageSlug(language: string) { + const employeePageSlug = + ( + await loadStudioQuery<{ slug: string } | null>(EMPLOYEE_PAGE_SLUG_QUERY, { + language, + }) + ).data?.slug ?? ""; + + return employeePageSlug; +} + +export async function fetchLocationsWithContact(): Promise< + EmployeeAndMetadata[] +> { + const companyLocations = ( + await loadStudioQuery(COMPANY_LOCATIONS_QUERY, {}) + ).data; + const contactByLocation = await fetchContactByLocationMap(companyLocations); + + return contactByLocation; +} + +async function fetchContactByLocationMap( + companyLocations: CompanyLocation[], +): Promise { + const uniqueEmails = Array.from( + new Set( + companyLocations + .map((location) => location.contactPerson) + .filter((contactPerson) => contactPerson !== undefined), + ), + ); + + const contactRes = await fetchEmployeesByEmails(uniqueEmails); + + if (!contactRes.ok) { + return []; + } + + const contactByLocation: EmployeeAndMetadata[] = []; + + for (const location of companyLocations) { + const employee = contactRes.value.find( + (employee) => employee.email === location.contactPerson, + ); + + if (!employee || !location?.companyLocationName) { + continue; + } + + contactByLocation.push({ + employee, + tag: location.companyLocationName, + tagSlug: location._id, + }); + } + return contactByLocation; +} diff --git a/src/components/customerCases/customerCase/contactInformation/contactInformation.module.css b/src/components/customerCases/customerCase/contactInformation/contactInformation.module.css deleted file mode 100644 index c79175a9d..000000000 --- a/src/components/customerCases/customerCase/contactInformation/contactInformation.module.css +++ /dev/null @@ -1,74 +0,0 @@ -.contactBox { - max-width: var(--max-content-width-large); -} - -.contactBox__inner { - --_contactBox__background: var(--background-bg-dark); - --_contactBox__color: var(--text-primary-light); - - display: grid; - grid-template-columns: repeat(auto-fit, minmax(260px, 1fr)); - - padding: 1.5rem 3rem; - flex-direction: column; - justify-content: flex-end; - align-items: flex-start; - gap: var(--Padding-l, 48px); - align-self: stretch; - - border-radius: var(--radius-small); - background: var(--_contactBox__background); - color: var(--_contactBox__color); -} -.contactBox__inner--light { - --_contactBox__background: var(--background-bg-light-secondary); - --_contactBox__color: var(--text-primary); -} - -.textContent { - height: 100%; - display: flex; - flex-direction: column; - gap: 1rem; - - justify-content: flex-end; -} - -.contactSelectorWrapper { - container-type: inline-size; - container-name: contactSelectorWrapper; -} - -.contactSelector { - display: flex; - gap: 1rem; -} - -.tagList { - display: flex; - flex-direction: column; - gap: 0.75rem; - - align-items: flex-start; - min-width: 120px; -} - -.employeeCard { - flex: 1; -} - -@media (max-width: 400px) { - .contactBox__inner { - padding: 1rem 1rem; - } -} - -@container contactSelectorWrapper (max-width: 350px) { - .contactSelector { - flex-direction: column; - } - - .tagList { - flex-direction: row; - } -} diff --git a/src/components/sections/contact-box/ContactBox.tsx b/src/components/sections/contact-box/ContactBox.tsx index 5928ed432..d319832a8 100644 --- a/src/components/sections/contact-box/ContactBox.tsx +++ b/src/components/sections/contact-box/ContactBox.tsx @@ -1,12 +1,7 @@ -import { Suspense } from "react"; - -import { EmployeeCardSkeleton } from "src/components/employeeCard/EmployeeCard"; -import Text from "src/components/text/Text"; import { ContactBoxSection } from "studio/lib/interfaces/pages"; import { fetchEmployeesWithTags, getEmployeesPageSlug } from "./api"; -import styles from "./contact-box.module.css"; -import ContactSelector from "./ContactSelector"; +import ContactBoxContent from "./ContactBoxContent"; export interface ContactBoxProps { section: ContactBoxSection; @@ -25,38 +20,12 @@ export default async function ContactBox({ const contactPoints = fetchEmployeesWithTags(section.contactPoints); - const backgroundClass = styles[`contactBox__inner--${section.background}`]; - const employeeCardBackground = - section.background === "transparent" ? "light" : section.background; - return ( -
-
-
- - {section.basicTitle} - - - {section.optionalSubtitle && ( - {section.optionalSubtitle} - )} -
- -
- - } - > - - -
-
-
+ ); } diff --git a/src/components/sections/contact-box/ContactBoxContent.tsx b/src/components/sections/contact-box/ContactBoxContent.tsx new file mode 100644 index 000000000..45cd64548 --- /dev/null +++ b/src/components/sections/contact-box/ContactBoxContent.tsx @@ -0,0 +1,61 @@ +import { Suspense } from "react"; + +import { EmployeeCardSkeleton } from "src/components/employeeCard/EmployeeCard"; +import Text from "src/components/text/Text"; + +import styles from "./contact-box.module.css"; +import ContactSelector from "./ContactSelector"; +import { EmployeeAndMetadata } from "./types"; + +export interface ContactBoxProps { + data: { + basicTitle: string; + optionalSubtitle?: string; + background: "light" | "dark" | "transparent"; + }; + language: string; + employeesPageSlug: string; + contactPoints: Promise; +} + +export default async function ContactBoxContent({ + data, + language, + employeesPageSlug, + contactPoints, +}: ContactBoxProps) { + const backgroundClass = styles[`contactBox__inner--${data.background}`]; + const employeeCardBackground = + data.background === "transparent" ? "light" : data.background; + + return ( +
+
+
+ + {data.basicTitle} + + + {data.optionalSubtitle && ( + {data.optionalSubtitle} + )} +
+ +
+ + } + > + + +
+
+
+ ); +} diff --git a/src/components/sections/contact-box/contact-box.module.css b/src/components/sections/contact-box/contact-box.module.css index 1c6650e8e..66d069bb6 100644 --- a/src/components/sections/contact-box/contact-box.module.css +++ b/src/components/sections/contact-box/contact-box.module.css @@ -1,6 +1,7 @@ .contactBox { margin: 1rem auto; padding: 0; + width: 100%; max-width: var(--max-content-width-large); } @@ -36,8 +37,6 @@ display: flex; flex-direction: column; gap: 1rem; - - justify-content: flex-end; } .contactSelectorWrapper { diff --git a/src/components/sections/contact-box/types.ts b/src/components/sections/contact-box/types.ts index bf454fc68..815b6dc48 100644 --- a/src/components/sections/contact-box/types.ts +++ b/src/components/sections/contact-box/types.ts @@ -4,5 +4,5 @@ export type EmployeeAndMetadata = { employee: ChewbaccaEmployee; tag: string; tagSlug: string; - overrideTitle: string; + overrideTitle?: string; }; From 425bd85b4774e432672c61ae31659512364f8fc8 Mon Sep 17 00:00:00 2001 From: Mikael Brevik Date: Thu, 12 Dec 2024 15:59:39 +0100 Subject: [PATCH 2/3] fix: contact box top align (#1034) From ee8ffa4dff58011b11787b9a18dbe602bdfb1a3e Mon Sep 17 00:00:00 2001 From: Mikael Brevik Date: Thu, 12 Dec 2024 16:01:42 +0100 Subject: [PATCH 3/3] fix: localtion label for job listings (#1032) --- src/components/jobPosting/JobPosting.tsx | 14 ++++++++++---- src/components/jobPosting/jobPosting.module.css | 1 - src/components/sections/jobs/JobPostingList.tsx | 6 +++++- 3 files changed, 15 insertions(+), 6 deletions(-) diff --git a/src/components/jobPosting/JobPosting.tsx b/src/components/jobPosting/JobPosting.tsx index a2dc38fb8..cd888b79c 100644 --- a/src/components/jobPosting/JobPosting.tsx +++ b/src/components/jobPosting/JobPosting.tsx @@ -11,9 +11,13 @@ function sortAlphabetically(list: string[]) { interface JobPostingProps { jobPosting: IJobPosting; + showLocations?: boolean; } -export default function JobPosting({ jobPosting }: JobPostingProps) { +export default function JobPosting({ + jobPosting, + showLocations = true, +}: JobPostingProps) { const jobPostingLocations = sortAlphabetically( jobPosting.locations.map( (location: CompanyLocation) => location.companyLocationName, @@ -30,9 +34,11 @@ export default function JobPosting({ jobPosting }: JobPostingProps) {
{jobPosting.role}
-
- {jobPostingLocations} -
+ {showLocations && ( +
+ {jobPostingLocations} +
+ )} ); diff --git a/src/components/jobPosting/jobPosting.module.css b/src/components/jobPosting/jobPosting.module.css index a75c301f4..324942653 100644 --- a/src/components/jobPosting/jobPosting.module.css +++ b/src/components/jobPosting/jobPosting.module.css @@ -33,7 +33,6 @@ } .role { - width: 250px; @media (max-width: 640px) { width: 100%; } diff --git a/src/components/sections/jobs/JobPostingList.tsx b/src/components/sections/jobs/JobPostingList.tsx index 70e0af7ac..ee71f4336 100644 --- a/src/components/sections/jobs/JobPostingList.tsx +++ b/src/components/sections/jobs/JobPostingList.tsx @@ -95,7 +95,11 @@ export default function JobPostingList({
{filteredJobPostings?.map((jobPosting: IJobPosting) => ( - + ))}