feat: announcement bar added

This commit is contained in:
Al Murad Uzzaman
2025-11-04 12:48:44 +06:00
parent 670162da19
commit 8cd72d878a
5 changed files with 106 additions and 10 deletions
+7 -7
View File
@@ -1,6 +1,6 @@
{
"name": "astroplate",
"version": "5.7.0",
"version": "5.8.0",
"description": "Astro and Tailwindcss boilerplate",
"author": "zeon.studio",
"license": "MIT",
@@ -17,12 +17,12 @@
},
"dependencies": {
"@astrojs/check": "0.9.5",
"@astrojs/mdx": "4.3.8",
"@astrojs/react": "4.4.0",
"@astrojs/mdx": "4.3.9",
"@astrojs/react": "4.4.1",
"@astrojs/sitemap": "3.6.0",
"@digi4care/astro-google-tagmanager": "^1.6.0",
"@justinribeiro/lite-youtube": "^1.8.2",
"astro": "5.15.1",
"@justinribeiro/lite-youtube": "^1.9.0",
"astro": "5.15.3",
"astro-auto-import": "^0.4.5",
"astro-font": "^1.1.0",
"date-fns": "^4.1.0",
@@ -43,10 +43,10 @@
"@tailwindcss/forms": "^0.5.10",
"@tailwindcss/typography": "^0.5.19",
"@tailwindcss/vite": "^4.1.16",
"@types/node": "24.9.1",
"@types/node": "24.10.0",
"@types/react": "19.2.2",
"@types/react-dom": "19.2.2",
"eslint": "^9.38.0",
"eslint": "^9.39.1",
"prettier": "^3.6.2",
"prettier-plugin-astro": "^0.14.1",
"prettier-plugin-tailwindcss": "^0.7.1",
+3 -3
View File
@@ -10,8 +10,8 @@
</h2>
<p align=center>
<a href="https://github.com/withastro/astro/releases/tag/astro%405.14.4">
<img src="https://img.shields.io/static/v1?label=ASTRO&message=5.14&color=000&logo=astro" alt="Astro Version 5.14"/>
<a href="https://github.com/withastro/astro/releases/tag/astro%405.15.3">
<img src="https://img.shields.io/static/v1?label=ASTRO&message=5.15&color=000&logo=astro" alt="Astro Version 5.15"/>
</a>
<a href="https://github.com/zeon-studio/astroplate/blob/main/LICENSE">
@@ -67,7 +67,7 @@
### 📦 Dependencies
- astro v5.7+
- astro v5.15+
- node v20.10+
- yarn v1.22+
- tailwind v4+
+6
View File
@@ -22,6 +22,12 @@
"blog_folder": "blog"
},
"announcement": {
"enable": true,
"content": "<span class='text-center block'>♥️ Loving Astroplate? <a class='underline' href='https://github.com/zeon-studio/astroplate' target='_blank' rel='noopener'>Please ⭐️ on Github</a></span>",
"expire_days": 7
},
"params": {
"contact_form_action": "#",
"copyright": "Designed And Developed by [Zeon Studio](https://zeon.studio)"
+2
View File
@@ -12,6 +12,7 @@ import {
} from "@digi4care/astro-google-tagmanager";
import { AstroFont } from "astro-font";
import { ClientRouter } from "astro:transitions";
import Announcement from "./helpers/Announcement";
import SearchModal from "./helpers/SearchModal";
// font families
@@ -189,6 +190,7 @@ const { title, meta_title, description, image, noindex, canonical } =
}
<TwSizeIndicator />
<Announcement client:load />
<Header />
<SearchModal client:load />
<main id="main-content">
+88
View File
@@ -0,0 +1,88 @@
import config from "@/config/config.json";
import { markdownify } from "@/lib/utils/textConverter";
import React, { useEffect, useState } from "react";
const { enable, content, expire_days } = config.announcement;
const Cookies = {
set: (name: string, value: string, options: any = {}) => {
if (typeof document === "undefined") return;
const defaults = { path: "/" };
const opts = { ...defaults, ...options };
if (typeof opts.expires === "number") {
opts.expires = new Date(Date.now() + opts.expires * 864e5);
}
if (opts.expires instanceof Date) {
opts.expires = opts.expires.toUTCString();
}
let cookieString = `${encodeURIComponent(name)}=${encodeURIComponent(value)}`;
for (let key in opts) {
if (!opts[key]) continue;
cookieString += `; ${key}`;
if (opts[key] !== true) {
cookieString += `=${opts[key]}`;
}
}
document.cookie = cookieString;
},
get: (name: string): string | null => {
if (typeof document === "undefined") return null;
const cookies = document.cookie.split("; ");
for (let cookie of cookies) {
const [key, value] = cookie.split("=");
if (decodeURIComponent(key) === name) {
return decodeURIComponent(value);
}
}
return null;
},
remove: (name: string, options: any = {}) => {
Cookies.set(name, "", { ...options, expires: -1 });
},
};
const Announcement: React.FC = () => {
const [isVisible, setIsVisible] = useState(false);
useEffect(() => {
if (enable && content && !Cookies.get("announcement-close")) {
setIsVisible(true);
}
}, []);
const handleClose = () => {
Cookies.set("announcement-close", "true", {
expires: expire_days,
});
setIsVisible(false);
};
if (!enable || !content || !isVisible) {
return null;
}
return (
<div className="relative z-999 bg-body dark:bg-darkmode-body shadow-[1px_0_10px_7px_rgba(154,154,154,0.11)] px-4 py-4 pr-12 md:text-lg transition-all duration-300">
<p
dangerouslySetInnerHTML={{ __html: markdownify(content) }}
/>
<button
onClick={handleClose}
className="absolute top-1/2 right-4 -translate-y-1/2 cursor-pointer flex items-center justify-center w-7 h-7 border border-border dark:border-darkmode-border rounded-full text-xl transition-colors duration-200"
aria-label="Close announcement"
>
&times;
</button>
</div>
);
};
export default Announcement;