We code better now
This commit is contained in:
parent
929519225a
commit
0a0020b5c0
@ -1,69 +1,48 @@
|
|||||||
import type { NextPage } from "next";
|
import type { NextPage } from "next";
|
||||||
import { useEffect, useRef, useState } from "react";
|
import { useContext, useEffect, useRef, useState } from "react";
|
||||||
import asciidoctor from "asciidoctor";
|
import asciidoctor from "asciidoctor";
|
||||||
import styles from "../styles/ProjectModal.module.css";
|
import styles from "../styles/ProjectModal.module.css";
|
||||||
import type { Project, Diary } from "../lib/projects/types";
|
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";
|
||||||
//import Link from "next/link";
|
//import Link from "next/link";
|
||||||
|
|
||||||
interface ModalInput {
|
const ProjectModal: NextPage = () => {
|
||||||
project: Project|Diary|undefined;
|
const [visible, setVisible] = useState<boolean>(false);
|
||||||
projectType: "project"|"diary";
|
const [diaryPages, setDiaryPages] = useState<string[]>([]);
|
||||||
visible: boolean;
|
const [content, setContent] = useState<string>(projectEmpty);
|
||||||
setVisible: CallableFunction;
|
|
||||||
}
|
|
||||||
|
|
||||||
const ad = asciidoctor();
|
const setModalContent = async (content: Project|Diary, selectedPage?: number) => {
|
||||||
|
if (content.type === "diary") setDiaryPages(content.entries.map(entry => entry.title));
|
||||||
|
setContent(await generateContent(content, selectedPage));
|
||||||
|
};
|
||||||
|
|
||||||
|
const { updateCallbacks: updateCmdCallbacks } = useCommands();
|
||||||
|
const { updateCallbacks: updateModalCallbacks } = useModalFunctions();
|
||||||
|
updateCmdCallbacks({ setModalVisible: setVisible, setModalContent});
|
||||||
|
updateModalCallbacks({ setVisible, setContent: setModalContent, setHtml: setContent });
|
||||||
|
|
||||||
const ProjectModal: NextPage<ModalInput> = ({ project, projectType, visible, setVisible }) => {
|
|
||||||
const projectEmpty = "<div>Kein Projekt ausgewählt.</div>";
|
|
||||||
const [projectData, setProjectData] = useState<string>(projectEmpty);
|
|
||||||
const containerRef = useRef<HTMLDivElement>(null);
|
const containerRef = useRef<HTMLDivElement>(null);
|
||||||
|
|
||||||
const projectNotFoundHtml = `<div class="${"error"}">Sorry! There is no data for this project. Please check back later to see if that changed!</div>`;
|
|
||||||
const projectServerErrorHtml = `<div class="${"error"}">Sorry! A server error happend when the project data was fetched!</div>`;
|
|
||||||
|
|
||||||
const generateFooter = (project: string, lastUpdate: string) => `<hr>
|
|
||||||
<div id="footer">
|
|
||||||
<div id="footer-text">
|
|
||||||
Last updated: ${lastUpdate} | <a href="https://git.c0ntroller.de/c0ntroller/frontpage-projects/src/branch/senpai/${project}.adoc">Document source</a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
`;
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (project && project.name) {
|
if (content && containerRef.current) {
|
||||||
// TODO
|
containerRef.current.innerHTML = content;
|
||||||
// set Spinner
|
|
||||||
setProjectData("Loading...");
|
|
||||||
fetch(`/api/${projectType === "diary" ? "diaries" : "projects"}/${project.name}`).then((res) => {
|
|
||||||
if (res.status === 404) setProjectData(projectNotFoundHtml);
|
|
||||||
if (res.status !== 200) setProjectData(projectServerErrorHtml);
|
|
||||||
res.text().then(data => {
|
|
||||||
try {
|
|
||||||
const adDoc = ad.load(data, { attributes: { showtitle: true } });
|
|
||||||
setProjectData(adDoc.convert(adDoc).toString() + generateFooter(project.name, adDoc.getAttribute("docdatetime")));
|
|
||||||
} catch {
|
|
||||||
setProjectData(projectServerErrorHtml);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
} else if (typeof project === "undefined") setProjectData(projectEmpty);
|
|
||||||
}, [project, projectType, projectEmpty, projectNotFoundHtml, projectServerErrorHtml]);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (projectData && containerRef.current && projectData !== "") {
|
|
||||||
containerRef.current.innerHTML = projectData;
|
|
||||||
}
|
}
|
||||||
|
}, [content]);
|
||||||
}, [projectData, visible]);
|
|
||||||
|
|
||||||
if (!visible) return <></>;
|
if (!visible) return <></>;
|
||||||
|
|
||||||
return <div className={styles.modal}>
|
return <div className={styles.modal}>
|
||||||
<a href="javascript:void(0);" onClick={() => setVisible(false)}>
|
<a onClick={() => setVisible(false)}>
|
||||||
<div className={styles.modalClose}><div className={styles.modalCloseAlign}>X</div></div>
|
<div className={styles.modalClose}><div className={styles.modalCloseAlign}>X</div></div>
|
||||||
</a>
|
</a>
|
||||||
<div className={styles.modalContainer}>
|
<div className={styles.modalContainer}>
|
||||||
|
{
|
||||||
|
// TODO
|
||||||
|
// If diaryPages
|
||||||
|
// Show page selector
|
||||||
|
}
|
||||||
<div className={`${styles.modalText} asciidoc`} ref={containerRef}>
|
<div className={`${styles.modalText} asciidoc`} ref={containerRef}>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import { NextPage } from "next";
|
import { NextPage } from "next";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { BaseSyntheticEvent, MutableRefObject, useEffect, useRef } from "react";
|
import type { BaseSyntheticEvent, MutableRefObject } from "react";
|
||||||
import styles from "../../styles/REPL/REPLHistory.module.css";
|
import styles from "../../styles/REPL/REPLHistory.module.css";
|
||||||
|
|
||||||
interface REPLHistoryParams {
|
interface REPLHistoryParams {
|
||||||
|
@ -1,27 +1,17 @@
|
|||||||
import type { NextPage } from "next";
|
import type { NextPage } from "next";
|
||||||
import useSWR from "swr";
|
import { MutableRefObject, useState, createRef } from "react";
|
||||||
import { MutableRefObject, useState, createRef, useEffect } from "react";
|
|
||||||
import { CommandInterface } from "../../lib/commands";
|
import { CommandInterface } from "../../lib/commands";
|
||||||
import styles from "../../styles/REPL/REPLInput.module.css";
|
import styles from "../../styles/REPL/REPLInput.module.css";
|
||||||
import type { ProjectList } from "../../lib/projects/types";
|
import { useCommands } from "../contexts/CommandInterface";
|
||||||
|
import { useModalFunctions } from "../contexts/ModalFunctions";
|
||||||
|
|
||||||
interface REPLInputParams {
|
interface REPLInputParams {
|
||||||
historyCallback: CallableFunction;
|
historyCallback: CallableFunction;
|
||||||
historyClear: CallableFunction;
|
historyClear: CallableFunction;
|
||||||
inputRef: MutableRefObject<HTMLInputElement|null>;
|
inputRef: MutableRefObject<HTMLInputElement|null>;
|
||||||
modalManipulation: {
|
|
||||||
setModalVisible: CallableFunction;
|
|
||||||
setModalProject: CallableFunction;
|
|
||||||
setModalProjectType: CallableFunction;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async function fetchProjects(endpoint: string): Promise<ProjectList> {
|
const REPLInput: NextPage<REPLInputParams> = ({historyCallback, historyClear, inputRef}) => {
|
||||||
const res = await fetch(endpoint);
|
|
||||||
return res.json();
|
|
||||||
}
|
|
||||||
|
|
||||||
const REPLInput: NextPage<REPLInputParams> = ({historyCallback, historyClear, inputRef, modalManipulation}) => {
|
|
||||||
const typed = createRef<HTMLSpanElement>();
|
const typed = createRef<HTMLSpanElement>();
|
||||||
const completion = createRef<HTMLSpanElement>();
|
const completion = createRef<HTMLSpanElement>();
|
||||||
const [currentCmd, setCurrentCmd] = useState<string[]>([]);
|
const [currentCmd, setCurrentCmd] = useState<string[]>([]);
|
||||||
@ -29,8 +19,8 @@ const REPLInput: NextPage<REPLInputParams> = ({historyCallback, historyClear, in
|
|||||||
const [inCmdHistory, setInCmdHistory] = useState<number>(-1);
|
const [inCmdHistory, setInCmdHistory] = useState<number>(-1);
|
||||||
const [cmdHistory, setCmdHistory] = useState<string[]>([]);
|
const [cmdHistory, setCmdHistory] = useState<string[]>([]);
|
||||||
const [usrInputTmp, setUsrInputTmp] = useState<string>("");
|
const [usrInputTmp, setUsrInputTmp] = useState<string>("");
|
||||||
const [cmdIf, setCmdIf] = useState<CommandInterface>(new CommandInterface(modalManipulation, {projects: [], diaries: []}));
|
const {cmdContext: cmdIf} = useCommands();
|
||||||
const { data: projects, error: projectsError } = useSWR("/api/projects?swr=1", fetchProjects);
|
const { modalFunctions } = useModalFunctions();
|
||||||
|
|
||||||
const setInput = (inputRef: HTMLInputElement, input: string) => {
|
const setInput = (inputRef: HTMLInputElement, input: string) => {
|
||||||
const nativeSetter = Object.getOwnPropertyDescriptor(window.HTMLInputElement.prototype, "value")?.set;
|
const nativeSetter = Object.getOwnPropertyDescriptor(window.HTMLInputElement.prototype, "value")?.set;
|
||||||
@ -38,8 +28,8 @@ const REPLInput: NextPage<REPLInputParams> = ({historyCallback, historyClear, in
|
|||||||
nativeSetter.call(inputRef, input);
|
nativeSetter.call(inputRef, input);
|
||||||
//if(typed.current) typed.current.innerHTML = input;
|
//if(typed.current) typed.current.innerHTML = input;
|
||||||
//if(completion.current) completion.current.innerHTML = "";
|
//if(completion.current) completion.current.innerHTML = "";
|
||||||
const event = new Event("input", { bubbles: true });
|
inputRef.dispatchEvent(new Event("input", { bubbles: true }));
|
||||||
inputRef.dispatchEvent(event);
|
inputRef.dispatchEvent(new Event("change", { bubbles: true }));
|
||||||
};
|
};
|
||||||
|
|
||||||
const clearInput = (inputRef: HTMLInputElement) => {
|
const clearInput = (inputRef: HTMLInputElement) => {
|
||||||
@ -83,74 +73,68 @@ const REPLInput: NextPage<REPLInputParams> = ({historyCallback, historyClear, in
|
|||||||
return false;
|
return false;
|
||||||
} else setJustTabbed(0);
|
} else setJustTabbed(0);
|
||||||
|
|
||||||
if (e.key === "Enter") {
|
switch (true) {
|
||||||
e.preventDefault();
|
case e.key === "Enter": {
|
||||||
const command = (e.target as HTMLInputElement).value.trim();
|
e.preventDefault();
|
||||||
if (cmdHistory.at(-1) !== command) setCmdHistory(cmdHistory.concat([command]).splice(0, 100));
|
const command = (e.target as HTMLInputElement).value.trim();
|
||||||
clearInput(input);
|
if (cmdHistory.at(-1) !== command) setCmdHistory(cmdHistory.concat([command]).splice(0, 100));
|
||||||
setInCmdHistory(-1);
|
clearInput(input);
|
||||||
setCurrentCmd([]);
|
setInCmdHistory(-1);
|
||||||
if (command === "clear") {
|
setCurrentCmd([]);
|
||||||
|
if (command === "clear") {
|
||||||
|
historyClear();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
const result = cmdIf.executeCommand(command);
|
||||||
|
historyCallback(result);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
case e.key === "d" && e.ctrlKey: {
|
||||||
|
e.preventDefault();
|
||||||
|
const result = cmdIf.executeCommand("exit");
|
||||||
|
clearInput(input);
|
||||||
|
historyCallback(result);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
case e.key === "l" && e.ctrlKey: {
|
||||||
|
e.preventDefault();
|
||||||
|
clearInput(input);
|
||||||
historyClear();
|
historyClear();
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
const result = cmdIf.executeCommand(command);
|
case (e.key === "c" || e.key === "u") && e.ctrlKey: {
|
||||||
historyCallback(result);
|
e.preventDefault();
|
||||||
return false;
|
clearInput(input);
|
||||||
}
|
return false;
|
||||||
|
|
||||||
if (e.key === "d" && e.ctrlKey) {
|
|
||||||
e.preventDefault();
|
|
||||||
const result = cmdIf.executeCommand("exit");
|
|
||||||
clearInput(input);
|
|
||||||
historyCallback(result);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (e.key === "l" && e.ctrlKey) {
|
|
||||||
e.preventDefault();
|
|
||||||
clearInput(input);
|
|
||||||
historyClear();
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ((e.key === "c" || e.key === "u") && e.ctrlKey) {
|
|
||||||
e.preventDefault();
|
|
||||||
clearInput(input);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (e.key === "ArrowUp") {
|
|
||||||
e.preventDefault();
|
|
||||||
const idx = inCmdHistory + 1;
|
|
||||||
if (idx < cmdHistory.length) {
|
|
||||||
if (inCmdHistory === -1) setUsrInputTmp(input.value);
|
|
||||||
|
|
||||||
const cmd = cmdHistory[cmdHistory.length - idx - 1];
|
|
||||||
setInput(input, cmd);
|
|
||||||
setInCmdHistory(idx);
|
|
||||||
}
|
}
|
||||||
}
|
case e.key === "ArrowUp": {
|
||||||
|
e.preventDefault();
|
||||||
|
const idx = inCmdHistory + 1;
|
||||||
|
if (idx < cmdHistory.length) {
|
||||||
|
if (inCmdHistory === -1) setUsrInputTmp(input.value);
|
||||||
|
|
||||||
if (e.key === "ArrowDown") {
|
const cmd = cmdHistory[cmdHistory.length - idx - 1];
|
||||||
e.preventDefault();
|
setInput(input, cmd);
|
||||||
const idx = inCmdHistory - 1;
|
setInCmdHistory(idx);
|
||||||
if (0 <= idx) {
|
}
|
||||||
const cmd = cmdHistory[cmdHistory.length - idx - 1];
|
return false;
|
||||||
setInput(input, cmd);
|
}
|
||||||
setInCmdHistory(idx);
|
case e.key === "ArrowDown": {
|
||||||
} else if (idx === -1) {
|
e.preventDefault();
|
||||||
setInput(input, usrInputTmp);
|
const idx = inCmdHistory - 1;
|
||||||
setInCmdHistory(-1);
|
if (0 <= idx) {
|
||||||
|
const cmd = cmdHistory[cmdHistory.length - idx - 1];
|
||||||
|
setInput(input, cmd);
|
||||||
|
setInCmdHistory(idx);
|
||||||
|
} else if (idx === -1) {
|
||||||
|
setInput(input, usrInputTmp);
|
||||||
|
setInCmdHistory(-1);
|
||||||
|
}
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (!projectsError && projects) setCmdIf(new CommandInterface(modalManipulation, projects));
|
|
||||||
// In any other case we just don't create another interface.
|
|
||||||
}, [projects, projectsError, modalManipulation]);
|
|
||||||
|
|
||||||
return <div className={styles.wrapperwrapper}>
|
return <div className={styles.wrapperwrapper}>
|
||||||
<span className={styles.inputstart}>$ </span>
|
<span className={styles.inputstart}>$ </span>
|
||||||
<div className={styles.wrapper}>
|
<div className={styles.wrapper}>
|
||||||
|
@ -6,14 +6,9 @@ import type { NextPage } from "next";
|
|||||||
|
|
||||||
interface IREPLProps {
|
interface IREPLProps {
|
||||||
inputRef: MutableRefObject<HTMLInputElement|null>;
|
inputRef: MutableRefObject<HTMLInputElement|null>;
|
||||||
modalManipulation: {
|
|
||||||
setModalVisible: CallableFunction;
|
|
||||||
setModalProject: CallableFunction;
|
|
||||||
setModalProjectType: CallableFunction;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const REPL: NextPage<IREPLProps> = ({ inputRef, modalManipulation }) => {
|
const REPL: NextPage<IREPLProps> = ({ inputRef }) => {
|
||||||
const date = new Date();
|
const date = new Date();
|
||||||
const [history, manipulateHistory] = useState<string[]>([`cer0 0S - ${date.toLocaleDateString()}`]);
|
const [history, manipulateHistory] = useState<string[]>([`cer0 0S - ${date.toLocaleDateString()}`]);
|
||||||
const containerRef = useRef<HTMLDivElement>(null);
|
const containerRef = useRef<HTMLDivElement>(null);
|
||||||
@ -30,7 +25,7 @@ const REPL: NextPage<IREPLProps> = ({ inputRef, modalManipulation }) => {
|
|||||||
|
|
||||||
return (<div className={styles.container} ref={containerRef}>
|
return (<div className={styles.container} ref={containerRef}>
|
||||||
<REPLHistory history={history} inputRef={inputRef} />
|
<REPLHistory history={history} inputRef={inputRef} />
|
||||||
<REPLInput historyCallback={onCommandExecuted} historyClear={onClearHistory} inputRef={inputRef} modalManipulation={modalManipulation} />
|
<REPLInput historyCallback={onCommandExecuted} historyClear={onClearHistory} inputRef={inputRef} />
|
||||||
<div style={{flexGrow: 2}} onClick={focusInput}></div>
|
<div style={{flexGrow: 2}} onClick={focusInput}></div>
|
||||||
</div>);
|
</div>);
|
||||||
};
|
};
|
||||||
|
18
components/contexts/CommandInterface.tsx
Normal file
18
components/contexts/CommandInterface.tsx
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
import { createContext, useContext } from "react";
|
||||||
|
import type { PropsWithChildren } from "react";
|
||||||
|
import { CommandInterface } from "../../lib/commands";
|
||||||
|
import type { Diary, Project, ContentList } from "../../lib/content/types";
|
||||||
|
|
||||||
|
interface CommandInterfaceCallbacks {
|
||||||
|
setModalVisible?: (visible: boolean) => void;
|
||||||
|
setModalContent?: (content: Project | Diary, selectedPage?: number) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
const commandInterface = new CommandInterface();
|
||||||
|
const CommandContext = createContext(commandInterface);
|
||||||
|
const setCommandCallbacks = (callbacks: CommandInterfaceCallbacks) => commandInterface.callbacks = {...commandInterface.callbacks, ...callbacks};
|
||||||
|
const setContents = (content: ContentList) => commandInterface.content = content;
|
||||||
|
const useCommands = () => ({cmdContext: useContext(CommandContext), updateCallbacks: setCommandCallbacks, setContents});
|
||||||
|
const CommandsProvider = (props: PropsWithChildren<{}>) => <CommandContext.Provider value={commandInterface} {...props} />;
|
||||||
|
|
||||||
|
export { CommandsProvider, useCommands };
|
17
components/contexts/ModalFunctions.tsx
Normal file
17
components/contexts/ModalFunctions.tsx
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
import { createContext, useContext } from "react";
|
||||||
|
import type { PropsWithChildren } from "react";
|
||||||
|
import type { Project, Diary } from "../../lib/content/types";
|
||||||
|
|
||||||
|
interface ModalFunctions {
|
||||||
|
setVisible?: CallableFunction;
|
||||||
|
setContent?: (content: Project| Diary) => void;
|
||||||
|
setHtml?: (html: string) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
const modalFunctions: ModalFunctions = {};
|
||||||
|
const ModalContext = createContext(modalFunctions);
|
||||||
|
const updateCallbacks = (callbacks: ModalFunctions) => Object.assign(modalFunctions, callbacks);
|
||||||
|
const useModalFunctions = () => ({modalFunctions: useContext(ModalContext), updateCallbacks});
|
||||||
|
const ModalFunctionProvider = (props: PropsWithChildren<{}>) => <ModalContext.Provider value={modalFunctions} {...props} />;
|
||||||
|
|
||||||
|
export { ModalFunctionProvider, useModalFunctions };
|
@ -1,3 +1,4 @@
|
|||||||
|
import type { Diary, Project } from "../content/types";
|
||||||
import type { Command, Flag } from "./types";
|
import type { Command, Flag } from "./types";
|
||||||
|
|
||||||
function getCommandByName(name: string): Command|undefined {
|
function getCommandByName(name: string): Command|undefined {
|
||||||
@ -146,9 +147,16 @@ const project: Command = {
|
|||||||
execute: (flags, args, _raw, cmdIf) => {
|
execute: (flags, args, _raw, cmdIf) => {
|
||||||
if (project.flags && checkFlagInclude(flags, project.flags.list)) {
|
if (project.flags && checkFlagInclude(flags, project.flags.list)) {
|
||||||
const result = ["Found the following projects:"];
|
const result = ["Found the following projects:"];
|
||||||
cmdIf.projects.projects.forEach(project => result.push(`\t${project.name}\t${project.short_desc}`));
|
|
||||||
|
const projects = cmdIf.content.filter(p => p.type === "project");
|
||||||
|
if(projects.length === 0) result.push("\tNo projects found.");
|
||||||
|
else projects.forEach(project => result.push(`\t${project.name}\t${project.short_desc}`));
|
||||||
|
|
||||||
result.push("And the following diaries:");
|
result.push("And the following diaries:");
|
||||||
cmdIf.projects.diaries.forEach(diary => result.push(`\t${diary.name}\t${diary.short_desc}`));
|
const diaries = cmdIf.content.filter(p => p.type === "diary");
|
||||||
|
if(diaries.length === 0) result.push("\tNo diaries found.");
|
||||||
|
else diaries.forEach(diary => result.push(`\t${diary.name}\t${diary.short_desc}`));
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -156,8 +164,7 @@ const project: Command = {
|
|||||||
|
|
||||||
if (args[0] === "this") args[0] = "homepage";
|
if (args[0] === "this") args[0] = "homepage";
|
||||||
|
|
||||||
let [pjt, ptype] = [cmdIf.projects.projects.find(p => p.name === args[0]), "project"];
|
let [pjt] = [cmdIf.content.find(p => p.name === args[0]) as Project|Diary|undefined];
|
||||||
if (!pjt) [pjt, ptype] = [cmdIf.projects.diaries.find(p => p.name === args[0]), "diary"];
|
|
||||||
if (!pjt) return [
|
if (!pjt) return [
|
||||||
`Cannot find project ${args[0]}!`,
|
`Cannot find project ${args[0]}!`,
|
||||||
"You can see available projects using 'project -l'."
|
"You can see available projects using 'project -l'."
|
||||||
@ -173,9 +180,8 @@ const project: Command = {
|
|||||||
}
|
}
|
||||||
if (project.flags && checkFlagInclude(flags, project.flags.minimal)) return pjt.desc;
|
if (project.flags && checkFlagInclude(flags, project.flags.minimal)) return pjt.desc;
|
||||||
|
|
||||||
cmdIf.callbacks.setModalProjectType(ptype);
|
if (cmdIf.callbacks?.setModalContent) cmdIf.callbacks.setModalContent(pjt);
|
||||||
cmdIf.callbacks.setModalProject(pjt);
|
if (cmdIf.callbacks?.setModalVisible) cmdIf.callbacks.setModalVisible(true);
|
||||||
cmdIf.callbacks.setModalVisible(true);
|
|
||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -203,4 +209,4 @@ const clear: Command = {
|
|||||||
execute: () => []
|
execute: () => []
|
||||||
};
|
};
|
||||||
|
|
||||||
export const commandList = [about, help, man, project, exitCmd, clear];
|
export const commandList = [about, help, man, project, exitCmd, clear].sort((a, b) => a.name.localeCompare(b.name));
|
@ -1,19 +1,18 @@
|
|||||||
import type { ProjectList } from "../projects/types";
|
import type { ContentList } from "../content/types";
|
||||||
import { printSyntax, commandList } from "./definitions";
|
import { printSyntax, commandList } from "./definitions";
|
||||||
|
|
||||||
interface CommandInterfaceCallbacks {
|
interface CommandInterfaceCallbacks {
|
||||||
setModalVisible: CallableFunction;
|
setModalVisible?: CallableFunction;
|
||||||
setModalProject: CallableFunction;
|
setModalContent?: CallableFunction;
|
||||||
setModalProjectType: CallableFunction;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export class CommandInterface {
|
export class CommandInterface {
|
||||||
callbacks: CommandInterfaceCallbacks;
|
callbacks?: CommandInterfaceCallbacks;
|
||||||
projects: ProjectList;
|
content: ContentList = [];
|
||||||
|
|
||||||
constructor(callbacks: CommandInterfaceCallbacks, projects: ProjectList) {
|
constructor(callbacks?: CommandInterfaceCallbacks, content?: ContentList) {
|
||||||
this.callbacks = callbacks;
|
this.callbacks = callbacks;
|
||||||
this.projects = projects;
|
this.content = content || [];
|
||||||
}
|
}
|
||||||
|
|
||||||
static commandCompletion(input: string): string[] {
|
static commandCompletion(input: string): string[] {
|
||||||
|
44
lib/content/generate.ts
Normal file
44
lib/content/generate.ts
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
import type { Project, Diary } from "./types";
|
||||||
|
import asciidoctor from "asciidoctor";
|
||||||
|
|
||||||
|
export const projectEmpty = "<div>Kein Projekt ausgewählt.</div>";
|
||||||
|
const projectNotFoundHtml = `<div class="${"error"}">Sorry! There is no data for this project. Please check back later to see if that changed!</div>`;
|
||||||
|
const projectServerErrorHtml = `<div class="${"error"}">Sorry! A server error happend when the project data was fetched!</div>`;
|
||||||
|
|
||||||
|
const ad = asciidoctor();
|
||||||
|
|
||||||
|
export async function generateContent(content: Project|Diary, selectedPage?: number): Promise<string> {
|
||||||
|
if(!content) return projectEmpty;
|
||||||
|
switch (content.type) {
|
||||||
|
case "project": return await generateProjectHTML(content);
|
||||||
|
case "diary": return await generateDiaryHTML(content, selectedPage);
|
||||||
|
default: return projectNotFoundHtml;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function generateProjectHTML(project: Project): Promise<string> {
|
||||||
|
const resp = await fetch(`/content/projects/${project.name}.adoc`);
|
||||||
|
if (resp.status !== 200) return projectServerErrorHtml;
|
||||||
|
const adDoc = ad.load(await resp.text(), { attributes: { showtitle: true } });
|
||||||
|
return `${adDoc.convert(adDoc).toString()}
|
||||||
|
<hr>
|
||||||
|
<div id="footer">
|
||||||
|
<div id="footer-text">
|
||||||
|
Last updated: ${adDoc.getAttribute("docdatetime")} | <a href="https://git.c0ntroller.de/c0ntroller/frontpage-content/src/branch/senpai/projects/${project.name}.adoc">Document source</a>
|
||||||
|
</div>
|
||||||
|
</div>`;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function generateDiaryHTML(diary: Diary, selectedPage?: number): Promise<string> {
|
||||||
|
const resp = selectedPage === undefined ? await fetch(`/content/diaries/${diary.name}.adoc`) : await fetch(`/content/diaries/${diary.name}/${diary.entries[selectedPage].filename}.adoc`);
|
||||||
|
if (resp.status !== 200) return projectServerErrorHtml;
|
||||||
|
const adDoc = ad.load(await resp.text(), { attributes: { showtitle: true } });
|
||||||
|
const gitfile = selectedPage === undefined ? `${diary.name}.adoc` : `${diary.name}/${diary.entries[selectedPage].filename}.adoc`;
|
||||||
|
return `${adDoc.convert(adDoc).toString()}
|
||||||
|
<hr>
|
||||||
|
<div id="footer">
|
||||||
|
<div id="footer-text">
|
||||||
|
Last updated: ${adDoc.getAttribute("docdatetime")} | <a href="https://git.c0ntroller.de/c0ntroller/frontpage-content/src/branch/senpai/diaries/${gitfile}">Document source</a>
|
||||||
|
</div>
|
||||||
|
</div>`;
|
||||||
|
}
|
@ -1,9 +1,9 @@
|
|||||||
export interface ProjectList {
|
export type ContentList = (Project|Diary)[];
|
||||||
projects: Project[];
|
|
||||||
diaries: Diary[];
|
export type ContentType = "project" | "diary";
|
||||||
}
|
|
||||||
|
|
||||||
export interface Project {
|
export interface Project {
|
||||||
|
type: "project";
|
||||||
name: string;
|
name: string;
|
||||||
desc: string[];
|
desc: string[];
|
||||||
short_desc: string;
|
short_desc: string;
|
||||||
@ -12,6 +12,7 @@ export interface Project {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export interface Diary {
|
export interface Diary {
|
||||||
|
type: "diary";
|
||||||
name: string;
|
name: string;
|
||||||
desc: string[];
|
desc: string[];
|
||||||
short_desc: string;
|
short_desc: string;
|
||||||
@ -19,6 +20,6 @@ export interface Diary {
|
|||||||
repo?: string;
|
repo?: string;
|
||||||
entries: {
|
entries: {
|
||||||
title: string;
|
title: string;
|
||||||
path: string;
|
filename: string;
|
||||||
}[];
|
}[];
|
||||||
}
|
}
|
@ -2,11 +2,13 @@ import type { AppProps } from "next/app";
|
|||||||
import Head from "next/head";
|
import Head from "next/head";
|
||||||
import "../styles/globals.css";
|
import "../styles/globals.css";
|
||||||
import "../styles/customAsciidoc.scss";
|
import "../styles/customAsciidoc.scss";
|
||||||
|
import { CommandsProvider } from "../components/contexts/CommandInterface";
|
||||||
|
import { ModalFunctionProvider } from "../components/contexts/ModalFunctions";
|
||||||
|
|
||||||
function MyApp({ Component, pageProps }: AppProps) {
|
function MyApp({ Component, pageProps }: AppProps) {
|
||||||
return <>
|
return <>
|
||||||
<Head>
|
<Head>
|
||||||
<meta charSet="utf-8"/>
|
<meta charSet="utf-8" />
|
||||||
<meta name="description" content="This is the homepage of C0ntroller." />
|
<meta name="description" content="This is the homepage of C0ntroller." />
|
||||||
<meta name="keyword" content="private, homepage, cli, hacker, terminal, javascript, js, nextjs, react, responsive" />
|
<meta name="keyword" content="private, homepage, cli, hacker, terminal, javascript, js, nextjs, react, responsive" />
|
||||||
<meta name="author" content="C0ntroller" />
|
<meta name="author" content="C0ntroller" />
|
||||||
@ -22,8 +24,8 @@ function MyApp({ Component, pageProps }: AppProps) {
|
|||||||
<link rel="apple-touch-icon" sizes="144x144" href="/apple-icon-144x144.png" />
|
<link rel="apple-touch-icon" sizes="144x144" href="/apple-icon-144x144.png" />
|
||||||
<link rel="apple-touch-icon" sizes="152x152" href="/apple-icon-152x152.png" />
|
<link rel="apple-touch-icon" sizes="152x152" href="/apple-icon-152x152.png" />
|
||||||
<link rel="apple-touch-icon" sizes="180x180" href="/apple-icon-180x180.png" />
|
<link rel="apple-touch-icon" sizes="180x180" href="/apple-icon-180x180.png" />
|
||||||
<link rel="icon" type="image/png" sizes="192x192" href="/android-chrome-192x192.png" />
|
<link rel="icon" type="image/png" sizes="192x192" href="/android-chrome-192x192.png" />
|
||||||
<link rel="icon" type="image/png" sizes="512x512" href="/android-chrome-512x512.png" />
|
<link rel="icon" type="image/png" sizes="512x512" href="/android-chrome-512x512.png" />
|
||||||
<link rel="icon" type="image/png" sizes="32x32" href="/favicon-32x32.png" />
|
<link rel="icon" type="image/png" sizes="32x32" href="/favicon-32x32.png" />
|
||||||
<link rel="icon" type="image/png" sizes="96x96" href="/favicon-96x96.png" />
|
<link rel="icon" type="image/png" sizes="96x96" href="/favicon-96x96.png" />
|
||||||
<link rel="icon" type="image/png" sizes="16x16" href="/favicon-16x16.png" />
|
<link rel="icon" type="image/png" sizes="16x16" href="/favicon-16x16.png" />
|
||||||
@ -32,7 +34,11 @@ function MyApp({ Component, pageProps }: AppProps) {
|
|||||||
<meta name="msapplication-TileImage" content="/mstile-310x310.png" />
|
<meta name="msapplication-TileImage" content="/mstile-310x310.png" />
|
||||||
<meta name="theme-color" content="#444444" />
|
<meta name="theme-color" content="#444444" />
|
||||||
</Head>
|
</Head>
|
||||||
<Component {...pageProps} />
|
<CommandsProvider>
|
||||||
|
<ModalFunctionProvider>
|
||||||
|
<Component {...pageProps} />
|
||||||
|
</ModalFunctionProvider>
|
||||||
|
</CommandsProvider>
|
||||||
</>;
|
</>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,32 +1,47 @@
|
|||||||
import type { NextPage } from "next";
|
import type { NextPage } from "next";
|
||||||
import Head from "next/head";
|
import Head from "next/head";
|
||||||
import { GithubLogo, InstagramLogo, DiscordLogo, GameController } from "phosphor-react";
|
import { GithubLogo, InstagramLogo, DiscordLogo, GameController } from "phosphor-react";
|
||||||
import { useRef, useState } from "react";
|
import { useEffect, useRef,useCallback } from "react";
|
||||||
|
import { useCommands } from "../components/contexts/CommandInterface";
|
||||||
|
import { useModalFunctions } from "../components/contexts/ModalFunctions";
|
||||||
import ProjectModal from "../components/ProjectModal";
|
import ProjectModal from "../components/ProjectModal";
|
||||||
import REPL from "../components/REPL";
|
import REPL from "../components/REPL";
|
||||||
import type { Project, Diary } from "../lib/projects/types";
|
|
||||||
import styles from "../styles/Home.module.css";
|
import styles from "../styles/Home.module.css";
|
||||||
|
|
||||||
const Home: NextPage = () => {
|
const Home: NextPage = () => {
|
||||||
const inputRef = useRef<HTMLInputElement>(null);
|
const inputRef = useRef<HTMLInputElement>(null);
|
||||||
const [modalVisible, setModalVisible] = useState<boolean>(false);
|
const { modalFunctions } = useModalFunctions();
|
||||||
const [modalProject, setModalProject] = useState<Project|Diary|undefined>(undefined);
|
const { setContents } = useCommands();
|
||||||
const [modalProjectType, setModalProjectType] = useState<"project"|"diary">("project");
|
|
||||||
|
const updateProjects = useCallback(async () => {
|
||||||
|
try {
|
||||||
|
const res = await fetch("/content/list.json");
|
||||||
|
const projects = await res.json();
|
||||||
|
setContents(projects);
|
||||||
|
} catch {}
|
||||||
|
}, [setContents]);
|
||||||
|
|
||||||
|
updateProjects();
|
||||||
|
|
||||||
const focusInput = () => { if (inputRef.current) inputRef.current.focus(); };
|
const focusInput = () => { if (inputRef.current) inputRef.current.focus(); };
|
||||||
|
|
||||||
const hideModalOnEsc = (e: React.KeyboardEvent) => {
|
const hideModalOnEsc = (e: React.KeyboardEvent) => {
|
||||||
if (e.key === "Escape") {
|
if (e.key === "Escape") {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
setModalVisible(false);
|
if(modalFunctions.setVisible) modalFunctions.setVisible(false);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const interval = setInterval(updateProjects, 30 * 1000);
|
||||||
|
return () => clearInterval(interval);
|
||||||
|
}, [updateProjects]);
|
||||||
|
|
||||||
return (<main onKeyDown={hideModalOnEsc} tabIndex={-1}>
|
return (<main onKeyDown={hideModalOnEsc} tabIndex={-1}>
|
||||||
<Head>
|
<Head>
|
||||||
<title>c0ntroller.de</title>
|
<title>c0ntroller.de</title>
|
||||||
</Head>
|
</Head>
|
||||||
<ProjectModal visible={modalVisible} project={modalProject} projectType={modalProjectType} setVisible={setModalVisible}/>
|
<ProjectModal />
|
||||||
<div className={styles.container}>
|
<div className={styles.container}>
|
||||||
<div className={styles.header}>
|
<div className={styles.header}>
|
||||||
<span className={styles.spacer} onClick={focusInput}> </span>
|
<span className={styles.spacer} onClick={focusInput}> </span>
|
||||||
@ -49,7 +64,7 @@ const Home: NextPage = () => {
|
|||||||
</span>
|
</span>
|
||||||
</a><span className={styles.spacer} onClick={focusInput}> </span>
|
</a><span className={styles.spacer} onClick={focusInput}> </span>
|
||||||
</div>
|
</div>
|
||||||
<REPL inputRef={inputRef} modalManipulation={{setModalVisible, setModalProject, setModalProjectType}}/>
|
<REPL inputRef={inputRef} />
|
||||||
</div>
|
</div>
|
||||||
</main>);
|
</main>);
|
||||||
};
|
};
|
||||||
|
Loading…
Reference in New Issue
Block a user