feat: improve the allow tool ui (#1473)

This commit is contained in:
Yingjie He
2025-03-03 13:42:49 -08:00
committed by GitHub
parent 426da76cd3
commit 7edb624bdc
3 changed files with 72 additions and 37 deletions

View File

@@ -9,7 +9,7 @@ import {
getTextContent,
getToolRequests,
getToolResponses,
getToolConfirmationRequestId,
getToolConfirmationContent,
} from '../types/message';
import ToolCallConfirmation from './ToolCallConfirmation';
@@ -36,7 +36,8 @@ export default function GooseMessage({ message, metadata, messages, append }: Go
const previousUrls = previousMessage ? extractUrls(getTextContent(previousMessage)) : [];
const urls = toolRequests.length === 0 ? extractUrls(textContent, previousUrls) : [];
const [toolConfirmationId, hasToolConfirmation] = getToolConfirmationRequestId(message);
const toolConfirmationContent = getToolConfirmationContent(message);
const hasToolConfirmation = toolConfirmationContent !== undefined;
// Find tool responses that correspond to the tool requests in this message
const toolResponsesMap = useMemo(() => {
@@ -72,7 +73,12 @@ export default function GooseMessage({ message, metadata, messages, append }: Go
</div>
)}
{hasToolConfirmation && <ToolCallConfirmation toolConfirmationId={toolConfirmationId} />}
{hasToolConfirmation && (
<ToolCallConfirmation
toolConfirmationId={toolConfirmationContent.id}
toolName={toolConfirmationContent.toolName}
/>
)}
{toolRequests.length > 0 && (
<div className="goose-message-tool bg-bgApp border border-borderSubtle dark:border-gray-700 rounded-b-2xl px-4 pt-4 pb-2 mt-1">

View File

@@ -1,11 +1,15 @@
import React, { useState } from 'react';
import { ConfirmToolRequest } from '../utils/toolConfirm';
import { snakeToTitleCase } from '../utils';
import Box from './ui/Box';
export default function ToolConfirmation({ toolConfirmationId }) {
const [disabled, setDisabled] = useState(false);
export default function ToolConfirmation({ toolConfirmationId, toolName }) {
const [clicked, setClicked] = useState(false);
const [status, setStatus] = useState('');
const handleButtonClick = (confirmed) => {
setDisabled(true);
setClicked(true);
setStatus(confirmed ? 'approved' : 'denied');
ConfirmToolRequest(toolConfirmationId, confirmed);
};
@@ -14,13 +18,45 @@ export default function ToolConfirmation({ toolConfirmationId }) {
<div className="goose-message-content bg-bgSubtle rounded-2xl px-4 py-2 rounded-b-none">
Goose would like to call the above tool. Allow?
</div>
{clicked ? (
<div className="goose-message-tool bg-bgApp border border-borderSubtle dark:border-gray-700 rounded-b-2xl px-4 pt-4 pb-2 flex gap-4 mt-1">
<div className="flex items-center">
{status === 'approved' && (
<svg
className="w-5 h-5 text-gray-500"
xmlns="http://www.w3.org/2000/svg"
fill="none"
viewBox="0 0 24 24"
stroke="currentColor"
strokeWidth={2}
>
<path strokeLinecap="round" strokeLinejoin="round" d="M5 13l4 4L19 7" />
</svg>
)}
{status === 'denied' && (
<svg
className="w-5 h-5 text-gray-500"
xmlns="http://www.w3.org/2000/svg"
fill="none"
viewBox="0 0 24 24"
stroke="currentColor"
strokeWidth={2}
>
<path strokeLinecap="round" strokeLinejoin="round" d="M6 18L18 6M6 6l12 12" />
</svg>
)}
<span className="ml-2 text-gray-500 dark:text-gray-400">
{snakeToTitleCase(toolName.substring(toolName.lastIndexOf('__') + 2))} is {status}
</span>
</div>
</div>
) : (
<div className="goose-message-tool bg-bgApp border border-borderSubtle dark:border-gray-700 rounded-b-2xl px-4 pt-4 pb-2 flex gap-4 mt-1">
<button
className={
'bg-black text-white dark:bg-white dark:text-black rounded-full px-6 py-2 transition'
}
onClick={() => handleButtonClick(true)}
disabled={disabled}
>
Allow tool
</button>
@@ -29,11 +65,11 @@ export default function ToolConfirmation({ toolConfirmationId }) {
'bg-white text-black dark:bg-black dark:text-white border border-gray-300 dark:border-gray-700 rounded-full px-6 py-2 transition'
}
onClick={() => handleButtonClick(false)}
disabled={disabled}
>
Deny
</button>
</div>
)}
</>
);
}

View File

@@ -187,20 +187,13 @@ export function getToolResponses(message: Message): ToolResponseMessageContent[]
);
}
export function getToolConfirmationRequestId(message: Message): [string, boolean] {
const hasToolConfirmationRequest = message.content.some(
export function getToolConfirmationContent(
message: Message
): ToolConfirmationRequestMessageContent {
return message.content.find(
(content): content is ToolConfirmationRequestMessageContent =>
content.type === 'toolConfirmationRequest'
);
const contentId = hasToolConfirmationRequest
? message.content.find(
(content): content is ToolConfirmationRequestMessageContent =>
content.type === 'toolConfirmationRequest'
)?.id || ''
: '';
return [contentId, hasToolConfirmationRequest];
}
export function hasCompletedToolCalls(message: Message): boolean {