diff --git a/app/components/layout/Footer/Footer.tsx b/app/components/layout/Footer/Footer.tsx index e97ee25..4a1fae1 100644 --- a/app/components/layout/Footer/Footer.tsx +++ b/app/components/layout/Footer/Footer.tsx @@ -1,11 +1,32 @@ +"use client"; + +import { useEffect, useState } from 'react'; import FooterTop from './FooterTop'; import FooterBottom from './FooterBottom'; +import { footerApi, FooterData } from "../../../../api/footerApi"; +import footerData from "./footer.json"; const Footer = () => { + const [data, setData] = useState(footerData as FooterData); + + useEffect(() => { + const loadFooterData = async () => { + try { + const apiData = await footerApi.getFooter(); + setData(apiData); + } catch (error) { + console.error("Failed to load footer data from API, using static data:", error); + // Keep using static data as fallback + } + }; + + loadFooterData(); + }, []); + return ( <> - - + + ); }; diff --git a/app/components/layout/Footer/FooterBottom.tsx b/app/components/layout/Footer/FooterBottom.tsx index 7bcd406..1087960 100644 --- a/app/components/layout/Footer/FooterBottom.tsx +++ b/app/components/layout/Footer/FooterBottom.tsx @@ -1,29 +1,18 @@ "use client"; import Link from "next/link"; -import { useEffect, useState } from "react"; -import { footerApi, FooterData } from "../../../../api/footerApi"; +import { FooterData } from "../../../../api/footerApi"; import footerData from "./footer.json"; -const FooterBottom = () => { - const [data, setData] = useState(footerData as FooterData); +interface FooterBottomProps { + data: FooterData; +} - useEffect(() => { - const loadFooterData = async () => { - try { - const apiData = await footerApi.getFooter(); - setData(apiData); - } catch (error) { - console.error("Failed to load footer data from API, using static data:", error); - // Keep using static data as fallback - } - }; - - loadFooterData(); - }, []); +const FooterBottom = ({ data }: FooterBottomProps) => { + const effectiveData = data || footerData; // Ensure we always have a valid `bottom` object, even if API shape changes - const bottom = data?.bottom || footerData.bottom; + const bottom = effectiveData?.bottom || footerData.bottom; // If bottom is still missing, avoid rendering to prevent runtime errors if (!bottom) { diff --git a/app/components/layout/Footer/FooterTop.tsx b/app/components/layout/Footer/FooterTop.tsx index 6cc9d72..d04dd17 100644 --- a/app/components/layout/Footer/FooterTop.tsx +++ b/app/components/layout/Footer/FooterTop.tsx @@ -1,29 +1,19 @@ "use client"; import Link from "next/link"; -import { useEffect, useState } from "react"; -import { footerApi, FooterData } from "../../../../api/footerApi"; +import { FooterData } from "../../../../api/footerApi"; import footerData from "./footer.json"; -const FooterTop = () => { - const [data, setData] = useState(footerData as FooterData); +interface FooterTopProps { + data: FooterData; +} - useEffect(() => { - const loadFooterData = async () => { - try { - const apiData = await footerApi.getFooter(); - setData(apiData); - } catch (error) { - console.error("Failed to load footer data from API, using static data:", error); - // Keep using static data as fallback - } - }; - - loadFooterData(); - }, []); +const FooterTop = ({ data }: FooterTopProps) => { + // Use passed data, fallback to static json if needed + const effectiveData = data || footerData; // Ensure we always have a valid `top` object, even if API shape changes - const top = data?.top || footerData.top; + const top = effectiveData?.top || footerData.top; // If for some reason `top` is still missing, avoid rendering to prevent runtime errors if (!top) { diff --git a/app/components/layout/Header/Header.tsx b/app/components/layout/Header/Header.tsx index b40de99..7fb025d 100644 --- a/app/components/layout/Header/Header.tsx +++ b/app/components/layout/Header/Header.tsx @@ -13,6 +13,7 @@ const Header = () => { const [isMobileMenuOpen, setIsMobileMenuOpen] = useState(false); const [isSearchOpen, setIsSearchOpen] = useState(false); const [menuItems, setMenuItems] = useState([]); + const [headerData, setHeaderData] = useState(null); const [isLoading, setIsLoading] = useState(true); const toggleOffcanvas = () => setIsOffcanvasOpen(!isOffcanvasOpen); @@ -33,31 +34,48 @@ const Header = () => { ); useEffect(() => { - const fetchMenu = async () => { + const fetchData = async () => { try { setIsLoading(true); - const data = await headerMenuService.getHeaderMenu(); - const mappedData = data.map((item) => adaptMenu(item)); - setMenuItems(mappedData); + + // Fetch Menu + const menuPromise = headerMenuService.getHeaderMenu(); + + // Fetch Header Data (Logo, Topbar) + const apiUrl = process.env.NEXT_PUBLIC_API_URL || "http://localhost:3000"; + const headerPromise = fetch(`${apiUrl}/api/header`).then(res => res.json()); + + const [menuData, headerResult] = await Promise.all([menuPromise, headerPromise]); + + // Process Menu + const mappedMenu = menuData.map((item) => adaptMenu(item)); + setMenuItems(mappedMenu); + + // Process Header Data + if (headerResult.success && headerResult.data) { + setHeaderData(headerResult.data); + } + } catch (error) { - console.error("Error fetching menu in Header:", error); + console.error("Error fetching header data:", error); } finally { setIsLoading(false); } }; - fetchMenu(); + fetchData(); }, [adaptMenu]); return ( <> - + setIsOffcanvasOpen(false)} menuItems={menuItems} /> diff --git a/app/components/layout/Header/HeaderBottom.tsx b/app/components/layout/Header/HeaderBottom.tsx index 6fb0af7..ca30c21 100644 --- a/app/components/layout/Header/HeaderBottom.tsx +++ b/app/components/layout/Header/HeaderBottom.tsx @@ -12,6 +12,7 @@ interface HeaderBottomProps { onToggleSearch: () => void; menuItems: any[]; isLoading: boolean; + logo: { light: string; dark: string; alt: string } | null; } const HeaderBottom: React.FC = ({ @@ -20,7 +21,18 @@ const HeaderBottom: React.FC = ({ onToggleSearch, menuItems, isLoading, + logo, }) => { + // Helper function to resolve logo URL + const getLogoUrl = (path: string | undefined) => { + if (!path) return "/assets/img/logo/black-logo.svg"; + if (path.startsWith("http")) return path; + const apiUrl = process.env.NEXT_PUBLIC_API_URL || "http://localhost:3001"; + return `${apiUrl}${path}`; + }; + + const logoSrc = getLogoUrl(logo?.light); + return (
@@ -29,7 +41,11 @@ const HeaderBottom: React.FC = ({
- logo-img + {logo?.alt
diff --git a/app/components/layout/Header/HeaderTop.tsx b/app/components/layout/Header/HeaderTop.tsx index 87b85d9..14da639 100644 --- a/app/components/layout/Header/HeaderTop.tsx +++ b/app/components/layout/Header/HeaderTop.tsx @@ -18,39 +18,15 @@ interface HeaderData { name: string; value: string; }>; - }; + } | null; } -const HeaderTop = () => { - const [data, setData] = useState(headerData); - const [loading, setLoading] = useState(true); - - useEffect(() => { - const fetchHeaderData = async () => { - try { - const apiUrl = process.env.NEXT_PUBLIC_API_URL || "http://localhost:3000"; - const response = await fetch(`${apiUrl}/api/header`); - - if (response.ok) { - const result = await response.json(); - if (result.success && result.data && result.data.top) { - setData({ - top: result.data.top, - }); - } - } - } catch (error) { - console.warn("Failed to fetch header data from API, using fallback:", error); - // Use fallback data (already set as initial state) - } finally { - setLoading(false); - } - }; - - fetchHeaderData(); - }, []); - - const { phone, email, location, socialLinks, languages } = data.top; +const HeaderTop: React.FC<{ data: HeaderData['top'] }> = ({ data }) => { + // Use passed data or fallback to local JSON if data is null (though parent should handle fetching) + // If data is null (initial load), we can use headerData fallback or render nothing/skeleton + + const displayData = data || headerData.top; + const { phone, email, location, socialLinks, languages } = displayData; return (
diff --git a/public/assets/img/favicon.png b/public/assets/img/favicon.png index 207b12b..820c256 100644 Binary files a/public/assets/img/favicon.png and b/public/assets/img/favicon.png differ