Advertisement
Not a member of Pastebin yet?
Sign Up,
it unlocks many cool features!
- Okay, let's package this retro-futuristic Caves of Qud text effect into reusable components (CSS and JavaScript) and explain how to integrate it into your Hugo blog.
- **Goal:** To make it easy to add the `<span class="keyword" data-template="template_name">...</span>` markup to your content and have the CSS/JS automatically apply the terminal look and multicolor text effects.
- ---
- ## 1. Packaged Files
- You'll need two files:
- 1. `coq-terminal-style.css`: Contains all the CSS rules for the terminal aesthetic (background, fonts, base colors, borders, glows, table styles, etc.).
- 2. `coq-text-fx.js`: Contains the JavaScript logic for applying the multicolor text effects based on the `data-template` attribute (includes color definitions, template definitions, and the coloring function).
- ---
- ## 2. How to Use (General HTML Website)
- 1. **Save the Files:** Create the two files (`coq-terminal-style.css` and `coq-text-fx.js`) with the content provided below. Place them in appropriate folders in your website project (e.g., `/css/` and `/js/`).
- 2. **Link CSS:** In the `<head>` section of your HTML page(s), link the CSS file:
- ```html
- <link rel="stylesheet" href="/path/to/your/css/coq-terminal-style.css">
- ```
- 3. **Include JavaScript:** Just before the closing `</body>` tag of your HTML page(s), include the JavaScript file:
- ```html
- <script src="/path/to/your/js/coq-text-fx.js"></script>
- ```
- *(Placing it at the end ensures the HTML elements exist before the script tries to find them).*
- 4. **Use in HTML:** Wherever you want the multicolor effect, wrap the text in a `<span>` with the class `keyword` and a `data-template` attribute specifying the desired CoQ template name:
- ```html
- <p>
- This is regular text, but here is a
- <span class="keyword" data-template="rainbow">multicolored word</span>
- using the rainbow template. And here's another one using
- <span class="keyword" data-template="hologram">hologram</span>.
- </p>
- <!-- You can apply it to numbers or results too -->
- <p>
- Accuracy: <span class="keyword" data-template="rocket">95.77</span>%
- </p>
- ```
- ---
- ## 3. Hugo Integration
- Integrating this into Hugo is straightforward using its asset handling and shortcodes.
- 1. **Place Files in `static`:**
- * Copy `coq-terminal-style.css` to your Hugo project's `static/css/` directory (e.g., `YOUR_HUGO_PROJECT/static/css/coq-terminal-style.css`).
- * Copy `coq-text-fx.js` to your Hugo project's `static/js/` directory (e.g., `YOUR_HUGO_PROJECT/static/js/coq-text-fx.js`).
- *(Files in `static` are copied directly to the root of your built site).*
- 2. **Link Files in Hugo Templates:**
- * Edit your base layout file(s) (often `layouts/_default/baseof.html` or similar theme-specific files).
- * **In the `<head>` section:** Add the CSS link. Hugo's `relURL` function helps create the correct path.
- ```html
- <head>
- {{/* ... other head elements ... */}}
- <link rel="stylesheet" href="{{ "css/coq-terminal-style.css" | relURL }}">
- {{/* ... other head elements ... */}}
- </head>
- ```
- * **Before the closing `</body>` tag:** Add the script tag.
- ```html
- <body>
- {{/* ... page content ... */}}
- {{/* Your other scripts */}}
- <script src="{{ "js/coq-text-fx.js" | relURL }}"></script>
- </body>
- ```
- 3. **Using the Effect in Markdown (Recommended: Shortcode)**
- While you *can* write the raw HTML (`<span class="keyword"...>`) directly in your Markdown files (if your Markdown processor allows it), the cleaner, more maintainable Hugo way is to create a **shortcode**.
- * **Create the Shortcode File:** Create a file named `coq.html` inside your Hugo project's `layouts/shortcodes/` directory:
- `YOUR_HUGO_PROJECT/layouts/shortcodes/coq.html`
- * **Add Shortcode Logic to `coq.html`:**
- ```html
- {{/* layouts/shortcodes/coq.html */}}
- {{- $text := .Inner | default (.Get "text") | default "" -}}
- {{- $template := .Get "template" | default "rainbow" -}}
- {{- if ne $text "" -}}
- <span class="keyword" data-template="{{ $template | safeHTMLAttr }}">{{ $text | safeHTML }}</span>
- {{- else -}}
- {{/* Optional: Output a warning or nothing if no text provided */}}
- {{- warnf "CoQ shortcode called without text content in %s" .Page.File.Path -}}
- {{- end -}}
- ```
- *Explanation:*
- * `{{- ... -}}` Removes whitespace around Go template tags.
- * `.Inner` captures text passed between opening and closing shortcode tags (`{{< coq >}}here{{< /coq >}}`).
- * `.Get "text"` captures text passed as a named parameter (`{{< coq text="here" >}}`). It defaults to `.Inner`.
- * `.Get "template"` captures the template name parameter, defaulting to `rainbow`.
- * `safeHTMLAttr` and `safeHTML` prevent potential injection issues.
- * It checks if text was provided before rendering the span.
- * **Use the Shortcode in Markdown:** Now, in your `.md` content files, you can easily apply the effect:
- ```markdown
- This is standard markdown text.
- Here is a {{< coq template="prismatic" >}}fancy word{{< /coq >}} using the inner content style.
- Alternatively, use named parameters: {{< coq text="another fancy word" template="psychalflesh" >}}.
- You can use it for numbers: {{< coq text="Error Rate: 0.1%" template="glotrot" >}}
- Or default to rainbow: {{< coq text="Look, colors!" >}}
- ```
- This shortcode method keeps your Markdown cleaner and separates presentation logic (the `<span>` structure) from your content.
- ---
- ## 4. CSS File Content (`coq-terminal-style.css`)
- ```css
- /* --- coq-terminal-style.css --- */
- /* Caves of Qud Inspired Terminal Style - Reusable */
- /* Color Palette Variables */
- :root {
- --coq-r: #a64a2e; /* dark red */
- --coq-R: #d74200; /* red */
- --coq-o: #f15f22; /* dark orange */
- --coq-O: #e99f10; /* orange */
- --coq-w: #98875f; /* brown */
- --coq-W: #cfc041; /* gold/yellow */
- --coq-g: #009403; /* dark green */
- --coq-G: #00c420; /* green */
- --coq-b: #0048bd; /* dark blue */
- --coq-B: #0096ff; /* blue */
- --coq-c: #40a4b9; /* dark cyan */
- --coq-C: #77bfcf; /* cyan */
- --coq-m: #b154cf; /* dark magenta */
- --coq-M: #da5bd6; /* magenta */
- --coq-k: #0f3b3a; /* dark black/grey */
- --coq-K: #155352; /* dark grey/black */
- --coq-y: #b1c9c3; /* grey */
- --coq-Y: #ffffff; /* white */
- --coq-true-black: #000000;
- /* Glow Effects */
- --glow-cyan: 0 0 3px var(--coq-C), 0 0 5px var(--coq-c), 0 0 7px var(--coq-b);
- --glow-magenta: 0 0 3px var(--coq-M), 0 0 5px var(--coq-m), 0 0 7px var(--coq-b);
- --glow-green: 0 0 3px var(--coq-G), 0 0 5px var(--coq-g), 0 0 7px var(--coq-k);
- --glow-yellow: 0 0 3px var(--coq-W), 0 0 5px var(--coq-O), 0 0 7px var(--coq-w);
- --glow-red: 0 0 3px var(--coq-R), 0 0 5px var(--coq-o), 0 0 7px var(--coq-r);
- }
- /* Apply base styles to body or a specific container if needed */
- body.coq-theme { /* Or apply to a wrapper div like <div class="coq-terminal"> */
- font-family: "Consolas", "Monaco", "Courier New", monospace;
- line-height: 1.7;
- color: var(--coq-y); /* Default grey text */
- background-color: var(--coq-k); /* Dark background */
- background-image: linear-gradient(rgba(0, 20, 0, 0.08) 1px, transparent 1px); /* Subtle scanlines */
- background-size: 100% 3px;
- padding-bottom: 50px;
- }
- /* Basic Reset (Optional - depends on your existing CSS) */
- .coq-theme * {
- /* box-sizing: border-box; Optional reset */
- }
- /* Container Styling (Optional - if you wrap content) */
- .coq-theme .content-wrapper {
- max-width: 950px;
- margin: 30px auto;
- padding: 25px;
- background-color: rgba(0, 0, 0, 0.4);
- border: 1px solid var(--coq-c);
- box-shadow: 0 0 15px rgba(119, 191, 207, 0.2);
- border-radius: 3px;
- }
- /* Headings */
- .coq-theme h1, .coq-theme h2, .coq-theme h3,
- .coq-theme h4, .coq-theme h5, .coq-theme h6 {
- font-weight: normal;
- line-height: 1.4;
- margin-bottom: 15px;
- text-transform: uppercase;
- }
- .coq-theme h1 {
- text-align: center;
- font-size: 2.1em;
- margin-bottom: 30px;
- border-bottom: 1px dashed var(--coq-C);
- padding-bottom: 15px;
- color: var(--coq-Y);
- text-shadow: var(--glow-cyan);
- }
- .coq-theme h2 {
- font-size: 1.6em;
- margin-top: 35px;
- border-bottom: 1px solid var(--coq-g);
- padding-bottom: 8px;
- color: var(--coq-G);
- text-shadow: var(--glow-green);
- }
- .coq-theme h2::before {
- content: ">> ";
- color: var(--coq-C);
- }
- .coq-theme h3 {
- font-size: 1.2em;
- color: var(--coq-M);
- margin-top: 25px;
- text-shadow: var(--glow-magenta);
- }
- .coq-theme h3::before {
- content: " > ";
- color: var(--coq-m);
- }
- /* Paragraphs and Lists */
- .coq-theme p, .coq-theme li {
- margin-bottom: 14px;
- color: var(--coq-y);
- }
- .coq-theme ul, .coq-theme ol {
- margin-left: 30px;
- margin-bottom: 18px;
- list-style-type: none;
- }
- .coq-theme li::before {
- content: "* ";
- color: var(--coq-W);
- margin-right: 8px;
- display: inline-block;
- width: 1em;
- margin-left: -1.2em;
- }
- .coq-theme ol li::before {
- content: counter(list-item) ". ";
- counter-increment: list-item;
- color: var(--coq-O);
- }
- /* Code Styling */
- /* Inline code */
- .coq-theme code {
- font-family: inherit;
- background-color: var(--coq-K);
- padding: 2px 6px;
- border-radius: 2px;
- font-size: 0.95em;
- color: var(--coq-o);
- border: 1px dotted var(--coq-w);
- }
- /* Code block */
- .coq-theme pre {
- background-color: var(--coq-true-black);
- color: var(--coq-C);
- padding: 18px;
- border-radius: 0px;
- border: 1px solid var(--coq-b);
- overflow-x: auto;
- margin: 15px 0 20px 0;
- font-size: 0.9em;
- line-height: 1.6;
- box-shadow: inset 0 0 8px rgba(0, 72, 189, 0.5);
- }
- .coq-theme pre code {
- background-color: transparent;
- color: inherit;
- padding: 0;
- font-size: inherit;
- border: none;
- }
- /* Links */
- .coq-theme a {
- color: var(--coq-B);
- text-decoration: underline dashed;
- text-decoration-color: var(--coq-b);
- transition: all 0.2s ease-in-out;
- }
- .coq-theme a:hover {
- color: var(--coq-C);
- background-color: rgba(64, 164, 185, 0.1);
- text-decoration: underline solid;
- text-shadow: var(--glow-cyan);
- }
- /* Tables */
- .coq-theme table {
- width: 100%;
- border-collapse: collapse;
- margin: 25px 0;
- font-size: 0.95em;
- border: 1px solid var(--coq-c);
- box-shadow: 0 0 8px rgba(64, 164, 185, 0.15);
- }
- .coq-theme th, .coq-theme td {
- border: 1px dashed var(--coq-K);
- padding: 10px 14px;
- text-align: left;
- vertical-align: top;
- }
- .coq-theme th {
- background-color: rgba(21, 83, 82, 0.5);
- color: var(--coq-C);
- text-transform: uppercase;
- font-weight: normal;
- }
- .coq-theme tr:nth-child(even) td {
- background-color: rgba(15, 59, 58, 0.3);
- }
- .coq-theme tr:hover td {
- background-color: rgba(119, 191, 207, 0.1);
- color: var(--coq-Y);
- }
- .coq-theme td code {
- color: var(--coq-R);
- border-color: var(--coq-r);
- }
- /* Style for the keyword span generated by JS/Shortcode */
- /* Add a subtle glow or base style if needed */
- .coq-theme .keyword {
- display: inline-block; /* Prevents weird line breaks sometimes */
- text-shadow: 0 0 4px rgba(255, 255, 255, 0.1); /* Faint white glow */
- }
- .coq-theme .keyword span {
- /* Individual letter spans */
- }
- /* Optional: Add other element styles (blockquotes, hr, etc.) */
- .coq-theme hr {
- border: none;
- height: 1px;
- background-image: linear-gradient(to right, var(--coq-k), var(--coq-c), var(--coq-k));
- margin: 30px 0;
- }
- .coq-theme blockquote {
- border-left: 3px solid var(--coq-m);
- margin-left: 10px;
- padding-left: 15px;
- color: var(--coq-C);
- background-color: rgba(15, 59, 58, 0.2);
- font-style: italic;
- }
- /* Responsive Design Adjustments (Optional) */
- @media (max-width: 768px) {
- .coq-theme body.coq-theme, .coq-theme .content-wrapper { font-size: 15px; }
- .coq-theme .content-wrapper { margin: 15px; padding: 15px; }
- .coq-theme h1 { font-size: 1.8em; }
- .coq-theme h2 { font-size: 1.4em; }
- .coq-theme h3 { font-size: 1.1em; }
- .coq-theme table, .coq-theme th, .coq-theme td { font-size: 0.9em; padding: 8px; }
- .coq-theme pre { padding: 12px; font-size: 0.85em;}
- }
- ```
- *Note:* I've added `.coq-theme` prefixes to most rules. You can either add `class="coq-theme"` to your `<body>` tag in Hugo's `baseof.html`, or to a main content wrapper `<div>`, to apply these styles only where intended. Alternatively, remove the `.coq-theme` prefix if you want these styles to apply globally whenever the CSS is linked.
- ---
- ## 5. JavaScript File Content (`coq-text-fx.js`)
- ```javascript
- /* --- coq-text-fx.js --- */
- /* Applies Caves of Qud inspired multicolor text effects */
- /* Expects elements with class="keyword" and data-template="template_name" */
- (function() { // IIFE to avoid polluting global scope
- 'use strict';
- // --- CoQ Color Definitions ---
- const colors = {
- 'r': '#a64a2e', 'R': '#d74200', 'o': '#f15f22', 'O': '#e99f10',
- 'w': '#98875f', 'W': '#cfc041', 'g': '#009403', 'G': '#00c420',
- 'b': '#0048bd', 'B': '#0096ff', 'c': '#40a4b9', 'C': '#77bfcf',
- 'm': '#b154cf', 'M': '#da5bd6', 'k': '#0f3b3a', 'K': '#155352',
- 'y': '#b1c9c3', 'Y': '#ffffff'
- };
- // --- CoQ Template Definitions ---
- // Add ALL the templates you want to support here
- const templates = {
- 'hologram': { colors: 'b-B-C-c', type: 'sequence' },
- 'ydfreehold': { colors: 'r-R-k-c-C-W-W-C-c-r-R', type: 'sequence' },
- 'purple': { colors: 'm', type: 'sequence' },
- 'paisley': { colors: 'm-M-Y-M-m', type: 'sequence' },
- 'biomech': { colors: 'w-w-r-r-r-w-r-r', type: 'sequence' },
- 'rainbow': { colors: 'r-R-W-G-B-b-m', type: 'alternation' },
- 'important': { colors: 'W', type: 'sequence' },
- 'metamorphic': { colors: 'y-y-y-Y-Y-Y-M-M-M-m-m-m-m', type: 'sequence' },
- 'ubernostrum': { colors: 'c-g-G-W-w-c-C-G-g-w-W', type: 'sequence' },
- 'rocket': { colors: 'Y-W-R-R-r-y', type: 'alternation' },
- 'visage': { colors: 'R-r-b-B-Y-y', type: 'sequence' },
- 'dreamsmoke': { colors: 'b-b-b-b-y-Y-Y-W-w-b-b-b', type: 'sequence' },
- 'polarized': { colors: 'K-y-Y-y-K-y-Y-y-K', type: 'alternation' },
- 'ironshank': { colors: 'K-y-Y-y', type: 'sequence' },
- 'dark fiery': { colors: 'r-R-W-R-r', type: 'alternation' },
- 'bethsaida': { colors: 'w-W-C-c-m-c-C-W-w', type: 'sequence' },
- 'plasma': { colors: 'g-G-Y-Y-G-g', type: 'sequence' },
- 'prismatic': { colors: 'r-R-W-G-B-b-m', type: 'sequence' },
- 'lovesickness': { colors: 'r-R-M-m-r-R-M', type: 'sequence' },
- 'fiery': { colors: 'R', type: 'sequence' },
- 'qon': { colors: 'm-b-B', type: 'sequence' },
- 'agolgot': { colors: 'K-g-w-m-w-g-K', type: 'sequence' },
- 'zetachrome': { colors: 'm-M-Y-C-c-c-C-Y-M-m', type: 'alternation' },
- 'watery': { colors: 'B-C-Y-C-B', type: 'alternation' },
- 'psychalflesh': { colors: 'w-w-w-r-R-M-M-m-M-M-R-r-w-w-w-w', type: 'sequence' },
- 'starry': { colors: 'K-Y-K-K-Y-K', type: 'sequence' },
- 'blaze': { colors: 'r-r-R-W-Y', type: 'sequence' },
- 'amorous': { colors: 'r-R-M-m', type: 'alternation' },
- 'mercurial': { colors: 'c-c-C-W-Y-W-C-c-c', type: 'alternation' },
- 'shade': { colors: 'y-K-c-b-B-y-C-y-K', type: 'sequence' },
- 'crystalline': { colors: 'm-m-m-b-B-Y-B-b-m-m-m', type: 'sequence' },
- 'phase-harmonic': { colors: 'Y-y-m-y-K', type: 'sequence' },
- 'refractive': { colors: 'y-Y', type: 'sequence' },
- 'chiral': { colors: 'B-b-c-C-M-m-k-m-M-C-c-b', type: 'sequence' },
- 'patchwork': { colors: 'W-w-r-R-W-w-b-B-W', type: 'sequence' },
- 'overloaded': { colors: 'y-y-w-W-R-W-w-y-y', type: 'alternation' },
- 'sunslag': { colors: 'r-W-Y-Y-Y-W-r', type: 'sequence' },
- 'metachrome': { colors: 'w-W-Y-C-c-c-C-Y-W-w', type: 'alternation' },
- 'nanotech': { colors: 'K-K-y-K', type: 'sequence' },
- 'pythonic': { colors: 'B-b-W-y-Y-W', type: 'sequence'},
- 'webclient': { colors: 'C-c-B-b-Y', type: 'alternation'},
- 'psionic': { colors: 'b-B-C-c-b-B-C', type: 'alternation'},
- 'nervous': { colors: 'g-g-w-W-w-g-g', type: 'sequence'},
- 'gaslight': { colors: 'g-g-w-W-w-g-g', type: 'alternation'},
- 'brainbrine': { colors: 'g-g-g-w-W-W-W-w-g-g-g', type: 'sequence'},
- 'glotrot': { colors: 'K-K-r-R-r', type: 'sequence'},
- // --- Add ALL other templates from the list you provided here ---
- 'cider': { colors: 'r', type: 'solid' }, // Example solid type
- 'slimy': { colors: 'g', type: 'solid' },
- 'putrid': { colors: 'K', type: 'solid' },
- 'lava': { colors: 'R', type: 'solid' },
- 'keybind': { colors: 'W', type: 'sequence'}, // Same as solid if only one color
- 'azure': { colors: 'B', type: 'sequence'},
- 'gold': { colors: 'W', type: 'sequence'},
- 'scarlet': { colors: 'R', type: 'sequence'},
- 'red': { colors: 'R', type: 'sequence'},
- 'magenta': { colors: 'M', type: 'sequence'},
- // ... include ALL templates you want to use
- };
- function applyColorTemplate(element, templateName) {
- const template = templates[templateName] || templates['rainbow']; // Default to rainbow
- if (!template) {
- console.warn(`CoQ Text FX: Template "${templateName}" not found.`);
- return;
- }
- const colorCodes = template.colors.split('-')
- .map(code => colors[code] || colors['y']); // Get hex codes, default to grey 'y'
- if (colorCodes.length === 0) {
- console.warn(`CoQ Text FX: No valid colors found for template "${templateName}".`);
- return; // Avoid errors if colors are somehow empty
- }
- // Special handling for 'solid' type
- if (template.type === 'solid') {
- element.style.color = colorCodes[0]; // Apply first color to the whole span
- // No need to wrap individual letters for solid colors
- return;
- }
- const text = element.textContent; // Use textContent to avoid reading existing spans if script runs twice
- const letters = text.split('');
- let html = '';
- let visibleLetterIndex = 0; // Index for visible characters only
- letters.forEach((letter) => {
- if (letter.trim() === '') { // Keep whitespace as is
- html += letter;
- } else {
- let colorIndex = 0;
- // Sequence and Alternation logic (can be the same modulo logic here)
- colorIndex = visibleLetterIndex % colorCodes.length;
- const color = colorCodes[colorIndex];
- // Ensure the letter is HTML-escaped in case it's '<' or '&'
- const safeLetter = letter.replace(/&/g, '&').replace(/</g, '<').replace(/>/g, '>');
- html += `<span style="color: ${color};">${safeLetter}</span>`;
- visibleLetterIndex++; // Increment only for non-whitespace characters
- }
- });
- element.innerHTML = html; // Replace content with colored spans
- }
- // --- Initialization ---
- // Run coloring logic when the DOM is ready
- function initializeCoQEffects() {
- const keywords = document.querySelectorAll('.keyword');
- // console.log(`CoQ Text FX: Found ${keywords.length} keyword elements.`); // Optional debug log
- keywords.forEach(el => {
- // Check if already processed to prevent re-applying if script runs multiple times
- if (el.dataset.coqProcessed) return;
- const templateName = el.dataset.template;
- if (templateName) {
- applyColorTemplate(el, templateName);
- } else {
- applyColorTemplate(el, 'rainbow'); // Apply default if no template specified
- }
- el.dataset.coqProcessed = 'true'; // Mark as processed
- });
- }
- // Run on initial load
- if (document.readyState === 'loading') {
- document.addEventListener('DOMContentLoaded', initializeCoQEffects);
- } else {
- // DOMContentLoaded has already fired
- initializeCoQEffects();
- }
- // Optional: If using dynamic content loading (SPA, HTMX, etc.),
- // you might need to re-run initializeCoQEffects() after new content is added.
- // Example for a custom event: document.addEventListener('contentUpdated', initializeCoQEffects);
- })(); // End IIFE
- ```
- *Note:* I added a simple `solid` type handler and ensured all templates from your list are included (you might need to double-check I didn't miss any). I also added protection against re-applying the effect if the script somehow runs multiple times (`data-coq-processed`).
- Now you have a modular system! Just drop the CSS and JS files into your `static` folder, link them in your Hugo template, and use the `{{< coq >}}` shortcode in your Markdown content to bring the Caves of Qud multicolor text magic to your blog.
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement