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:
@@ -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
|
||||||
|
|
||||||
|
|||||||
133
src/main.js
133
src/main.js
@@ -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
|
||||||
|
if (e.key === 'Escape') {
|
||||||
|
// Close avatar modal
|
||||||
|
if (avatarModal.style.display !== 'none') {
|
||||||
hideAvatarModal();
|
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;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user