refactor: Modularize UI components and utilities
- Extract UI components into separate JS files - Centralize configuration values in config.js - Introduce a dedicated logger module - Improve file tree drag-and-drop and undo functionality - Refactor modal handling to a single manager - Add URL routing support for SPA navigation - Implement view mode for read-only access
This commit is contained in:
@@ -1,5 +1,5 @@
|
||||
// Markdown Editor Application with File Tree
|
||||
(function() {
|
||||
(function () {
|
||||
'use strict';
|
||||
|
||||
// State management
|
||||
@@ -26,13 +26,13 @@
|
||||
document.body.classList.add('dark-mode');
|
||||
document.getElementById('darkModeIcon').textContent = '☀️';
|
||||
localStorage.setItem('darkMode', 'true');
|
||||
|
||||
mermaid.initialize({
|
||||
|
||||
mermaid.initialize({
|
||||
startOnLoad: false,
|
||||
theme: 'dark',
|
||||
securityLevel: 'loose'
|
||||
});
|
||||
|
||||
|
||||
if (editor && editor.getValue()) {
|
||||
updatePreview();
|
||||
}
|
||||
@@ -43,13 +43,13 @@
|
||||
document.body.classList.remove('dark-mode');
|
||||
document.getElementById('darkModeIcon').textContent = '🌙';
|
||||
localStorage.setItem('darkMode', 'false');
|
||||
|
||||
mermaid.initialize({
|
||||
|
||||
mermaid.initialize({
|
||||
startOnLoad: false,
|
||||
theme: 'default',
|
||||
securityLevel: 'loose'
|
||||
});
|
||||
|
||||
|
||||
if (editor && editor.getValue()) {
|
||||
updatePreview();
|
||||
}
|
||||
@@ -64,7 +64,7 @@
|
||||
}
|
||||
|
||||
// Initialize Mermaid
|
||||
mermaid.initialize({
|
||||
mermaid.initialize({
|
||||
startOnLoad: false,
|
||||
theme: 'default',
|
||||
securityLevel: 'loose'
|
||||
@@ -86,15 +86,15 @@
|
||||
async function uploadImage(file) {
|
||||
const formData = new FormData();
|
||||
formData.append('file', file);
|
||||
|
||||
|
||||
try {
|
||||
const response = await fetch('/api/upload-image', {
|
||||
method: 'POST',
|
||||
body: formData
|
||||
});
|
||||
|
||||
|
||||
if (!response.ok) throw new Error('Upload failed');
|
||||
|
||||
|
||||
const result = await response.json();
|
||||
return result.url;
|
||||
} catch (error) {
|
||||
@@ -107,44 +107,44 @@
|
||||
// Handle drag and drop for images
|
||||
function setupDragAndDrop() {
|
||||
const editorElement = document.querySelector('.CodeMirror');
|
||||
|
||||
|
||||
['dragenter', 'dragover', 'dragleave', 'drop'].forEach(eventName => {
|
||||
editorElement.addEventListener(eventName, preventDefaults, false);
|
||||
});
|
||||
|
||||
|
||||
function preventDefaults(e) {
|
||||
e.preventDefault();
|
||||
e.stopPropagation();
|
||||
}
|
||||
|
||||
|
||||
['dragenter', 'dragover'].forEach(eventName => {
|
||||
editorElement.addEventListener(eventName, () => {
|
||||
editorElement.classList.add('drag-over');
|
||||
}, false);
|
||||
});
|
||||
|
||||
|
||||
['dragleave', 'drop'].forEach(eventName => {
|
||||
editorElement.addEventListener(eventName, () => {
|
||||
editorElement.classList.remove('drag-over');
|
||||
}, false);
|
||||
});
|
||||
|
||||
|
||||
editorElement.addEventListener('drop', async (e) => {
|
||||
const files = e.dataTransfer.files;
|
||||
|
||||
|
||||
if (files.length === 0) return;
|
||||
|
||||
const imageFiles = Array.from(files).filter(file =>
|
||||
|
||||
const imageFiles = Array.from(files).filter(file =>
|
||||
file.type.startsWith('image/')
|
||||
);
|
||||
|
||||
|
||||
if (imageFiles.length === 0) {
|
||||
showNotification('Please drop image files only', 'warning');
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
showNotification(`Uploading ${imageFiles.length} image(s)...`, 'info');
|
||||
|
||||
|
||||
for (const file of imageFiles) {
|
||||
const url = await uploadImage(file);
|
||||
if (url) {
|
||||
@@ -156,11 +156,11 @@
|
||||
}
|
||||
}
|
||||
}, false);
|
||||
|
||||
|
||||
editorElement.addEventListener('paste', async (e) => {
|
||||
const items = e.clipboardData?.items;
|
||||
if (!items) return;
|
||||
|
||||
|
||||
for (const item of items) {
|
||||
if (item.type.startsWith('image/')) {
|
||||
e.preventDefault();
|
||||
@@ -189,15 +189,15 @@
|
||||
lineWrapping: true,
|
||||
autofocus: true,
|
||||
extraKeys: {
|
||||
'Ctrl-S': function() { saveFile(); },
|
||||
'Cmd-S': function() { saveFile(); }
|
||||
'Ctrl-S': function () { saveFile(); },
|
||||
'Cmd-S': function () { saveFile(); }
|
||||
}
|
||||
});
|
||||
|
||||
editor.on('change', debounce(updatePreview, 300));
|
||||
|
||||
|
||||
setTimeout(setupDragAndDrop, 100);
|
||||
|
||||
|
||||
setupScrollSync();
|
||||
}
|
||||
|
||||
@@ -217,13 +217,13 @@
|
||||
// Setup synchronized scrolling
|
||||
function setupScrollSync() {
|
||||
const previewDiv = document.getElementById('preview');
|
||||
|
||||
|
||||
editor.on('scroll', () => {
|
||||
if (!isScrollingSynced) return;
|
||||
|
||||
|
||||
const scrollInfo = editor.getScrollInfo();
|
||||
const scrollPercentage = scrollInfo.top / (scrollInfo.height - scrollInfo.clientHeight);
|
||||
|
||||
|
||||
const previewScrollHeight = previewDiv.scrollHeight - previewDiv.clientHeight;
|
||||
previewDiv.scrollTop = previewScrollHeight * scrollPercentage;
|
||||
});
|
||||
@@ -233,7 +233,7 @@
|
||||
async function updatePreview() {
|
||||
const markdown = editor.getValue();
|
||||
const previewDiv = document.getElementById('preview');
|
||||
|
||||
|
||||
if (!markdown.trim()) {
|
||||
previewDiv.innerHTML = `
|
||||
<div class="text-muted text-center mt-5">
|
||||
@@ -243,17 +243,17 @@
|
||||
`;
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
try {
|
||||
let html = marked.parse(markdown);
|
||||
|
||||
|
||||
html = html.replace(
|
||||
/<pre><code class="language-mermaid">([\s\S]*?)<\/code><\/pre>/g,
|
||||
'<div class="mermaid">$1</div>'
|
||||
);
|
||||
|
||||
|
||||
previewDiv.innerHTML = html;
|
||||
|
||||
|
||||
const codeBlocks = previewDiv.querySelectorAll('pre code');
|
||||
codeBlocks.forEach(block => {
|
||||
const languageClass = Array.from(block.classList).find(cls => cls.startsWith('language-'));
|
||||
@@ -261,7 +261,7 @@
|
||||
Prism.highlightElement(block);
|
||||
}
|
||||
});
|
||||
|
||||
|
||||
const mermaidElements = previewDiv.querySelectorAll('.mermaid');
|
||||
if (mermaidElements.length > 0) {
|
||||
try {
|
||||
@@ -291,7 +291,7 @@
|
||||
try {
|
||||
const response = await fetch('/api/tree');
|
||||
if (!response.ok) throw new Error('Failed to load file tree');
|
||||
|
||||
|
||||
fileTree = await response.json();
|
||||
renderFileTree();
|
||||
} catch (error) {
|
||||
@@ -303,12 +303,12 @@
|
||||
function renderFileTree() {
|
||||
const container = document.getElementById('fileTree');
|
||||
container.innerHTML = '';
|
||||
|
||||
|
||||
if (fileTree.length === 0) {
|
||||
container.innerHTML = '<div class="text-muted text-center p-3">No files yet</div>';
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
fileTree.forEach(node => {
|
||||
container.appendChild(createTreeNode(node));
|
||||
});
|
||||
@@ -317,13 +317,13 @@
|
||||
function createTreeNode(node, level = 0) {
|
||||
const nodeDiv = document.createElement('div');
|
||||
nodeDiv.className = 'tree-node-wrapper';
|
||||
|
||||
|
||||
const nodeContent = document.createElement('div');
|
||||
nodeContent.className = 'tree-node';
|
||||
nodeContent.dataset.path = node.path;
|
||||
nodeContent.dataset.type = node.type;
|
||||
nodeContent.dataset.name = node.name;
|
||||
|
||||
|
||||
// Make draggable
|
||||
nodeContent.draggable = true;
|
||||
nodeContent.addEventListener('dragstart', handleDragStart);
|
||||
@@ -331,14 +331,13 @@
|
||||
nodeContent.addEventListener('dragover', handleDragOver);
|
||||
nodeContent.addEventListener('dragleave', handleDragLeave);
|
||||
nodeContent.addEventListener('drop', handleDrop);
|
||||
|
||||
|
||||
const contentWrapper = document.createElement('div');
|
||||
contentWrapper.className = 'tree-node-content';
|
||||
|
||||
|
||||
if (node.type === 'directory') {
|
||||
const toggle = document.createElement('span');
|
||||
toggle.className = 'tree-node-toggle';
|
||||
toggle.innerHTML = '▶';
|
||||
toggle.addEventListener('click', (e) => {
|
||||
e.stopPropagation();
|
||||
toggleNode(nodeDiv);
|
||||
@@ -349,56 +348,56 @@
|
||||
spacer.style.width = '16px';
|
||||
contentWrapper.appendChild(spacer);
|
||||
}
|
||||
|
||||
|
||||
const icon = document.createElement('i');
|
||||
icon.className = node.type === 'directory' ? 'bi bi-folder tree-node-icon' : 'bi bi-file-earmark-text tree-node-icon';
|
||||
contentWrapper.appendChild(icon);
|
||||
|
||||
|
||||
const name = document.createElement('span');
|
||||
name.className = 'tree-node-name';
|
||||
name.textContent = node.name;
|
||||
contentWrapper.appendChild(name);
|
||||
|
||||
|
||||
if (node.type === 'file' && node.size) {
|
||||
const size = document.createElement('span');
|
||||
size.className = 'file-size-badge';
|
||||
size.textContent = formatFileSize(node.size);
|
||||
contentWrapper.appendChild(size);
|
||||
}
|
||||
|
||||
|
||||
nodeContent.appendChild(contentWrapper);
|
||||
|
||||
|
||||
nodeContent.addEventListener('click', (e) => {
|
||||
if (node.type === 'file') {
|
||||
loadFile(node.path);
|
||||
}
|
||||
});
|
||||
|
||||
|
||||
nodeContent.addEventListener('contextmenu', (e) => {
|
||||
e.preventDefault();
|
||||
showContextMenu(e, node);
|
||||
});
|
||||
|
||||
|
||||
nodeDiv.appendChild(nodeContent);
|
||||
|
||||
|
||||
if (node.children && node.children.length > 0) {
|
||||
const childrenDiv = document.createElement('div');
|
||||
childrenDiv.className = 'tree-children collapsed';
|
||||
|
||||
|
||||
node.children.forEach(child => {
|
||||
childrenDiv.appendChild(createTreeNode(child, level + 1));
|
||||
});
|
||||
|
||||
|
||||
nodeDiv.appendChild(childrenDiv);
|
||||
}
|
||||
|
||||
|
||||
return nodeDiv;
|
||||
}
|
||||
|
||||
function toggleNode(nodeWrapper) {
|
||||
const toggle = nodeWrapper.querySelector('.tree-node-toggle');
|
||||
const children = nodeWrapper.querySelector('.tree-children');
|
||||
|
||||
|
||||
if (children) {
|
||||
children.classList.toggle('collapsed');
|
||||
toggle.classList.toggle('expanded');
|
||||
@@ -437,10 +436,10 @@
|
||||
|
||||
function handleDragOver(e) {
|
||||
if (!draggedNode) return;
|
||||
|
||||
|
||||
e.preventDefault();
|
||||
e.dataTransfer.dropEffect = 'move';
|
||||
|
||||
|
||||
const targetType = e.currentTarget.dataset.type;
|
||||
if (targetType === 'directory') {
|
||||
e.currentTarget.classList.add('drag-over');
|
||||
@@ -454,18 +453,18 @@
|
||||
async function handleDrop(e) {
|
||||
e.preventDefault();
|
||||
e.currentTarget.classList.remove('drag-over');
|
||||
|
||||
|
||||
if (!draggedNode) return;
|
||||
|
||||
|
||||
const targetPath = e.currentTarget.dataset.path;
|
||||
const targetType = e.currentTarget.dataset.type;
|
||||
|
||||
|
||||
if (targetType !== 'directory') return;
|
||||
if (draggedNode.path === targetPath) return;
|
||||
|
||||
|
||||
const sourcePath = draggedNode.path;
|
||||
const destPath = targetPath + '/' + draggedNode.name;
|
||||
|
||||
|
||||
try {
|
||||
const response = await fetch('/api/file/move', {
|
||||
method: 'POST',
|
||||
@@ -475,16 +474,16 @@
|
||||
destination: destPath
|
||||
})
|
||||
});
|
||||
|
||||
|
||||
if (!response.ok) throw new Error('Move failed');
|
||||
|
||||
|
||||
showNotification(`Moved ${draggedNode.name}`, 'success');
|
||||
loadFileTree();
|
||||
} catch (error) {
|
||||
console.error('Error moving file:', error);
|
||||
showNotification('Error moving file', 'danger');
|
||||
}
|
||||
|
||||
|
||||
draggedNode = null;
|
||||
}
|
||||
|
||||
@@ -496,18 +495,18 @@
|
||||
contextMenuTarget = node;
|
||||
const menu = document.getElementById('contextMenu');
|
||||
const pasteItem = document.getElementById('pasteMenuItem');
|
||||
|
||||
|
||||
// Show paste option only if clipboard has something and target is a directory
|
||||
if (clipboard && node.type === 'directory') {
|
||||
pasteItem.style.display = 'flex';
|
||||
} else {
|
||||
pasteItem.style.display = 'none';
|
||||
}
|
||||
|
||||
|
||||
menu.style.display = 'block';
|
||||
menu.style.left = e.pageX + 'px';
|
||||
menu.style.top = e.pageY + 'px';
|
||||
|
||||
|
||||
document.addEventListener('click', hideContextMenu);
|
||||
}
|
||||
|
||||
@@ -525,20 +524,20 @@
|
||||
try {
|
||||
const response = await fetch(`/api/file?path=${encodeURIComponent(path)}`);
|
||||
if (!response.ok) throw new Error('Failed to load file');
|
||||
|
||||
|
||||
const data = await response.json();
|
||||
currentFile = data.filename;
|
||||
currentFilePath = path;
|
||||
|
||||
|
||||
document.getElementById('filenameInput').value = path;
|
||||
editor.setValue(data.content);
|
||||
updatePreview();
|
||||
|
||||
|
||||
document.querySelectorAll('.tree-node').forEach(node => {
|
||||
node.classList.remove('active');
|
||||
});
|
||||
document.querySelector(`[data-path="${path}"]`)?.classList.add('active');
|
||||
|
||||
|
||||
showNotification(`Loaded ${data.filename}`, 'info');
|
||||
} catch (error) {
|
||||
console.error('Error loading file:', error);
|
||||
@@ -548,27 +547,27 @@
|
||||
|
||||
async function saveFile() {
|
||||
const path = document.getElementById('filenameInput').value.trim();
|
||||
|
||||
|
||||
if (!path) {
|
||||
showNotification('Please enter a filename', 'warning');
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
const content = editor.getValue();
|
||||
|
||||
|
||||
try {
|
||||
const response = await fetch('/api/file', {
|
||||
method: 'POST',
|
||||
headers: { 'Content-Type': 'application/json' },
|
||||
body: JSON.stringify({ path, content })
|
||||
});
|
||||
|
||||
|
||||
if (!response.ok) throw new Error('Failed to save file');
|
||||
|
||||
|
||||
const result = await response.json();
|
||||
currentFile = path.split('/').pop();
|
||||
currentFilePath = result.path;
|
||||
|
||||
|
||||
showNotification(`Saved ${currentFile}`, 'success');
|
||||
loadFileTree();
|
||||
} catch (error) {
|
||||
@@ -582,26 +581,26 @@
|
||||
showNotification('No file selected', 'warning');
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
if (!confirm(`Are you sure you want to delete ${currentFile}?`)) {
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
try {
|
||||
const response = await fetch(`/api/file?path=${encodeURIComponent(currentFilePath)}`, {
|
||||
method: 'DELETE'
|
||||
});
|
||||
|
||||
|
||||
if (!response.ok) throw new Error('Failed to delete file');
|
||||
|
||||
|
||||
showNotification(`Deleted ${currentFile}`, 'success');
|
||||
|
||||
|
||||
currentFile = null;
|
||||
currentFilePath = null;
|
||||
document.getElementById('filenameInput').value = '';
|
||||
editor.setValue('');
|
||||
updatePreview();
|
||||
|
||||
|
||||
loadFileTree();
|
||||
} catch (error) {
|
||||
console.error('Error deleting file:', error);
|
||||
@@ -617,27 +616,27 @@
|
||||
document.getElementById('filenameInput').focus();
|
||||
editor.setValue('');
|
||||
updatePreview();
|
||||
|
||||
|
||||
document.querySelectorAll('.tree-node').forEach(node => {
|
||||
node.classList.remove('active');
|
||||
});
|
||||
|
||||
|
||||
showNotification('Enter filename and start typing', 'info');
|
||||
}
|
||||
|
||||
async function createFolder() {
|
||||
const folderName = prompt('Enter folder name:');
|
||||
if (!folderName) return;
|
||||
|
||||
|
||||
try {
|
||||
const response = await fetch('/api/directory', {
|
||||
method: 'POST',
|
||||
headers: { 'Content-Type': 'application/json' },
|
||||
body: JSON.stringify({ path: folderName })
|
||||
});
|
||||
|
||||
|
||||
if (!response.ok) throw new Error('Failed to create folder');
|
||||
|
||||
|
||||
showNotification(`Created folder ${folderName}`, 'success');
|
||||
loadFileTree();
|
||||
} catch (error) {
|
||||
@@ -652,32 +651,32 @@
|
||||
|
||||
async function handleContextMenuAction(action) {
|
||||
if (!contextMenuTarget) return;
|
||||
|
||||
|
||||
switch (action) {
|
||||
case 'open':
|
||||
if (contextMenuTarget.type === 'file') {
|
||||
loadFile(contextMenuTarget.path);
|
||||
}
|
||||
break;
|
||||
|
||||
|
||||
case 'rename':
|
||||
await renameItem();
|
||||
break;
|
||||
|
||||
|
||||
case 'copy':
|
||||
clipboard = { ...contextMenuTarget, operation: 'copy' };
|
||||
showNotification(`Copied ${contextMenuTarget.name}`, 'info');
|
||||
break;
|
||||
|
||||
|
||||
case 'move':
|
||||
clipboard = { ...contextMenuTarget, operation: 'move' };
|
||||
showNotification(`Cut ${contextMenuTarget.name}`, 'info');
|
||||
break;
|
||||
|
||||
|
||||
case 'paste':
|
||||
await pasteItem();
|
||||
break;
|
||||
|
||||
|
||||
case 'delete':
|
||||
await deleteItem();
|
||||
break;
|
||||
@@ -687,10 +686,10 @@
|
||||
async function renameItem() {
|
||||
const newName = prompt(`Rename ${contextMenuTarget.name}:`, contextMenuTarget.name);
|
||||
if (!newName || newName === contextMenuTarget.name) return;
|
||||
|
||||
|
||||
const oldPath = contextMenuTarget.path;
|
||||
const newPath = oldPath.substring(0, oldPath.lastIndexOf('/') + 1) + newName;
|
||||
|
||||
|
||||
try {
|
||||
const endpoint = contextMenuTarget.type === 'directory' ? '/api/directory/rename' : '/api/file/rename';
|
||||
const response = await fetch(endpoint, {
|
||||
@@ -701,9 +700,9 @@
|
||||
new_path: newPath
|
||||
})
|
||||
});
|
||||
|
||||
|
||||
if (!response.ok) throw new Error('Rename failed');
|
||||
|
||||
|
||||
showNotification(`Renamed to ${newName}`, 'success');
|
||||
loadFileTree();
|
||||
} catch (error) {
|
||||
@@ -714,12 +713,12 @@
|
||||
|
||||
async function pasteItem() {
|
||||
if (!clipboard) return;
|
||||
|
||||
|
||||
const destDir = contextMenuTarget.path;
|
||||
const sourcePath = clipboard.path;
|
||||
const fileName = clipboard.name;
|
||||
const destPath = destDir + '/' + fileName;
|
||||
|
||||
|
||||
try {
|
||||
if (clipboard.operation === 'copy') {
|
||||
// Copy operation
|
||||
@@ -731,7 +730,7 @@
|
||||
destination: destPath
|
||||
})
|
||||
});
|
||||
|
||||
|
||||
if (!response.ok) throw new Error('Copy failed');
|
||||
showNotification(`Copied ${fileName} to ${contextMenuTarget.name}`, 'success');
|
||||
} else if (clipboard.operation === 'move') {
|
||||
@@ -744,12 +743,12 @@
|
||||
destination: destPath
|
||||
})
|
||||
});
|
||||
|
||||
|
||||
if (!response.ok) throw new Error('Move failed');
|
||||
showNotification(`Moved ${fileName} to ${contextMenuTarget.name}`, 'success');
|
||||
clipboard = null; // Clear clipboard after move
|
||||
}
|
||||
|
||||
|
||||
loadFileTree();
|
||||
} catch (error) {
|
||||
console.error('Error pasting:', error);
|
||||
@@ -761,7 +760,7 @@
|
||||
if (!confirm(`Are you sure you want to delete ${contextMenuTarget.name}?`)) {
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
try {
|
||||
let response;
|
||||
if (contextMenuTarget.type === 'directory') {
|
||||
@@ -773,9 +772,9 @@
|
||||
method: 'DELETE'
|
||||
});
|
||||
}
|
||||
|
||||
|
||||
if (!response.ok) throw new Error('Delete failed');
|
||||
|
||||
|
||||
showNotification(`Deleted ${contextMenuTarget.name}`, 'success');
|
||||
loadFileTree();
|
||||
} catch (error) {
|
||||
@@ -793,7 +792,7 @@
|
||||
if (!toastContainer) {
|
||||
toastContainer = createToastContainer();
|
||||
}
|
||||
|
||||
|
||||
const toast = document.createElement('div');
|
||||
toast.className = `toast align-items-center text-white bg-${type} border-0`;
|
||||
toast.setAttribute('role', 'alert');
|
||||
@@ -803,12 +802,12 @@
|
||||
<button type="button" class="btn-close btn-close-white me-2 m-auto" data-bs-dismiss="toast" aria-label="Close"></button>
|
||||
</div>
|
||||
`;
|
||||
|
||||
|
||||
toastContainer.appendChild(toast);
|
||||
|
||||
|
||||
const bsToast = new bootstrap.Toast(toast, { delay: 3000 });
|
||||
bsToast.show();
|
||||
|
||||
|
||||
toast.addEventListener('hidden.bs.toast', () => {
|
||||
toast.remove();
|
||||
});
|
||||
@@ -831,13 +830,13 @@
|
||||
initDarkMode();
|
||||
initEditor();
|
||||
loadFileTree();
|
||||
|
||||
|
||||
document.getElementById('saveBtn').addEventListener('click', saveFile);
|
||||
document.getElementById('deleteBtn').addEventListener('click', deleteFile);
|
||||
document.getElementById('newFileBtn').addEventListener('click', newFile);
|
||||
document.getElementById('newFolderBtn').addEventListener('click', createFolder);
|
||||
document.getElementById('darkModeToggle').addEventListener('click', toggleDarkMode);
|
||||
|
||||
|
||||
// Context menu actions
|
||||
document.querySelectorAll('.context-menu-item').forEach(item => {
|
||||
item.addEventListener('click', () => {
|
||||
@@ -846,14 +845,14 @@
|
||||
hideContextMenu();
|
||||
});
|
||||
});
|
||||
|
||||
|
||||
document.addEventListener('keydown', (e) => {
|
||||
if ((e.ctrlKey || e.metaKey) && e.key === 's') {
|
||||
e.preventDefault();
|
||||
saveFile();
|
||||
}
|
||||
});
|
||||
|
||||
|
||||
console.log('Markdown Editor with File Tree initialized');
|
||||
}
|
||||
|
||||
|
||||
Reference in New Issue
Block a user