-
-
+
+
+ {isLoading
+ ?
+ : }
- {isLoading ?
Loading...
: (
-
- {data.map((d) => (
-
{JSON.stringify(d.document, null, 2)}
- ))}
-
- )}
+ {data?.hits?.length && !isLoading
+ ?
+ : isLoading
+ ?
+ : (
+
+
+ No Results
+
+ )}
);
};
diff --git a/lib/crud.ts b/lib/crud.ts
index 4f998b7..fbc936b 100644
--- a/lib/crud.ts
+++ b/lib/crud.ts
@@ -1,6 +1,5 @@
import {
createDocument,
- Document,
getDocument,
getDocuments,
transformDocument,
@@ -20,7 +19,6 @@ export function createCrud
(
async function read(id: string) {
const path = pathFromId(id);
const content = await getDocument(path);
-
return parse(content, id);
}
function create(id: string, content: string | ArrayBuffer) {
diff --git a/lib/highlight.ts b/lib/highlight.ts
deleted file mode 100644
index 6c2bf60..0000000
--- a/lib/highlight.ts
+++ /dev/null
@@ -1,79 +0,0 @@
-import { refractor } from "https://esm.sh/refractor@4.8.1";
-import { visit } from "https://esm.sh/unist-util-visit@5.0.0";
-import { toString } from "https://esm.sh/hast-util-to-string@2.0.0";
-
-import jsx from "https://esm.sh/refractor/lang/jsx";
-import javascript from "https://esm.sh/refractor/lang/javascript";
-import css from "https://esm.sh/refractor/lang/css";
-import cssExtras from "https://esm.sh/refractor/lang/css-extras";
-import jsExtras from "https://esm.sh/refractor/lang/js-extras";
-import sql from "https://esm.sh/refractor/lang/sql";
-import typescript from "https://esm.sh/refractor/lang/typescript";
-import swift from "https://esm.sh/refractor/lang/swift";
-import objectivec from "https://esm.sh/refractor/lang/objectivec";
-import markdown from "https://esm.sh/refractor/lang/markdown";
-import json from "https://esm.sh/refractor/lang/json";
-
-refractor.register(jsx);
-refractor.register(json);
-refractor.register(typescript);
-refractor.register(javascript);
-refractor.register(css);
-refractor.register(cssExtras);
-refractor.register(jsExtras);
-refractor.register(sql);
-refractor.register(swift);
-refractor.register(objectivec);
-refractor.register(markdown);
-
-refractor.alias({ jsx: ["js"] });
-refractor.alias({typescript:["ts"]})
-
-const getLanguage = (node) => {
- const className = node.properties.className || [];
-
- for (const classListItem of className) {
- if (classListItem.slice(0, 9) === "language-") {
- return classListItem.slice(9).toLowerCase();
- }
- }
-
- return null;
-};
-
-const rehypePrism = (options) => {
- options = options || {};
-
- return (tree) => {
- visit(tree, "element", visitor);
- };
-
- function visitor(node, index, parent) {
- if (!parent || parent.tagName !== "pre" || node.tagName !== "code") {
- return;
- }
-
- const lang = getLanguage(node);
-
- if (lang === null) {
- return;
- }
-
- let result;
- try {
- parent.properties.className = (parent.properties.className || []).concat(
- "language-" + lang,
- );
- result = refractor.highlight(toString(node), lang);
- } catch (err) {
- if (options.ignoreMissing && /Unknown language/.test(err.message)) {
- return;
- }
- throw err;
- }
-
- node.children = result;
- }
-};
-
-export default rehypePrism;
diff --git a/lib/hooks/useDebouncedCallback.ts b/lib/hooks/useDebouncedCallback.ts
index 7a0c05f..9a223fb 100644
--- a/lib/hooks/useDebouncedCallback.ts
+++ b/lib/hooks/useDebouncedCallback.ts
@@ -1,24 +1,286 @@
-// useDebouncedCallback.tsx
-import { useEffect, useState } from "preact/hooks";
+import { useEffect, useMemo, useRef } from "preact/hooks";
-const useDebouncedCallback = (
- callback: (...args: any[]) => void,
- delay: number,
-) => {
- const [debouncedCallback, setDebouncedCallback] = useState(() => callback);
+export interface CallOptions {
+ /**
+ * Controls if the function should be invoked on the leading edge of the timeout.
+ */
+ leading?: boolean;
+ /**
+ * Controls if the function should be invoked on the trailing edge of the timeout.
+ */
+ trailing?: boolean;
+}
+
+export interface Options extends CallOptions {
+ /**
+ * The maximum time the given function is allowed to be delayed before it's invoked.
+ */
+ maxWait?: number;
+}
+
+export interface ControlFunctions {
+ /**
+ * Cancel pending function invocations
+ */
+ cancel: () => void;
+ /**
+ * Immediately invoke pending function invocations
+ */
+ flush: () => void;
+ /**
+ * Returns `true` if there are any pending function invocations
+ */
+ isPending: () => boolean;
+}
+
+/**
+ * Subsequent calls to the debounced function `debounced.callback` return the result of the last func invocation.
+ * Note, that if there are no previous invocations it's mean you will get undefined. You should check it in your code properly.
+ */
+export interface DebouncedState ReturnType>
+ extends ControlFunctions {
+ (...args: Parameters): ReturnType | undefined;
+}
+
+/**
+ * Creates a debounced function that delays invoking `func` until after `wait`
+ * milliseconds have elapsed since the last time the debounced function was
+ * invoked, or until the next browser frame is drawn.
+ *
+ * The debounced function comes with a `cancel` method to cancel delayed `func`
+ * invocations and a `flush` method to immediately invoke them.
+ *
+ * Provide `options` to indicate whether `func` should be invoked on the leading
+ * and/or trailing edge of the `wait` timeout. The `func` is invoked with the
+ * last arguments provided to the debounced function.
+ *
+ * Subsequent calls to the debounced function return the result of the last
+ * `func` invocation.
+ *
+ * **Note:** If `leading` and `trailing` options are `true`, `func` is
+ * invoked on the trailing edge of the timeout only if the debounced function
+ * is invoked more than once during the `wait` timeout.
+ *
+ * If `wait` is `0` and `leading` is `false`, `func` invocation is deferred
+ * until the next tick, similar to `setTimeout` with a timeout of `0`.
+ *
+ * If `wait` is omitted in an environment with `requestAnimationFrame`, `func`
+ * invocation will be deferred until the next frame is drawn (typically about
+ * 16ms).
+ *
+ * See [David Corbacho's article](https://css-tricks.com/debouncing-throttling-explained-examples/)
+ * for details over the differences between `debounce` and `throttle`.
+ *
+ * @category Function
+ * @param {Function} func The function to debounce.
+ * @param {number} [wait=0]
+ * The number of milliseconds to delay; if omitted, `requestAnimationFrame` is
+ * used (if available, otherwise it will be setTimeout(...,0)).
+ * @param {Object} [options={}] The options object.
+ * Controls if `func` should be invoked on the leading edge of the timeout.
+ * @param {boolean} [options.leading=false]
+ * The maximum time `func` is allowed to be delayed before it's invoked.
+ * @param {number} [options.maxWait]
+ * Controls if `func` should be invoked the trailing edge of the timeout.
+ * @param {boolean} [options.trailing=true]
+ * @returns {Function} Returns the new debounced function.
+ * @example
+ *
+ * // Avoid costly calculations while the window size is in flux.
+ * const resizeHandler = useDebouncedCallback(calculateLayout, 150);
+ * window.addEventListener('resize', resizeHandler)
+ *
+ * // Invoke `sendMail` when clicked, debouncing subsequent calls.
+ * const clickHandler = useDebouncedCallback(sendMail, 300, {
+ * leading: true,
+ * trailing: false,
+ * })
+ *
+ *
+ * // Ensure `batchLog` is invoked once after 1 second of debounced calls.
+ * const debounced = useDebouncedCallback(batchLog, 250, { 'maxWait': 1000 })
+ * const source = new EventSource('/stream')
+ * source.addEventListener('message', debounced)
+ *
+ * // Cancel the trailing debounced invocation.
+ * window.addEventListener('popstate', debounced.cancel)
+ *
+ * // Check for pending invocations.
+ * const status = debounced.pending() ? "Pending..." : "Ready"
+ */
+export default function useDebouncedCallback<
+ T extends (...args: any) => ReturnType,
+>(
+ func: T,
+ wait?: number,
+ options?: Options,
+): DebouncedState {
+ const lastCallTime = useRef(null);
+ const lastInvokeTime = useRef(0);
+ const timerId = useRef(null);
+ const lastArgs = useRef([]);
+ const lastThis = useRef();
+ const result = useRef>();
+ const funcRef = useRef(func);
+ const mounted = useRef(true);
useEffect(() => {
- const debounceHandler = setTimeout(() => {
- setDebouncedCallback(() => callback);
- }, delay);
+ funcRef.current = func;
+ }, [func]);
+ // Bypass `requestAnimationFrame` by explicitly setting `wait=0`.
+ const useRAF = !wait && wait !== 0 && typeof window !== "undefined";
+
+ if (typeof func !== "function") {
+ throw new TypeError("Expected a function");
+ }
+
+ wait = +wait || 0;
+ options = options || {};
+
+ const leading = !!options.leading;
+ const trailing = "trailing" in options ? !!options.trailing : true; // `true` by default
+ const maxing = "maxWait" in options;
+ const maxWait = maxing ? Math.max(+options.maxWait || 0, wait) : null;
+
+ useEffect(() => {
+ mounted.current = true;
return () => {
- clearTimeout(debounceHandler);
+ mounted.current = false;
};
- }, [callback, delay]);
+ }, []);
- return debouncedCallback;
-};
+ // You may have a question, why we have so many code under the useMemo definition.
+ //
+ // This was made as we want to escape from useCallback hell and
+ // not to initialize a number of functions each time useDebouncedCallback is called.
+ //
+ // It means that we have less garbage for our GC calls which improves performance.
+ // Also, it makes this library smaller.
+ //
+ // And the last reason, that the code without lots of useCallback with deps is easier to read.
+ // You have only one place for that.
+ const debounced = useMemo(() => {
+ const invokeFunc = (time: number) => {
+ const args = lastArgs.current;
+ const thisArg = lastThis.current;
-export default useDebouncedCallback;
+ lastArgs.current = lastThis.current = null;
+ lastInvokeTime.current = time;
+ return (result.current = funcRef.current.apply(thisArg, args));
+ };
+
+ const startTimer = (pendingFunc: () => void, wait: number) => {
+ if (useRAF) cancelAnimationFrame(timerId.current);
+ timerId.current = useRAF
+ ? requestAnimationFrame(pendingFunc)
+ : setTimeout(pendingFunc, wait);
+ };
+
+ const shouldInvoke = (time: number) => {
+ if (!mounted.current) return false;
+
+ const timeSinceLastCall = time - lastCallTime.current;
+ const timeSinceLastInvoke = time - lastInvokeTime.current;
+
+ // Either this is the first call, activity has stopped and we're at the
+ // trailing edge, the system time has gone backwards and we're treating
+ // it as the trailing edge, or we've hit the `maxWait` limit.
+ return (
+ !lastCallTime.current ||
+ timeSinceLastCall >= wait ||
+ timeSinceLastCall < 0 ||
+ (maxing && timeSinceLastInvoke >= maxWait)
+ );
+ };
+
+ const trailingEdge = (time: number) => {
+ timerId.current = null;
+
+ // Only invoke if we have `lastArgs` which means `func` has been
+ // debounced at least once.
+ if (trailing && lastArgs.current) {
+ return invokeFunc(time);
+ }
+ lastArgs.current = lastThis.current = null;
+ return result.current;
+ };
+
+ const timerExpired = () => {
+ const time = Date.now();
+ if (shouldInvoke(time)) {
+ return trailingEdge(time);
+ }
+ // https://github.com/xnimorz/use-debounce/issues/97
+ if (!mounted.current) {
+ return;
+ }
+ // Remaining wait calculation
+ const timeSinceLastCall = time - lastCallTime.current;
+ const timeSinceLastInvoke = time - lastInvokeTime.current;
+ const timeWaiting = wait - timeSinceLastCall;
+ const remainingWait = maxing
+ ? Math.min(timeWaiting, maxWait - timeSinceLastInvoke)
+ : timeWaiting;
+
+ // Restart the timer
+ startTimer(timerExpired, remainingWait);
+ };
+
+ const func: DebouncedState = (...args: Parameters): ReturnType => {
+ const time = Date.now();
+ const isInvoking = shouldInvoke(time);
+
+ lastArgs.current = args;
+ lastThis.current = this;
+ lastCallTime.current = time;
+
+ if (isInvoking) {
+ if (!timerId.current && mounted.current) {
+ // Reset any `maxWait` timer.
+ lastInvokeTime.current = lastCallTime.current;
+ // Start the timer for the trailing edge.
+ startTimer(timerExpired, wait);
+ // Invoke the leading edge.
+ return leading ? invokeFunc(lastCallTime.current) : result.current;
+ }
+ if (maxing) {
+ // Handle invocations in a tight loop.
+ startTimer(timerExpired, wait);
+ return invokeFunc(lastCallTime.current);
+ }
+ }
+ if (!timerId.current) {
+ startTimer(timerExpired, wait);
+ }
+ return result.current;
+ };
+
+ func.cancel = () => {
+ if (timerId.current) {
+ useRAF
+ ? cancelAnimationFrame(timerId.current)
+ : clearTimeout(timerId.current);
+ }
+ lastInvokeTime.current = 0;
+ lastArgs.current =
+ lastCallTime.current =
+ lastThis.current =
+ timerId.current =
+ null;
+ };
+
+ func.isPending = () => {
+ return !!timerId.current;
+ };
+
+ func.flush = () => {
+ return !timerId.current ? result.current : trailingEdge(Date.now());
+ };
+
+ return func;
+ }, [leading, maxing, wait, maxWait, trailing, useRAF]);
+
+ return debounced;
+}
diff --git a/lib/hooks/useTraceProps.ts b/lib/hooks/useTraceProps.ts
new file mode 100644
index 0000000..e76f4e2
--- /dev/null
+++ b/lib/hooks/useTraceProps.ts
@@ -0,0 +1,17 @@
+import { useEffect, useRef } from "preact/hooks";
+
+export function useTraceUpdate(props) {
+ const prev = useRef(props);
+ useEffect(() => {
+ const changedProps = Object.entries(props).reduce((ps, [k, v]) => {
+ if (prev.current[k] !== v) {
+ ps[k] = [prev.current[k], v];
+ }
+ return ps;
+ }, {});
+ if (Object.keys(changedProps).length > 0) {
+ console.log("Changed props:", changedProps);
+ }
+ prev.current = props;
+ });
+}
diff --git a/lib/menus.ts b/lib/menus.ts
deleted file mode 100644
index 83db52b..0000000
--- a/lib/menus.ts
+++ /dev/null
@@ -1,18 +0,0 @@
-export const menu = [
- {
- name: "🏡 Home",
- link: "/",
- },
- {
- name: "🍽️ Recipes",
- link: "/recipes",
- },
- {
- name: "🍿 Movies",
- link: "/movies",
- },
- {
- name: "📝 Articles",
- link: "/articles",
- },
-];
diff --git a/lib/resource/recipes.ts b/lib/resource/recipes.ts
index 62ddfe6..404f2fa 100644
--- a/lib/resource/recipes.ts
+++ b/lib/resource/recipes.ts
@@ -3,10 +3,9 @@ import {
getTextOfChild,
getTextOfRange,
parseDocument,
- renderMarkdown,
} from "@lib/documents.ts";
import { parse } from "yaml";
-import { parseIngredient } from "https://esm.sh/parse-ingredient";
+import { parseIngredient } from "https://esm.sh/parse-ingredient@1.0.1";
import { createCrud } from "@lib/crud.ts";
import { extractHashTags } from "@lib/string.ts";
diff --git a/lib/resources.ts b/lib/resources.ts
new file mode 100644
index 0000000..7e2899f
--- /dev/null
+++ b/lib/resources.ts
@@ -0,0 +1,26 @@
+export const resources = {
+ "home": {
+ emoji: "🏡",
+ name: "Home",
+ link: "/",
+ prefix: "",
+ },
+ "recipe": {
+ emoji: "🍽️",
+ name: "Recipes",
+ link: "/recipes",
+ prefix: "Recipes/",
+ },
+ "movie": {
+ emoji: "🍿",
+ name: "Movies",
+ link: "/movies",
+ prefix: "Media/movies/",
+ },
+ "article": {
+ emoji: "📝",
+ name: "Articles",
+ link: "/articles",
+ prefix: "Media/articles/",
+ },
+} as const;
diff --git a/lib/string.ts b/lib/string.ts
index 8160220..b7bc057 100644
--- a/lib/string.ts
+++ b/lib/string.ts
@@ -1,3 +1,5 @@
+import { resources } from "@lib/resources.ts";
+
export function formatDate(date: Date): string {
const options = { year: "numeric", month: "long", day: "numeric" } as const;
return new Intl.DateTimeFormat("en-US", options).format(date);
@@ -92,3 +94,9 @@ export function getCookie(name: string): string | null {
return decodeURIComponent(cookie.substring(nameLenPlus));
})[0] || null;
}
+
+const resourcePrefixes = Object.values(resources).map((v) => v.prefix).filter(
+ (s) => s.length > 2,
+);
+export const isLocalImage = (src: string) =>
+ resourcePrefixes.some((p) => src.startsWith(p));
diff --git a/lib/types.ts b/lib/types.ts
index 5631f8e..114104a 100644
--- a/lib/types.ts
+++ b/lib/types.ts
@@ -1,3 +1,6 @@
+import { SearchResponse } from "https://raw.githubusercontent.com/bradenmacdonald/typesense-deno/main/src/Typesense/Documents.ts";
+import { resources } from "@lib/resources.ts";
+
export interface TMDBMovie {
adult: boolean;
backdrop_path: string;
@@ -23,3 +26,14 @@ export interface GiteaOauthUser {
picture: string;
groups: any;
}
+
+export type SearchResult = SearchResponse<{
+ id:string;
+ name: string;
+ type: keyof typeof resources;
+ date?: string;
+ rating: number;
+ tags: string[];
+ description?: string;
+ image?: string;
+}>;
diff --git a/lib/typesense.ts b/lib/typesense.ts
index e2f5071..4277949 100644
--- a/lib/typesense.ts
+++ b/lib/typesense.ts
@@ -29,13 +29,6 @@ function sanitizeStringForTypesense(input: string) {
// Use a promise to initialize the client as needed, rather than at import time.
let clientPromise: Promise | undefined;
-clientPromise?.then((client) => {
- client?.collections().create({
- name: "resources",
- fields: [],
- });
-});
-
export function getTypeSenseClient(): Promise {
if (clientPromise === undefined) {
let typesenseUrl: URL;
@@ -83,6 +76,7 @@ async function initializeTypesense() {
{ name: "rating", type: "int32", facet: true },
{ name: "tags", type: "string[]", facet: true },
{ name: "description", type: "string", optional: true },
+ { name: "image", type: "string", optional: true },
],
default_sorting_field: "rating", // Default field for sorting
});
@@ -117,6 +111,7 @@ async function synchronizeWithTypesense() {
description: sanitizeStringForTypesense(
resource?.description || resource?.content || "",
),
+ image: resource?.meta?.image,
tags: resource?.tags || [],
rating: resource?.meta?.rating || 0,
date: resource?.meta?.date?.toString() || "",
@@ -130,23 +125,29 @@ async function synchronizeWithTypesense() {
);
// Get all the IDs of documents currently indexed in Typesense
- // const allTypesenseDocuments = await client.collections("resources")
- // .documents()
- // .search({ q: "*", query_by: "name" });
+ const allTypesenseDocuments = await client.collections("resources")
+ .documents().search({
+ q: "*",
+ query_by: "name,type,date,description",
+ per_page: 250,
+ limit_hits: 9999,
+ });
- // const documentIds = allTypesenseDocuments.hits?.map((doc) => doc.id);
+ const documentIds = allTypesenseDocuments.hits?.map((doc) =>
+ doc?.document?.id
+ ) as string[];
// Find deleted document IDs by comparing the Typesense document IDs with the current list of resources
- // const deletedDocumentIds = documentIds?.filter((id) =>
- // !allResources.some((resource) => resource.id.toString() === id)
- // );
+ const deletedDocumentIds = documentIds?.filter((id) =>
+ !allResources.some((resource) => resource.id.toString() === id)
+ );
// Delete the documents with IDs found in deletedDocumentIds
- // await Promise.all(
- // deletedDocumentIds?.map((id) =>
- // client.collections("resources").documents()
- // ),
- // );
+ await Promise.all(
+ deletedDocumentIds?.map((id) =>
+ client.collections("resources").documents(id).delete()
+ ),
+ );
log.info("data synchronized");
} catch (error) {
diff --git a/routes/_app.tsx b/routes/_app.tsx
index 8417772..c8a8372 100644
--- a/routes/_app.tsx
+++ b/routes/_app.tsx
@@ -7,6 +7,7 @@ export default function App({ Component }: AppProps) {
+
>
diff --git a/routes/api/resources.ts b/routes/api/resources.ts
index 0c87e28..ed0e0a9 100644
--- a/routes/api/resources.ts
+++ b/routes/api/resources.ts
@@ -2,6 +2,10 @@ import { Handlers } from "$fresh/server.ts";
import { BadRequestError } from "@lib/errors.ts";
import { getTypeSenseClient } from "@lib/typesense.ts";
import { json } from "@lib/helpers.ts";
+import { getArticle } from "@lib/resource/articles.ts";
+import { getMovie } from "@lib/tmdb.ts";
+import { getRecipe } from "@lib/resource/recipes.ts";
+import { getDocument } from "@lib/documents.ts";
export const handler: Handlers = {
async GET(req, _ctx) {
@@ -30,8 +34,9 @@ export const handler: Handlers = {
q: query,
query_by,
filter_by,
+ per_page: 50,
});
- return json(searchResults.hits);
+ return json(searchResults);
},
};
diff --git a/routes/articles/[name].tsx b/routes/articles/[name].tsx
index 847e171..f46a774 100644
--- a/routes/articles/[name].tsx
+++ b/routes/articles/[name].tsx
@@ -6,6 +6,7 @@ import { KMenu } from "@islands/KMenu.tsx";
import { YoutubePlayer } from "@components/Youtube.tsx";
import { HashTags } from "@components/HashTags.tsx";
import { isYoutubeLink } from "@lib/string.ts";
+import { renderMarkdown } from "@lib/documents.ts";
export const handler: Handlers = {
async GET(_, ctx) {
@@ -19,6 +20,8 @@ export default function Greet(props: PageProps) {
const { author = "", date = "" } = article.meta;
+ const content = renderMarkdown(article.content);
+
return (
${article.name}`}>
) {
class="whitespace-break-spaces markdown-body"
data-color-mode="dark"
data-dark-theme="dark"
- dangerouslySetInnerHTML={{ __html: article.content || "" }}
+ dangerouslySetInnerHTML={{ __html: content || "" }}
>
- {article.content||""}
+ {content||""}
diff --git a/routes/index.tsx b/routes/index.tsx
index eeb82d1..a044397 100644
--- a/routes/index.tsx
+++ b/routes/index.tsx
@@ -2,7 +2,7 @@ import { Head } from "$fresh/runtime.ts";
import { MainLayout } from "@components/layouts/main.tsx";
import { Card } from "@components/Card.tsx";
import { PageProps } from "$fresh/server.ts";
-import { menu } from "@lib/menus.ts";
+import { resources } from "@lib/resources.ts";
import { KMenu } from "@islands/KMenu.tsx";
export default function Home(props: PageProps) {
@@ -14,10 +14,10 @@ export default function Home(props: PageProps) {
- {menu.map((m) => {
+ {Object.values(resources).map((m) => {
return (
diff --git a/routes/movies/[name].tsx b/routes/movies/[name].tsx
index 26eab98..f0e68ae 100644
--- a/routes/movies/[name].tsx
+++ b/routes/movies/[name].tsx
@@ -4,6 +4,7 @@ import { getMovie, Movie } from "@lib/resource/movies.ts";
import { RecipeHero } from "@components/RecipeHero.tsx";
import { KMenu } from "@islands/KMenu.tsx";
import { HashTags } from "@components/HashTags.tsx";
+import { renderMarkdown } from "@lib/documents.ts";
export const handler: Handlers
= {
async GET(_, ctx) {
@@ -17,6 +18,8 @@ export default function Greet(props: PageProps) {
const { author = "", date = "" } = movie.meta;
+ const content = renderMarkdown(movie.description || "");
+
return (
${movie.name}`}>
) {
: <>>}
- {movie.description}
+ {content}
diff --git a/static/favicon.png b/static/favicon.png
new file mode 100644
index 0000000..f0e3432
Binary files /dev/null and b/static/favicon.png differ
diff --git a/static/global.css b/static/global.css
index fa239c8..31d124d 100644
--- a/static/global.css
+++ b/static/global.css
@@ -17,7 +17,7 @@
}
body {
- background: #141217;
+ background: #141218;
padding: 0px 20px;
}
pre {
@@ -25,7 +25,8 @@ pre {
}
a {
- color: cadetblue
+ color: cadetblue;
+ font-family: Work Sans;
}
.custom-grid {
@@ -38,6 +39,19 @@ a {
}
}
+.animate-hover {
+ animation: hover 4s infinite;
+
+}
+@keyframes hover {
+ 0%, 100% {
+ transform: translateY(-15%);
+ }
+ 50% {
+ transform: translateY(0);
+ }
+}
+
.noisy-gradient::after {
content: "";
top: 0;