Portfolio/src/components/ProjectDetails/ProjectDetails.jsx

101 lines
4.6 KiB
JavaScript
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

import "./ProjectDetails.css"
import SkillCard from "../SkillCard/SkillCard.jsx";
import React, {useEffect, useState} from "react";
import NavBar from "../NavBar/NavBar.jsx";
import ReactMarkdown from 'react-markdown';
import remarkGfm from 'remark-gfm';
import {useNavigate} from "react-router-dom";
function ProjectDetails({project}) {
const [mdTitle, setMdTitle] = useState([]);
const navigate = useNavigate();
useEffect(() =>{
if (!project.long_description) return;
const matches = [...project.long_description.matchAll(/^## (?!#)(.+)$/gm)];
setMdTitle(matches.map(match => match[1].trim()));
}, [project.long_description])
const handleSummaryClick = (event, anchor) => {
event.preventDefault();
const element = document.getElementById(anchor);
if (element) {
element.scrollIntoView({ behavior: 'smooth', block: 'start' });
}
};
return (
<div>
<button className="return-button" onClick={() => navigate(-1)}>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 -960 960 960" width="24px" height="24px" fill="currentColor">
<path d="m313-440 224 224-57 56-320-320 320-320 57 56-224 224h487v80H313Z"/>
</svg>
</button>
<div className="nav-bar">
<NavBar/>
</div>
<div className="project-details">
<div className="project-details-header">
<h1>
{project.title}
{project.school ? (
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 640 640" fill="#D95F46" width="1em" height="1em">
<path d="M80 259.8L289.2 345.9C299 349.9 309.4 352 320 352C330.6 352 341 349.9 350.8 345.9L593.2 246.1C602.2 242.4 608 233.7 608 224C608 214.3 602.2 205.6 593.2 201.9L350.8 102.1C341 98.1 330.6 96 320 96C309.4 96 299 98.1 289.2 102.1L46.8 201.9C37.8 205.6 32 214.3 32 224L32 520C32 533.3 42.7 544 56 544C69.3 544 80 533.3 80 520L80 259.8zM128 331.5L128 448C128 501 214 544 320 544C426 544 512 501 512 448L512 331.4L369.1 390.3C353.5 396.7 336.9 400 320 400C303.1 400 286.5 396.7 270.9 390.3L128 331.4z"/>
</svg>
) : null}
</h1>
{project.end_year === null ? (
<p className="project-details-years">{project.beginning_year + ' in progress'}</p>
) : project.beginning_year === project.end_year ? (
<p className="project-details-years">{project.beginning_year}</p>
) : (
<p className="project-details-years">{project.beginning_year + ' ' + project.end_year}</p>
)}
<p className="project-details-short-description">{project.short_description}</p>
<ul>
{project.skills && project.skills.map((skill) => (
<li>
<SkillCard text={skill} />
</li>
))}
</ul>
</div>
<div className="project-details-container">
<div className="project-details-content">
<ReactMarkdown
remarkPlugins={[remarkGfm]}
components={{
h2({node, ...props}) {
const id = String(props.children).toLowerCase().replace(/\s+/g, "-");
return <h2 id={id} {...props} />;
}
}}
>
{project.long_description}
</ReactMarkdown>
</div>
<aside>
<ul>
{mdTitle.map(title => {
const anchor = title.toLowerCase().replace(/\s+/g, "-");
return (
<li key={anchor}>
<a href={`#${anchor}`} onClick={(e) => handleSummaryClick(e, anchor)}>
{title}
</a>
</li>
);
})}
</ul>
</aside>
</div>
</div>
</div>
);
}
export default ProjectDetails