feat: add keyboard shortcuts and copy message buttons

Keyboard shortcuts:
- Up Arrow: Edit last user message (when input at start)
- Left/Right Arrow: Navigate swipes (when not in input)
- Escape: Close panels/modals, cancel editing
- Ctrl+K: Focus message input
- Ctrl+/: Toggle Roleplay Tools
- Ctrl+Enter: Send message (alternative)

Message controls:
- Add Copy Message button for both user and assistant messages
- Visual feedback (checkmark) after copying
- 2-second feedback duration

Updated README to reflect actual implemented shortcuts.

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
2025-10-16 22:12:14 -07:00
parent 0bd1590681
commit 8c70e0558f
2 changed files with 139 additions and 4 deletions

View File

@@ -57,8 +57,12 @@ Config stored in `~/.config/claudia/config.json`
- **Enter** - Send message - **Enter** - Send message
- **Shift+Enter** - New line - **Shift+Enter** - New line
- **Up Arrow** - Edit last user message - **Ctrl+Enter** - Send message (alternative)
- **Left/Right Arrow** - Swipe between responses - **Up Arrow** - Edit last user message (when input is at start)
- **Left/Right Arrow** - Navigate between response alternatives
- **Escape** - Close panels/modals, cancel editing
- **Ctrl+K** - Focus message input
- **Ctrl+/** - Toggle Roleplay Tools panel
## Roadmap ## Roadmap

View File

@@ -481,6 +481,27 @@ function addMessage(content, isUser = false, skipActions = false, timestamp = nu
editBtn.addEventListener('click', () => handleEditMessage(messageDiv, content)); editBtn.addEventListener('click', () => handleEditMessage(messageDiv, content));
actionsDiv.appendChild(editBtn); actionsDiv.appendChild(editBtn);
// Copy message button
const copyBtn = document.createElement('button');
copyBtn.className = 'message-action-btn';
copyBtn.innerHTML = `<svg width="14" height="14" viewBox="0 0 14 14" fill="none">
<rect x="3" y="3" width="7" height="7" rx="1" stroke="currentColor" stroke-width="1.5" fill="none"/>
<path d="M5 3V2C5 1.44772 5.44772 1 6 1H12C12.5523 1 13 1.44772 13 2V8C13 8.55228 12.5523 9 12 9H11" stroke="currentColor" stroke-width="1.5" stroke-linecap="round"/>
</svg>`;
copyBtn.title = 'Copy message';
copyBtn.addEventListener('click', () => {
navigator.clipboard.writeText(content);
// Visual feedback
const originalHTML = copyBtn.innerHTML;
copyBtn.innerHTML = `<svg width="14" height="14" viewBox="0 0 14 14" fill="none">
<path d="M2 7l3 3 7-7" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round"/>
</svg>`;
setTimeout(() => {
copyBtn.innerHTML = originalHTML;
}, 2000);
});
actionsDiv.appendChild(copyBtn);
// Pin button // Pin button
const pinBtn = document.createElement('button'); const pinBtn = document.createElement('button');
pinBtn.className = 'message-action-btn message-pin-btn'; pinBtn.className = 'message-action-btn message-pin-btn';
@@ -557,6 +578,27 @@ function addMessage(content, isUser = false, skipActions = false, timestamp = nu
hideBtn.addEventListener('click', () => handleToggleHidden(messageDiv)); hideBtn.addEventListener('click', () => handleToggleHidden(messageDiv));
actionsDiv.appendChild(hideBtn); actionsDiv.appendChild(hideBtn);
// Copy message button
const copyMsgBtn = document.createElement('button');
copyMsgBtn.className = 'message-action-btn';
copyMsgBtn.innerHTML = `<svg width="14" height="14" viewBox="0 0 14 14" fill="none">
<rect x="3" y="3" width="7" height="7" rx="1" stroke="currentColor" stroke-width="1.5" fill="none"/>
<path d="M5 3V2C5 1.44772 5.44772 1 6 1H12C12.5523 1 13 1.44772 13 2V8C13 8.55228 12.5523 9 12 9H11" stroke="currentColor" stroke-width="1.5" stroke-linecap="round"/>
</svg>`;
copyMsgBtn.title = 'Copy message';
copyMsgBtn.addEventListener('click', () => {
navigator.clipboard.writeText(content);
// Visual feedback
const originalHTML = copyMsgBtn.innerHTML;
copyMsgBtn.innerHTML = `<svg width="14" height="14" viewBox="0 0 14 14" fill="none">
<path d="M2 7l3 3 7-7" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round"/>
</svg>`;
setTimeout(() => {
copyMsgBtn.innerHTML = originalHTML;
}, 2000);
});
actionsDiv.appendChild(copyMsgBtn);
// Delete button // Delete button
const deleteBtn = document.createElement('button'); const deleteBtn = document.createElement('button');
deleteBtn.className = 'message-action-btn message-delete-btn'; deleteBtn.className = 'message-action-btn message-delete-btn';
@@ -3337,9 +3379,98 @@ window.addEventListener('DOMContentLoaded', () => {
avatarModalOverlay.addEventListener('click', hideAvatarModal); avatarModalOverlay.addEventListener('click', hideAvatarModal);
// ESC key to close modal // ESC key to close modal
// Global keyboard shortcuts
document.addEventListener('keydown', (e) => { document.addEventListener('keydown', (e) => {
if (e.key === 'Escape' && avatarModal.style.display !== 'none') { // Escape key handling
hideAvatarModal(); if (e.key === 'Escape') {
// Close avatar modal
if (avatarModal.style.display !== 'none') {
hideAvatarModal();
return;
}
// Close roleplay panel
const roleplayPanel = document.getElementById('roleplay-panel');
if (roleplayPanel && roleplayPanel.classList.contains('active')) {
document.getElementById('close-roleplay-btn').click();
return;
}
// Close settings panel
const settingsPanel = document.getElementById('settings-panel');
if (settingsPanel && settingsPanel.classList.contains('active')) {
document.getElementById('close-settings-btn').click();
return;
}
// Cancel message editing
const editActions = document.querySelector('.message-edit-actions');
if (editActions) {
const cancelBtn = editActions.querySelector('.message-edit-cancel');
if (cancelBtn) cancelBtn.click();
return;
}
}
// Up Arrow - Edit last user message (when input is focused and empty/at start)
if (e.key === 'ArrowUp' && e.target === messageInput && messageInput.selectionStart === 0) {
const messages = document.querySelectorAll('.message.user');
if (messages.length > 0) {
const lastUserMessage = messages[messages.length - 1];
const editBtn = lastUserMessage.querySelector('.message-action-btn[title="Edit message"]');
if (editBtn) {
e.preventDefault();
editBtn.click();
}
}
return;
}
// Left Arrow - Previous swipe (when not in input)
if (e.key === 'ArrowLeft' && e.target !== messageInput && !e.target.matches('input, textarea, select')) {
const lastAssistantMessage = [...document.querySelectorAll('.message.assistant')].pop();
if (lastAssistantMessage) {
const prevBtn = lastAssistantMessage.querySelector('.swipe-prev');
if (prevBtn && !prevBtn.disabled) {
e.preventDefault();
prevBtn.click();
}
}
return;
}
// Right Arrow - Next swipe (when not in input)
if (e.key === 'ArrowRight' && e.target !== messageInput && !e.target.matches('input, textarea, select')) {
const lastAssistantMessage = [...document.querySelectorAll('.message.assistant')].pop();
if (lastAssistantMessage) {
const nextBtn = lastAssistantMessage.querySelector('.swipe-next');
if (nextBtn && !nextBtn.disabled) {
e.preventDefault();
nextBtn.click();
}
}
return;
}
// Ctrl/Cmd + K - Focus message input
if ((e.ctrlKey || e.metaKey) && e.key === 'k') {
e.preventDefault();
messageInput.focus();
return;
}
// Ctrl/Cmd + / - Toggle roleplay panel
if ((e.ctrlKey || e.metaKey) && e.key === '/') {
e.preventDefault();
document.getElementById('roleplay-btn').click();
return;
}
// Ctrl/Cmd + Enter - Send message (alternative to Enter)
if ((e.ctrlKey || e.metaKey) && e.key === 'Enter' && e.target === messageInput) {
e.preventDefault();
handleSubmit(e);
return;
} }
}); });