commit 81741f0c7cfb41cadb0ba486bbc5173025ccbbba
Author: ffff:72.234.190.31 <ffff:72.234.190.31@importer.scroll.pub>
Date: 2024-11-16 00:07:17 +0000
Subject: Updated index.scroll
diff --git a/index.scroll b/index.scroll
index dcd1248..9e7d4c2 100644
--- a/index.scroll
+++ b/index.scroll
@@ -1,4 +1,3 @@
buildHtml
claude.html
editButton /edit.html
-scrollVersionLink
commit 8dac0a7b23b5c6ee742246bc9f24dfc1baefea80
Author: ffff:72.234.190.31 <ffff:72.234.190.31@importer.scroll.pub>
Date: 2024-11-16 00:07:07 +0000
Subject: Updated index.scroll
diff --git a/index.scroll b/index.scroll
index 9e7d4c2..dcd1248 100644
--- a/index.scroll
+++ b/index.scroll
@@ -1,3 +1,4 @@
buildHtml
claude.html
editButton /edit.html
+scrollVersionLink
commit 1ac6288c67fee526ddee5197c2a0cccfa16e353b
Author: ffff:72.234.190.31 <ffff:72.234.190.31@importer.scroll.pub>
Date: 2024-11-16 00:06:56 +0000
Subject: Updated index.scroll
diff --git a/index.scroll b/index.scroll
index 7f3dc1b..9e7d4c2 100644
--- a/index.scroll
+++ b/index.scroll
@@ -1,3 +1,3 @@
buildHtml
claude.html
-editButton
+editButton /edit.html
commit 5577d0b1e8c729a0143d7965f434e45eccecdb2c
Author: ffff:72.234.190.31 <ffff:72.234.190.31@importer.scroll.pub>
Date: 2024-11-16 00:06:47 +0000
Subject: Updated index.scroll
diff --git a/index.scroll b/index.scroll
index 8e7cd22..7f3dc1b 100644
--- a/index.scroll
+++ b/index.scroll
@@ -1,2 +1,3 @@
buildHtml
claude.html
+editButton
commit 65c8feb964ff35bb1b6f94fe417332853e7a3d6d
Author: Breck Yunits <breck7@gmail.com>
Date: 2024-11-15 14:05:21 -1000
Subject:
diff --git a/latex.js b/latex.js
index 652fcdf..536672e 100644
--- a/latex.js
+++ b/latex.js
@@ -1,192 +1,197 @@
class LaTeXToScroll {
constructor() {
this.blockCommands = {
- section: "# ",
- subsection: "## ",
- subsubsection: "### ",
- paragraph: "*",
- begin: {
- itemize: "- ",
- enumerate: "1. ",
- quote: "> ",
- verbatim: "code\n",
- center: "center\n",
- figure: "figure\n",
- },
+ 'section': '# ',
+ 'subsection': '## ',
+ 'subsubsection': '### ',
+ 'paragraph': '*',
+ 'begin': {
+ 'itemize': '- ',
+ 'enumerate': '1. ',
+ 'quote': '> ',
+ 'verbatim': 'code\n',
+ 'center': 'center\n',
+ 'figure': 'figure\n',
+ }
};
this.inlineCommands = {
- textbf: "*",
- textit: "_",
- texttt: "`",
- emph: "_",
- href: "link",
+ 'textbf': '*',
+ 'textit': '_',
+ 'texttt': '`',
+ 'emph': '_',
+ 'href': 'link'
};
}
convert(latex) {
let scroll = latex;
-
+
// Remove comments
- scroll = scroll.replace(/%.*$/gm, "");
-
+ scroll = scroll.replace(/%.*$/gm, '');
+
+ // Handle \input commands first
+ scroll = this.convertInputCommands(scroll);
+
// Handle block environments
scroll = this.convertEnvironments(scroll);
-
+
// Handle section commands
scroll = this.convertSections(scroll);
-
+
// Handle inline formatting
scroll = this.convertInlineCommands(scroll);
-
+
// Handle special characters
scroll = this.convertSpecialCharacters(scroll);
-
- // Handle lists
- scroll = this.convertLists(scroll);
-
- // Handle images
- scroll = this.convertImages(scroll);
-
+
// Handle citations and references
scroll = this.convertCitations(scroll);
-
+
// Clean up extra whitespace
scroll = this.cleanupWhitespace(scroll);
-
+
return scroll;
}
+ convertInputCommands(text) {
+ // Match \input{filename} with or without .tex extension
+ const inputRegex = /\\input\{([^}]+)\}/g;
+ return text.replace(inputRegex, (match, filename) => {
+ // Remove .tex extension if present and add .scroll
+ const scrollFile = filename.replace(/\.tex$/, '') + '.scroll';
+ return `${scrollFile}`;
+ });
+ }
+
convertEnvironments(text) {
let result = text;
-
+
// Match begin/end environment blocks
const envRegex = /\\begin\{(\w+)\}([\s\S]*?)\\end\{\1\}/g;
-
+
result = result.replace(envRegex, (match, env, content) => {
- const prefix = this.blockCommands.begin[env] || "";
-
- if (env === "itemize" || env === "enumerate") {
+ const prefix = this.blockCommands.begin[env] || '';
+
+ if (env === 'itemize' || env === 'enumerate') {
return this.convertListItems(content, env);
- } else if (env === "verbatim") {
- return `${prefix}${content.trim()}`;
- } else if (env === "figure") {
+ } else if (env === 'verbatim') {
+ return `code\n${content.trim()}`;
+ } else if (env === 'figure') {
return this.convertFigure(content);
}
-
+
// Indent content for environments that need it
const indentedContent = content
.trim()
- .split("\n")
- .map((line) => " " + line)
- .join("\n");
-
+ .split('\n')
+ .map(line => ' ' + line)
+ .join('\n');
+
return `${prefix}${indentedContent}\n`;
});
-
+
return result;
}
+ convertListItems(content, type) {
+ const marker = type === 'itemize' ? '- ' : '1. ';
+ return content
+ .split('\\item')
+ .slice(1) // Skip first empty element
+ .map(item => marker + item.trim())
+ .join('\n') + '\n';
+ }
+
convertSections(text) {
let result = text;
-
+
// Convert section commands
- Object.keys(this.blockCommands).forEach((cmd) => {
- if (cmd === "begin") return;
- const regex = new RegExp(`\\\\${cmd}{([^}]+)}`, "g");
+ Object.keys(this.blockCommands).forEach(cmd => {
+ if (cmd === 'begin') return;
+ const regex = new RegExp(`\\\\${cmd}{([^}]+)}`, 'g');
result = result.replace(regex, (match, content) => {
return `${this.blockCommands[cmd]}${content.trim()}\n`;
});
});
-
+
return result;
}
convertInlineCommands(text) {
let result = text;
-
+
// Convert inline formatting commands
- Object.keys(this.inlineCommands).forEach((cmd) => {
- const regex = new RegExp(`\\\\${cmd}{([^}]+)}`, "g");
+ Object.keys(this.inlineCommands).forEach(cmd => {
+ const regex = new RegExp(`\\\\${cmd}{([^}]+)}`, 'g');
result = result.replace(regex, (match, content) => {
- if (cmd === "href") {
- const [url, text] = content.split("}{");
- return `${text}\n link ${url} ${text}`;
+ if (cmd === 'href') {
+ // Handle \href{url}{text}
+ const [url, linkText] = content.split('}{');
+ return `${linkText.replace('}', '')}\n link ${url} ${linkText.replace('}', '')}`;
}
const marker = this.inlineCommands[cmd];
return `${marker}${content}${marker}`;
});
});
-
+
return result;
}
convertSpecialCharacters(text) {
const charMap = {
- "\\&": "&",
- "\\%": "%",
- "\\$": "$",
- "\\#": "#",
- "\\_": "_",
- "\\{": "{",
- "\\}": "}",
- "~": " ",
- "``": '"',
- "''": '"',
+ '\\&': '&',
+ '\\%': '%',
+ '\\$': '$',
+ '\\#': '#',
+ '\\_': '_',
+ '\\{': '{',
+ '\\}': '}',
+ '~': ' ',
+ '``': '"',
+ "''": '"'
};
-
+
let result = text;
- Object.keys(charMap).forEach((char) => {
- result = result.replace(
- new RegExp(char.replace(/[.*+?^${}()|[\]\\]/g, "\\$&"), "g"),
- charMap[char],
- );
+ Object.keys(charMap).forEach(char => {
+ result = result.replace(new RegExp(char.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'), 'g'), charMap[char]);
});
-
+
return result;
}
- convertListItems(content, type) {
- const marker = type === "itemize" ? "- " : "1. ";
- const lines = content
- .split("\\item")
- .slice(1) // Skip first empty element
- .map((item) => marker + item.trim())
- .join("\n");
- return lines + "\n";
- }
-
convertFigure(content) {
// Extract image path and caption
const includegraphicsRegex = /\\includegraphics(?:\[.*?\])?\{([^}]+)\}/;
const captionRegex = /\\caption\{([^}]+)\}/;
-
- const imagePath = content.match(includegraphicsRegex)?.[1] || "";
- const caption = content.match(captionRegex)?.[1] || "";
-
+
+ const imagePath = content.match(includegraphicsRegex)?.[1] || '';
+ const caption = content.match(captionRegex)?.[1] || '';
+
return `image ${imagePath}\n caption ${caption}\n`;
}
convertCitations(text) {
let result = text;
-
+
// Convert \cite{key} to ^key
- result = result.replace(/\\cite\{([^}]+)\}/g, "^$1");
-
+ result = result.replace(/\\cite\{([^}]+)\}/g, '^$1');
+
// Convert \ref{key} to ^key
- result = result.replace(/\\ref\{([^}]+)\}/g, "^$1");
-
+ result = result.replace(/\\ref\{([^}]+)\}/g, '^$1');
+
return result;
}
cleanupWhitespace(text) {
- return (
- text
- .replace(/\n\s*\n\s*\n/g, "\n\n") // Remove extra blank lines
- .replace(/[ \t]+$/gm, "") // Remove trailing whitespace
- .trim() + "\n"
- ); // Ensure single newline at end
+ return text
+ .replace(/\n\s*\n\s*\n/g, '\n\n') // Remove extra blank lines
+ .replace(/[ \t]+$/gm, '') // Remove trailing whitespace
+ .trim() + '\n'; // Ensure single newline at end
}
}
-// module.exports = LaTeXToScroll;
+// For Node.js environment
+if (typeof module !== 'undefined' && module.exports) {
+ module.exports = LaTeXToScroll;
+}
diff --git a/markdown.js b/markdown.js
index b417866..7c13a78 100644
--- a/markdown.js
+++ b/markdown.js
@@ -1,6 +1,5 @@
class MarkdownToScroll {
constructor() {
- // Mapping of common Markdown patterns
this.blockPatterns = {
header: /^(#{1,6})\s+(.+)$/,
horizontalRule: /^[\*\-_]{3,}$/,
@@ -19,118 +18,76 @@ class MarkdownToScroll {
strikethrough: /~~(.+?)~~/g,
link: /\[([^\]]+)\]\(([^)]+)\)(?:\{([^}]+)\})?/g,
image: /!\[([^\]]*)\]\(([^)]+)\)(?:\{([^}]+)\})?/g,
- footnoteRef: /\[\^(\d+)\]/g,
- footnoteDef: /^\[\^(\d+)\]:\s*(.+)$/,
};
}
convert(markdown) {
- // Split into lines for block-level processing
- let lines = markdown.trim().split("\n");
- let scroll = "";
+ if (!markdown) return '';
+
+ let lines = markdown.toString().trim().split('\n');
+ let scroll = '';
let inCodeBlock = false;
- let codeBlockContent = "";
- let inTable = false;
- let tableContent = [];
+ let codeBlockContent = '';
+ let inList = false;
+ let listIndentLevel = 0;
for (let i = 0; i < lines.length; i++) {
- const line = lines[i];
-
- // Handle code blocks first
- if (line.startsWith("```")) {
+ let line = lines[i].trimEnd();
+
+ // Handle code blocks
+ if (line.startsWith('```')) {
if (!inCodeBlock) {
inCodeBlock = true;
- codeBlockContent = "";
+ codeBlockContent = '';
continue;
} else {
- scroll += this.convertCodeBlock(codeBlockContent.trim()) + "\n\n";
+ scroll += 'code\n' + codeBlockContent.trim() + '\n\n';
inCodeBlock = false;
continue;
}
}
if (inCodeBlock) {
- codeBlockContent += line + "\n";
+ codeBlockContent += line + '\n';
continue;
}
- // Handle tables
- if (line.startsWith("|")) {
- if (!inTable) {
- inTable = true;
- tableContent = [];
- }
- if (!this.blockPatterns.tableDelimiter.test(line)) {
- tableContent.push(line);
- }
+ // Handle horizontal rules
+ if (this.blockPatterns.horizontalRule.test(line)) {
+ scroll += '---\n\n';
continue;
- } else if (inTable) {
- scroll += this.convertTable(tableContent) + "\n\n";
- inTable = false;
- tableContent = [];
}
- // Process block-level patterns
- let processed = false;
-
- // Headers
+ // Handle headers
const headerMatch = line.match(this.blockPatterns.header);
if (headerMatch) {
- scroll +=
- "#".repeat(headerMatch[1].length) +
- " " +
- this.convertInlineElements(headerMatch[2]) +
- "\n\n";
- processed = true;
- }
-
- // Horizontal Rule
- if (this.blockPatterns.horizontalRule.test(line)) {
- scroll += "---\n\n";
- processed = true;
+ scroll += '#'.repeat(headerMatch[1].length) + ' ' +
+ this.convertInlineElements(headerMatch[2]) + '\n\n';
+ continue;
}
- // Blockquotes
+ // Handle blockquotes
const blockquoteMatch = line.match(this.blockPatterns.blockquote);
if (blockquoteMatch) {
- scroll += "> " + this.convertInlineElements(blockquoteMatch[1]) + "\n";
- processed = true;
+ scroll += '> ' + this.convertInlineElements(blockquoteMatch[1]) + '\n';
+ continue;
}
- // Unordered Lists
+ // Handle lists
const ulMatch = line.match(this.blockPatterns.unorderedList);
- if (ulMatch) {
- scroll += "- " + this.convertInlineElements(ulMatch[1]) + "\n";
- processed = true;
- }
-
- // Ordered Lists
const olMatch = line.match(this.blockPatterns.orderedList);
- if (olMatch) {
- scroll += "1. " + this.convertInlineElements(olMatch[1]) + "\n";
- processed = true;
- }
-
- // Footnote Definitions
- const footnoteMatch = line.match(this.blockPatterns.footnoteDef);
- if (footnoteMatch) {
- scroll +=
- "^" +
- footnoteMatch[1] +
- " " +
- this.convertInlineElements(footnoteMatch[2]) +
- "\n";
- processed = true;
- }
-
- // Regular paragraphs
- if (!processed && line.trim()) {
- scroll += "* " + this.convertInlineElements(line) + "\n\n";
+ if (ulMatch || olMatch) {
+ const content = (ulMatch || olMatch)[1];
+ const marker = ulMatch ? '- ' : '1. ';
+ scroll += marker + this.convertInlineElements(content) + '\n';
+ continue;
}
- // Preserve blank lines
- if (!line.trim()) {
- scroll += "\n";
+ // Handle regular paragraphs
+ if (line.trim()) {
+ scroll += '* ' + this.convertInlineElements(line) + '\n\n';
+ } else {
+ scroll += '\n';
}
}
@@ -138,69 +95,42 @@ class MarkdownToScroll {
}
convertInlineElements(text) {
- let result = text;
-
- // Convert images before links (to avoid nested parsing issues)
- result = result.replace(
- this.inlinePatterns.image,
- (match, alt, src, title) => {
- let scroll = `\nimage ${src}`;
- if (alt) scroll += `\n caption ${alt}`;
- return scroll;
- },
- );
+ if (!text) return '';
+ let result = text.toString();
+
+ // Convert images before links
+ result = result.replace(this.inlinePatterns.image, (match, alt, src, title) => {
+ let scroll = `\nimage ${src}`;
+ if (alt) scroll += `\n caption ${alt}`;
+ return scroll;
+ });
// Convert links
- result = result.replace(
- this.inlinePatterns.link,
- (match, text, url, title) => {
- let scroll = text + "\n link " + url + " " + text;
- if (title) scroll += "\n title " + title;
- return scroll;
- },
- );
+ result = result.replace(this.inlinePatterns.link, (match, text, url) => {
+ return `${text}\n link ${url} ${text}`;
+ });
// Convert other inline elements
result = result
- .replace(this.inlinePatterns.bold, "*$1*")
- .replace(this.inlinePatterns.italic, "_$1_")
- .replace(this.inlinePatterns.code, "`$1`")
- .replace(this.inlinePatterns.strikethrough, "strike $1")
- .replace(this.inlinePatterns.footnoteRef, "^$1");
+ .replace(this.inlinePatterns.bold, '*$1*')
+ .replace(this.inlinePatterns.italic, '_$1_')
+ .replace(this.inlinePatterns.code, '`$1`')
+ .replace(this.inlinePatterns.strikethrough, 'strike $1');
return result;
}
- convertCodeBlock(content) {
- return "code\n" + content;
- }
-
- convertTable(tableContent) {
- let scroll = "table\n";
- scroll += " data\n";
-
- // Convert each table row to CSV format
- tableContent.forEach((row) => {
- const cells = row
- .split("|")
- .filter((cell) => cell.trim()) // Remove empty cells from start/end
- .map((cell) => cell.trim())
- .map((cell) => this.convertInlineElements(cell))
- .join(",");
- scroll += " " + cells + "\n";
- });
-
- return scroll;
- }
-
cleanup(scroll) {
- return (
- scroll
- .replace(/\n{3,}/g, "\n\n") // Remove extra blank lines
- .replace(/[ \t]+$/gm, "") // Remove trailing whitespace
- .trim() + "\n"
- ); // Ensure single newline at end
+ return scroll
+ .replace(/\n{3,}/g, '\n\n') // Remove extra blank lines
+ .replace(/[ \t]+$/gm, '') // Remove trailing whitespace
+ .replace(/\^undefined\n/g, '') // Remove undefined references
+ .replace(/\n\n+/g, '\n\n') // Normalize multiple newlines
+ .trim() + '\n'; // Ensure single newline at end
}
}
-// module.exports = MarkdownToScroll;
+// For Node.js environment
+if (typeof module !== 'undefined' && module.exports) {
+ module.exports = MarkdownToScroll;
+}
commit 091f117da86925c7598a457df6c85398d8532f74
Author: ffff:72.234.190.31 <ffff:72.234.190.31@hub.scroll.pub>
Date: 2024-11-15 22:32:37 +0000
Subject: Updated claude.html
diff --git a/claude.html b/claude.html
index 0491b68..a6170bf 100644
--- a/claude.html
+++ b/claude.html
@@ -23,7 +23,7 @@
"Helvetica Neue", Arial, sans-serif;
line-height: 1.6;
padding: 2rem;
- padding-top: 1rem;
+ padding-top: 0.2rem;
max-width: 1400px;
margin: 0 auto;
background-color: var(--gray-100);
commit a9f73861d3625c0b5838f2897fd7af44ed831b88
Author: ffff:72.234.190.31 <ffff:72.234.190.31@hub.scroll.pub>
Date: 2024-11-15 22:32:19 +0000
Subject: Updated claude.html
diff --git a/claude.html b/claude.html
index 5aad4c3..0491b68 100644
--- a/claude.html
+++ b/claude.html
@@ -33,7 +33,7 @@
h2 {
text-align: center;
color: var(--gray-700);
- margin-bottom: 1rem;
+ margin-bottom: 0.2rem;
margin-top: 0;
}
commit c751f7b9004e4deb7574ed040280137302269e2a
Author: ffff:72.234.190.31 <ffff:72.234.190.31@hub.scroll.pub>
Date: 2024-11-15 22:32:07 +0000
Subject: Updated claude.html
diff --git a/claude.html b/claude.html
index b19388f..5aad4c3 100644
--- a/claude.html
+++ b/claude.html
@@ -23,6 +23,7 @@
"Helvetica Neue", Arial, sans-serif;
line-height: 1.6;
padding: 2rem;
+ padding-top: 1rem;
max-width: 1400px;
margin: 0 auto;
background-color: var(--gray-100);
commit e0541c6b4c5c80e9d420f628ef7c0f04c5b38bcf
Author: ffff:72.234.190.31 <ffff:72.234.190.31@hub.scroll.pub>
Date: 2024-11-15 22:31:52 +0000
Subject: Updated claude.html
diff --git a/claude.html b/claude.html
index 3f9fb3b..b19388f 100644
--- a/claude.html
+++ b/claude.html
@@ -33,6 +33,7 @@
text-align: center;
color: var(--gray-700);
margin-bottom: 1rem;
+ margin-top: 0;
}
.container {
commit f050db2e8eab55421729c8f43ed3e0103e52389b
Author: ffff:72.234.190.31 <ffff:72.234.190.31@hub.scroll.pub>
Date: 2024-11-15 22:31:42 +0000
Subject: Updated claude.html
diff --git a/claude.html b/claude.html
index c85b4aa..3f9fb3b 100644
--- a/claude.html
+++ b/claude.html
@@ -32,7 +32,7 @@
h2 {
text-align: center;
color: var(--gray-700);
- margin-bottom: 2rem;
+ margin-bottom: 1rem;
}
.container {