Skip to content
40 changes: 38 additions & 2 deletions package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@
"preview": "vite preview"
},
"dependencies": {
"framer-motion": "^11.15.0",
"prismjs": "^1.29.0",
"react": "^18.3.1",
"react-dom": "^18.3.1",
Expand Down
76 changes: 52 additions & 24 deletions src/components/SnippetList.tsx
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import { useState } from "react";
import { motion, AnimatePresence } from "framer-motion";
import { SnippetType } from "../types";
import { useAppContext } from "../contexts/AppContext";
import { useSnippets } from "../hooks/useSnippets";
Expand Down Expand Up @@ -30,31 +31,58 @@ const SnippetList = () => {

return (
<>
<ul role="list" className="snippets">
{fetchedSnippets.map((snippet, idx) => (
<li key={idx}>
<button
className="snippet | flow"
data-flow-space="sm"
onClick={() => handleOpenModal(snippet)}
<motion.ul
role="list"
className="snippets"
>
<AnimatePresence mode="popLayout">
{fetchedSnippets.map((snippet, idx) => (
<motion.li
key={idx}
initial={{ opacity: 0, y: 20 }}
animate={{
opacity: 1,
y: 0,
transition: {
delay: idx * 0.05,
duration: 0.2
}
}}
exit={{
opacity: 0,
y: -20,
transition: {
delay: (fetchedSnippets.length - 1 - idx) * 0.01,
duration: 0.09
}
}}
>
<div className="snippet__preview">
<img src={language.icon} alt={language.lang} />
</div>

<h3 className="snippet__title">{snippet.title}</h3>
</button>
</li>
))}
</ul>

{isModalOpen && snippet && (
<SnippetModal
snippet={snippet}
handleCloseModal={handleCloseModal}
language={language.lang}
/>
)}
<motion.button
className="snippet | flow"
data-flow-space="sm"
onClick={() => handleOpenModal(snippet)}
whileHover={{ scale: 1.01 }}
whileTap={{ scale: 0.98 }}
>
<div className="snippet__preview">
<img src={language.icon} alt={language.lang} />
</div>
<h3 className="snippet__title">{snippet.title}</h3>
</motion.button>
</motion.li>
))}
</AnimatePresence>
</motion.ul>

<AnimatePresence mode="wait">
{isModalOpen && snippet && (
<SnippetModal
snippet={snippet}
handleCloseModal={handleCloseModal}
language={language.lang}
/>
)}
</AnimatePresence>
</>
);
};
Expand Down
22 changes: 18 additions & 4 deletions src/components/SnippetModal.tsx
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import React from "react";
import ReactDOM from "react-dom";
import { motion } from "framer-motion";
import Button from "./Button";
import { CloseIcon } from "./Icons";
import CodePreview from "./CodePreview";
Expand All @@ -23,15 +24,28 @@ const SnippetModal: React.FC<Props> = ({
useEscapeKey(handleCloseModal);

return ReactDOM.createPortal(
<div
<motion.div
key="modal-overlay"
className="modal-overlay"
onClick={(e) => {
if (e.target === e.currentTarget) {
handleCloseModal();
}
}}
initial={{ opacity: 0 }}
animate={{ opacity: 1 }}
exit={{ opacity: 0 }}
transition={{ duration: 0.2 }}
>
<div className="modal | flow" data-flow-space="lg">
<motion.div
key="modal-content"
className="modal | flow"
data-flow-space="lg"
initial={{ scale: 0.8, opacity: 0, y: 20 }}
animate={{ scale: 1, opacity: 1, y: 0 }}
exit={{ scale: 0.8, opacity: 0, y: 20 }}
transition={{ type: "spring", duration: 0.5 }}
>
<div className="modal__header">
<h2 className="section-title">{snippet.title}</h2>
<Button isIcon={true} onClick={handleCloseModal}>
Expand Down Expand Up @@ -61,8 +75,8 @@ const SnippetModal: React.FC<Props> = ({
</li>
))}
</ul>
</div>
</div>,
</motion.div>
</motion.div>,
modalRoot
);
};
Expand Down
3 changes: 2 additions & 1 deletion src/types/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,8 @@ export type CategoryType = {
snippets: SnippetType[];
};

export type SnippetType = {
export interface SnippetType {
id?: string | number;
title: string;
description: string;
code: string[];
Expand Down