Added an header and other components
This commit is contained in:
BIN
assets/img/logo.png
Normal file
BIN
assets/img/logo.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 4.5 KiB |
50
components/blocks/anchoring.tsx
Normal file
50
components/blocks/anchoring.tsx
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
import * as React from "react";
|
||||||
|
import {useEffect, useRef, useState} from "react";
|
||||||
|
|
||||||
|
interface AnchoringProps {
|
||||||
|
text: string, // Default: Découvrez-en plus !
|
||||||
|
linkTo: string // Default: #main-page
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Anchoring = ({ data }: { data: AnchoringProps }) => {
|
||||||
|
const [opacity, setOpacity] = useState(1);
|
||||||
|
const anchoringRef = useRef(null);
|
||||||
|
|
||||||
|
const handleScroll = () => {
|
||||||
|
const scrollTop = window.scrollY || window.pageYOffset;
|
||||||
|
const elementHeight = anchoringRef.current.clientHeight;
|
||||||
|
const newOpacity = (elementHeight - scrollTop) / elementHeight;
|
||||||
|
setOpacity(newOpacity);
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
handleScroll(); // Initialize opacity on mount
|
||||||
|
|
||||||
|
window.addEventListener('scroll', handleScroll);
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
window.removeEventListener('scroll', handleScroll);
|
||||||
|
};
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const anchoringStyle = {
|
||||||
|
opacity: opacity,
|
||||||
|
transition: 'opacity 0.3s', // You can add a transition for a smooth effect
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="mx-auto">
|
||||||
|
<div
|
||||||
|
className="anchoring"
|
||||||
|
ref={anchoringRef}
|
||||||
|
style={anchoringStyle}
|
||||||
|
>
|
||||||
|
<a href={`${ data.linkTo }`}>
|
||||||
|
<h1>{ data.text }</h1>
|
||||||
|
<img
|
||||||
|
src=""/>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -45,15 +45,6 @@ export const Footer = ({ data, icon, rawData }) => {
|
|||||||
href="/"
|
href="/"
|
||||||
className="group mx-2 flex items-center font-bold tracking-tight text-gray-400 dark:text-gray-300 opacity-50 hover:opacity-100 transition duration-150 ease-out whitespace-nowrap"
|
className="group mx-2 flex items-center font-bold tracking-tight text-gray-400 dark:text-gray-300 opacity-50 hover:opacity-100 transition duration-150 ease-out whitespace-nowrap"
|
||||||
>
|
>
|
||||||
<Icon
|
|
||||||
parentColor={data.color}
|
|
||||||
data={{
|
|
||||||
name: icon.name,
|
|
||||||
color: data.color === "primary" ? "primary" : icon.color,
|
|
||||||
style: icon.style,
|
|
||||||
}}
|
|
||||||
className="inline-block h-10 w-auto group-hover:text-orange-500"
|
|
||||||
/>
|
|
||||||
</Link>
|
</Link>
|
||||||
<div className="flex gap-4">
|
<div className="flex gap-4">
|
||||||
{data.social && data.social.facebook && (
|
{data.social && data.social.facebook && (
|
||||||
|
|||||||
@@ -1,36 +1,23 @@
|
|||||||
import React from "react";
|
import React from "react";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { useRouter } from "next/router";
|
|
||||||
import { Icon } from "../util/icon";
|
|
||||||
import { tinaField } from "tinacms/dist/react";
|
import { tinaField } from "tinacms/dist/react";
|
||||||
|
import defaultLogo from "../../assets/img/logo.png";
|
||||||
import {GlobalHeader} from "../../tina/__generated__/types";
|
import {GlobalHeader} from "../../tina/__generated__/types";
|
||||||
|
|
||||||
export const Header = ({ data }: { data: GlobalHeader }) => {
|
export const Header = ({ data }: { data: GlobalHeader }) => {
|
||||||
const router = useRouter();
|
|
||||||
|
|
||||||
const [isClient, setIsClient] = React.useState(false);
|
|
||||||
React.useEffect(() => {
|
|
||||||
setIsClient(true);
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<nav className="navbar navbar-expand-lg navbar-dark bg-primary sticky-top">
|
<nav className="navbar flex flex-wrap content-between sticky top-0 py-2">
|
||||||
<div className="container d-flex flex-column flex-md-row justify-content-between">
|
<div className="flex grow justify-between px-6">
|
||||||
<div className="flex">
|
<div className="flex navbar-brand">
|
||||||
<Link
|
<Link
|
||||||
href="/"
|
href="/"
|
||||||
className="navbar-brand container d-flex align-items-md-center"
|
className="container flex items-center"
|
||||||
>
|
>
|
||||||
<Icon
|
<img src={ data.logoSrc || defaultLogo.src }/>
|
||||||
tinaField={tinaField(data, "icon")}
|
<div className="flex flex-col">
|
||||||
parentColor={data.color}
|
<h3 className="title" data-tina-field={tinaField(data, "title")}>{data.title}</h3>
|
||||||
data={{
|
<div className="subtitle" data-tina-field={tinaField(data, "subtitle")}>{data.subtitle}</div>
|
||||||
name: data.icon.name,
|
</div>
|
||||||
color: data.icon.color,
|
|
||||||
style: data.icon.style,
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
<span data-tina-field={tinaField(data, "name")}>{data.name}</span>
|
|
||||||
</Link>
|
</Link>
|
||||||
</div>
|
</div>
|
||||||
<button className="navbar-toggler" type="button" data-bs-toggle="collapse"
|
<button className="navbar-toggler" type="button" data-bs-toggle="collapse"
|
||||||
@@ -38,8 +25,8 @@ export const Header = ({ data }: { data: GlobalHeader }) => {
|
|||||||
aria-label="Toggle navigation">
|
aria-label="Toggle navigation">
|
||||||
<span className="navbar-toggler-icon"></span>
|
<span className="navbar-toggler-icon"></span>
|
||||||
</button>
|
</button>
|
||||||
<div className="collapse navbar-collapse justify-content-end" id="navbarSupportedContent">
|
<div className={`flex items-center` } id="navbarSupportedContent">
|
||||||
<ul className="navbar-nav d-flex flex-column flex-md-row">
|
<ul className="flex md:flex-row">
|
||||||
{data.nav &&
|
{data.nav &&
|
||||||
data.nav.map((item, i) => {
|
data.nav.map((item, i) => {
|
||||||
/*const activeItem =
|
/*const activeItem =
|
||||||
@@ -53,24 +40,16 @@ export const Header = ({ data }: { data: GlobalHeader }) => {
|
|||||||
>
|
>
|
||||||
<Link
|
<Link
|
||||||
data-tina-field={tinaField(item, "label")}
|
data-tina-field={tinaField(item, "label")}
|
||||||
href={`/${item.href}`}
|
href={`${ item.external? "" : "/" }${ item.href }`}
|
||||||
className="nav-link p-2"
|
className={`nav-link p-2 ${ item.external ? "external-link-icon" : ""}`}
|
||||||
>
|
>
|
||||||
{item.label}
|
{item.label}
|
||||||
</Link>
|
</Link>
|
||||||
</li>
|
</li>
|
||||||
);
|
);
|
||||||
})}
|
})}
|
||||||
<li className="nav-item">
|
|
||||||
<a id="redirecting" className="nav-link p-2 pe-4" href="https://www.rdv360.com/psychartherapie">Rendez-vous</a>
|
|
||||||
</li>
|
|
||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
<div
|
|
||||||
className={`absolute h-1 bg-gradient-to-r from-transparent ${
|
|
||||||
data.color === "primary" ? `via-white` : `via-black dark:via-white`
|
|
||||||
} to-transparent bottom-0 left-4 right-4 -z-1 opacity-5`}
|
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
</nav>
|
</nav>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -5,8 +5,9 @@
|
|||||||
"color": "orange",
|
"color": "orange",
|
||||||
"style": "float"
|
"style": "float"
|
||||||
},
|
},
|
||||||
"name": "Tina Starter",
|
"title": "PsychARThérapie",
|
||||||
"color": "default",
|
"subtitle": "Pauline Vince",
|
||||||
|
"logoSrc": "/uploads/logo.png",
|
||||||
"nav": [
|
"nav": [
|
||||||
{
|
{
|
||||||
"href": "",
|
"href": "",
|
||||||
@@ -19,6 +20,11 @@
|
|||||||
{
|
{
|
||||||
"href": "posts",
|
"href": "posts",
|
||||||
"label": "Blog"
|
"label": "Blog"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"href": "https://www.rdv360.com/psychartherapie",
|
||||||
|
"label": "Rendez-vous",
|
||||||
|
"external": true
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import "../styles.css";
|
import "../styles/styles.css";
|
||||||
|
|
||||||
const App = ({ Component, pageProps }) => {
|
const App = ({ Component, pageProps }) => {
|
||||||
return <Component {...pageProps} />;
|
return <Component {...pageProps} />;
|
||||||
|
|||||||
BIN
public/uploads/logo.png
Normal file
BIN
public/uploads/logo.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 4.5 KiB |
@@ -1,3 +0,0 @@
|
|||||||
@tailwind base;
|
|
||||||
@tailwind components;
|
|
||||||
@tailwind utilities;
|
|
||||||
16
styles/anchoring.css
Normal file
16
styles/anchoring.css
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
.anchoring {
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.anchoring h1 {
|
||||||
|
margin-bottom: -35px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.anchoring h1, .anchoring h1:hover {
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
.anchoring img {
|
||||||
|
width: 128px;
|
||||||
|
height: 128px;
|
||||||
|
}
|
||||||
35
styles/header.css
Normal file
35
styles/header.css
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
.navbar {
|
||||||
|
font-family: 'Nunito', sans-serif;
|
||||||
|
background: var(--header-BackgroundColor);
|
||||||
|
color: var(--header-textColor);
|
||||||
|
z-index: 9999;
|
||||||
|
}
|
||||||
|
|
||||||
|
.navbar .navbar-brand h3 {
|
||||||
|
margin-bottom: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.navbar .navbar-brand .title {
|
||||||
|
font-size: 1.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.navbar .navbar-brand .subtitle {
|
||||||
|
font-size: .9rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.navbar .nav-item .nav-link {
|
||||||
|
font-size: 1.4rem;
|
||||||
|
font-weight: 400;
|
||||||
|
}
|
||||||
|
|
||||||
|
.external-link-icon:after {
|
||||||
|
content: "";
|
||||||
|
position: relative;
|
||||||
|
display: inline-block;
|
||||||
|
left: 5px;
|
||||||
|
top: 2px;
|
||||||
|
background: url('') center center;
|
||||||
|
background-size: contain;
|
||||||
|
height: 16px;
|
||||||
|
width: 16px;
|
||||||
|
}
|
||||||
21
styles/styles.css
Normal file
21
styles/styles.css
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
@import url('https://fonts.googleapis.com/css2?family=Nunito:wght@300&display=swap');
|
||||||
|
@import url('https://fonts.googleapis.com/css2?family=Questrial&display=swap');
|
||||||
|
@import "header.css";
|
||||||
|
@import "anchoring.css";
|
||||||
|
|
||||||
|
@tailwind base;
|
||||||
|
@tailwind components;
|
||||||
|
@tailwind utilities;
|
||||||
|
|
||||||
|
html {
|
||||||
|
scroll-padding-top: 7.25rem;
|
||||||
|
|
||||||
|
--body-BackgroundColor: #f5f5f5;
|
||||||
|
--primaryColor: #124498;
|
||||||
|
--header-BackgroundColor: var(--primaryColor);
|
||||||
|
--header-textColor: #ffd39c;
|
||||||
|
}
|
||||||
|
|
||||||
|
body {
|
||||||
|
font-family: 'Questrial', sans-serif;
|
||||||
|
}
|
||||||
@@ -4,7 +4,6 @@ import { featureBlockSchema } from "../components/blocks/features";
|
|||||||
import { heroBlockSchema } from "../components/blocks/hero";
|
import { heroBlockSchema } from "../components/blocks/hero";
|
||||||
import { testimonialBlockSchema } from "../components/blocks/testimonial";
|
import { testimonialBlockSchema } from "../components/blocks/testimonial";
|
||||||
import { ColorPickerInput } from "./fields/color";
|
import { ColorPickerInput } from "./fields/color";
|
||||||
import { iconSchema } from "../components/util/icon";
|
|
||||||
|
|
||||||
const config = defineConfig({
|
const config = defineConfig({
|
||||||
clientId: process.env.NEXT_PUBLIC_TINA_CLIENT_ID!,
|
clientId: process.env.NEXT_PUBLIC_TINA_CLIENT_ID!,
|
||||||
@@ -159,20 +158,20 @@ const config = defineConfig({
|
|||||||
label: "Header",
|
label: "Header",
|
||||||
name: "header",
|
name: "header",
|
||||||
fields: [
|
fields: [
|
||||||
iconSchema as any,
|
|
||||||
{
|
{
|
||||||
type: "string",
|
type: "string",
|
||||||
label: "Name",
|
label: "Title",
|
||||||
name: "name",
|
name: "title",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: "string",
|
type: "string",
|
||||||
label: "Color",
|
label: "Subtitle",
|
||||||
name: "color",
|
name: "subtitle",
|
||||||
options: [
|
},
|
||||||
{ label: "Default", value: "default" },
|
{
|
||||||
{ label: "Primary", value: "primary" },
|
type: "image",
|
||||||
],
|
label: "Logo",
|
||||||
|
name: "logoSrc"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: "object",
|
type: "object",
|
||||||
@@ -186,6 +185,7 @@ const config = defineConfig({
|
|||||||
defaultItem: {
|
defaultItem: {
|
||||||
href: "home",
|
href: "home",
|
||||||
label: "Home",
|
label: "Home",
|
||||||
|
external: false
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
fields: [
|
fields: [
|
||||||
@@ -199,6 +199,11 @@ const config = defineConfig({
|
|||||||
label: "Label",
|
label: "Label",
|
||||||
name: "label",
|
name: "label",
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
type: "boolean",
|
||||||
|
label: "External",
|
||||||
|
name: "external",
|
||||||
|
}
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
|
|||||||
File diff suppressed because one or more lines are too long
Reference in New Issue
Block a user