Button toggle.
This commit is contained in:
parent
4f988959ce
commit
79a4f9832c
@ -73,11 +73,12 @@
|
||||
margin-bottom: 1rem;
|
||||
}
|
||||
.compose-wrapper textarea { resize: vertical; min-height: 60px; }
|
||||
input,textarea {
|
||||
input,textarea,tag-input {
|
||||
padding: .5rem;
|
||||
border: 1px solid #ccc;
|
||||
border-radius: 4px;
|
||||
width: 100%;
|
||||
margin-bottom: 5px;
|
||||
}
|
||||
button {
|
||||
padding: .5rem 1rem;
|
||||
@ -85,6 +86,49 @@
|
||||
border-radius: 4px;
|
||||
cursor: pointer;
|
||||
}
|
||||
/* Context Menu Styling */
|
||||
.context-menu {
|
||||
position: absolute;
|
||||
background-color: #fff;
|
||||
border: 1px solid #ccc;
|
||||
border-radius: 5px;
|
||||
box-shadow: 0 2px 5px rgba(0, 0, 0, 0.15);
|
||||
padding: 5px;
|
||||
z-index: 1000;
|
||||
display: none; /* Initially hidden */
|
||||
min-width: 150px;
|
||||
}
|
||||
|
||||
.context-menu-item {
|
||||
padding: 8px 12px;
|
||||
cursor: pointer;
|
||||
user-select: none;
|
||||
}
|
||||
|
||||
.context-menu-item:hover {
|
||||
background-color: #f0f0f0;
|
||||
}
|
||||
|
||||
.context-menu-item span {
|
||||
display: block;
|
||||
}
|
||||
/* Animations */
|
||||
@keyframes fadeIn {
|
||||
from {
|
||||
opacity: 0;
|
||||
transform: translateY(-5px);
|
||||
}
|
||||
to {
|
||||
opacity: 1;
|
||||
transform: translateY(0);
|
||||
}
|
||||
}
|
||||
|
||||
.context-menu.show {
|
||||
animation: fadeIn 0.2s ease-out;
|
||||
display: block;
|
||||
}
|
||||
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
@ -106,7 +150,68 @@
|
||||
</main>
|
||||
</div>
|
||||
|
||||
<div id="context-menu" class="context-menu">
|
||||
<div class="context-menu-item"><span>Edit Tag</span></div>
|
||||
<div class="context-menu-item"><span>Rename Tag</span></div>
|
||||
<div class="context-menu-item"><span>Delete Tag</span></div>
|
||||
</div>
|
||||
|
||||
<script>
|
||||
class ContextMenu extends HTMLElement {
|
||||
constructor() {
|
||||
super();
|
||||
this.menu = document.createElement('div');
|
||||
this.menu.classList.add('context-menu');
|
||||
document.body.appendChild(this.menu);
|
||||
|
||||
document.body.addEventListener('contextmenu', (event) => {
|
||||
event.preventDefault(); // Prevent default context menu
|
||||
this.showContextMenu(event);
|
||||
});
|
||||
|
||||
document.addEventListener('click', () => {
|
||||
this.hideContextMenu();
|
||||
});
|
||||
}
|
||||
addItem(text) {
|
||||
const item = document.createElement('div');
|
||||
item.classList.add('context-menu-item');
|
||||
item.innerHTML = `<span>${text}</span>`;
|
||||
this.menu.appendChild(item);
|
||||
}
|
||||
|
||||
showContextMenu(event) {
|
||||
|
||||
let clicked = event.target.closest('.right-clickable');
|
||||
if (!clicked){
|
||||
clicked = event.target;
|
||||
if (!clicked.classList.contains('right-clickable'))
|
||||
return;
|
||||
}
|
||||
|
||||
//Get position
|
||||
const x = event.clientX;
|
||||
const y = event.clientY;
|
||||
|
||||
// Set the clicked tag's data to a property so it's available for the menu items.
|
||||
this.currentElement = clicked;
|
||||
|
||||
this.menu.style.left = `${x}px`;
|
||||
this.menu.style.top = `${y}px`;
|
||||
this.menu.classList.add('show'); // Add show class for animation
|
||||
}
|
||||
|
||||
|
||||
hideContextMenu() {
|
||||
this.menu.classList.remove('show');
|
||||
}
|
||||
}
|
||||
|
||||
customElements.define('context-menu', ContextMenu);
|
||||
|
||||
const contextMenu = new ContextMenu();
|
||||
contextMenu.addItem('Edit Tag');
|
||||
</script>
|
||||
|
||||
<script src="https://cdn.jsdelivr.net/npm/marked/marked.min.js"></script>
|
||||
<!--
|
||||
@ -379,7 +484,7 @@ class NoteCard extends HTMLElement {
|
||||
? `<img src="${a.url}">`
|
||||
: `<a href="${a.url}" target="_blank">${a.url}</a>`).join('')}
|
||||
</div>` : ''}
|
||||
${tags.length ? `<div class="tag-list">${tags.map(t => `<span class="tag">${t}</span>`).join('')}</div>` : ''}
|
||||
${tags.length ? `<div class="tag-list">${tags.map(t => `<span class="tag right-clickable" data-tag="${t}">${t}</span>`).join('')}</div>` : ''}
|
||||
</div>`;
|
||||
}
|
||||
}
|
||||
@ -391,14 +496,15 @@ class NoteCompose extends HTMLElement {
|
||||
super();
|
||||
this.noteId = null; // null = new
|
||||
this.innerHTML = `
|
||||
<form class="compose-wrapper">
|
||||
<button class="compose-button">New note</button>
|
||||
<form class="compose-wrapper" style="display:none;">
|
||||
<input name="title" placeholder="Title" required>
|
||||
<textarea name="body" placeholder="Take a note…" required></textarea>
|
||||
<input name="files" type="file" multiple>
|
||||
<tag-input name="tags" placeholder="Tags (comma-separated)"></tag-input>
|
||||
<div style="display:flex;gap:.5rem;">
|
||||
<button type="submit">Save</button>
|
||||
<button type="button" id="cancel" style="display:none;">Cancel</button>
|
||||
<button type="button" class="cancel">Cancel</button>
|
||||
</div>
|
||||
</form>`;
|
||||
}
|
||||
@ -408,18 +514,27 @@ class NoteCompose extends HTMLElement {
|
||||
e.preventDefault();
|
||||
this.save();
|
||||
});
|
||||
|
||||
|
||||
this.querySelector('#cancel').addEventListener('click', () => this.reset());
|
||||
this.composeButton = this.querySelector('.compose-button')
|
||||
this.composeButton.addEventListener('click', () => this.show());
|
||||
this.querySelector('.cancel').addEventListener('click', () => this.reset());
|
||||
document.addEventListener('edit-note', e => this.load(e.detail));
|
||||
}
|
||||
show(){
|
||||
this.form.style.display = 'block';
|
||||
this.q('title').focus();
|
||||
this.composeButton.style.display = "none";
|
||||
}
|
||||
hide(){
|
||||
this.form.style.display = 'none';
|
||||
this.composeButton.style.display = "block";
|
||||
}
|
||||
q(s) { return this.querySelector(s); }
|
||||
load(n) { /* pre-fill for edit */
|
||||
this.noteId = n.id;
|
||||
this.q('title').value = n.title;
|
||||
this.q('body').value = n.body;
|
||||
this.q('tags').value = (n.tags || []).join(', ');
|
||||
this.q('#cancel').style.display = 'inline-block';
|
||||
this.show();
|
||||
this.scrollIntoView({behavior:'smooth'});
|
||||
}
|
||||
async save() {
|
||||
@ -446,7 +561,7 @@ class NoteCompose extends HTMLElement {
|
||||
if (res.ok) { this.reset(); document.dispatchEvent(new CustomEvent('notes-changed')); }
|
||||
else alert('Save failed');
|
||||
}
|
||||
reset() { this.noteId=null; this.form.reset(); this.q('#cancel').style.display='none'; this.q('tags').value=''; }
|
||||
reset() { this.noteId=null; this.form.reset(); this.hide(); this.q('tags').value=''; }
|
||||
q(sel){ return this.querySelector(sel.includes('#') ? sel : `[name=\"${sel}\"]`); }
|
||||
}
|
||||
customElements.define('note-compose', NoteCompose);
|
||||
|
1
main.py
1
main.py
@ -273,7 +273,6 @@ async def _upsert_note(note_id: Optional[int], payload: Dict[str, Any]):
|
||||
db['notes'].update({"id": note_id, "title": title, "body": body, "updated_at": now}, ["id"])
|
||||
db['attachments'].delete(note_id=note_id)
|
||||
db['note_tags'].delete(note_id=note_id)
|
||||
db.query("DELETE FROM notes_fts WHERE note_id = :nid", nid=note_id)
|
||||
|
||||
# (Re‑)insert into FTS table
|
||||
for t in tags:
|
||||
|
Loading…
Reference in New Issue
Block a user