Compare commits

..

1 Commits

Author SHA1 Message Date
MikeHughes-BIN 8f6ffa702c Add documentation folder 2026-01-17 14:42:36 +01:00
7 changed files with 258 additions and 454 deletions
+39 -60
View File
@@ -97,33 +97,31 @@
<!-- Here starts code from step 2--> <!-- Here starts code from step 2-->
<div class="step" id="step2" style="display:none;"> <div class="step" id="step2" style="display:none;">
<h2 class="h2">Choose your preferences:</h2> <h2 class="h2">Choose your preferences:</h2>
<div class="KI-wrapper">
<label id="labelKI">Select ki:</label>
<select name="ai_type" id="ai_type"></select>
</div>
<div class="step2-form"> <div class="transcript-wrap">
<label id="labelTranscription">Select transcription:</label>
<select name="transkript_type" id="transkript_type"></select>
</div>
<div class="step2-row"> <div class="type-wrapper">
<label>Select ki:</label> <label id="labelType">Select type:</label>
<select id="ai_type"></select> <select name="output_type" id="output_type">
</div> <option value="pdf">.pdf</option>
<option value="word">.docx</option>
<option value="txt">.txt</option>
</select>
</div>
<div class="step2-row"> <div class="language-wrapper">
<label>Select transcription:</label> <label id="labelLanguage">Select language:</label>
<select id="transkript_type"></select>
</div>
<div class="step2-row"> <select name="document_language_option" id="document_language_option">
<label>Select type:</label>
<select id="output_type">
<option>.pdf</option>
<option>.docx</option>
<option>.txt</option>
</select>
</div>
<div class="step2-row">
<label>Select language:</label>
<select id="document_language_option"></select>
</div>
</select>
</div> </div>
</div> </div>
@@ -196,49 +194,30 @@
<!-- Here starts code from step 5--> <!-- Here starts code from step 5-->
<div class="step" id="step5" style="display:none;"> <div class="step" id="step5" style="display:none;">
<h2 class="h2">Change names of the speakers:</h2> <h2 class="h2">Change names of the speakers:</h2>
<div class="speakerView" id="speakerView">
<div class="speaker-container"> <label id="labelSpeaker">Select Speaker:</label>
<table class="speaker-table"> <select name="cur_speaker" id="cur_speaker">
<tbody> </select>
<tr> </div>
<td class="label-cell"> <div class="speakerAudio" id="speakerAutio">
<label id="labelSpeaker" for="cur_speaker">Select Speaker:</label> <label id="labelSpeakerAudio">Selected Speaker:</label>
</td> <audio controls id="speakerAudioViewer">
<td class="input-cell"> Currently there is no audio file here.
<select name="cur_speaker" id="cur_speaker"></select> </audio>
</td> </div>
</tr> <div class="speakerWrite" id="speakerWrite">
<tr> <label id="labelSpeakerWriter">Write name:</label>
<td class="label-cell"> <input type="text" id="newSpeaker">
<label id="labelSpeakerAudio">Speaker Audio:</label> </div>
</td> <div class="speakerButton-group">
<td class="input-cell"> <button id="speakerLocker" onclick="rewriteSpeakerName()">Rename Speaker</button>
<audio controls id="speakerAudioViewer"> <button id="speakerResender" onclick="sendSpeakerPackages()">Rewrite document</button>
Currently there is no audio file here.
</audio>
</td>
</tr>
<tr>
<td class="label-cell">
<label id="labelSpeakerWriter" for="newSpeaker">New Name:</label>
</td>
<td class="input-cell">
<input type="text" id="newSpeaker" placeholder="Enter new speaker name">
</td>
</tr>
</tbody>
</table>
<div class="speaker-button-group">
<button id="speakerLocker" onclick="rewriteSpeakerName()">Rename Speaker</button>
<button id="speakerResender" onclick="sendSpeakerPackages()">Rewrite Document</button>
</div>
</div> </div>
</div> </div>
<!-- Here starts code from step 6--> <!-- Here starts code from step 6-->
<div class="step" id="step6" style="display:none;"> <div class="step" id="step6" style="display:none;">
<h2 class="h2">Click to download your document:</h2> <h2 class="h2">Klick to download your document:</h2>
<button class="download-btn" id="downloadButton" onclick="fileDownload()">Download</button> <button class="download-btn" id="downloadButton" onclick="fileDownload()">Download</button>
</div> </div>
</div> </div>
+22 -22
View File
@@ -1,7 +1,7 @@
var languageOptions = { var languageOptions = {
"eng":{ "eng":{
"flagPath": "flags/united-kingdom-flag-png-large.jpg", "flagPath": "flags/united-kingdom-flag-png-large.jpg",
"labelKI": "Select AI:", "labelKI": "Select ki:",
"labelTranscription": "Select transcription:", "labelTranscription": "Select transcription:",
"labelLanguage": "Select language:", "labelLanguage": "Select language:",
"title": "Video to document", "title": "Video to document",
@@ -9,7 +9,7 @@ var languageOptions = {
"p1": "Drag and drop video file", "p1": "Drag and drop video file",
"fileName": "No video chosen", "fileName": "No video chosen",
"manualUploadBtn": "Search video", "manualUploadBtn": "Search video",
"checkbox_group": "Choose preferred document style:", "checkbox_group": "Choose prefered document style:",
"label_format": "Meeting report", "label_format": "Meeting report",
"label_summary": "Summary with timestamps", "label_summary": "Summary with timestamps",
"submitButton": "Submit", "submitButton": "Submit",
@@ -27,7 +27,7 @@ var languageOptions = {
"speakerResender": "Rewrite document", "speakerResender": "Rewrite document",
"downloadButton": "Download", "downloadButton": "Download",
"box1_p1": "---Starting---", "box1_p1": "---Starting---",
"box2_p2": "---Transcribing---", "box2_p2": "---Transkribing---",
"box3_p3": "---Document creation---", "box3_p3": "---Document creation---",
"labelType": "Select document type:", "labelType": "Select document type:",
@@ -46,13 +46,13 @@ var languageOptions = {
}, },
"de":{ "de":{
"flagPath": "flags/germany-flag-png-large.jpg", "flagPath": "flags/germany-flag-png-large.jpg",
"labelKI": "Wähle KI:", "labelKI": "Waehle KI:",
"labelTranscription": "Wähle Transkription:", "labelTranscription": "Waehle Transkription:",
"labelLanguage": "Wähle Sprache:", "labelLanguage": "Waehle Sprache:",
"title": "Video zu Dokument", "title": "Video zu Dokument",
"h1": "Video zu Dokument", "h1": "Video zu Dokument",
"p1": "Video per Drag & Drop ablegen", "p1": "Video per Drag & Drop ablegen",
"fileName": "Kein Video ausgewählt", "fileName": "Kein Video ausgewaehlt",
"manualUploadBtn": "Video suchen", "manualUploadBtn": "Video suchen",
"checkbox_group": "Bevorzugte Dokumentvarianten:", "checkbox_group": "Bevorzugte Dokumentvarianten:",
"label_format": "Meeting Bericht", "label_format": "Meeting Bericht",
@@ -64,7 +64,7 @@ var languageOptions = {
"step_nav4": "Schritt 4", "step_nav4": "Schritt 4",
"step_nav5": "Schritt 5", "step_nav5": "Schritt 5",
"step_nav6": "Schritt 6", "step_nav6": "Schritt 6",
"h2": "Lade dein Video hier hoch:", "h2": "Uploade dein Video hier:",
"labelSpeaker": "Wähle Sprecher:", "labelSpeaker": "Wähle Sprecher:",
"labelSpeakerAudio": "Ausgewählter Sprecher:", "labelSpeakerAudio": "Ausgewählter Sprecher:",
"labelSpeakerWriter": "Schreib Namen:", "labelSpeakerWriter": "Schreib Namen:",
@@ -72,27 +72,27 @@ var languageOptions = {
"speakerResender": "Überschreibe Dokument", "speakerResender": "Überschreibe Dokument",
"downloadButton": "Download", "downloadButton": "Download",
"box1_p1": "---Startet---", "box1_p1": "---Startet---",
"box2_p2": "---Transkribierung---", "box2_p2": "---Transkribing---",
"box3_p3": "---Dokument erstellen---", "box3_p3": "---Dokument kreieren---",
"labelType": "Wähle Dokumenttyp:", "labelType": "Wähle Dokumenttype:",
"customDocBtn": "Dokumenttypen verwalten", "customDocBtn": "Dokumenttypen verwalten",
"cd_h1": "Dokumenttypen verwalten", "cd_h1": "Dokumenttypen verwalten",
"cd_existingDocs": "Vorhandene Dokumente auswählen (optional):", "cd_existingDocs": "Vorhandene Dokumente auswählen (optional):",
"cd_docName": "Dokumentname", "cd_docName": "Dokument Name",
"docName": "Geben Sie hier den Dokumentnamen ein", "docName": "Geben Sie hier den Dokumentnamen ein",
"cd_promt": "Ihr Prompt:", "cd_promt": "Ihr Prompt:",
"prompt": "Geben Sie hier die Eingabeaufforderung für Ihr Dokument ein...", "prompt": "Geben Sie hier die Eingabeaufforderung für Ihr Dokument ein...",
"goBackBtn": "Zurück", "goBackBtn": "Zurück",
"deleteBtn": "Lösche Dokument", "deleteBtn": "Lösche Dokument",
"generateBtn": "Speichere Dokument", "generateBtn": "Speicher Dokument",
"newDoc": "-- Neues Dokument erstellen --" "newDoc": "-- Neues Dokument erstellen --"
}, },
"in":{ "in":{
"flagPath": "flags/india-flag-png-large.png", "flagPath": "flags/india-flag-png-large.png",
"labelKI": "KI का चयन करें:", "labelKI": "की का चयन करें:",
"labelTranscription": "प्रतिलेखन चुनें:", "labelTranscription": "प्रतिलेखन चुनें:",
"labelLanguage": "भाषा चुने:", "labelLanguage": "भाषा चुने:",
"title": "दस्तावेज़ के लिए वीडियो", "title": "दस्तावेज़ के लिए वीडियो",
"h1": "दस्तावेज़ के लिए वीडियो", "h1": "दस्तावेज़ के लिए वीडियो",
"p1": "वीडियो फ़ाइल खींचें और छोड़ें", "p1": "वीडियो फ़ाइल खींचें और छोड़ें",
@@ -101,7 +101,7 @@ var languageOptions = {
"checkbox_group": "पसंदीदा दस्तावेज़ शैली चुनें:", "checkbox_group": "पसंदीदा दस्तावेज़ शैली चुनें:",
"label_format": "बैठक रिपोर्ट", "label_format": "बैठक रिपोर्ट",
"label_summary": "टाइमस्टैम्प के साथ सारांश", "label_summary": "टाइमस्टैम्प के साथ सारांश",
"submitButton": "जमा करें", "submitButton": "जमा करना",
"step_nav1": "स्टेप 1", "step_nav1": "स्टेप 1",
"step_nav2": "स्टेप 2", "step_nav2": "स्टेप 2",
"step_nav3": "स्टेप 3", "step_nav3": "स्टेप 3",
@@ -110,11 +110,11 @@ var languageOptions = {
"step_nav6": "स्टेप 6", "step_nav6": "स्टेप 6",
"h2": "अपना वीडियो यहां अपलोड करें:", "h2": "अपना वीडियो यहां अपलोड करें:",
"labelSpeaker": "स्पीकर चुनें:", "labelSpeaker": "स्पीकर चुनें:",
"labelSpeakerAudio": "चयनित स्पीकर:", "labelSpeakerAudio": "चयनित वक्ता:",
"labelSpeakerWriter": "नाम लिखें:", "labelSpeakerWriter": "नाम लिखें:",
"speakerLocker": "स्पीकर का नाम बदलें", "speakerLocker": "स्पीकर का नाम बदलें",
"speakerResender": "दस्तावेज़ फिर से लिखें", "speakerResender": "दस्तावेज़ पुनः लिखें",
"downloadButton": "डाउनलोड करें", "downloadButton": "डाउनलोड करना",
"box1_p1": "---प्रारंभ---", "box1_p1": "---प्रारंभ---",
"box2_p2": "---प्रतिलेखन---", "box2_p2": "---प्रतिलेखन---",
"box3_p3": "---दस्तावेज़ निर्माण---", "box3_p3": "---दस्तावेज़ निर्माण---",
@@ -127,10 +127,10 @@ var languageOptions = {
"docName": "यहां दस्तावेज़ का नाम दर्ज करें", "docName": "यहां दस्तावेज़ का नाम दर्ज करें",
"cd_promt": "आपका संकेत:", "cd_promt": "आपका संकेत:",
"prompt": "अपने दस्तावेज़ के लिए प्रॉम्प्ट यहां टाइप करें...", "prompt": "अपने दस्तावेज़ के लिए प्रॉम्प्ट यहां टाइप करें...",
"goBackBtn": "वापस जाएं", "goBackBtn": "वापस करना",
"deleteBtn": "दस्तावेज़ हटाए", "deleteBtn": "दस्तावेज़ हटाए",
"generateBtn": "दस्तावेज़ सहेजें", "generateBtn": "दस्तावेज़ सहेजें",
"newDoc": "-- नया दस्तावेज़ बनाए --" "newDoc": "-- नया दस्तावेज़ बनाए --"
} }
+12 -23
View File
@@ -432,30 +432,19 @@ function setSpeakerAudiosValue(valy) {
//Function to rewrite the speaker name in the json //Function to rewrite the speaker name in the json
function rewriteSpeakerName() { function rewriteSpeakerName() {
try { try {
const select = document.getElementById("cur_speaker"); var tempy = document.getElementById("cur_speaker").value;
const newName = document.getElementById("newSpeaker").value.trim(); speakerAudios[tempy].name = document.getElementById("newSpeaker").value;
loadSpeakerOptions(speakerAudios);
if (!newName) {
alert("Please enter a new speaker name");
return;
}
const selectedIndex = select.selectedIndex;
const selectedValue = select.value;
// Update speakerAudios data
speakerAudios[selectedValue].name = newName;
// Update the specific option text and keep value
select.options[selectedIndex].text = newName;
select.options[selectedIndex].value = selectedValue;
// Keep it selected
select.selectedIndex = selectedIndex;
console.log("Speaker renamed:", newName);
} catch (error) { } catch (error) {
console.log("Error renaming speaker:", error); console.log("\n\n\n" + error + "\n\n\n")
}
}
//Function to send the json with the given names back to the program to rewrite the document file
function sendSpeakerPackages() {
try {
window.submitSpeaker.speaker_submit(speakerAudios);
} catch (error) {
console.log(error);
} }
} }
+22 -127
View File
@@ -11,12 +11,12 @@ body {
} }
#h1 { #h1 {
position: static; position: absolute;
transform: none; left: 50%;
top: 50%;
transform: translate(-50%, -50%);
margin: 0; margin: 0;
z-index: 20; z-index: 20;
flex: 1;
text-align: center;
} }
#h1-wrapper { #h1-wrapper {
@@ -30,26 +30,6 @@ body {
margin-bottom: 10px; margin-bottom: 10px;
display: flex; display: flex;
align-items: center; align-items: center;
justify-content: space-between;
padding: 0 20px;
box-sizing: border-box;
}
.gui-language {
position: absolute;
right: 20px;
top: 50%;
transform: translateY(-50%);
z-index: 100;
pointer-events: auto;
}
#language_option {
padding: 8px 12px;
border-radius: 4px;
border: 1px solid #ccc;
font-size: 14px;
cursor: pointer;
} }
.upload-container { .upload-container {
@@ -105,6 +85,7 @@ body {
#previewThumbnail { #previewThumbnail {
width: 150px; width: 150px;
height: 100px; height: 100px;
/*border: 1px dashed black;*/
} }
.custom-btn { .custom-btn {
@@ -127,9 +108,8 @@ body {
background-color: #0056b3; background-color: #0056b3;
} }
.step h2 { #step2 {
width: 100%; gap: 25px;
text-align: center;
} }
.KI-wrapper { .KI-wrapper {
@@ -376,10 +356,7 @@ input[type="file"] {
#ai_type, #ai_type,
#transkript_type, #transkript_type,
#language_option { #language_option {
padding: 8px 12px; padding: 3px;
border-radius: 4px;
border: 1px solid #ccc;
font-size: 14px;
} }
.labelDiv { .labelDiv {
@@ -427,6 +404,7 @@ input[type="file"] {
.step { .step {
margin-top: 40px; margin-top: 40px;
margin-bottom: 40px; margin-bottom: 40px;
;
display: flex; display: flex;
flex-direction: column; flex-direction: column;
min-height: 425px; min-height: 425px;
@@ -601,27 +579,7 @@ li {
transition: all 0.3s ease; transition: all 0.3s ease;
} }
#step2,
#step2 {
font-size: larger;
align-items: center;
}
.step2-form {
width: 100%;
max-width: 420px;
display: flex;
flex-direction: column;
gap: 24px; /* DAS ist dein Spacing */
}
.step2-row {
display: flex;
flex-direction: column;
gap: 6px;
}
#step3, #step3,
#step5 { #step5 {
font-size: larger; font-size: larger;
@@ -632,7 +590,7 @@ li {
} }
#step5 { #step5 {
align-items: center; align-items: flex-start;
} }
.button-group { .button-group {
@@ -656,84 +614,27 @@ li {
font-size: 14px; font-size: 14px;
} }
.h2 {
font-size: 25px;
}
.speaker-container {
width: 100%;
max-width: 700px;
margin-top: 30px;
}
.speaker-table {
width: 100%;
border-collapse: collapse;
background: white;
}
.speaker-table tbody tr {
display: flex;
align-items: center;
gap: 20px;
margin-bottom: 25px;
padding: 10px 0;
}
.label-cell {
flex: 0 0 150px;
text-align: left;
}
.label-cell label {
font-weight: 400;
display: block;
}
.input-cell {
flex: 1;
}
#cur_speaker,
#newSpeaker {
width: 100%;
padding: 10px;
border-radius: 6px;
border: 1px solid #ccc;
font-size: 14px;
box-sizing: border-box;
}
#speakerAudioViewer {
width: 100%;
height: 35px;
border-radius: 6px;
}
.speaker-button-group {
display: flex;
gap: 15px;
justify-content: center;
margin-top: 30px;
}
#speakerLocker, #speakerLocker,
#speakerResender { #speakerResender {
padding: 12px 25px; padding: 10px 20px;
margin: 20px auto;
background-color: #007BFF; background-color: #007BFF;
color: white; color: white;
border: none; border: none;
border-radius: 8px; border-radius: 8px;
cursor: pointer; cursor: pointer;
font-size: 14px; font-size: 14px;
font-weight: 500;
transition: background-color 0.2s;
} }
#speakerLocker:hover, .h2 {
#speakerResender:hover { font-size: 25px;
background-color: #0056b3; }
.speakerView,
.speakerAudio,
.speakerWrite {
margin-top: auto;
margin-bottom: auto;
} }
.container { .container {
@@ -802,9 +703,3 @@ button:hover {
color: #333; color: #333;
word-break: break-word; word-break: break-word;
} }
.container input,
.container textarea,
.container select {
width: 100%;
}
+13 -33
View File
@@ -168,39 +168,9 @@ electron.ipcMain.on("file_submit", async (event, args) => {
throw new Error("Unknown document type: " + args.document.type); throw new Error("Unknown document type: " + args.document.type);
} }
electron.ipcMain.on("file_download", async (event) => { console.log(args);
try { let audiopath = "";
if (!globalFinalHtmlPath) { let transcriptpath = "";
throw new Error("No document generated yet");
}
const format = String(globalArgs?.document?.outputType || "")
.replace('.', '')
.toLowerCase();
if (!format) {
throw new Error("No output format selected");
}
const outputPath = await mapFunctions
.get("htmlDocumentConverter")
.convert({
inputPath: globalFinalHtmlPath,
format,
showDialog: true
});
event.sender.send("download_success", {
path: outputPath,
format
});
} catch (err) {
console.error("file_download failed:", err);
event.sender.send("error", err.message || String(err));
}
});
console.log("\n\n Running the Video to Audio Extractor"); console.log("\n\n Running the Video to Audio Extractor");
// This code handles the Video to Audio extraction module call // This code handles the Video to Audio extraction module call
@@ -316,6 +286,16 @@ electron.ipcMain.on("file_download", async (event) => {
} }
}); });
electron.ipcMain.on("file_download", async () => {
await mapFunctions
.get("htmlDocumentConverter")
.convert({
inputPath: globalFinalHtmlPath,
format: globalArgs.document.outputType,
showDialog: true,
});
});
electron.ipcMain.on("speaker_submit", async (event, args) => { electron.ipcMain.on("speaker_submit", async (event, args) => {
console.log("\n\n\nJa also hier kam was an \n\n\n"); console.log("\n\n\nJa also hier kam was an \n\n\n");
console.log(args); console.log(args);
+144 -183
View File
@@ -1,37 +1,35 @@
const fs = require("fs"); const fs = require('fs');
const path = require("path"); const path = require('path');
const puppeteer = require("puppeteer"); const puppeteer = require('puppeteer');
const htmlToDocx = require("html-to-docx"); const htmlToDocx = require('html-to-docx');
const { execSync } = require("child_process"); const { execSync } = require('child_process');
const os = require("os"); const os = require('os');
const outputDir = path.join(__dirname, "../../../storage/documents"); const outputDir = path.join(__dirname, "../../../storage/documents");
if (!fs.existsSync(outputDir)) { if (!fs.existsSync(outputDir)) {
fs.mkdirSync(outputDir, { recursive: true }); fs.mkdirSync(outputDir, { recursive: true });
} }
async function showSaveDialog(defaultName, format) { async function showSaveDialog(defaultName, format) {
const platform = os.platform(); const platform = os.platform();
if (platform === "darwin") { if (platform === 'darwin') {
// macOS // macOS
const applescript = ` const applescript = `
set defaultName to "${defaultName}.${format}" set defaultName to "${defaultName}.${format}"
set theFile to choose file name with prompt "Dokument speichern als:" default name defaultName set theFile to choose file name with prompt "Dokument speichern als:" default name defaultName
POSIX path of theFile POSIX path of theFile
`; `;
try { try {
const result = execSync(`osascript -e '${applescript}'`, { const result = execSync(`osascript -e '${applescript}'`, { encoding: 'utf8' });
encoding: "utf8", return result.trim();
}); } catch (err) {
return result.trim(); if (err.status === 1) return null; // User canceled
} catch (err) { throw err;
if (err.status === 1) return null; // User canceled }
throw err; } else if (platform === 'win32') {
}
} else if (platform === "win32") {
const safeName = decodeURIComponent(defaultName); const safeName = decodeURIComponent(defaultName);
const powershell = ` const powershell = `
@@ -45,192 +43,155 @@ async function showSaveDialog(defaultName, format) {
`; `;
try { try {
const result = execSync(
`powershell -NoProfile -Command "${powershell.replace(/\r?\n/g, " ")}"`,
{ encoding: "utf8" },
);
return result.trim() || null;
} catch (err) {
if (err.status === 1) return null; // User cancelled
throw new Error("Save dialog failed: " + err.message);
}
} else {
// Linux - zenity oder kdialog
try {
const result = execSync(
`zenity --file-selection --save --confirm-overwrite --filename="${defaultName}.${format}"`,
{ encoding: "utf8" },
);
return result.trim();
} catch (err) {
try {
const result = execSync( const result = execSync(
`kdialog --getsavefilename . "${defaultName}.${format}"`, `powershell -NoProfile -Command "${powershell.replace(/\r?\n/g, ' ')}"`,
{ encoding: "utf8" }, { encoding: 'utf8' }
); );
return result.trim(); return result.trim() || null;
} catch (err2) { } catch (err) {
// Fallback if (err.status === 1) return null; // User cancelled
return path.join(os.homedir(), "Downloads", `${defaultName}.${format}`); throw new Error("Save dialog failed: " + err.message);
} }
} else {
// Linux - zenity oder kdialog
try {
const result = execSync(
`zenity --file-selection --save --confirm-overwrite --filename="${defaultName}.${format}"`,
{ encoding: 'utf8' }
);
return result.trim();
} catch (err) {
try {
const result = execSync(
`kdialog --getsavefilename . "${defaultName}.${format}"`,
{ encoding: 'utf8' }
);
return result.trim();
} catch (err2) {
// Fallback
return path.join(os.homedir(), 'Downloads', `${defaultName}.${format}`);
}
}
} }
}
} }
const module_exports = { const module_exports = {
name: "htmlDocumentConverter", name: "htmlDocumentConverter",
type: "converter", type: "converter",
displayname: "HTML Document Converter", displayname: "HTML Document Converter",
description: "Converts LLM-generated HTML to PDF, DOCX, TXT, or HTML", description: "Converts LLM-generated HTML to PDF, DOCX, TXT, or HTML",
/** /**
* Main conversion function * Main conversion function
* @param {Object} options * @param {Object} options
* @param {string} options.inputPath - Path to the HTML input * @param {string} options.inputPath - Path to the HTML input
* @param {string} options.format - 'pdf' | 'docx' | 'html' | 'txt' * @param {string} options.format - 'pdf' | 'docx' | 'html' | 'txt'
* @param {string} [options.outputName] - Optional output filename (without extension) * @param {string} [options.outputName] - Optional output filename (without extension)
* @param {boolean} [options.showDialog] - Show save dialog (default: false in module mode, true in CLI mode) * @param {boolean} [options.showDialog] - Show save dialog (default: false in module mode, true in CLI mode)
*/ */
async convert({ inputPath, format = "pdf", outputName, showDialog = false }) { async convert({ inputPath, format = 'pdf', outputName, showDialog = false }) {
format = format.toLowerCase().replace(".", ""); // <-- FIX if (!fs.existsSync(inputPath)) {
throw new Error(`Input file not found: ${inputPath}`);
}
if (!["pdf", "docx", "html", "txt"].includes(format)) { const ext = path.extname(inputPath).toLowerCase();
throw new Error(`Unsupported format: ${format}`); const baseName = outputName || path.basename(inputPath, ext);
}
if (!fs.existsSync(inputPath)) {
throw new Error(`Input file not found: ${inputPath}`);
}
const ext = path.extname(inputPath).toLowerCase(); let outputFile;
const baseName = outputName || path.basename(inputPath, ext);
let outputFile; if (showDialog) {
// Zeige nativen Dialog
outputFile = await showSaveDialog(baseName, format);
if (!outputFile) {
console.log('Speichervorgang abgebrochen.');
return null;
}
} else {
// Nutze Standard-Ausgabeverzeichnis
outputFile = path.join(outputDir, `${baseName}.${format.toLowerCase()}`);
}
if (showDialog) { let htmlContent = fs.readFileSync(inputPath, 'utf8');
// Zeige nativen Dialog
outputFile = await showSaveDialog(baseName, format);
if (!outputFile) {
console.log("Speichervorgang abgebrochen.");
return null;
}
} else {
// Nutze Standard-Ausgabeverzeichnis
outputFile = path.join(outputDir, `${baseName}.${format.toLowerCase()}`);
}
let htmlContent = fs.readFileSync(inputPath, "utf8"); // Remove <think> tags if present
htmlContent = htmlContent.replace(/<think>[\s\S]*?<\/think>/gi, '');
// Remove <think> tags if present switch (format.toLowerCase()) {
htmlContent = htmlContent.replace(/<think>[\s\S]*?<\/think>/gi, ""); case 'html':
fs.writeFileSync(outputFile, htmlContent, 'utf8');
break;
case 'pdf':
await this.htmlToPDF(htmlContent, outputFile);
break;
case 'docx':
await this.htmlToDOCX(htmlContent, outputFile);
break;
case 'txt':
fs.writeFileSync(outputFile, this.htmlToTXT(htmlContent), 'utf8');
break;
default:
throw new Error(`Unsupported format: ${format}`);
}
switch (format.toLowerCase()) { console.log(`Erfolgreich gespeichert: ${outputFile}`);
case "html": return outputFile;
fs.writeFileSync(outputFile, htmlContent, "utf8"); },
break;
case "pdf":
await this.htmlToPDF(htmlContent, outputFile);
break;
case "docx":
await this.htmlToDOCX(htmlContent, outputFile);
break;
case "txt":
fs.writeFileSync(outputFile, this.htmlToTXT(htmlContent), "utf8");
break;
default:
throw new Error(`Unsupported format: ${format}`);
}
console.log(`Erfolgreich gespeichert: ${outputFile}`); // HTML → PDF
return outputFile; async htmlToPDF(html, outputPath) {
}, const browser = await puppeteer.launch({
headless: true,
// HTML → PDF args: ['--no-sandbox', '--disable-setuid-sandbox']
async htmlToPDF(html, outputPath) { });
let browser; const page = await browser.newPage();
try { await page.setContent(html, { waitUntil: 'networkidle0' });
browser = await puppeteer.launch({ await page.pdf({
headless: true, path: outputPath,
args: ["--no-sandbox", "--disable-setuid-sandbox"], format: 'A4',
}); printBackground: true,
margin: { top: '20mm', right: '20mm', bottom: '20mm', left: '20mm' }
const page = await browser.newPage(); });
await page.setContent(html, { waitUntil: "networkidle0" });
await page.pdf({
path: outputPath,
format: "A4",
printBackground: true,
margin: {
top: "20mm",
right: "20mm",
bottom: "20mm",
left: "20mm",
},
});
} finally {
if (browser) {
await browser.close(); await browser.close();
} },
}
},
// HTML → DOCX // HTML → DOCX
async htmlToDOCX(html, outputPath) { async htmlToDOCX(html, outputPath) {
try { const buffer = await htmlToDocx(html);
// htmltodocx library converts HTML string into a Word .docx buffer fs.writeFileSync(outputPath, buffer);
// Usage from htmltodocx docs: },
// await HTMLtoDOCX(htmlString, headerHTMLString, documentOptions, footerHTMLString) [oai_citation:0‡GitHub](https://github.com/privateOmega/html-to-docx?utm_source=chatgpt.com)
const buffer = await htmlToDocx(html, null, {
table: { row: { cantSplit: true } },
});
fs.writeFileSync(outputPath, buffer);
} catch (err) {
throw new Error(`DOCX conversion failed: ${err.message}`);
}
},
// HTML → TXT // HTML → TXT (rudimentär)
htmlToTXT(html) { htmlToTXT(html) {
// A decent plain text conversion: strip tags and collapse whitespace return html.replace(/<[^>]*>/g, '').replace(/\s+\n/g, '\n').trim();
// If you want more advanced extraction consider using a library like `html-to-text` or `strip-html` [oai_citation:1‡GitHub](https://github.com/html-to-text/node-html-to-text?utm_source=chatgpt.com) }
return (
html
// Remove all tags
.replace(/<[^>]+>/g, "")
// Convert multiple whitespace into single spaces
.replace(/\s+/g, " ")
.trim()
);
},
}; };
module.exports = module_exports; module.exports = module_exports;
// CLI usage mit Dialog // CLI usage mit Dialog
if (require.main === module) { if (require.main === module) {
(async () => { (async () => {
const args = process.argv.slice(2); const args = process.argv.slice(2);
if (args.length < 1) { if (args.length < 1) {
console.log("Usage: node htmlDocumentConverter.js <input.html> [format]"); console.log('Usage: node htmlDocumentConverter.js <input.html> [format]');
console.log("Formats: pdf (default), docx, html, txt"); console.log('Formats: pdf (default), docx, html, txt');
console.log(""); console.log('');
console.log( console.log('Ein nativer "Speichern unter" Dialog wird automatisch geöffnet.');
'Ein nativer "Speichern unter" Dialog wird automatisch geöffnet.', process.exit(1);
); }
process.exit(1);
}
const inputPath = args[0]; const inputPath = args[0];
const format = args[1] || "pdf"; const format = args[1] || 'pdf';
try { try {
await module_exports.convert({ await module_exports.convert({
inputPath, inputPath,
format, format,
showDialog: true, showDialog: true
}); });
} catch (err) { } catch (err) {
console.error("Konvertierung fehlgeschlagen:", err.message); console.error('Konvertierung fehlgeschlagen:', err.message);
process.exit(1); process.exit(1);
} }
})(); })();
} }