Phase 1 (Foundation): - Project restructure (presenton-main → backend/ + frontend/) - Database schema (8 new models, Alembic config, seed script) - Auth (Azure AD SSO + dev bypass, JWT sessions, AuthMiddleware) - RBAC (access_service, rbac_middleware, admin routers) - Audit logging (fire-and-forget, AuditMiddleware, admin router) - i18n (react-i18next with 5 namespace files) Phase 2 (Admin Panel & Client Management): - Admin panel shell (sidebar layout, role guard, 12 pages) - Redux admin slice with 18 async thunks - User management (role changes, deactivation) - Client management (CRUD, brand config, team management) - Brand config editor (colors, fonts, logos, voice rules) - Master deck upload & parser (PPTX → HTML → React pipeline) - Audit log viewer with filters and CSV/JSON export Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
139 lines
No EOL
6.3 KiB
TypeScript
139 lines
No EOL
6.3 KiB
TypeScript
import React from 'react'
|
|
// charts removed
|
|
import * as z from "zod";
|
|
|
|
const ImageSchema = z.object({
|
|
__image_url__: z.string().url().default("https://images.pexels.com/photos/31527637/pexels-photo-31527637.jpeg").meta({
|
|
description: "URL to image",
|
|
}),
|
|
__image_prompt__: z.string().min(10).max(150).default("Small decorative photo partially behind the card showing a business theme").meta({
|
|
description: "Prompt used to generate the image. Max 30 words",
|
|
}),
|
|
})
|
|
|
|
const IconSchema = z.object({
|
|
__icon_url__: z.string().default("").meta({
|
|
description: "URL to icon",
|
|
}),
|
|
__icon_query__: z.string().min(3).max(30).default("").meta({
|
|
description: "Query used to search the icon. Max 5 words",
|
|
}),
|
|
})
|
|
|
|
const layoutId = "split-left-strip-header-title-subtitle-cards-slide"
|
|
const layoutName = "Heading Bullet Image Description"
|
|
const layoutDescription = "A slide with a left strip, top label with rule, right header, right description, floating small image, and a centered card with ...cards."
|
|
|
|
const Schema = z.object({
|
|
metaMaxWords: z.number().default(200).meta({
|
|
description: "Maximum number of words the text areas can handle collectively.",
|
|
}),
|
|
pageNumber: z.string().min(1).max(3).default("7").meta({
|
|
description: "Top-right page number text. Max 3 chars",
|
|
}),
|
|
|
|
heading: z.string().min(16).max(38).default("A Blueprint for\nSuccess").meta({
|
|
description: "Main heading across up to 2 lines. Max 7 words",
|
|
}),
|
|
subheading: z.string().min(60).max(200).default("Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna").meta({
|
|
description: "Supporting paragraph under the heading. Max 35 words",
|
|
}),
|
|
smallImage: ImageSchema.default({
|
|
__image_url__: "https://images.pexels.com/photos/327533/pexels-photo-327533.jpeg",
|
|
__image_prompt__: "A small landscape image suitable for a business slide"
|
|
}).meta({
|
|
description: "Small image partially behind the main card",
|
|
}),
|
|
cards: z.array(z.object({
|
|
title: z.string().min(8).max(16).default("Strategy 01").meta({
|
|
description: "Card ribbon title. Max 3 words",
|
|
}),
|
|
body: z.string().min(60).max(160).default("Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor").meta({
|
|
description: "Card body text. Max 28 words",
|
|
}),
|
|
})).min(1).max(4).default([
|
|
{ title: "Strategy 01", body: "Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor" },
|
|
{ title: "Strategy 02", body: "Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor" },
|
|
{ title: "Strategy 03", body: "Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor" },
|
|
{ title: "Strategy 04", body: "Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor" },
|
|
]).meta({
|
|
description: "Array of strategy cards",
|
|
}),
|
|
|
|
})
|
|
|
|
type SlideData = z.infer<typeof Schema>
|
|
|
|
interface SlideLayoutProps {
|
|
data?: Partial<SlideData>
|
|
}
|
|
|
|
const dynamicSlideLayout: React.FC<SlideLayoutProps> = ({ data: slideData }) => {
|
|
const cards = slideData?.cards || []
|
|
|
|
return (
|
|
<>
|
|
<link
|
|
href="https://fonts.googleapis.com/css2?family=Playfair+Display:wght@400;500;600;700&display=swap"
|
|
rel="stylesheet"
|
|
/>
|
|
<div className=" w-full rounded-sm max-w-[1280px] shadow-lg max-h-[720px] aspect-video relative z-20 mx-auto overflow-hidden" style={{ fontFamily: "var(--heading-font-family,Playfair Display)", backgroundColor: 'var(--background-color, #FFFFFF)' }}>
|
|
{/* page number removed */}
|
|
|
|
<div className="grid grid-cols-2 h-full">
|
|
<div className="relative bg-[#efefef]">
|
|
{slideData?.smallImage?.__image_url__ ? (
|
|
<>
|
|
<img
|
|
src={slideData.smallImage.__image_url__}
|
|
alt={slideData.smallImage.__image_prompt__ || "image"}
|
|
className="absolute inset-0 w-full h-full object-cover"
|
|
/>
|
|
{/* overlay removed */}
|
|
</>
|
|
) : null}
|
|
<div className="pt-6 pl-10 pr-6 relative z-[1]">
|
|
<div className="flex items-center gap-6">
|
|
<div className="flex items-center gap-1">
|
|
|
|
{(slideData as any)?._logo_url__ && <img src={(slideData as any)?._logo_url__} alt="logo" className="w-6 h-6" />}
|
|
{(slideData as any)?.__companyName__ && <span className="text-[18px] font-semibold" style={{ color: 'var(--background-text, #111827)' }}>{(slideData as any)?.__companyName__ || "Pitchdeck"}</span>}
|
|
</div>
|
|
<div className="h-[2px] w-[220px] rounded-full" style={{ backgroundColor: 'var(--background-text, #111827)' }}></div>
|
|
</div>
|
|
</div>
|
|
<div className="absolute bottom-0 left-0 w-full h-[88px] bg-gradient-to-t from-black/20 to-transparent z-[1]"></div>
|
|
</div>
|
|
|
|
<div className="relative px-12 pt-16">
|
|
<h1 className=" leading-[72px] text-[64px] tracking-tight whitespace-pre-line font-semibold" style={{ color: 'var(--background-text, #111827)' }}>
|
|
{slideData?.heading || "A Blueprint for\nSuccess"}
|
|
</h1>
|
|
|
|
</div>
|
|
</div>
|
|
<div className="absolute left-10 right-10 top-[320px] z-10">
|
|
<div className="w-fit max-w-full rounded-md shadow-[0_20px_60px_rgba(0,0,0,0.12)] ml-auto" style={{ backgroundColor: 'var(--card-color, #F3F4F6)' }}>
|
|
<div className="px-8 py-10">
|
|
<div className="grid grid-flow-col auto-cols-max gap-6">
|
|
{cards.map((card, idx) => (
|
|
<div key={idx} className="flex flex-col items-center">
|
|
<div className="w-[240px] h-[64px] rounded-sm text-white flex items-center justify-center text-[22px]" style={{ backgroundColor: 'var(--primary-color, #1B8C2D)', color: 'var(--primary-text, #FFFFFF)' }}>
|
|
{card.title}
|
|
</div>
|
|
<p className="mt-6 text-center text-[16px] leading-[28px] max-w-[240px]" style={{ color: 'var(--background-text, #6B7280)' }}>
|
|
{card.body}
|
|
</p>
|
|
</div>
|
|
))}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</>
|
|
)
|
|
}
|
|
|
|
export { Schema, layoutId, layoutName, layoutDescription }
|
|
export default dynamicSlideLayout |