88 lines
4.3 KiB
TypeScript
88 lines
4.3 KiB
TypeScript
import type { NextPage } from "next";
|
|
import { useEffect, useRef, useState } from "react";
|
|
import { useRouter } from "next/router";
|
|
import styles from "../styles/ProjectModal.module.css";
|
|
import type { Project, Diary } from "../lib/content/types";
|
|
import { useCommands } from "./contexts/CommandInterface";
|
|
import { generateContent, projectEmpty } from "../lib/content/generate";
|
|
import { useModalFunctions } from "./contexts/ModalFunctions";
|
|
|
|
// Code Highlighting
|
|
import hljs from "highlight.js";
|
|
import rust from "highlight.js/lib/languages/rust";
|
|
import bash from "highlight.js/lib/languages/shell";
|
|
hljs.registerLanguage("rust", rust);
|
|
hljs.registerLanguage("bash", bash);
|
|
hljs.registerLanguage("console", bash);
|
|
hljs.registerLanguage("shell", bash);
|
|
|
|
const ProjectModal: NextPage = () => {
|
|
const [visible, setVisible] = useState<boolean>(false);
|
|
const [currentContent, setCurrentContent] = useState<Project | Diary | undefined>(undefined);
|
|
const [currentPage, setCurrentPage] = useState<number>(0);
|
|
const [HTMLContent, setHTMLContent] = useState<string>(projectEmpty);
|
|
const router = useRouter();
|
|
|
|
const { updateCallbacks: updateCmdCallbacks, cmdContext } = useCommands();
|
|
const { updateCallbacks: updateModalCallbacks } = useModalFunctions();
|
|
|
|
const setModalContent = async (content: Project | Diary, selectedPage?: number) => {
|
|
setCurrentContent(content);
|
|
router.replace("#", `#/${content.type}/${content.name}${content.type === "diary" && selectedPage ? `/${selectedPage}`: ""}`, {shallow: true});
|
|
if (content.type === "diary") setCurrentPage(selectedPage === undefined ? 0 : selectedPage);
|
|
setHTMLContent(await generateContent(content, selectedPage));
|
|
};
|
|
|
|
const setModalVisible = async (visible: boolean) => {
|
|
if (!visible) router.replace("#", undefined, {shallow: true});
|
|
setVisible(visible);
|
|
};
|
|
|
|
const onContentReady = () => {
|
|
const selected = window.location.hash.split("/");
|
|
if (selected[2]) cmdContext.executeCommand(`project ${selected[2]}`);
|
|
};
|
|
|
|
updateCmdCallbacks({ setModalVisible, setModalContent, setModalHTML: setHTMLContent });
|
|
updateModalCallbacks({ setVisible: setModalVisible, setContent: setModalContent, setHtml: setHTMLContent, onContentReady });
|
|
|
|
useEffect(() => {
|
|
hljs.highlightAll();
|
|
}, [HTMLContent]);
|
|
|
|
const containerRef = useRef<HTMLDivElement>(null);
|
|
|
|
if (!visible) return <></>;
|
|
|
|
const nextPageSelector = (() => {
|
|
if (!currentContent || currentContent?.type !== "diary" || currentContent.entries.length === 0) return null;
|
|
|
|
const prev = <span className={styles.leftSelectSpace}>{currentPage > 0 ? <a className={"fake-link"} onClick={() => setModalContent(currentContent, currentPage - 1)}>< {currentPage - 1 > 0 ? currentContent.entries[currentPage - 2].title : "Main page"}</a> : null}</span>;
|
|
const next = <span className={styles.rightSelectSpace}>{currentPage < currentContent.entries.length ? <a className={"fake-link"} onClick={() => setModalContent(currentContent, currentPage + 1)}>{currentContent.entries[currentPage].title} ></a> : null}</span>;
|
|
|
|
const select = (
|
|
<select onChange={(e) => setModalContent(currentContent, Number.parseInt(e.target.value))} value={currentPage}>
|
|
<option key={-1} value={0}>Main page</option>
|
|
{currentContent.entries.map((entry, i) => <option key={i} value={i + 1}>{entry.title}</option>)}
|
|
</select>
|
|
);
|
|
|
|
return <div className={styles.pageSelector}>{prev}<span style={{visibility: currentPage > 0 ? "visible" : "hidden"}}> | </span>{select}<span style={{visibility: currentPage < currentContent.entries.length ? "visible" : "hidden"}}> | </span>{next}</div>;
|
|
})();
|
|
|
|
return <div className={styles.modal}>
|
|
<a onClick={() => setModalVisible(false)} className={"fake-link"}>
|
|
<div className={styles.modalClose}><div className={styles.modalCloseAlign}>X</div></div>
|
|
</a>
|
|
<div className={styles.modalContainer}>
|
|
{nextPageSelector}
|
|
<div className={`${styles.modalText} asciidoc`} ref={containerRef} dangerouslySetInnerHTML={{ __html: HTMLContent ? HTMLContent : projectEmpty }}>
|
|
|
|
</div>
|
|
{nextPageSelector}
|
|
</div>
|
|
</div>;
|
|
};
|
|
|
|
|
|
export default ProjectModal; |