mirror of
https://github.com/open-webui/open-webui
synced 2025-04-05 05:10:46 +00:00
enh: rich text input toggle option
This commit is contained in:
parent
8b61b39c75
commit
adede5480d
@ -509,6 +509,7 @@
|
|||||||
</InputMenu>
|
</InputMenu>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{#if $settings?.richTextInput ?? true}
|
||||||
<div
|
<div
|
||||||
bind:this={chatInputContainerElement}
|
bind:this={chatInputContainerElement}
|
||||||
id="chat-input-container"
|
id="chat-input-container"
|
||||||
@ -676,6 +677,180 @@
|
|||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
{:else}
|
||||||
|
<textarea
|
||||||
|
id="chat-input"
|
||||||
|
bind:this={chatInputElement}
|
||||||
|
class="scrollbar-hidden bg-gray-50 dark:bg-gray-850 dark:text-gray-100 outline-none w-full py-3 px-1 rounded-xl resize-none h-[48px]"
|
||||||
|
placeholder={placeholder ? placeholder : $i18n.t('Send a Message')}
|
||||||
|
bind:value={prompt}
|
||||||
|
on:keypress={(e) => {
|
||||||
|
if (
|
||||||
|
!$mobile ||
|
||||||
|
!(
|
||||||
|
'ontouchstart' in window ||
|
||||||
|
navigator.maxTouchPoints > 0 ||
|
||||||
|
navigator.msMaxTouchPoints > 0
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
// Prevent Enter key from creating a new line
|
||||||
|
if (e.key === 'Enter' && !e.shiftKey) {
|
||||||
|
e.preventDefault();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Submit the prompt when Enter key is pressed
|
||||||
|
if (prompt !== '' && e.key === 'Enter' && !e.shiftKey) {
|
||||||
|
dispatch('submit', prompt);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
on:keydown={async (e) => {
|
||||||
|
const isCtrlPressed = e.ctrlKey || e.metaKey; // metaKey is for Cmd key on Mac
|
||||||
|
const commandsContainerElement =
|
||||||
|
document.getElementById('commands-container');
|
||||||
|
|
||||||
|
// Command/Ctrl + Shift + Enter to submit a message pair
|
||||||
|
if (isCtrlPressed && e.key === 'Enter' && e.shiftKey) {
|
||||||
|
e.preventDefault();
|
||||||
|
createMessagePair(prompt);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if Ctrl + R is pressed
|
||||||
|
if (prompt === '' && isCtrlPressed && e.key.toLowerCase() === 'r') {
|
||||||
|
e.preventDefault();
|
||||||
|
console.log('regenerate');
|
||||||
|
|
||||||
|
const regenerateButton = [
|
||||||
|
...document.getElementsByClassName('regenerate-response-button')
|
||||||
|
]?.at(-1);
|
||||||
|
|
||||||
|
regenerateButton?.click();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (prompt === '' && e.key == 'ArrowUp') {
|
||||||
|
e.preventDefault();
|
||||||
|
|
||||||
|
const userMessageElement = [
|
||||||
|
...document.getElementsByClassName('user-message')
|
||||||
|
]?.at(-1);
|
||||||
|
|
||||||
|
const editButton = [
|
||||||
|
...document.getElementsByClassName('edit-user-message-button')
|
||||||
|
]?.at(-1);
|
||||||
|
|
||||||
|
console.log(userMessageElement);
|
||||||
|
|
||||||
|
userMessageElement.scrollIntoView({ block: 'center' });
|
||||||
|
editButton?.click();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (commandsContainerElement && e.key === 'ArrowUp') {
|
||||||
|
e.preventDefault();
|
||||||
|
commandsElement.selectUp();
|
||||||
|
|
||||||
|
const commandOptionButton = [
|
||||||
|
...document.getElementsByClassName('selected-command-option-button')
|
||||||
|
]?.at(-1);
|
||||||
|
commandOptionButton.scrollIntoView({ block: 'center' });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (commandsContainerElement && e.key === 'ArrowDown') {
|
||||||
|
e.preventDefault();
|
||||||
|
commandsElement.selectDown();
|
||||||
|
|
||||||
|
const commandOptionButton = [
|
||||||
|
...document.getElementsByClassName('selected-command-option-button')
|
||||||
|
]?.at(-1);
|
||||||
|
commandOptionButton.scrollIntoView({ block: 'center' });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (commandsContainerElement && e.key === 'Enter') {
|
||||||
|
e.preventDefault();
|
||||||
|
|
||||||
|
const commandOptionButton = [
|
||||||
|
...document.getElementsByClassName('selected-command-option-button')
|
||||||
|
]?.at(-1);
|
||||||
|
|
||||||
|
if (e.shiftKey) {
|
||||||
|
prompt = `${prompt}\n`;
|
||||||
|
} else if (commandOptionButton) {
|
||||||
|
commandOptionButton?.click();
|
||||||
|
} else {
|
||||||
|
document.getElementById('send-message-button')?.click();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (commandsContainerElement && e.key === 'Tab') {
|
||||||
|
e.preventDefault();
|
||||||
|
|
||||||
|
const commandOptionButton = [
|
||||||
|
...document.getElementsByClassName('selected-command-option-button')
|
||||||
|
]?.at(-1);
|
||||||
|
|
||||||
|
commandOptionButton?.click();
|
||||||
|
} else if (e.key === 'Tab') {
|
||||||
|
const words = findWordIndices(prompt);
|
||||||
|
|
||||||
|
if (words.length > 0) {
|
||||||
|
const word = words.at(0);
|
||||||
|
const fullPrompt = prompt;
|
||||||
|
|
||||||
|
prompt = prompt.substring(0, word?.endIndex + 1);
|
||||||
|
await tick();
|
||||||
|
|
||||||
|
e.target.scrollTop = e.target.scrollHeight;
|
||||||
|
prompt = fullPrompt;
|
||||||
|
await tick();
|
||||||
|
|
||||||
|
e.preventDefault();
|
||||||
|
e.target.setSelectionRange(word?.startIndex, word.endIndex + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
e.target.style.height = '';
|
||||||
|
e.target.style.height = Math.min(e.target.scrollHeight, 200) + 'px';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (e.key === 'Escape') {
|
||||||
|
console.log('Escape');
|
||||||
|
atSelectedModel = undefined;
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
rows="1"
|
||||||
|
on:input={async (e) => {
|
||||||
|
e.target.style.height = '';
|
||||||
|
e.target.style.height = Math.min(e.target.scrollHeight, 200) + 'px';
|
||||||
|
user = null;
|
||||||
|
}}
|
||||||
|
on:focus={async (e) => {
|
||||||
|
e.target.style.height = '';
|
||||||
|
e.target.style.height = Math.min(e.target.scrollHeight, 200) + 'px';
|
||||||
|
}}
|
||||||
|
on:paste={async (e) => {
|
||||||
|
const clipboardData = e.clipboardData || window.clipboardData;
|
||||||
|
|
||||||
|
if (clipboardData && clipboardData.items) {
|
||||||
|
for (const item of clipboardData.items) {
|
||||||
|
if (item.type.indexOf('image') !== -1) {
|
||||||
|
const blob = item.getAsFile();
|
||||||
|
const reader = new FileReader();
|
||||||
|
|
||||||
|
reader.onload = function (e) {
|
||||||
|
files = [
|
||||||
|
...files,
|
||||||
|
{
|
||||||
|
type: 'image',
|
||||||
|
url: `${e.target.result}`
|
||||||
|
}
|
||||||
|
];
|
||||||
|
};
|
||||||
|
|
||||||
|
reader.readAsDataURL(blob);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
{/if}
|
||||||
|
|
||||||
<div class="self-end mb-2 flex space-x-1 mr-1">
|
<div class="self-end mb-2 flex space-x-1 mr-1">
|
||||||
{#if !history?.currentId || history.messages[history.currentId]?.done == true}
|
{#if !history?.currentId || history.messages[history.currentId]?.done == true}
|
||||||
|
@ -120,13 +120,16 @@
|
|||||||
const chatInputElement = document.getElementById('chat-input');
|
const chatInputElement = document.getElementById('chat-input');
|
||||||
|
|
||||||
await tick();
|
await tick();
|
||||||
|
|
||||||
if (chatInputContainerElement) {
|
if (chatInputContainerElement) {
|
||||||
chatInputContainerElement.style.height = '';
|
chatInputContainerElement.style.height = '';
|
||||||
chatInputContainerElement.style.height =
|
chatInputContainerElement.style.height =
|
||||||
Math.min(chatInputContainerElement.scrollHeight, 200) + 'px';
|
Math.min(chatInputContainerElement.scrollHeight, 200) + 'px';
|
||||||
|
}
|
||||||
|
|
||||||
chatInputElement?.focus();
|
await tick();
|
||||||
|
if (chatInputElement) {
|
||||||
|
chatInputElement.focus();
|
||||||
|
chatInputElement.dispatchEvent(new Event('input'));
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
|
@ -58,13 +58,18 @@
|
|||||||
await tick();
|
await tick();
|
||||||
|
|
||||||
const chatInputContainerElement = document.getElementById('chat-input-container');
|
const chatInputContainerElement = document.getElementById('chat-input-container');
|
||||||
|
const chatInputElement = document.getElementById('chat-input');
|
||||||
|
|
||||||
if (chatInputContainerElement) {
|
if (chatInputContainerElement) {
|
||||||
chatInputContainerElement.style.height = '';
|
chatInputContainerElement.style.height = '';
|
||||||
chatInputContainerElement.style.height =
|
chatInputContainerElement.style.height =
|
||||||
Math.min(chatInputContainerElement.scrollHeight, 200) + 'px';
|
Math.min(chatInputContainerElement.scrollHeight, 200) + 'px';
|
||||||
|
}
|
||||||
|
|
||||||
const chatInputElement = document.getElementById('chat-input');
|
await tick();
|
||||||
chatInputElement?.focus();
|
if (chatInputElement) {
|
||||||
|
chatInputElement.focus();
|
||||||
|
chatInputElement.dispatchEvent(new Event('input'));
|
||||||
}
|
}
|
||||||
|
|
||||||
await tick();
|
await tick();
|
||||||
|
@ -30,6 +30,7 @@
|
|||||||
// Interface
|
// Interface
|
||||||
let defaultModelId = '';
|
let defaultModelId = '';
|
||||||
let showUsername = false;
|
let showUsername = false;
|
||||||
|
let richTextInput = true;
|
||||||
|
|
||||||
let landingPageMode = '';
|
let landingPageMode = '';
|
||||||
let chatBubble = true;
|
let chatBubble = true;
|
||||||
@ -125,6 +126,11 @@
|
|||||||
saveSettings({ autoTags });
|
saveSettings({ autoTags });
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const toggleRichTextInput = async () => {
|
||||||
|
richTextInput = !richTextInput;
|
||||||
|
saveSettings({ richTextInput });
|
||||||
|
};
|
||||||
|
|
||||||
const toggleResponseAutoCopy = async () => {
|
const toggleResponseAutoCopy = async () => {
|
||||||
const permission = await navigator.clipboard
|
const permission = await navigator.clipboard
|
||||||
.readText()
|
.readText()
|
||||||
@ -172,6 +178,7 @@
|
|||||||
showEmojiInCall = $settings.showEmojiInCall ?? false;
|
showEmojiInCall = $settings.showEmojiInCall ?? false;
|
||||||
voiceInterruption = $settings.voiceInterruption ?? false;
|
voiceInterruption = $settings.voiceInterruption ?? false;
|
||||||
|
|
||||||
|
richTextInput = $settings.richTextInput ?? true;
|
||||||
landingPageMode = $settings.landingPageMode ?? '';
|
landingPageMode = $settings.landingPageMode ?? '';
|
||||||
chatBubble = $settings.chatBubble ?? true;
|
chatBubble = $settings.chatBubble ?? true;
|
||||||
widescreenMode = $settings.widescreenMode ?? false;
|
widescreenMode = $settings.widescreenMode ?? false;
|
||||||
@ -422,6 +429,28 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<div class=" py-0.5 flex w-full justify-between">
|
||||||
|
<div class=" self-center text-xs">
|
||||||
|
{$i18n.t('Rich Text Input for Chat')}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<button
|
||||||
|
class="p-1 px-3 text-xs flex rounded transition"
|
||||||
|
on:click={() => {
|
||||||
|
toggleRichTextInput();
|
||||||
|
}}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
{#if richTextInput === true}
|
||||||
|
<span class="ml-2 self-center">{$i18n.t('On')}</span>
|
||||||
|
{:else}
|
||||||
|
<span class="ml-2 self-center">{$i18n.t('Off')}</span>
|
||||||
|
{/if}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
<div class=" py-0.5 flex w-full justify-between">
|
<div class=" py-0.5 flex w-full justify-between">
|
||||||
<div class=" self-center text-xs">
|
<div class=" self-center text-xs">
|
||||||
|
Loading…
Reference in New Issue
Block a user