Commit 04ac208e by Madhankumar

blog app issue

parent 923886bb
/** @type { import('@storybook/nextjs').StorybookConfig } */ // // .storybook/main.js or .storybook/main.ts
const config = { // module.exports = {
// stories: [
// "../components/**/*.mdx",
// "../components/**/*.stories.@(js|jsx|mjs|ts|tsx)",
// ],
// addons: [
// "@storybook/addon-links",
// "@storybook/addon-essentials",
// "@storybook/addon-onboarding",
// "@storybook/addon-interactions",
// "storybook-dark-mode",
// ],
// frameworks: {
// "@storybook/nextjs": {
// options: {},
// },
// },
// docs: {
// autodocs: "tag",
// },
// framework: {
// name: "@storybook/nextjs",
// options: {},
// },
// };
import path from "path";
module.exports = {
stories: [ stories: [
"../components/**/*.mdx", "../components/**/*.mdx",
"../components/**/*.stories.@(js|jsx|mjs|ts|tsx)", "../components/**/*.stories.@(js|jsx|mjs|ts|tsx)",
...@@ -11,12 +43,24 @@ const config = { ...@@ -11,12 +43,24 @@ const config = {
"@storybook/addon-interactions", "@storybook/addon-interactions",
"storybook-dark-mode", "storybook-dark-mode",
], ],
webpackFinal: (config) => {
config.resolve = {
...config.resolve,
fallback: {
fs: false,
path: false,
os: false,
},
alias: {
...config.resolve.alias,
"@components": path.resolve(__dirname, "../components"),
"@img": path.resolve(__dirname, "../public/images"),
},
};
return config;
},
framework: { framework: {
name: "@storybook/nextjs", name: "@storybook/nextjs",
options: {}, options: {},
}, },
docs: {
autodocs: "tag",
},
}; };
export default config;
/** @type { import('@storybook/react').Preview } */ /** @type { import('@storybook/react').Preview } */
import "../app/globals.css"; import "../app/globals.css";
import { useDarkMode } from "storybook-dark-mode"; import { useDarkMode } from "storybook-dark-mode";
import { themes } from "@storybook/theming";
import "@fortawesome/fontawesome-svg-core/styles.css"; // Import the CSS import "@fortawesome/fontawesome-svg-core/styles.css"; // Import the CSS
const preview = { const preview = {
parameters: { parameters: {
actions: { argTypesRegex: "^on[A-Z].*" }, docs: {
theme: themes.dark,
},
actions: {
argTypesRegex: "^on[A-Z].*",
},
controls: { controls: {
matchers: { matchers: {
color: /(background|color)$/i, color: /(background|color)$/i,
date: /Date$/i, date: /Date$/,
}, },
}, },
darkMode: { darkMode: {
...@@ -23,7 +29,6 @@ const preview = { ...@@ -23,7 +29,6 @@ const preview = {
export const decorators = [ export const decorators = [
(Story) => { (Story) => {
const currentTheme = useDarkMode() ? "light" : "dark"; const currentTheme = useDarkMode() ? "light" : "dark";
return <Story theme={currentTheme} />; return <Story theme={currentTheme} />;
}, },
......
import "@styles/global.css"; import "@styles/global.css";
// import "@fortawesome/fontawesome-svg-core/styles.css"; // Import the CSS
// import { library } from "@fortawesome/fontawesome-svg-core";
// import { faNewspaper } from "@fortawesome/free-solid-svg-icons";
// library.add(faNewspaper);
export default function App({ Component, pageProps }) { export default function App({ Component, pageProps }) {
return <Component {...pageProps} />; return <Component {...pageProps} />;
} }
import React, { createContext, useState, useContext, useEffect } from "react"; "use client";
import { useDarkMode } from "storybook-dark-mode";
const ThemeContext = createContext(); import { createContext, useContext, useState, useEffect } from "react";
export const ThemeProvider = ({ children }) => { export const ThemeContext = createContext();
const isDarkMode = useDarkMode(); export const useAppContext = () => useContext(ThemeContext);
export default function ThemeProvider({ children, ...props }) {
// State for theme
const [theme, setTheme] = useState("light"); const [theme, setTheme] = useState("light");
const toggleTheme = () => { useEffect(() => {
setTheme((prevTheme) => (prevTheme === "light" ? "dark" : "light")); handleThemeChange();
}, [theme]);
// Function to handle theme changes
const handleThemeChange = () => {
const storedTheme = localStorage.getItem("theme");
if (!storedTheme) {
// Set default theme to light if not stored
setDefaultTheme();
} else {
// Set theme from storage
setThemeFromStorage(storedTheme);
}
}; };
useEffect(() => { // Function to set default theme to light
// You can use localStorage to persist the theme preference const setDefaultTheme = () => {
const savedTheme = localStorage.getItem("theme"); const defaultTheme = "light";
setTheme((isDarkMode ? "dark" : "light") || savedTheme); localStorage.setItem("theme", defaultTheme);
}, [isDarkMode]); setThemeAndApplyClass(defaultTheme);
};
useEffect(() => { // Function to set theme from storage
document.documentElement.setAttribute("data-theme", theme); const setThemeFromStorage = (storedTheme) => {
localStorage.setItem("theme", theme); setThemeAndApplyClass(storedTheme);
}, [theme]); };
// Function to set theme and apply class to document element
const setThemeAndApplyClass = (newTheme) => {
localStorage.setItem("theme", newTheme);
setTheme(newTheme);
document.documentElement.setAttribute("class", newTheme);
};
const contextValue = {
theme,
setTheme,
...props,
};
return ( return (
<ThemeContext.Provider value={{ theme, toggleTheme }}> <ThemeContext.Provider value={contextValue}>
{children} {children}
</ThemeContext.Provider> </ThemeContext.Provider>
); );
}; }
export const useTheme = () => {
const context = useContext(ThemeContext);
if (!context) {
throw new Error("useTheme must be used within a ThemeProvider");
}
return context;
};
export const decorators = [
(renderStory) => <ThemeProvider>{renderStory()}</ThemeProvider>,
];
...@@ -56,7 +56,7 @@ html { ...@@ -56,7 +56,7 @@ html {
padding: 0; padding: 0;
margin: 0; margin: 0;
box-sizing: border-box; box-sizing: border-box;
background-color: var(--bg-color); background: var(--bg-color);
min-height: 100dvh; min-height: 100dvh;
max-width: 1920px; max-width: 1920px;
margin: 0 auto; margin: 0 auto;
...@@ -70,9 +70,11 @@ h1 { ...@@ -70,9 +70,11 @@ h1 {
} }
h2 { h2 {
font-size: var(--heading2); font-size: var(--heading2);
color: var(--font-color-900);
} }
h3 { h3 {
font-size: var(--heading3); font-size: var(--heading3);
color: var(--font-color-900);
} }
h4 { h4 {
font-size: var(--heading4); font-size: var(--heading4);
...@@ -103,8 +105,8 @@ p { ...@@ -103,8 +105,8 @@ p {
display: flex; display: flex;
gap: 1em; gap: 1em;
} }
@media (prefers-color-scheme: dark) { /* @media (prefers-color-scheme: dark) {
html { html {
background-color: var(--bg-color); background-color: var(--bg-color);
} }
} } */
import { Inter } from "next/font/google"; import ThemeProvider from "./context";
import "./globals.css"; import "./globals.css";
import { useTheme, ThemeProvider } from "./context";
import { useEffect } from "react";
const inter = Inter({ subsets: ["latin"] });
export const metadata = {
title: "Home",
description: "Next Blog",
};
export default function RootLayout({ children }) { export default function RootLayout({ children }) {
const { theme } = useTheme();
useEffect(() => {
console.log("Theme in RootLayout:", theme);
}, [theme]);
return ( return (
<ThemeProvider> <ThemeProvider>
<html lang="en" className={theme}> <html lang="en">
<body>{children}</body> <body>{children}</body>
</html> </html>
</ThemeProvider> </ThemeProvider>
......
import React from "react";
const loading = () => {
return <div>Loading ...</div>;
};
export default loading;
import React from "react";
const PageNotFound = () => {
return (
<div>
<h1>404 Not Found</h1>
</div>
);
};
export default PageNotFound;
import React from "react"; import BlogLists from "@components/top-level/blog-lists";
import Header from "@components/base/header";
import { getAllPosts } from "@lib/posts";
import { useAppContext } from "./context";
const Home = () => { const Home = () => {
const { theme, setTheme } = useAppContext();
let blogs;
const fetchAllBlogs = () => {
blogs = getAllPosts("posts");
};
fetchAllBlogs();
return ( return (
<div> <div>
<h1>Home</h1> <Header name="NewsBlog" currentTheme={theme} onThemeChange={setTheme} />
<BlogLists
title={blogs?.title}
description={blogs?.description}
blogs={blogs}
/>
</div> </div>
); );
}; };
......
import Button from "."; import Button from "@components/base/button/index";
export default { export default {
title: "Base/Button", title: "Base/Button",
......
...@@ -3,14 +3,14 @@ import cn from "classnames"; ...@@ -3,14 +3,14 @@ import cn from "classnames";
import styles from "./styles.module.css"; import styles from "./styles.module.css";
const Button = ({ children, isDisabled, variant, className, ...props }) => { const Button = ({ children, isDisabled, variant, className, ...props }) => {
const classNames = cn({ const classNames = cn(styles["btn"], {
[styles[variant]]: variant, [styles[variant]]: variant,
[styles[className]]: className, [styles[className]]: className,
[styles.disabled]: isDisabled === true, [styles.disabled]: isDisabled === true,
}); });
return ( return (
<button className={`${classNames}`} {...props}> <button className={`${classNames}`} {...props} disabled={isDisabled}>
{children} {children}
</button> </button>
); );
...@@ -18,7 +18,7 @@ const Button = ({ children, isDisabled, variant, className, ...props }) => { ...@@ -18,7 +18,7 @@ const Button = ({ children, isDisabled, variant, className, ...props }) => {
Button.propTypes = { Button.propTypes = {
variant: PropTypes.oneOf(["primary", "secondary"]), variant: PropTypes.oneOf(["primary", "secondary"]),
children: PropTypes.element, children: PropTypes.any,
isDisabled: PropTypes.bool, isDisabled: PropTypes.bool,
}; };
......
button { .btn {
padding: 0.8rem 1.5rem; padding: 0.8rem 1.5rem;
outline: none; outline: none;
border: none; border: none;
...@@ -17,5 +17,4 @@ button { ...@@ -17,5 +17,4 @@ button {
} }
.disabled { .disabled {
opacity: 0.7; opacity: 0.7;
/* background-color: var(--secondary-bgcolor); */
} }
import React from "react"; import Card from "@components/base/card/index";
import Card from ".";
export default { export default {
title: "Base/Card", title: "Base/Card",
...@@ -26,4 +25,7 @@ export const card = { ...@@ -26,4 +25,7 @@ export const card = {
}, },
readingTime: "1 min", readingTime: "1 min",
}, },
render: (args) => {
return <Card {...args} />;
},
}; };
...@@ -54,16 +54,18 @@ Card.propTypes = { ...@@ -54,16 +54,18 @@ Card.propTypes = {
description: PropTypes.string, description: PropTypes.string,
image: PropTypes.shape({ image: PropTypes.shape({
url: PropTypes.string, url: PropTypes.string,
height: PropTypes.number,
width: PropTypes.number,
}), }),
publishedDate: PropTypes.string, publishedDate: PropTypes.string,
author: { author: PropTypes.shape({
name: PropTypes.string, name: PropTypes.string,
image: PropTypes.shape({ image: PropTypes.shape({
url: PropTypes.string, url: PropTypes.string,
height: PropTypes.number, height: PropTypes.number,
width: PropTypes.number, // Use PropTypes.string for images width: PropTypes.number, // Use PropTypes.string for images
}), }),
}, }),
readingTime: PropTypes.string, readingTime: PropTypes.string,
}; };
......
...@@ -5,62 +5,58 @@ ...@@ -5,62 +5,58 @@
flex-direction: column; flex-direction: column;
transition: all 0.3s ease-in-out; transition: all 0.3s ease-in-out;
word-wrap: break-word; word-wrap: break-word;
&:hover {
transform: translateY(-10px);
box-shadow: rgba(0, 0, 0, 0.1) 0px 3px 8px -1px,
rgba(0, 0, 0, 0.06) 0px 2px 8px -1px;
}
.img {
max-width: 100%;
height: auto;
border-top-right-radius: 20px;
border-top-left-radius: 20px;
}
} }
.card:hover {
transform: translateY(-10px);
box-shadow:
rgba(0, 0, 0, 0.1) 0px 3px 8px -1px,
rgba(0, 0, 0, 0.06) 0px 2px 8px -1px;
}
.card .img {
max-width: 100%;
height: auto;
border-top-right-radius: 20px;
border-top-left-radius: 20px;
}
.authorcontainer { .authorcontainer {
padding: 1.5rem; padding: 1.5rem;
padding-top: 0rem; padding-top: 0rem;
margin-top: -20px; margin-top: -20px;
& .description {
color: var(--font-color-300);
line-height: 1.5em;
font-weight: 600px;
word-wrap: break-word;
}
.cardbottom {
display: grid;
grid-template-columns: 0.5fr 2fr;
line-height: 0;
align-items: center;
grid-gap: 16px;
color: var(--font-color-900);
.img {
border-radius: 50%;
height: auto;
max-width: 100%;
}
.authordescription {
.authorName {
font-size: 18px;
font-weight: 500;
}
.author-desc {
display: flex;
align-items: center;
gap: 1rem;
color: var(--font-color-300);
line-height: 0.5em;
& span {
display: flex;
align-items: center;
}
}
}
}
} }
.authorcontainer .description {
color: var(--font-color-300);
line-height: 1.5em;
font-weight: 600px;
word-wrap: break-word;
}
.authorcontainer .cardbottom {
display: grid;
grid-template-columns: 0.5fr 2fr;
line-height: 0;
align-items: center;
grid-gap: 16px;
color: var(--font-color-900);
}
.authorcontainer .cardbottom .img {
border-radius: 50%;
height: auto;
max-width: 100%;
}
.authorcontainer .cardbottom .authordescription .authorName {
font-size: 18px;
font-weight: 500;
}
.authorcontainer .cardbottom .authordescription .author-desc {
display: flex;
align-items: center;
gap: 1rem;
color: var(--font-color-300);
line-height: 0.5em;
}
.authorcontainer .cardbottom .authordescription .author-desc span {
display: flex;
align-items: center;
}
.title { .title {
color: var(--font-color-900); color: var(--font-color-900);
font-size: 22px; font-size: 22px;
...@@ -68,9 +64,9 @@ ...@@ -68,9 +64,9 @@
font-weight: 600; font-weight: 600;
padding: 1.5rem; padding: 1.5rem;
padding-bottom: 0rem; padding-bottom: 0rem;
&:hover { }
color: #5a67d8; .title:hover {
} color: #5a67d8;
} }
@media screen and (min-width: 768px) { @media screen and (min-width: 768px) {
......
import Header from "."; import Header from "@components/base/header/index";
export default { export default {
title: "Base/Header", title: "Base/Header",
component: Header, component: Header,
......
...@@ -4,16 +4,16 @@ import Icons from "@components/base/icons"; ...@@ -4,16 +4,16 @@ import Icons from "@components/base/icons";
import styles from "./styles.module.css"; import styles from "./styles.module.css";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
function Header({ name, currentTheme, onThemeChange, onSearch, onClose }) { function Header({ name, currentTheme, onThemeChange }) {
const handleTheme = () => { const handleTheme = () => {
onThemeChange(currentTheme); onThemeChange(currentTheme);
}; };
const handleSearch = (value) => { // const handleSearch = (value) => {
onSearch(value); // onSearch(value);
}; // };
const handleClose = () => { // const handleClose = () => {
onClose(""); // onClose("");
}; // };
return ( return (
<header className={styles.header}> <header className={styles.header}>
<div className={styles.container}> <div className={styles.container}>
...@@ -24,7 +24,7 @@ function Header({ name, currentTheme, onThemeChange, onSearch, onClose }) { ...@@ -24,7 +24,7 @@ function Header({ name, currentTheme, onThemeChange, onSearch, onClose }) {
</Link> </Link>
<div className={styles.search}> <div className={styles.search}>
<Search onSearch={handleSearch} onClose={handleClose} /> <Search />
</div> </div>
<div onClick={handleTheme} className={styles.themelight}> <div onClick={handleTheme} className={styles.themelight}>
...@@ -40,6 +40,7 @@ function Header({ name, currentTheme, onThemeChange, onSearch, onClose }) { ...@@ -40,6 +40,7 @@ function Header({ name, currentTheme, onThemeChange, onSearch, onClose }) {
} }
Header.propTypes = { Header.propTypes = {
name: PropTypes.string, name: PropTypes.string,
currentTheme: PropTypes.string,
onThemeChange: PropTypes.func, onThemeChange: PropTypes.func,
}; };
......
...@@ -2,35 +2,34 @@ ...@@ -2,35 +2,34 @@
position: sticky; position: sticky;
top: 57px; top: 57px;
height: 179px; height: 179px;
.container {
background-color: var(--card-bg);
width: 100%;
.navbar {
display: flex;
align-items: center;
justify-content: space-between;
padding: 0.7em 10%;
.navbar-brand {
display: grid;
grid-template-columns: auto 1fr;
gap: 1rem;
align-items: center;
text-decoration: none;
h2 {
color: var(--title-color);
}
.themelight {
display: flex;
gap: 1rem;
}
.search {
display: none;
}
}
}
}
} }
.header .container {
background-color: var(--card-bg);
width: 100%;
}
.header .container .navbar {
display: flex;
align-items: center;
justify-content: space-between;
padding: 0.7em 10%;
}
.header .container .navbar .navbar-brand {
display: grid;
grid-template-columns: auto 1fr;
gap: 1rem;
align-items: center;
text-decoration: none;
}
.header .container .navbar .navbar-brand h2 {
color: var(--title-color);
}
.header .container .navbar .navbar-brand .themelight {
display: flex;
gap: 1rem;
}
.header .container .navbar .navbar-brand .search {
display: none;
}
@media screen and (min-width: 991px) { @media screen and (min-width: 991px) {
.search { .search {
display: block; display: block;
......
import Icons from "."; import Icons from "@components/base/icons/index";
export default { export default {
title: "Base/Icons", title: "Base/Icons",
......
...@@ -8,18 +8,6 @@ ...@@ -8,18 +8,6 @@
width: 38px; width: 38px;
color: var(--font-color-300); color: var(--font-color-300);
} }
/* .search svg,
.clock svg,
.close svg {
height: 18px;
width: 18px;
vertical-align: sub;
}
.moon svg, .moon svg,
.sun svg { .sun svg {
height: 35px; height: 35px;
...@@ -27,8 +15,6 @@ ...@@ -27,8 +15,6 @@
color: var(--font-color-300); color: var(--font-color-300);
vertical-align: sub; vertical-align: sub;
} }
*/
.menu-icon svg { .menu-icon svg {
background: #5a67d8; background: #5a67d8;
color: white; color: white;
......
"use client";
import { useState, useEffect } from "react"; import { useState, useEffect } from "react";
import Icons from "@components/base/icons"; import Icons from "@components/base/icons";
import styles from "./styles.module.css"; import styles from "./styles.module.css";
......
import Search from "."; import Search from "@components/base/search/index";
export default { export default {
title: "Base/Search", title: "Base/Search",
component: Search, component: Search,
......
.search {
.search{
position: relative; position: relative;
display: inline-block; display: inline-block;
border-radius: 3px; border-radius: 3px;
}
.search-input { .search .search-input {
height: 48px; height: 48px;
width: 30vw; width: 30vw;
background: var(--input-bg); background: var(--input-bg);
border-radius: 5px; border-radius: 5px;
border: .5px solid var(--border-color); border: 0.5px solid var(--border-color);
outline:none; outline: none;
padding-left: 3rem; padding-left: 3rem;
font-size: 16px; font-size: 16px;
&::placeholder{
color: #718096;
font-weight: 600;
color: var(--font-color-300);
}
} }
.search-icon { .search .search-input::placeholder {
color: #718096;
font-weight: 600;
color: var(--font-color-300);
}
.search .search-icon {
position: absolute; position: absolute;
top: 14px; top: 14px;
padding-left: 17px; padding-left: 17px;
color: var(--font-color-300); color: var(--font-color-300);
} }
.search .close-icon {
.close-icon {
cursor: pointer; cursor: pointer;
position: absolute; position: absolute;
color: #2b4c82; color: #2b4c82;
top: 15px; top: 15px;
right: 15px; right: 15px;
} }
}
import BlogLists from "."; import BlogLists from "@components/top-level/blog-lists/index";
export default { export default {
title: "Top-Level/Blog-Lists", title: "Top-Level/Blog-Lists",
component: BlogLists, component: BlogLists,
......
"use client";
import React, { useState } from "react"; import React, { useState } from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import Card from "@/components/base/card"; import Card from "@components/base/card";
import Pagination from "@components/top-level/pagination"; import Pagination from "@components/top-level/pagination";
import styles from "./styles.module.css"; import styles from "./styles.module.css";
......
.blog-container { .blog-container {
max-width: 1200px; max-width: 1200px;
margin: 0 auto; margin: 0 auto;
.title { }
.blog-container .title {
font-size: 38px; font-size: 38px;
line-height: 25px; line-height: 25px;
font-weight: 600; font-weight: 600;
padding-inline: 1.5rem; padding-inline: 1.5rem;
color: var(--title-color); color: var(--title-color);
} }
.description { .blog-container .description {
color: var(--font-color-300); color: var(--font-color-300);
padding-inline: 1.5rem; padding-inline: 1.5rem;
margin-bottom: 2em; margin-bottom: 2em;
} }
.row { .blog-container .row {
display: grid; display: grid;
grid-row-gap: 3rem; grid-row-gap: 3rem;
.col { }
.blog-container .row .col {
display: flex; display: flex;
align-self: normal; align-self: normal;
flex: 1 0 auto; flex: 1 0 auto;
} }
} .pagination {
}
.pagination{
margin-top: 3rem; margin-top: 3rem;
} }
@media screen and (min-width: 768px) { @media screen and (min-width: 768px) {
.row { .row {
grid-template-columns: repeat(auto-fit, minmax(20rem, 1fr)); grid-template-columns: repeat(auto-fit, minmax(20rem, 1fr));
...@@ -37,12 +33,11 @@ ...@@ -37,12 +33,11 @@
} }
} }
@media screen and (min-width: 1440px) { @media screen and (min-width: 1440px) {
.bloglist-container { .bloglist-container {
padding: 0 100px; padding: 0 100px;
.row { }
.bloglist-container .row {
grid-template-columns: repeat(auto-fill, minmax(25rem, 1fr)); grid-template-columns: repeat(auto-fill, minmax(25rem, 1fr));
justify-items: center; justify-items: center;
} }
}
} }
import { useState, useEffect } from "react"; import { useState, useEffect } from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import Button from "@/components/base/button"; import Button from "@components/base/button";
import styles from "./styles.module.css"; import styles from "./styles.module.css";
function Pagination({ total, currentPage, onPageChange, perPage }) { function Pagination({ total, currentPage, onPageChange, perPage }) {
......
import Pagination from "."; import Pagination from "@components/top-level/pagination/index";
export default { export default {
title: "Top-Level/Pagination", title: "Top-Level/Pagination",
component: Pagination, component: Pagination,
......
...@@ -2,29 +2,23 @@ ...@@ -2,29 +2,23 @@
display: flex; display: flex;
justify-content: center; justify-content: center;
align-items: center; align-items: center;
/* background: rgba(216, 73, 73, 0.2); */
width: 100%; width: 100%;
font-weight: 500; font-weight: 500;
font-size: 15px; font-size: 15px;
> span {
display: flex;
justify-content: center;
align-items: center;
margin: 0 0.2em;
padding-inline: 4px;
border-radius: 10px;
height: 40px;
text-decoration: none;
transition: background-color 0.2s;
cursor: pointer;
}
} }
/* .previous button:active, .pagination-container > span {
.next button:active { display: flex;
background-color: var(--primary-bgcolor); justify-content: center;
color: var(--primary-color); align-items: center;
} */ margin: 0 0.2em;
:is(.previous, .next) button:active { padding-inline: 4px;
border-radius: 10px;
height: 40px;
text-decoration: none;
transition: background-color 0.2s;
cursor: pointer;
}
:is(.previous, .next)button:active {
background-color: var(--primary-bgcolor); background-color: var(--primary-bgcolor);
color: var(--primary-color); color: var(--primary-color);
} }
import SingleBlog from "."; import SingleBlog from "@components/top-level/single-blog/index";
export default { export default {
title: "Top-Level/Single-Blog", title: "Top-Level/Single-Blog",
component: SingleBlog, component: SingleBlog,
...@@ -10,7 +10,7 @@ export const singleBlog = { ...@@ -10,7 +10,7 @@ export const singleBlog = {
author: { author: {
name: "Charlotte mia", name: "Charlotte mia",
image: { image: {
url: require("@/public/images/blogsingle.jpg").default.src, url: require("@img/blogsingle.jpg").default.src,
width: 1920, width: 1920,
height: 500, height: 500,
}, },
......
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
} }
.blog-content p, .blog-content p,
ol li { ol li {
font-size: clamp(0.8rem, 1rem + 1vw, 1.1rem); font-size: clamp(0.8rem, 2rem, 1.1rem);
} }
.blogimage { .blogimage {
overflow: hidden; overflow: hidden;
...@@ -21,7 +21,6 @@ ol li { ...@@ -21,7 +21,6 @@ ol li {
object-fit: cover; object-fit: cover;
display: block; display: block;
} }
blockquote p { blockquote p {
font-size: 20px; font-size: 20px;
} }
...@@ -39,31 +38,26 @@ blockquote p { ...@@ -39,31 +38,26 @@ blockquote p {
text-decoration: none; text-decoration: none;
color: var(--font-color-300); color: var(--font-color-300);
} }
/* .blog-list ul li a:hover {
color: var(--font-color-700);
} */
.blog-container { .blog-container {
width: min(90%, 100% - 2rem); width: 90%;
margin: 0 auto; margin: 0 auto;
} }
.tag { .tag {
display: flex; display: flex;
align-items: center; align-items: center;
gap: 0.2rem; gap: 0.2rem;
> p {
background: #e2e8f0;
padding: 0.3em 1em;
border-radius: 40px;
color: #718096;
font-weight: 400;
}
> span {
font-weight: bold;
font-size: 20px;
}
} }
.tag > p {
background: #e2e8f0;
padding: 0.3em 1em;
border-radius: 40px;
color: #718096;
font-weight: 400;
}
.tag > span {
font-weight: bold;
font-size: 20px;
}
.tag p:hover { .tag p:hover {
background: #5a67d8; background: #5a67d8;
color: white; color: white;
...@@ -77,10 +71,9 @@ blockquote p::before { ...@@ -77,10 +71,9 @@ blockquote p::before {
content: "\2014\00A0"; content: "\2014\00A0";
} }
.head-container { .head-container {
width: min(100% - 2rem, 90%); width: 90%;
margin-inline: auto; margin-inline: auto;
} }
.title { .title {
margin: 0; margin: 0;
font-weight: 600; font-weight: 600;
...@@ -89,16 +82,11 @@ blockquote p::before { ...@@ -89,16 +82,11 @@ blockquote p::before {
.list-bold { .list-bold {
font-weight: bold; font-weight: bold;
} }
h2,
h3 {
color: var(--font-color-900);
}
@media (min-width: 768px) { @media screen and (min-width: 768px) {
.blog-content { .blog-content {
padding: 3rem; padding: 3rem;
} }
.tag { .tag {
gap: 1rem; gap: 1rem;
} }
......
{ {
"compilerOptions": { "compilerOptions": {
"baseUrl": ".", "baseUrl": "./",
"paths": { "paths": {
"@/*": ["./*"], "@components/*": ["components/*"],
"@components/*": ["components/*"], "@img/*": ["public/images/*"],
"@img/*": ["public/images/*"], "@posts/*": ["posts/*"],
"@context/*": ["app/context/*"], "@context/*": ["app/context/*"],
} "@lib/*": ["lib/*"]
},
"experimentalDecorators": true
} }
} }
import fs from "fs";
import path from "path";
import matter from "gray-matter";
import { remark } from "remark";
import html from "remark-html";
export const dbDirectory = (filePaths) => {
return path.join(process.cwd(), filePaths);
};
export function getAllPosts(fileDirectory) {
const directory = path.join(process.cwd(), fileDirectory);
const fileNames = fs.readdirSync(directory);
const data = fileNames.map((fileName) => {
const id = fileName.replace(/\.md$/, "");
const fullPath = path.join(fileDirectory, fileName);
const fileContents = fs.readFileSync(fullPath, "utf8");
const matterResult = matter(fileContents);
return {
id,
...matterResult.data,
};
});
return data;
}
/** @type {import('next').NextConfig} */ /** @type {import('next').NextConfig} */
module.exports = { module.exports = {};
experimental: {
prefetch: false,
},
};
This source diff could not be displayed because it is too large. You can view the blob instead.
...@@ -16,15 +16,18 @@ ...@@ -16,15 +16,18 @@
"@fortawesome/react-fontawesome": "^0.1.14", "@fortawesome/react-fontawesome": "^0.1.14",
"classnames": "^2.3.2", "classnames": "^2.3.2",
"gray-matter": "^4.0.3", "gray-matter": "^4.0.3",
"next": "14.0.2", "next": "^14.0.3",
"next-themes": "^0.2.1", "next-themes": "^0.2.1",
"path": "^0.12.7",
"prettier": "^3.1.0", "prettier": "^3.1.0",
"qs": "^6.11.2",
"react": "^18", "react": "^18",
"react-dom": "^18", "react-dom": "^18",
"react-icons": "^4.12.0", "react-icons": "^4.12.0",
"react-markdown": "^9.0.1", "react-markdown": "^9.0.1",
"remark": "^15.0.1",
"remark-gfm": "^4.0.0", "remark-gfm": "^4.0.0",
"storybook-dark-mode": "^3.0.1" "remark-html": "^16.0.1"
}, },
"devDependencies": { "devDependencies": {
"@storybook/addon-essentials": "^7.5.3", "@storybook/addon-essentials": "^7.5.3",
...@@ -38,6 +41,7 @@ ...@@ -38,6 +41,7 @@
"eslint": "^8", "eslint": "^8",
"eslint-config-next": "14.0.2", "eslint-config-next": "14.0.2",
"eslint-plugin-storybook": "^0.6.15", "eslint-plugin-storybook": "^0.6.15",
"storybook": "^7.5.3" "storybook": "^7.5.3",
"storybook-dark-mode": "^3.0.3"
} }
} }
---
id: 1
image: { url: "/images/lifestyle1.jpg", width: 470, height: 300 }
title: "How to get a perfect start for beginning runners"
description: "Lorem ipsum dolor sir amet consectetur ipsum adipisicing elit. Quis vitae sit."
author:
{
name: "Charlotte mia",
image: { url: "/images/a1.jpg", width: 80, height: 80 },
}
publishedDate: "2020-07-16"
readingTime: "1 min read"
category: "Fashion"
categories: ["Fashion", "Beauty"]
---
---
id: 2
image: { url: "/images/lifestyle2.jpg", width: 470, height: 300 }
title: "How to get a perfect start for beginning runners"
description: "Lorem ipsum dolor sir amet consectetur ipsum adipisicing elit. Quis vitae sit."
author:
{
name: "Charlotte mia",
image: { url: "/images/a2.jpg", width: 80, height: 80 },
}
publishedDate: "2020-07-16"
readingTime: "1 min read"
category: "Fashion"
categories: ["Fashion", "Beauty"]
---
---
id: 3
image: { url: "/images/lifestyle3.jpg", width: 470, height: 300 }
title: "How to get a perfect start for beginning runners"
description: "Lorem ipsum dolor sir amet consectetur ipsum adipisicing elit. Quis vitae sit."
author:
{
name: "Charlotte mia",
image: { url: "/images/a3.jpg", width: 80, height: 80 },
}
publishedDate: "2020-07-16"
readingTime: "1 min read"
category: "Fashion"
categories: ["Fashion", "Beauty"]
---
---
id: 4
image: { url: "/images/lifestyle4.jpg", width: 470, height: 300 }
title: "How to get a perfect start for beginning runners"
description: "Lorem ipsum dolor sir amet consectetur ipsum adipisicing elit. Quis vitae sit."
author:
{
name: "Charlotte mia",
image: { url: "/images/a2.jpg", width: 80, height: 80 },
}
publishedDate: "2020-07-16"
readingTime: "1 min read"
category: "Fashion"
categories: ["Fashion", "Beauty"]
---
---
id: 5
image: { url: "/images/lifestyle5.jpg", width: 470, height: 300 }
title: "How to get a perfect start for beginning runners"
description: "Lorem ipsum dolor sir amet consectetur ipsum adipisicing elit. Quis vitae sit."
author:
{
name: "Charlotte mia",
image: { url: "/images/a1.jpg", width: 80, height: 80 },
}
publishedDate: "2020-07-16"
readingTime: "1 min read"
category: "Fashion"
categories: ["Fashion", "Beauty"]
---
---
id: 6
image: { url: "/images/lifestyle6.jpg", width: 470, height: 300 }
title: "How to get a perfect start for beginning runners"
description: "Lorem ipsum dolor sir amet consectetur ipsum adipisicing elit. Quis vitae sit."
author:
{
name: "Charlotte mia",
image: { url: "/images/a3.jpg", width: 80, height: 80 },
}
publishedDate: "2020-07-16"
readingTime: "1 min read"
category: "Fashion"
categories: ["Fashion", "Beauty"]
---
---
id: 7
image: { url: "/images/lifestyle7.jpg", width: 470, height: 300 }
title: "How to get a perfect start for beginning runners"
description: "Lorem ipsum dolor sir amet consectetur ipsum adipisicing elit. Quis vitae sit."
author:
{
name: "Charlotte mia",
image: { url: "/images/a1.jpg", width: 80, height: 80 },
}
publishedDate: "2020-07-16"
readingTime: "1 min read"
category: "Fashion"
categories: ["Fashion", "Beauty"]
---
---
id: 8
image: { url: "/images/lifestyle8.jpg", width: 470, height: 300 }
title: "How to get a perfect start for beginning runners"
description: "Lorem ipsum dolor sir amet consectetur ipsum adipisicing elit. Quis vitae sit."
author:
{
name: "Charlotte mia",
image: { url: "/images/a2.jpg", width: 80, height: 80 },
}
publishedDate: "2020-07-16"
readingTime: "1 min read"
category: "Fashion"
categories: ["Fashion", "Beauty"]
---
---
id: 9
image: { url: "/images/lifestyle2.jpg", width: 470, height: 300 }
title: "How to get a perfect start for beginning runners"
description: "Lorem ipsum dolor sir amet consectetur ipsum adipisicing elit. Quis vitae sit."
author:
{
name: "Charlotte mia",
image: { url: "/images/a1.jpg", width: 80, height: 80 },
}
publishedDate: "2020-07-16"
readingTime: "1 min read"
category: "Fashion"
categories: ["Fashion", "Beauty"]
---
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment