Compare commits

...

17 Commits

Author SHA1 Message Date
MikeHughes-BIN b6276b005e Removed Parakeet again as it requires PyTorch which is too large for this project 2026-01-18 19:13:18 +01:00
MikeHughes-BIN 8f2024df14 Refactor HTML structure for improved readability and maintainability 2026-01-18 19:12:04 +01:00
MikeHughes-BIN 0f548b3012 Add function to send speaker packages and handle errors 2026-01-18 19:05:55 +01:00
Hughes, Mike 7fbf0c59d1 Merge branch 'feature/ui-test' into 'develop'
Feature/ui test

See merge request proj-wise2526-video2document/video2document!96
2026-01-18 18:02:35 +01:00
Hughes, Mike ef20a08d9f Merge branch 'fix/speaker-selection' into 'develop'
Speakers that have been renamed will stay in the selection as requested by Oliver

See merge request proj-wise2526-video2document/video2document!97
2026-01-18 18:00:53 +01:00
MikeHughes-BIN 9a3f84efc8 Speakers that have been renamed will stay in the selection as requested by Oliver 2026-01-18 17:57:18 +01:00
Hughes, Mike 6813b45c80 Merge branch 'develop' into 'feature/ui-test'
# Conflicts:
#   main.js
2026-01-18 17:38:36 +01:00
MikeHughes-BIN 773e8b471c Remove notification for successful file download in ipcMain handler 2026-01-18 17:37:47 +01:00
MikeHughes-BIN 013c9b5f2c Refactor file download handling to improve error management and user notifications 2026-01-18 17:32:29 +01:00
MikeHughes-BIN 18e791d56e Refactor code formatting and improve error handling in htmlDocumentConverter 2026-01-18 17:28:41 +01:00
MikeHughes-BIN 1ed386fcf4 Enhance file download handling and format validation in htmlDocumentConverter 2026-01-18 17:02:29 +01:00
MikeHughes-BIN c98d7761b2 Refactor UI elements for step selection and speaker management; improve styling and accessibility 2026-01-18 16:55:42 +01:00
Hughes, Mike d09b75a6cd Merge branch 'feature/ui-test' into 'develop'
Feature/ui test

See merge request proj-wise2526-video2document/video2document!95
2026-01-17 14:35:38 +01:00
Verena Schulz 0427056f65 Small changes 2026-01-17 00:11:57 +01:00
Verena Schulz 8076fe92f5 Integrated customDocBtn better with css into the menu 2026-01-17 00:11:37 +01:00
Verena Schulz c2e6c4a186 Style changes for custom document css 2026-01-16 23:59:30 +01:00
Verena Schulz 826381d858 Added hover effects for other document types 2026-01-16 23:54:31 +01:00
8 changed files with 808 additions and 545 deletions
+204 -87
View File
@@ -1,15 +1,18 @@
<!DOCTYPE html>
<!doctype html>
<html lang="de">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title id="title">Video to document</title>
<link rel="stylesheet" href="style.css">
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/lc-select@1.3.0/themes/light.css">
</head>
<body>
<link rel="stylesheet" href="style.css" />
<link
rel="stylesheet"
href="https://cdn.jsdelivr.net/npm/lc-select@1.3.0/themes/light.css"
/>
</head>
<div id="h1-wrapper">
<body>
<div id="h1-wrapper">
<section class="p-menu1">
<nav id="navbar" class="navigation" role="navigation">
<input id="toggle1" type="checkbox" />
@@ -20,7 +23,9 @@
</label>
<nav class="menu1">
<button id="customDocBtn" onclick="showCD()">Manage document types</button>
<button id="customDocBtn" onclick="showCD()">
Manage document types
</button>
<a href="help_page.html" class="li1">Help</a>
</nav>
</nav>
@@ -28,45 +33,53 @@
<h1 id="h1">Video to document</h1>
<div class="gui-language">
<!-- to do: Ausprobieren mit li, a oder button, im Notfall ohne Flaggen Icons, kein hover-->
<select name="language_option" id="language_option"></select>
</div>
<div class="gui-language">
<select name="language_option" id="language_option"></select>
</div>
</div>
</div>
<div class="step-nav">
<div class="step-nav">
<div class="step-item active" data-step="1" id="step_nav1">1. Step</div>
<div class="step-item" data-step="2" id="step_nav2">2. Step</div>
<div class="step-item" data-step="3" id="step_nav3">3. Step</div>
<div class="step-item" data-step="4" id="step_nav4">4. Step</div>
<div class="step-item" data-step="5" id="step_nav5">5. Step</div>
<div class="step-item" data-step="6" id="step_nav6">6. Step</div>
</div>
</div>
<div id="middleContainerWrapper" class="middle-container-wrapper">
<div id="middleContainerWrapper" class="middle-container-wrapper">
<button id="prevBtn" class="navBtn" disabled>&larr;</button>
<!-- Visible middle part-->
<!-- Visible middle part-->
<div class="mitte" id="mitte">
<!--Costum document section-->
<div class="container" id="cdContainer" style="display:none;">
<div class="container" id="cdContainer" style="display: none">
<h1 id="cd_h1">Manage document types</h1>
<label for="existingDocs" id="cd_existingDocs">Select existing documents (optional):</label>
<label for="existingDocs" id="cd_existingDocs"
>Select existing documents (optional):</label
>
<!--Drop Down-->
<select name="existingDocs" id="existingDocs">
<option value="newDoc" id="newDoc">-- Create new document --</option>
<option value="newDoc" id="newDoc">
-- Create new document --
</option>
</select>
<div id="docNameWrapper">
<label for="docName" id="cd_docName">Document name:</label>
<input type="text" id="docName" placeholder="Enter the document name here">
<input
type="text"
id="docName"
placeholder="Enter the document name here"
/>
</div>
<label for="prompt" id="cd_promt">Your prompt:</label>
<textarea id="prompt" placeholder="Type the prompt for your document here..."></textarea>
<textarea
id="prompt"
placeholder="Type the prompt for your document here..."
></textarea>
<div class="buttons">
<button id="goBackBtn">Return</button>
@@ -82,20 +95,22 @@
<h2 class="h2">Upload your video here:</h2>
<div class="upload-container" id="uploadContainer">
<p id="p1">Drag and drop video file</p>
<video id="previewThumbnail" autoplay="false">
</video>
<video id="previewThumbnail" autoplay="false"></video>
<div class="file-name" id="fileName">No video chosen</div>
<div id="thumbnailContainer">
<img id="thumbnailImage" style="display:none;">
<img id="thumbnailImage" style="display: none" />
</div>
<button class="custom-btn" id="manualUploadBtn">Search video</button>
<input type="file" id="videoUpload" accept="video/*">
<button class="custom-btn" id="manualUploadBtn">
Search video
</button>
<input type="file" id="videoUpload" accept="video/*" />
</div>
</div>
<!-- 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>
<div class="step2-form">
<div class="KI-wrapper">
<label id="labelKI">Select ki:</label>
<select name="ai_type" id="ai_type"></select>
@@ -110,7 +125,7 @@
<label id="labelType">Select type:</label>
<select name="output_type" id="output_type">
<option value="pdf">.pdf</option>
<option value="word">.docx</option>
<option value="word">.word</option>
<option value="txt">.txt</option>
</select>
</div>
@@ -118,106 +133,208 @@
<div class="language-wrapper">
<label id="labelLanguage">Select language:</label>
<select name="document_language_option" id="document_language_option">
</select>
<select
name="document_language_option"
id="document_language_option"
></select>
</div>
</div>
</div>
<!-- Here starts code from step 3-->
<!-- Hover Effekt für Dokumentenvorschau, Fragezeichen hinter Text, drüber hoven zeigt Beispieldokument -->
<div class="step" id="step3" style="display:none;">
<div class="step" id="step3" style="display: none">
<div class="checkbox-group">
<h2 class="h2">Choose prefered document style:</h2>
<div class="checkbox-container">
<input type="checkbox" name ="docFormat" id="docFormat" value="followup-report">
<input
type="checkbox"
name="docFormat"
id="docFormat"
value="followup-report"
/>
<label id="label_format" for="docFormat">Follow-up Report</label>
<div class="figure1">
<img class="img-icon" src="icons/question-mark-button-icon--free-clip-art-30.png">
<img class="img-hover1" src="flags/germany-flag-png-large.jpg">
<img
class="img-icon"
src="icons/question-mark-button-icon--free-clip-art-30.png"
/>
<img
class="img-hover1"
src="flags/germany-flag-png-large.jpg"
/>
</div>
</div>
<div class="checkbox-container">
<input type="checkbox" name="docFormat" id="docFormatSummary1" value="agenda">
<input
type="checkbox"
name="docFormat"
id="docFormatSummary1"
value="agenda"
/>
<label id="label_summary" for="docFormatSummary">Agenda</label>
<img class="img-icon" src="icons/question-mark-button-icon--free-clip-art-30.png">
<div class="figure2">
<img
class="img-icon"
src="icons/question-mark-button-icon--free-clip-art-30.png"
/>
<img class="img-hover2" src="flags/india-flag-png-large.png" />
</div>
</div>
<div class="checkbox-container">
<input type="checkbox" name="docFormat" id="docFormatSummary2" value="result-protocol">
<label id="label_summary" for="docFormatSummary">Resultprotocol</label>
<img class="img-icon" src="icons/question-mark-button-icon--free-clip-art-30.png">
<input
type="checkbox"
name="docFormat"
id="docFormatSummary2"
value="result-protocol"
/>
<label id="label_summary" for="docFormatSummary"
>Resultprotocol</label
>
<div class="figure3">
<img
class="img-icon"
src="icons/question-mark-button-icon--free-clip-art-30.png"
/>
<img
class="img-hover3"
src="flags/united-kingdom-flag-png-large.jpg"
/>
</div>
</div>
<div class="checkbox-container">
<input type="checkbox" name="docFormat" id="docFormatSummary3" value="sprint-planning">
<label id="label_summary" for="docFormatSummary">Sprint Planning Note</label>
<img class="img-icon" src="icons/question-mark-button-icon--free-clip-art-30.png">
<input
type="checkbox"
name="docFormat"
id="docFormatSummary3"
value="sprint-planning"
/>
<label id="label_summary" for="docFormatSummary"
>Sprint Planning Note</label
>
<div class="figure4">
<img
class="img-icon"
src="icons/question-mark-button-icon--free-clip-art-30.png"
/>
<img
class="img-hover4"
src="flags/germany-flag-png-large.jpg"
/>
</div>
</div>
<div class="checkbox-container">
<input type="checkbox" name="docFormat" id="docFormatCustom" value="custom">
<select name="customDocumentTypes" id="customDocumentTypes">
</select>
<input
type="checkbox"
name="docFormat"
id="docFormatCustom"
value="custom"
/>
<select
name="customDocumentTypes"
id="customDocumentTypes"
></select>
</div>
</div>
</div>
<!-- Here starts code from step 4-->
<div class="step" id="step4" style="display:none;">
<div class="step" id="step4" style="display: none">
<h2 class="h2">Click to submit:</h2>
<button class="submit-btn" id="submitButton" onclick="checkBoxes()" disabled>Submit</button>
<button
class="submit-btn"
id="submitButton"
onclick="checkBoxes()"
disabled
>
Submit
</button>
<div class="testy" id="testy">
<div class="box2" id="box1">
</div>
<div class="box2" id="box1"></div>
<p id="box1_p1">---Starting---</p>
<div class="box2" id="box2">
</div>
<div class="box2" id="box2"></div>
<p id="box2_p2">---Transkribing---</p>
<div class="box2" id="box3">
</div>
<div class="box2" id="box3"></div>
<p id="box3_p3">---Document creation---</p>
<div class="box2" id="box4">
</div>
<div class="box2" id="box4"></div>
</div>
</div>
<!-- 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>
<div class="speakerView" id="speakerView">
<label id="labelSpeaker">Select Speaker:</label>
<select name="cur_speaker" id="cur_speaker">
</select>
</div>
<div class="speakerAudio" id="speakerAutio">
<label id="labelSpeakerAudio">Selected Speaker:</label>
<div class="speaker-container">
<table class="speaker-table">
<tbody>
<tr>
<td class="label-cell">
<label id="labelSpeaker" for="cur_speaker"
>Select Speaker:</label
>
</td>
<td class="input-cell">
<select name="cur_speaker" id="cur_speaker"></select>
</td>
</tr>
<tr>
<td class="label-cell">
<label id="labelSpeakerAudio">Speaker Audio:</label>
</td>
<td class="input-cell">
<audio controls id="speakerAudioViewer">
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 class="speakerWrite" id="speakerWrite">
<label id="labelSpeakerWriter">Write name:</label>
<input type="text" id="newSpeaker">
</div>
<div class="speakerButton-group">
<button id="speakerLocker" onclick="rewriteSpeakerName()">Rename Speaker</button>
<button id="speakerResender" onclick="sendSpeakerPackages()">Rewrite document</button>
</div>
</div>
<!-- Here starts code from step 6-->
<div class="step" id="step6" style="display:none;">
<h2 class="h2">Klick to download your document:</h2>
<button class="download-btn" id="downloadButton" onclick="fileDownload()">Download</button>
<div class="step" id="step6" style="display: none">
<h2 class="h2">Click to download your document:</h2>
<button
class="download-btn"
id="downloadButton"
onclick="fileDownload()"
>
Download
</button>
</div>
</div>
<button id ="nextBtn" class="navBtn">&rarr;</button>
</div>
<script src="https://cdn.jsdelivr.net/npm/lc-select@1.3.0/lc_select.min.js"></script>
<script src="languages.js"></script>
<script src="script.js"></script>
<script src="./renderer.js"></script>
</body>
<button id="nextBtn" class="navBtn">&rarr;</button>
</div>
<script src="https://cdn.jsdelivr.net/npm/lc-select@1.3.0/lc_select.min.js"></script>
<script src="languages.js"></script>
<script src="script.js"></script>
<script src="./renderer.js"></script>
</body>
</html>
+22 -22
View File
@@ -1,7 +1,7 @@
var languageOptions = {
"eng":{
"flagPath": "flags/united-kingdom-flag-png-large.jpg",
"labelKI": "Select ki:",
"labelKI": "Select AI:",
"labelTranscription": "Select transcription:",
"labelLanguage": "Select language:",
"title": "Video to document",
@@ -9,7 +9,7 @@ var languageOptions = {
"p1": "Drag and drop video file",
"fileName": "No video chosen",
"manualUploadBtn": "Search video",
"checkbox_group": "Choose prefered document style:",
"checkbox_group": "Choose preferred document style:",
"label_format": "Meeting report",
"label_summary": "Summary with timestamps",
"submitButton": "Submit",
@@ -27,7 +27,7 @@ var languageOptions = {
"speakerResender": "Rewrite document",
"downloadButton": "Download",
"box1_p1": "---Starting---",
"box2_p2": "---Transkribing---",
"box2_p2": "---Transcribing---",
"box3_p3": "---Document creation---",
"labelType": "Select document type:",
@@ -46,13 +46,13 @@ var languageOptions = {
},
"de":{
"flagPath": "flags/germany-flag-png-large.jpg",
"labelKI": "Waehle KI:",
"labelTranscription": "Waehle Transkription:",
"labelLanguage": "Waehle Sprache:",
"labelKI": "Wähle KI:",
"labelTranscription": "Wähle Transkription:",
"labelLanguage": "Wähle Sprache:",
"title": "Video zu Dokument",
"h1": "Video zu Dokument",
"p1": "Video per Drag & Drop ablegen",
"fileName": "Kein Video ausgewaehlt",
"fileName": "Kein Video ausgewählt",
"manualUploadBtn": "Video suchen",
"checkbox_group": "Bevorzugte Dokumentvarianten:",
"label_format": "Meeting Bericht",
@@ -64,7 +64,7 @@ var languageOptions = {
"step_nav4": "Schritt 4",
"step_nav5": "Schritt 5",
"step_nav6": "Schritt 6",
"h2": "Uploade dein Video hier:",
"h2": "Lade dein Video hier hoch:",
"labelSpeaker": "Wähle Sprecher:",
"labelSpeakerAudio": "Ausgewählter Sprecher:",
"labelSpeakerWriter": "Schreib Namen:",
@@ -72,27 +72,27 @@ var languageOptions = {
"speakerResender": "Überschreibe Dokument",
"downloadButton": "Download",
"box1_p1": "---Startet---",
"box2_p2": "---Transkribing---",
"box3_p3": "---Dokument kreieren---",
"labelType": "Wähle Dokumenttype:",
"box2_p2": "---Transkribierung---",
"box3_p3": "---Dokument erstellen---",
"labelType": "Wähle Dokumenttyp:",
"customDocBtn": "Dokumenttypen verwalten",
"cd_h1": "Dokumenttypen verwalten",
"cd_existingDocs": "Vorhandene Dokumente auswählen (optional):",
"cd_docName": "Dokument Name",
"cd_docName": "Dokumentname",
"docName": "Geben Sie hier den Dokumentnamen ein",
"cd_promt": "Ihr Prompt:",
"prompt": "Geben Sie hier die Eingabeaufforderung für Ihr Dokument ein...",
"goBackBtn": "Zurück",
"deleteBtn": "Lösche Dokument",
"generateBtn": "Speicher Dokument",
"generateBtn": "Speichere Dokument",
"newDoc": "-- Neues Dokument erstellen --"
},
"in":{
"flagPath": "flags/india-flag-png-large.png",
"labelKI": "की का चयन करें:",
"labelKI": "KI का चयन करें:",
"labelTranscription": "प्रतिलेखन चुनें:",
"labelLanguage": "भाषा चुने:",
"labelLanguage": "भाषा चुने:",
"title": "दस्तावेज़ के लिए वीडियो",
"h1": "दस्तावेज़ के लिए वीडियो",
"p1": "वीडियो फ़ाइल खींचें और छोड़ें",
@@ -101,7 +101,7 @@ var languageOptions = {
"checkbox_group": "पसंदीदा दस्तावेज़ शैली चुनें:",
"label_format": "बैठक रिपोर्ट",
"label_summary": "टाइमस्टैम्प के साथ सारांश",
"submitButton": "जमा करना",
"submitButton": "जमा करें",
"step_nav1": "स्टेप 1",
"step_nav2": "स्टेप 2",
"step_nav3": "स्टेप 3",
@@ -110,11 +110,11 @@ var languageOptions = {
"step_nav6": "स्टेप 6",
"h2": "अपना वीडियो यहां अपलोड करें:",
"labelSpeaker": "स्पीकर चुनें:",
"labelSpeakerAudio": "चयनित वक्ता:",
"labelSpeakerAudio": "चयनित स्पीकर:",
"labelSpeakerWriter": "नाम लिखें:",
"speakerLocker": "स्पीकर का नाम बदलें",
"speakerResender": "दस्तावेज़ पुनः लिखें",
"downloadButton": "डाउनलोड करना",
"speakerResender": "दस्तावेज़ फिर से लिखें",
"downloadButton": "डाउनलोड करें",
"box1_p1": "---प्रारंभ---",
"box2_p2": "---प्रतिलेखन---",
"box3_p3": "---दस्तावेज़ निर्माण---",
@@ -127,10 +127,10 @@ var languageOptions = {
"docName": "यहां दस्तावेज़ का नाम दर्ज करें",
"cd_promt": "आपका संकेत:",
"prompt": "अपने दस्तावेज़ के लिए प्रॉम्प्ट यहां टाइप करें...",
"goBackBtn": "वापस करना",
"deleteBtn": "दस्तावेज़ हटाए",
"goBackBtn": "वापस जाएं",
"deleteBtn": "दस्तावेज़ हटाए",
"generateBtn": "दस्तावेज़ सहेजें",
"newDoc": "-- नया दस्तावेज़ बनाए --"
"newDoc": "-- नया दस्तावेज़ बनाए --"
}
+32 -11
View File
@@ -432,19 +432,30 @@ function setSpeakerAudiosValue(valy) {
//Function to rewrite the speaker name in the json
function rewriteSpeakerName() {
try {
var tempy = document.getElementById("cur_speaker").value;
speakerAudios[tempy].name = document.getElementById("newSpeaker").value;
loadSpeakerOptions(speakerAudios);
} catch (error) {
console.log("\n\n\n" + error + "\n\n\n")
const select = document.getElementById("cur_speaker");
const newName = document.getElementById("newSpeaker").value.trim();
if (!newName) {
alert("Please enter a new speaker name");
return;
}
}
//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);
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) {
console.log(error);
console.log("Error renaming speaker:", error);
}
}
@@ -489,3 +500,13 @@ function reloadDocuments() {
});
});
}
function sendSpeakerPackages() {
try {
window.submitSpeaker.speaker_submit(speakerAudios);
} catch (error) {
console.log(error);
}
}
window.sendSpeakerPackages = sendSpeakerPackages;
+209 -28
View File
@@ -11,12 +11,12 @@ body {
}
#h1 {
position: absolute;
left: 50%;
top: 50%;
transform: translate(-50%, -50%);
position: static;
transform: none;
margin: 0;
z-index: 20;
flex: 1;
text-align: center;
}
#h1-wrapper {
@@ -30,6 +30,26 @@ body {
margin-bottom: 10px;
display: flex;
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 {
@@ -85,7 +105,6 @@ body {
#previewThumbnail {
width: 150px;
height: 100px;
/*border: 1px dashed black;*/
}
.custom-btn {
@@ -108,8 +127,9 @@ body {
background-color: #0056b3;
}
#step2 {
gap: 25px;
.step h2 {
width: 100%;
text-align: center;
}
.KI-wrapper {
@@ -186,6 +206,8 @@ input[type="file"] {
gap: 5px;
}
/* Hover effects for all different document options (with placeholders)*/
.figure1 {
position: relative;
}
@@ -201,12 +223,79 @@ input[type="file"] {
object-fit: contain;
display: none;
transition: opacity .2s;
z-index: 999;
}
.figure1:hover .img-hover1 {
display: flex;
}
.figure2 {
position: relative;
}
.img-hover2 {
position: absolute;
width: 200px;
height: 200px;
top: 0;
right: 40%;
left: 0;
bottom: 0;
object-fit: contain;
display: none;
transition: opacity .2s;
z-index: 999;
}
.figure2:hover .img-hover2 {
display: flex;
}
.figure3 {
position: relative;
}
.img-hover3 {
position: absolute;
width: 200px;
height: 200px;
top: 0;
right: 40%;
left: 0;
bottom: 0;
object-fit: contain;
display: none;
transition: opacity .2s;
z-index: 999;
}
.figure3:hover .img-hover3 {
display: flex;
}
.figure4 {
position: relative;
}
.img-hover4 {
position: absolute;
width: 200px;
height: 200px;
top: 0;
right: 40%;
left: 0;
bottom: 0;
object-fit: contain;
display: none;
transition: opacity .2s;
z-index: 999;
}
.figure4:hover .img-hover4 {
display: flex;
}
.img-icon {
width: 15px;
height: 15px;
@@ -236,7 +325,7 @@ input[type="file"] {
background-color: #FFF;
display: flex;
width: 780px;
height: 500px;
height: 550px;
flex-direction: column;
align-items: center;
gap: 10px;
@@ -245,6 +334,7 @@ input[type="file"] {
border-style: solid;
border-radius: 6px;
box-shadow: 0px 4px 10px rgba(0, 0, 0, 0.1);
padding-top: 50px;
}
.progressbar {
@@ -286,7 +376,10 @@ input[type="file"] {
#ai_type,
#transkript_type,
#language_option {
padding: 3px;
padding: 8px 12px;
border-radius: 4px;
border: 1px solid #ccc;
font-size: 14px;
}
.labelDiv {
@@ -334,7 +427,6 @@ input[type="file"] {
.step {
margin-top: 40px;
margin-bottom: 40px;
;
display: flex;
flex-direction: column;
min-height: 425px;
@@ -477,7 +569,14 @@ li {
-webkit-transition: all 0.3s ease;
}
.menu1 a:first-child {
#customDocBtn {
border: none;
background-color:#1C3B69;
font: 700 20px 'Oswald', sans-serif;
border-radius: 0%;
}
.menu1 button:first-child {
margin-top: 30px;
}
@@ -494,7 +593,7 @@ li {
text-decoration: none;
}
.li1:hover {
.li1:hover, #customDocBtn:hover{
background-color: #FFF;
color: rgb(61, 61, 61);
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1);
@@ -502,7 +601,27 @@ li {
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,
#step5 {
font-size: larger;
@@ -513,7 +632,7 @@ li {
}
#step5 {
align-items: flex-start;
align-items: center;
}
.button-group {
@@ -537,35 +656,91 @@ li {
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,
#speakerResender {
padding: 10px 20px;
margin: 20px auto;
padding: 12px 25px;
background-color: #007BFF;
color: white;
border: none;
border-radius: 8px;
cursor: pointer;
font-size: 14px;
font-weight: 500;
transition: background-color 0.2s;
}
.h2 {
font-size: 25px;
}
.speakerView,
.speakerAudio,
.speakerWrite {
margin-top: auto;
margin-bottom: auto;
#speakerLocker:hover,
#speakerResender:hover {
background-color: #0056b3;
}
.container {
background: white;
padding: 30px;
margin-top: 50px;
padding: 10px;
margin-top: 30px;
border-radius: 12px;
box-shadow: 0 4px 20px rgba(0, 0, 0, 0.1);
width: 90%;
max-width: 650px;
}
@@ -627,3 +802,9 @@ button:hover {
color: #333;
word-break: break-word;
}
.container input,
.container textarea,
.container select {
width: 100%;
}
+33 -3
View File
@@ -168,9 +168,39 @@ electron.ipcMain.on("file_submit", async (event, args) => {
throw new Error("Unknown document type: " + args.document.type);
}
console.log(args);
let audiopath = "";
let transcriptpath = "";
electron.ipcMain.on("file_download", async (event) => {
try {
if (!globalFinalHtmlPath) {
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");
// This code handles the Video to Audio extraction module call
+79 -40
View File
@@ -1,9 +1,9 @@
const fs = require('fs');
const path = require('path');
const puppeteer = require('puppeteer');
const htmlToDocx = require('html-to-docx');
const { execSync } = require('child_process');
const os = require('os');
const fs = require("fs");
const path = require("path");
const puppeteer = require("puppeteer");
const htmlToDocx = require("html-to-docx");
const { execSync } = require("child_process");
const os = require("os");
const outputDir = path.join(__dirname, "../../../storage/documents");
@@ -14,7 +14,7 @@ if (!fs.existsSync(outputDir)) {
async function showSaveDialog(defaultName, format) {
const platform = os.platform();
if (platform === 'darwin') {
if (platform === "darwin") {
// macOS
const applescript = `
set defaultName to "${defaultName}.${format}"
@@ -23,13 +23,15 @@ async function showSaveDialog(defaultName, format) {
`;
try {
const result = execSync(`osascript -e '${applescript}'`, { encoding: 'utf8' });
const result = execSync(`osascript -e '${applescript}'`, {
encoding: "utf8",
});
return result.trim();
} catch (err) {
if (err.status === 1) return null; // User canceled
throw err;
}
} else if (platform === 'win32') {
} else if (platform === "win32") {
const safeName = decodeURIComponent(defaultName);
const powershell = `
@@ -44,8 +46,8 @@ async function showSaveDialog(defaultName, format) {
try {
const result = execSync(
`powershell -NoProfile -Command "${powershell.replace(/\r?\n/g, ' ')}"`,
{ encoding: 'utf8' }
`powershell -NoProfile -Command "${powershell.replace(/\r?\n/g, " ")}"`,
{ encoding: "utf8" },
);
return result.trim() || null;
} catch (err) {
@@ -57,19 +59,19 @@ async function showSaveDialog(defaultName, format) {
try {
const result = execSync(
`zenity --file-selection --save --confirm-overwrite --filename="${defaultName}.${format}"`,
{ encoding: 'utf8' }
{ encoding: "utf8" },
);
return result.trim();
} catch (err) {
try {
const result = execSync(
`kdialog --getsavefilename . "${defaultName}.${format}"`,
{ encoding: 'utf8' }
{ encoding: "utf8" },
);
return result.trim();
} catch (err2) {
// Fallback
return path.join(os.homedir(), 'Downloads', `${defaultName}.${format}`);
return path.join(os.homedir(), "Downloads", `${defaultName}.${format}`);
}
}
}
@@ -89,7 +91,12 @@ const module_exports = {
* @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)
*/
async convert({ inputPath, format = 'pdf', outputName, showDialog = false }) {
async convert({ inputPath, format = "pdf", outputName, showDialog = false }) {
format = format.toLowerCase().replace(".", ""); // <-- FIX
if (!["pdf", "docx", "html", "txt"].includes(format)) {
throw new Error(`Unsupported format: ${format}`);
}
if (!fs.existsSync(inputPath)) {
throw new Error(`Input file not found: ${inputPath}`);
}
@@ -103,7 +110,7 @@ const module_exports = {
// Zeige nativen Dialog
outputFile = await showSaveDialog(baseName, format);
if (!outputFile) {
console.log('Speichervorgang abgebrochen.');
console.log("Speichervorgang abgebrochen.");
return null;
}
} else {
@@ -111,23 +118,23 @@ const module_exports = {
outputFile = path.join(outputDir, `${baseName}.${format.toLowerCase()}`);
}
let htmlContent = fs.readFileSync(inputPath, 'utf8');
let htmlContent = fs.readFileSync(inputPath, "utf8");
// Remove <think> tags if present
htmlContent = htmlContent.replace(/<think>[\s\S]*?<\/think>/gi, '');
htmlContent = htmlContent.replace(/<think>[\s\S]*?<\/think>/gi, "");
switch (format.toLowerCase()) {
case 'html':
fs.writeFileSync(outputFile, htmlContent, 'utf8');
case "html":
fs.writeFileSync(outputFile, htmlContent, "utf8");
break;
case 'pdf':
case "pdf":
await this.htmlToPDF(htmlContent, outputFile);
break;
case 'docx':
case "docx":
await this.htmlToDOCX(htmlContent, outputFile);
break;
case 'txt':
fs.writeFileSync(outputFile, this.htmlToTXT(htmlContent), 'utf8');
case "txt":
fs.writeFileSync(outputFile, this.htmlToTXT(htmlContent), "utf8");
break;
default:
throw new Error(`Unsupported format: ${format}`);
@@ -139,31 +146,61 @@ const module_exports = {
// HTML → PDF
async htmlToPDF(html, outputPath) {
const browser = await puppeteer.launch({
let browser;
try {
browser = await puppeteer.launch({
headless: true,
args: ['--no-sandbox', '--disable-setuid-sandbox']
args: ["--no-sandbox", "--disable-setuid-sandbox"],
});
const page = await browser.newPage();
await page.setContent(html, { waitUntil: 'networkidle0' });
await page.setContent(html, { waitUntil: "networkidle0" });
await page.pdf({
path: outputPath,
format: 'A4',
format: "A4",
printBackground: true,
margin: { top: '20mm', right: '20mm', bottom: '20mm', left: '20mm' }
margin: {
top: "20mm",
right: "20mm",
bottom: "20mm",
left: "20mm",
},
});
} finally {
if (browser) {
await browser.close();
}
}
},
// HTML → DOCX
async htmlToDOCX(html, outputPath) {
const buffer = await htmlToDocx(html);
try {
// htmltodocx library converts HTML string into a Word .docx 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 (rudimentär)
// HTML → TXT
htmlToTXT(html) {
return html.replace(/<[^>]*>/g, '').replace(/\s+\n/g, '\n').trim();
}
// A decent plain text conversion: strip tags and collapse whitespace
// 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;
@@ -173,24 +210,26 @@ if (require.main === module) {
(async () => {
const args = process.argv.slice(2);
if (args.length < 1) {
console.log('Usage: node htmlDocumentConverter.js <input.html> [format]');
console.log('Formats: pdf (default), docx, html, txt');
console.log('');
console.log('Ein nativer "Speichern unter" Dialog wird automatisch geöffnet.');
console.log("Usage: node htmlDocumentConverter.js <input.html> [format]");
console.log("Formats: pdf (default), docx, html, txt");
console.log("");
console.log(
'Ein nativer "Speichern unter" Dialog wird automatisch geöffnet.',
);
process.exit(1);
}
const inputPath = args[0];
const format = args[1] || 'pdf';
const format = args[1] || "pdf";
try {
await module_exports.convert({
inputPath,
format,
showDialog: true
showDialog: true,
});
} catch (err) {
console.error('Konvertierung fehlgeschlagen:', err.message);
console.error("Konvertierung fehlgeschlagen:", err.message);
process.exit(1);
}
})();
@@ -1,54 +0,0 @@
// -----------------------------------------------------------
// Parakeet (Step 3A: spawn Python minimal integration)
// -----------------------------------------------------------
const fs = require("fs");
const path = require("path");
const { spawn } = require("child_process");
module.exports = {
name: "parakeet",
type: "transcription",
displayname: "NVIDIA Parakeet",
async function(audioFilePath) {
console.log("🦜 [Parakeet] Starting test integration (spawn only)...");
console.log("🦜 Input audio:", audioFilePath);
// Check audio exists
if (!fs.existsSync(audioFilePath)) {
throw new Error("Audio file does not exist: " + audioFilePath);
}
// Output path in storage/transcripts
const sessionId = path.basename(audioFilePath).replace(/\.[^.]+$/, "");
const outputDir = path.join(__dirname, "../../../storage/transcripts");
fs.mkdirSync(outputDir, { recursive: true });
const outputPath = path.join(outputDir, `${sessionId}.json`);
// -------------------------------------------------------
// SPAWN PYTHON SCRIPT (step 3A — dummy script)
// -------------------------------------------------------
return new Promise((resolve, reject) => {
const python310 = "C:\\Users\\smith\\AppData\\Local\\Programs\\Python\\Python310\\python.exe";
const py = spawn(python310, [
path.join(__dirname, "parakeet_transcribe.py"),
audioFilePath,
outputPath
]);
py.stdout.on("data", data => console.log("🦜 [Python]", data.toString().trim()));
py.stderr.on("data", data => console.error("🦜 [Python ERR]", data.toString().trim()));
py.on("close", code => {
if (code === 0) {
console.log("🦜 [Parakeet] Done (spawn test). Output:", outputPath);
resolve(outputPath);
} else {
reject(new Error("Python script failed with exit code " + code));
}
});
});
}
};
@@ -1,71 +0,0 @@
# -----------------------------------------------------------
# Parakeet Real Transcriber (NVIDIA NeMo + PyTorch GPU)
# -----------------------------------------------------------
import sys
import json
import soundfile as sf
import torch
from nemo.collections.asr.models import ASRModel
# Args:
# sys.argv[1] = input audio path
# sys.argv[2] = output JSON path
audio_path = sys.argv[1]
output_path = sys.argv[2]
print("🔥 Starting Parakeet model...")
device = "cuda" if torch.cuda.is_available() else "cpu"
print("🔥 Using device:", device)
# -----------------------------------------------------------
# Load Parakeet model (NVIDIA pretrained ASR)
# -----------------------------------------------------------
model = ASRModel.from_pretrained(model_name="nvidia/parakeet-ctc-0.6b")
model = model.to(device)
model.eval()
# -----------------------------------------------------------
# Load audio
# -----------------------------------------------------------
print("🎧 Loading audio:", audio_path)
audio, sr = sf.read(audio_path)
# model expects mono float32
if len(audio.shape) > 1:
audio = audio.mean(axis=1)
audio = audio.astype("float32")
# -----------------------------------------------------------
# Run inference
# -----------------------------------------------------------
print("🧠 Running inference...")
with torch.no_grad():
hyp = model.transcribe([audio])[0]
# Extract only the text
if hasattr(hyp, "text"):
transcript = hyp.text
else:
# fallback: convert to string (rare)
transcript = str(hyp)
print("📄 Transcript:", transcript)
# -----------------------------------------------------------
# Save JSON format compatible with V2D pipeline
# -----------------------------------------------------------
result = {
"id": output_path.split("/")[-1].replace(".json", ""),
"tool": "nemo_parakeet",
"status": "completed",
"text": transcript,
"words": [] # Parakeet XS doesnt return word timestamps
}
with open(output_path, "w", encoding="utf-8") as f:
json.dump(result, f, indent=2, ensure_ascii=False)
print("✔ JSON saved at:", output_path)