Create new workspace, and more optimizations

This commit is contained in:
Gregory Schier
2023-03-18 19:36:31 -07:00
parent b1835561a8
commit 9ac5572094
20 changed files with 272 additions and 133 deletions

View File

@@ -5,7 +5,7 @@
<link rel="icon" type="image/svg+xml" href="/vite.svg" /> <link rel="icon" type="image/svg+xml" href="/vite.svg" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" /> <meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Yaak App</title> <title>Yaak App</title>
<!-- <script src="http://localhost:8097"></script>--> <!-- <script src="http://localhost:8097"></script>-->
<style> <style>
body { body {
background-color: white; background-color: white;

View File

@@ -226,6 +226,23 @@ async fn set_key_value(
Ok(()) Ok(())
} }
#[tauri::command]
async fn create_workspace(
name: &str,
app_handle: AppHandle<Wry>,
db_instance: State<'_, Mutex<Pool<Sqlite>>>,
) -> Result<String, String> {
let pool = &*db_instance.lock().await;
let created_workspace =
models::create_workspace(name, "", pool).await.expect("Failed to create workspace");
app_handle
.emit_all("updated_workspace", &created_workspace)
.unwrap();
Ok(created_workspace.id)
}
#[tauri::command] #[tauri::command]
async fn create_request( async fn create_request(
workspace_id: &str, workspace_id: &str,
@@ -478,6 +495,7 @@ fn main() {
requests, requests,
send_request, send_request,
create_request, create_request,
create_workspace,
update_request, update_request,
delete_request, delete_request,
responses, responses,

View File

@@ -8,9 +8,10 @@ import { matchPath } from 'react-router-dom';
import { keyValueQueryKey } from '../hooks/useKeyValue'; import { keyValueQueryKey } from '../hooks/useKeyValue';
import { requestsQueryKey } from '../hooks/useRequests'; import { requestsQueryKey } from '../hooks/useRequests';
import { responsesQueryKey } from '../hooks/useResponses'; import { responsesQueryKey } from '../hooks/useResponses';
import { workspacesQueryKey } from '../hooks/useWorkspaces';
import { DEFAULT_FONT_SIZE } from '../lib/constants'; import { DEFAULT_FONT_SIZE } from '../lib/constants';
import { extractKeyValue } from '../lib/keyValueStore'; import { extractKeyValue } from '../lib/keyValueStore';
import type { HttpRequest, HttpResponse, KeyValue } from '../lib/models'; import type { HttpRequest, HttpResponse, KeyValue, Workspace } from '../lib/models';
import { convertDates } from '../lib/models'; import { convertDates } from '../lib/models';
import { AppRouter, WORKSPACE_REQUEST_PATH } from './AppRouter'; import { AppRouter, WORKSPACE_REQUEST_PATH } from './AppRouter';
@@ -70,6 +71,25 @@ await listen('updated_response', ({ payload: response }: { payload: HttpResponse
); );
}); });
await listen('updated_workspace', ({ payload: workspace }: { payload: Workspace }) => {
queryClient.setQueryData(workspacesQueryKey(), (workspaces: Workspace[] = []) => {
const newWorkspaces = [];
let found = false;
for (const w of workspaces) {
if (w.id === workspace.id) {
found = true;
newWorkspaces.push(convertDates(workspace));
} else {
newWorkspaces.push(w);
}
}
if (!found) {
newWorkspaces.push(convertDates(workspace));
}
return newWorkspaces;
});
});
await listen('send_request', async () => { await listen('send_request', async () => {
const params = matchPath(WORKSPACE_REQUEST_PATH, window.location.pathname); const params = matchPath(WORKSPACE_REQUEST_PATH, window.location.pathname);
const requestId = params?.params.requestId; const requestId = params?.params.requestId;

View File

@@ -0,0 +1,35 @@
import { memo, useCallback } from 'react';
import { Button } from './core/Button';
import type { DropdownMenuRadioItem } from './core/Dropdown';
import { DropdownMenuRadio, DropdownMenuTrigger } from './core/Dropdown';
type Props = {
method: string;
onChange: (method: string) => void;
};
const items = [
{ label: 'GET', value: 'GET' },
{ label: 'PUT', value: 'PUT' },
{ label: 'POST', value: 'POST' },
{ label: 'PATCH', value: 'PATCH' },
{ label: 'DELETE', value: 'DELETE' },
{ label: 'OPTIONS', value: 'OPTIONS' },
{ label: 'HEAD', value: 'HEAD' },
];
export const RequestMethodDropdown = memo(function RequestMethodDropdown({
method,
onChange,
}: Props) {
const handleChange = useCallback((i: DropdownMenuRadioItem) => onChange(i.value), [onChange]);
return (
<DropdownMenuRadio onValueChange={handleChange} value={method.toUpperCase()} items={items}>
<DropdownMenuTrigger>
<Button type="button" size="sm" className="mx-0.5" justify="start">
{method.toUpperCase()}
</Button>
</DropdownMenuTrigger>
</DropdownMenuRadio>
);
});

View File

@@ -1,9 +1,7 @@
import classnames from 'classnames'; import classnames from 'classnames';
import { useCallback, useMemo } from 'react'; import { useCallback, useMemo } from 'react';
import { useActiveRequest } from '../hooks/useActiveRequest'; import { useActiveRequest } from '../hooks/useActiveRequest';
import { useIsResponseLoading } from '../hooks/useIsResponseLoading';
import { useKeyValue } from '../hooks/useKeyValue'; import { useKeyValue } from '../hooks/useKeyValue';
import { useSendRequest } from '../hooks/useSendRequest';
import { useUpdateRequest } from '../hooks/useUpdateRequest'; import { useUpdateRequest } from '../hooks/useUpdateRequest';
import { tryFormatJson } from '../lib/formatters'; import { tryFormatJson } from '../lib/formatters';
import type { HttpHeader } from '../lib/models'; import type { HttpHeader } from '../lib/models';
@@ -23,8 +21,6 @@ export function RequestPane({ fullHeight, className }: Props) {
const activeRequest = useActiveRequest(); const activeRequest = useActiveRequest();
const activeRequestId = activeRequest?.id ?? null; const activeRequestId = activeRequest?.id ?? null;
const updateRequest = useUpdateRequest(activeRequestId); const updateRequest = useUpdateRequest(activeRequestId);
const sendRequest = useSendRequest(activeRequestId);
const responseLoading = useIsResponseLoading();
const activeTab = useKeyValue<string>({ const activeTab = useKeyValue<string>({
key: ['active_request_body_tab', activeRequestId ?? 'n/a'], key: ['active_request_body_tab', activeRequestId ?? 'n/a'],
initialValue: 'body', initialValue: 'body',
@@ -49,11 +45,9 @@ export function RequestPane({ fullHeight, className }: Props) {
{ value: 'headers', label: 'Headers' }, { value: 'headers', label: 'Headers' },
{ value: 'auth', label: 'Auth' }, { value: 'auth', label: 'Auth' },
], ],
[], [activeRequest?.bodyType],
); );
const handleMethodChange = useCallback((method: string) => updateRequest.mutate({ method }), []);
const handleUrlChange = useCallback((url: string) => updateRequest.mutate({ url }), []);
const handleBodyChange = useCallback((body: string) => updateRequest.mutate({ body }), []); const handleBodyChange = useCallback((body: string) => updateRequest.mutate({ body }), []);
const handleHeadersChange = useCallback( const handleHeadersChange = useCallback(
(headers: HttpHeader[]) => updateRequest.mutate({ headers }), (headers: HttpHeader[]) => updateRequest.mutate({ headers }),
@@ -64,15 +58,7 @@ export function RequestPane({ fullHeight, className }: Props) {
return ( return (
<div className={classnames(className, 'p-2 grid grid-rows-[auto_minmax(0,1fr)] grid-cols-1')}> <div className={classnames(className, 'p-2 grid grid-rows-[auto_minmax(0,1fr)] grid-cols-1')}>
<UrlBar <UrlBar request={activeRequest} />
key={activeRequest.id}
method={activeRequest.method}
url={activeRequest.url}
onMethodChange={handleMethodChange}
onUrlChange={handleUrlChange}
sendRequest={sendRequest}
loading={responseLoading}
/>
<Tabs <Tabs
value={activeTab.value} value={activeTab.value}
onChangeValue={activeTab.set} onChangeValue={activeTab.set}

View File

@@ -1,5 +1,6 @@
import classnames from 'classnames'; import classnames from 'classnames';
import { memo, useEffect, useMemo, useState } from 'react'; import { memo, useEffect, useMemo, useState } from 'react';
import { useActiveRequestId } from '../hooks/useActiveRequestId';
import { useDeleteResponses } from '../hooks/useDeleteResponses'; import { useDeleteResponses } from '../hooks/useDeleteResponses';
import { useDeleteResponse } from '../hooks/useResponseDelete'; import { useDeleteResponse } from '../hooks/useResponseDelete';
import { useResponses } from '../hooks/useResponses'; import { useResponses } from '../hooks/useResponses';
@@ -21,7 +22,8 @@ interface Props {
export const ResponsePane = memo(function ResponsePane({ className }: Props) { export const ResponsePane = memo(function ResponsePane({ className }: Props) {
const [activeResponseId, setActiveResponseId] = useState<string | null>(null); const [activeResponseId, setActiveResponseId] = useState<string | null>(null);
const responses = useResponses(); const activeRequestId = useActiveRequestId();
const responses = useResponses(activeRequestId);
const activeResponse: HttpResponse | null = activeResponseId const activeResponse: HttpResponse | null = activeResponseId
? responses.find((r) => r.id === activeResponseId) ?? null ? responses.find((r) => r.id === activeResponseId) ?? null
: responses[responses.length - 1] ?? null; : responses[responses.length - 1] ?? null;

View File

@@ -1,4 +1,5 @@
import classnames from 'classnames'; import classnames from 'classnames';
import type { MouseEvent as ReactMouseEvent } from 'react';
import React, { memo, useCallback, useEffect, useMemo, useRef, useState } from 'react'; import React, { memo, useCallback, useEffect, useMemo, useRef, useState } from 'react';
import { DndProvider, useDrag, useDrop } from 'react-dnd'; import { DndProvider, useDrag, useDrop } from 'react-dnd';
import { HTML5Backend } from 'react-dnd-html5-backend'; import { HTML5Backend } from 'react-dnd-html5-backend';
@@ -7,7 +8,6 @@ import { useCreateRequest } from '../hooks/useCreateRequest';
import { useDeleteRequest } from '../hooks/useDeleteRequest'; import { useDeleteRequest } from '../hooks/useDeleteRequest';
import { useKeyValue } from '../hooks/useKeyValue'; import { useKeyValue } from '../hooks/useKeyValue';
import { useRequests } from '../hooks/useRequests'; import { useRequests } from '../hooks/useRequests';
import { useTheme } from '../hooks/useTheme';
import { useUpdateRequest } from '../hooks/useUpdateRequest'; import { useUpdateRequest } from '../hooks/useUpdateRequest';
import { clamp } from '../lib/clamp'; import { clamp } from '../lib/clamp';
import type { HttpRequest } from '../lib/models'; import type { HttpRequest } from '../lib/models';
@@ -17,6 +17,7 @@ import { Icon } from './core/Icon';
import { IconButton } from './core/IconButton'; import { IconButton } from './core/IconButton';
import { HStack, VStack } from './core/Stacks'; import { HStack, VStack } from './core/Stacks';
import { WindowDragRegion } from './core/WindowDragRegion'; import { WindowDragRegion } from './core/WindowDragRegion';
import { ToggleThemeButton } from './ToggleThemeButton';
interface Props { interface Props {
className?: string; className?: string;
@@ -45,7 +46,6 @@ export function Container({ className }: Props) {
const requests = useRequests(); const requests = useRequests();
const activeRequest = useActiveRequest(); const activeRequest = useActiveRequest();
const createRequest = useCreateRequest({ navigateAfter: true }); const createRequest = useCreateRequest({ navigateAfter: true });
const { appearance, toggleAppearance } = useTheme();
const moveState = useRef<{ move: (e: MouseEvent) => void; up: () => void } | null>(null); const moveState = useRef<{ move: (e: MouseEvent) => void; up: () => void } | null>(null);
const unsub = () => { const unsub = () => {
@@ -59,7 +59,7 @@ export function Container({ className }: Props) {
width.set(INITIAL_WIDTH); width.set(INITIAL_WIDTH);
}, []); }, []);
const handleResizeStart = useCallback((e: React.MouseEvent<HTMLDivElement>) => { const handleResizeStart = useCallback((e: ReactMouseEvent<HTMLDivElement>) => {
unsub(); unsub();
const mouseStartX = e.clientX; const mouseStartX = e.clientX;
const startWidth = width.value; const startWidth = width.value;
@@ -77,7 +77,9 @@ export function Container({ className }: Props) {
document.documentElement.addEventListener('mouseup', moveState.current.up); document.documentElement.addEventListener('mouseup', moveState.current.up);
setIsRisizing(true); setIsRisizing(true);
}, []); }, []);
const sidebarStyles = useMemo(() => ({ width: width.value }), [width.value]); const sidebarStyles = useMemo(() => ({ width: width.value }), [width.value]);
const sidebarWidth = width.value - 1; // Minus 1 for the border
return ( return (
<div <div
@@ -115,17 +117,13 @@ export function Container({ className }: Props) {
</HStack> </HStack>
<VStack as="ul" className="py-3 overflow-auto h-full" space={1}> <VStack as="ul" className="py-3 overflow-auto h-full" space={1}>
<SidebarItems <SidebarItems
sidebarWidth={sidebarRef.current?.clientWidth ?? 0} sidebarWidth={sidebarWidth}
activeRequestId={activeRequest?.id} activeRequestId={activeRequest?.id}
requests={requests} requests={requests}
/> />
</VStack> </VStack>
<HStack className="mx-1 pb-1" alignItems="center" justifyContent="end"> <HStack className="mx-1 pb-1" alignItems="center" justifyContent="end">
<IconButton <ToggleThemeButton />
title={appearance === 'dark' ? 'Enable light mode' : 'Enable dark mode'}
icon={appearance === 'dark' ? 'moon' : 'sun'}
onClick={toggleAppearance}
/>
</HStack> </HStack>
</div> </div>
); );

View File

@@ -0,0 +1,15 @@
import React from 'react';
import { useTheme } from '../hooks/useTheme';
import { toggleAppearance } from '../lib/theme/window';
import { IconButton } from './core/IconButton';
export function ToggleThemeButton() {
const { appearance, toggleAppearance } = useTheme();
return (
<IconButton
title={appearance === 'dark' ? 'Enable light mode' : 'Enable dark mode'}
icon={appearance === 'dark' ? 'moon' : 'sun'}
onClick={toggleAppearance}
/>
);
}

View File

@@ -1,26 +1,22 @@
import { useCallback } from 'react'; import { memo, useCallback } from 'react';
import { Button } from './core/Button'; import { useIsResponseLoading } from '../hooks/useIsResponseLoading';
import { DropdownMenuRadio, DropdownMenuTrigger } from './core/Dropdown'; import { useSendRequest } from '../hooks/useSendRequest';
import { useUpdateRequest } from '../hooks/useUpdateRequest';
import type { HttpRequest } from '../lib/models';
import { IconButton } from './core/IconButton'; import { IconButton } from './core/IconButton';
import { Input } from './core/Input'; import { Input } from './core/Input';
import type { TabItem } from './core/Tabs/Tabs'; import { RequestMethodDropdown } from './RequestMethodDropdown';
interface Props { interface Props {
sendRequest: () => void; request: HttpRequest;
loading: boolean;
method: string;
url: string;
onMethodChange: (method: string) => void;
onUrlChange: (url: string) => void;
} }
export function UrlBar({ sendRequest, loading, onMethodChange, method, onUrlChange, url }: Props) { export const UrlBar = memo(function UrlBar({ request }: Props) {
const handleMethodChange = useCallback( const sendRequest = useSendRequest(request.id);
(v: TabItem) => { const updateRequest = useUpdateRequest(request.id);
onMethodChange(v.value); const handleMethodChange = useCallback((method: string) => updateRequest.mutate({ method }), []);
}, const handleUrlChange = useCallback((url: string) => updateRequest.mutate({ url }), []);
[onMethodChange], const loading = useIsResponseLoading(request.id);
);
return ( return (
<form <form
onSubmit={async (e) => { onSubmit={async (e) => {
@@ -40,30 +36,10 @@ export function UrlBar({ sendRequest, loading, onMethodChange, method, onUrlChan
name="url" name="url"
label="Enter URL" label="Enter URL"
containerClassName="shadow shadow-gray-100 dark:shadow-gray-0" containerClassName="shadow shadow-gray-100 dark:shadow-gray-0"
onChange={onUrlChange} onChange={handleUrlChange}
defaultValue={url} defaultValue={request.url}
placeholder="Enter a URL..." placeholder="Enter a URL..."
leftSlot={ leftSlot={<RequestMethodDropdown method={request.method} onChange={handleMethodChange} />}
<DropdownMenuRadio
onValueChange={handleMethodChange}
value={method.toUpperCase()}
items={[
{ label: 'GET', value: 'GET' },
{ label: 'PUT', value: 'PUT' },
{ label: 'POST', value: 'POST' },
{ label: 'PATCH', value: 'PATCH' },
{ label: 'DELETE', value: 'DELETE' },
{ label: 'OPTIONS', value: 'OPTIONS' },
{ label: 'HEAD', value: 'HEAD' },
]}
>
<DropdownMenuTrigger>
<Button type="button" disabled={loading} size="sm" className="mx-0.5" justify="start">
{method.toUpperCase()}
</Button>
</DropdownMenuTrigger>
</DropdownMenuRadio>
}
rightSlot={ rightSlot={
<IconButton <IconButton
title="Send Request" title="Send Request"
@@ -78,4 +54,4 @@ export function UrlBar({ sendRequest, loading, onMethodChange, method, onUrlChan
/> />
</form> </form>
); );
} });

View File

@@ -1,3 +1,4 @@
import { DropdownMenuSeparator } from '@radix-ui/react-dropdown-menu';
import classnames from 'classnames'; import classnames from 'classnames';
import { useNavigate } from 'react-router-dom'; import { useNavigate } from 'react-router-dom';
import { useWindowSize } from 'react-use'; import { useWindowSize } from 'react-use';
@@ -14,6 +15,7 @@ import { WindowDragRegion } from './core/WindowDragRegion';
import { RequestPane } from './RequestPane'; import { RequestPane } from './RequestPane';
import { ResponsePane } from './ResponsePane'; import { ResponsePane } from './ResponsePane';
import { Sidebar } from './Sidebar'; import { Sidebar } from './Sidebar';
import { WorkspaceDropdown } from './WorkspaceDropdown';
export default function Workspace() { export default function Workspace() {
const navigate = useNavigate(); const navigate = useNavigate();
@@ -38,19 +40,7 @@ export default function Workspace() {
alignItems="center" alignItems="center"
> >
<div className="flex-1 -ml-2"> <div className="flex-1 -ml-2">
<DropdownMenuRadio <WorkspaceDropdown />
onValueChange={(v) => {
navigate(`/workspaces/${v.value}`);
}}
value={activeWorkspace?.id}
items={workspaces.map((w) => ({ label: w.name, value: w.id }))}
>
<DropdownMenuTrigger>
<Button size="sm" className="!px-2 truncate" forDropdown>
{activeWorkspace?.name ?? 'Unknown'}
</Button>
</DropdownMenuTrigger>
</DropdownMenuRadio>
</div> </div>
<div className="flex-[2] text-center text-gray-700 text-sm truncate"> <div className="flex-[2] text-center text-gray-700 text-sm truncate">
{activeRequest?.name} {activeRequest?.name}

View File

@@ -0,0 +1,46 @@
import { useMemo } from 'react';
import { useNavigate } from 'react-router-dom';
import { useActiveWorkspace } from '../hooks/useActiveWorkspace';
import { useCreateWorkspace } from '../hooks/useCreateWorkspace';
import { useWorkspaces } from '../hooks/useWorkspaces';
import { Button } from './core/Button';
import type { DropdownItem } from './core/Dropdown';
import { Dropdown, DropdownMenuTrigger } from './core/Dropdown';
import { Icon } from './core/Icon';
export function WorkspaceDropdown() {
const navigate = useNavigate();
const workspaces = useWorkspaces();
const activeWorkspace = useActiveWorkspace();
const createWorkspace = useCreateWorkspace({ navigateAfter: true });
const items: DropdownItem[] = useMemo(() => {
const workspaceItems = workspaces.map((w) => ({
label: w.name,
value: w.id,
leftSlot: activeWorkspace?.id === w.id ? <Icon icon="check" /> : <Icon icon="empty" />,
onSelect: () => navigate(`/workspaces/${w.id}`),
}));
return [
...workspaceItems,
'-----',
{
label: 'New Workspace',
value: 'new',
leftSlot: <Icon icon="plus" />,
onSelect: () => createWorkspace.mutate({ name: 'New Workspace' }),
},
];
}, [workspaces, activeWorkspace]);
return (
<Dropdown items={items}>
<DropdownMenuTrigger>
<Button size="sm" className="!px-2 truncate" forDropdown>
{activeWorkspace?.name ?? 'Unknown'}
</Button>
</DropdownMenuTrigger>
</Dropdown>
);
}

View File

@@ -1,6 +1,6 @@
import classnames from 'classnames'; import classnames from 'classnames';
import type { HTMLAttributes } from 'react'; import type { HTMLAttributes } from 'react';
import { forwardRef, useMemo } from 'react'; import { forwardRef, memo, useMemo } from 'react';
import { Link } from 'react-router-dom'; import { Link } from 'react-router-dom';
import { Icon } from './Icon'; import { Icon } from './Icon';
@@ -26,7 +26,7 @@ export type ButtonProps = HTMLAttributes<HTMLElement> & {
}; };
// eslint-disable-next-line @typescript-eslint/no-explicit-any // eslint-disable-next-line @typescript-eslint/no-explicit-any
export const Button = forwardRef<any, ButtonProps>(function Button( const _Button = forwardRef<any, ButtonProps>(function Button(
{ {
to, to,
className, className,
@@ -71,3 +71,5 @@ export const Button = forwardRef<any, ButtonProps>(function Button(
); );
} }
}); });
export const Button = memo(_Button);

View File

@@ -9,6 +9,7 @@ import {
useCallback, useCallback,
useImperativeHandle, useImperativeHandle,
useLayoutEffect, useLayoutEffect,
useMemo,
useState, useState,
} from 'react'; } from 'react';
@@ -61,17 +62,18 @@ export const DropdownMenuRadio = memo(function DropdownMenuRadio({
); );
}); });
export type DropdownItem =
| {
label: string;
onSelect?: () => void;
disabled?: boolean;
leftSlot?: ReactNode;
}
| '-----';
export interface DropdownProps { export interface DropdownProps {
children: ReactElement<typeof DropdownMenuTrigger>; children: ReactElement<typeof DropdownMenuTrigger>;
items: ( items: DropdownItem[];
| {
label: string;
onSelect?: () => void;
disabled?: boolean;
leftSlot?: ReactNode;
}
| '-----'
)[];
} }
export const Dropdown = memo(function Dropdown({ children, items }: DropdownProps) { export const Dropdown = memo(function Dropdown({ children, items }: DropdownProps) {
@@ -106,19 +108,21 @@ interface DropdownMenuPortalProps {
children: ReactNode; children: ReactNode;
} }
function DropdownMenuPortal({ children }: DropdownMenuPortalProps) { const DropdownMenuPortal = memo(function DropdownMenuPortal({ children }: DropdownMenuPortalProps) {
const container = document.querySelector<Element>('#radix-portal'); const container = document.querySelector<Element>('#radix-portal');
if (container === null) return null; if (container === null) return null;
const initial = useMemo(() => ({ opacity: 0 }), []);
const animate = useMemo(() => ({ opacity: 1 }), []);
return ( return (
<D.Portal> <D.Portal>
<motion.div initial={{ opacity: 0 }} animate={{ opacity: 1 }}> <motion.div initial={initial} animate={animate}>
{children} {children}
</motion.div> </motion.div>
</D.Portal> </D.Portal>
); );
} });
const DropdownMenuContent = forwardRef<HTMLDivElement, D.DropdownMenuContentProps>( const _DropdownMenuContent = forwardRef<HTMLDivElement, D.DropdownMenuContentProps>(
function DropdownMenuContent( function DropdownMenuContent(
{ className, children, ...props }: D.DropdownMenuContentProps, { className, children, ...props }: D.DropdownMenuContentProps,
ref: ForwardedRef<HTMLDivElement>, ref: ForwardedRef<HTMLDivElement>,
@@ -127,33 +131,34 @@ const DropdownMenuContent = forwardRef<HTMLDivElement, D.DropdownMenuContentProp
const [divRef, setDivRef] = useState<HTMLDivElement | null>(null); const [divRef, setDivRef] = useState<HTMLDivElement | null>(null);
useImperativeHandle<HTMLDivElement | null, HTMLDivElement | null>(ref, () => divRef); useImperativeHandle<HTMLDivElement | null, HTMLDivElement | null>(ref, () => divRef);
const initDivRef = (ref: HTMLDivElement | null) => { const initDivRef = useCallback((ref: HTMLDivElement | null) => {
setDivRef(ref); setDivRef(ref);
}; }, []);
// Calculate the max height so we can scroll // Calculate the max height so we can scroll
useLayoutEffect(() => { useLayoutEffect(() => {
if (divRef === null) return; if (divRef === null) return;
// Needs to be in a setTimeout because the ref is not positioned yet // Needs to be in a setTimeout because the ref is not positioned yet
// TODO: Make this better? // TODO: Make this better?
setTimeout(() => { const t = setTimeout(() => {
const windowBox = document.documentElement.getBoundingClientRect(); const windowBox = document.documentElement.getBoundingClientRect();
const menuBox = divRef.getBoundingClientRect(); const menuBox = divRef.getBoundingClientRect();
const styles = { maxHeight: windowBox.height - menuBox.top - 5 - 45 }; const styles = { maxHeight: windowBox.height - menuBox.top - 5 - 45 };
setStyles(styles); setStyles(styles);
}); });
return () => clearTimeout(t);
}, [divRef]); }, [divRef]);
return ( return (
<D.Content <D.Content
ref={initDivRef} ref={initDivRef}
align="start" align="start"
style={styles}
className={classnames( className={classnames(
className, className,
'bg-gray-50 rounded-md shadow-lg dark:shadow-gray-0 p-1.5 border border-gray-200', 'bg-gray-50 rounded-md shadow-lg dark:shadow-gray-0 p-1.5 border border-gray-200',
'overflow-auto m-1', 'overflow-auto m-1',
)} )}
style={styles}
{...props} {...props}
> >
{children} {children}
@@ -161,10 +166,11 @@ const DropdownMenuContent = forwardRef<HTMLDivElement, D.DropdownMenuContentProp
); );
}, },
); );
const DropdownMenuContent = memo(_DropdownMenuContent);
type DropdownMenuItemProps = D.DropdownMenuItemProps & ItemInnerProps; type DropdownMenuItemProps = D.DropdownMenuItemProps & ItemInnerProps;
function DropdownMenuItem({ const DropdownMenuItem = memo(function DropdownMenuItem({
leftSlot, leftSlot,
rightSlot, rightSlot,
className, className,
@@ -184,7 +190,7 @@ function DropdownMenuItem({
</ItemInner> </ItemInner>
</D.Item> </D.Item>
); );
} });
// type DropdownMenuCheckboxItemProps = DropdownMenu.DropdownMenuCheckboxItemProps & ItemInnerProps; // type DropdownMenuCheckboxItemProps = DropdownMenu.DropdownMenuCheckboxItemProps & ItemInnerProps;
// //
@@ -230,12 +236,12 @@ const DropdownMenuRadioItem = memo(function DropdownMenuRadioItem({
return ( return (
<D.RadioItem asChild {...props}> <D.RadioItem asChild {...props}>
<ItemInner <ItemInner
rightSlot={rightSlot}
leftSlot={ leftSlot={
<D.ItemIndicator> <D.ItemIndicator>
<CheckIcon /> <CheckIcon />
</D.ItemIndicator> </D.ItemIndicator>
} }
rightSlot={rightSlot}
> >
{children} {children}
</ItemInner> </ItemInner>
@@ -260,7 +266,11 @@ const DropdownMenuRadioItem = memo(function DropdownMenuRadioItem({
// }, // },
// ); // );
function DropdownMenuLabel({ className, children, ...props }: D.DropdownMenuLabelProps) { const DropdownMenuLabel = memo(function DropdownMenuLabel({
className,
children,
...props
}: D.DropdownMenuLabelProps) {
return ( return (
<D.Label asChild {...props}> <D.Label asChild {...props}>
<ItemInner noHover className={classnames(className, 'opacity-50 uppercase text-sm')}> <ItemInner noHover className={classnames(className, 'opacity-50 uppercase text-sm')}>
@@ -268,16 +278,19 @@ function DropdownMenuLabel({ className, children, ...props }: D.DropdownMenuLabe
</ItemInner> </ItemInner>
</D.Label> </D.Label>
); );
} });
function DropdownMenuSeparator({ className, ...props }: D.DropdownMenuSeparatorProps) { const DropdownMenuSeparator = memo(function DropdownMenuSeparator({
className,
...props
}: D.DropdownMenuSeparatorProps) {
return ( return (
<D.Separator <D.Separator
className={classnames(className, 'h-[1px] bg-gray-400 bg-opacity-30 my-1')} className={classnames(className, 'h-[1px] bg-gray-400 bg-opacity-30 my-1')}
{...props} {...props}
/> />
); );
} });
type DropdownMenuTriggerProps = D.DropdownMenuTriggerProps & { type DropdownMenuTriggerProps = D.DropdownMenuTriggerProps & {
children: ReactNode; children: ReactNode;
@@ -304,7 +317,7 @@ interface ItemInnerProps {
className?: string; className?: string;
} }
const ItemInner = forwardRef<HTMLDivElement, ItemInnerProps>(function ItemInner( const _ItemInner = forwardRef<HTMLDivElement, ItemInnerProps>(function ItemInner(
{ leftSlot, rightSlot, children, className, noHover, ...props }: ItemInnerProps, { leftSlot, rightSlot, children, className, noHover, ...props }: ItemInnerProps,
ref, ref,
) { ) {
@@ -324,3 +337,5 @@ const ItemInner = forwardRef<HTMLDivElement, ItemInnerProps>(function ItemInner(
</div> </div>
); );
}); });
const ItemInner = memo(_ItemInner);

View File

@@ -1,5 +1,6 @@
import classnames from 'classnames'; import classnames from 'classnames';
import { forwardRef } from 'react'; import type { MouseEvent } from 'react';
import { forwardRef, memo, useCallback } from 'react';
import { useTimedBoolean } from '../../hooks/useTimedBoolean'; import { useTimedBoolean } from '../../hooks/useTimedBoolean';
import type { ButtonProps } from './Button'; import type { ButtonProps } from './Button';
import { Button } from './Button'; import { Button } from './Button';
@@ -14,7 +15,7 @@ type Props = IconProps &
title: string; title: string;
}; };
export const IconButton = forwardRef<HTMLButtonElement, Props>(function IconButton( const _IconButton = forwardRef<HTMLButtonElement, Props>(function IconButton(
{ {
showConfirm, showConfirm,
icon, icon,
@@ -30,16 +31,20 @@ export const IconButton = forwardRef<HTMLButtonElement, Props>(function IconButt
ref, ref,
) { ) {
const [confirmed, setConfirmed] = useTimedBoolean(); const [confirmed, setConfirmed] = useTimedBoolean();
const handleClick = useCallback(
(e: MouseEvent<HTMLElement>) => {
if (showConfirm) setConfirmed();
onClick?.(e);
},
[onClick],
);
return ( return (
<Button <Button
ref={ref} ref={ref}
aria-hidden={icon === 'empty'} aria-hidden={icon === 'empty'}
disabled={icon === 'empty'} disabled={icon === 'empty'}
tabIndex={tabIndex ?? icon === 'empty' ? -1 : undefined} tabIndex={tabIndex ?? icon === 'empty' ? -1 : undefined}
onClick={(e) => { onClick={handleClick}
if (showConfirm) setConfirmed();
onClick?.(e);
}}
className={classnames( className={classnames(
className, className,
'text-gray-700 hover:text-gray-1000', 'text-gray-700 hover:text-gray-1000',
@@ -63,3 +68,5 @@ export const IconButton = forwardRef<HTMLButtonElement, Props>(function IconButt
</Button> </Button>
); );
}); });
export const IconButton = memo(_IconButton);

View File

@@ -1,5 +1,5 @@
import classnames from 'classnames'; import classnames from 'classnames';
import { useEffect, useMemo, useState } from 'react'; import { memo, useEffect, useMemo, useState } from 'react';
import type { GenericCompletionOption } from './Editor/genericCompletion'; import type { GenericCompletionOption } from './Editor/genericCompletion';
import { IconButton } from './IconButton'; import { IconButton } from './IconButton';
import { Input } from './Input'; import { Input } from './Input';
@@ -21,7 +21,11 @@ interface PairContainer {
id: string; id: string;
} }
export function PairEditor({ pairs: originalPairs, className, onChange }: Props) { export const PairEditor = memo(function PairEditor({
pairs: originalPairs,
className,
onChange,
}: Props) {
const newPairContainer = (): PairContainer => { const newPairContainer = (): PairContainer => {
return { pair: { name: '', value: '' }, id: Math.random().toString() }; return { pair: { name: '', value: '' }, id: Math.random().toString() };
}; };
@@ -79,9 +83,9 @@ export function PairEditor({ pairs: originalPairs, className, onChange }: Props)
</VStack> </VStack>
</div> </div>
); );
} });
function FormRow({ const FormRow = memo(function FormRow({
pairContainer, pairContainer,
onChange, onChange,
onDelete, onDelete,
@@ -146,4 +150,4 @@ function FormRow({
)} )}
</div> </div>
); );
} });

View File

@@ -1,5 +1,6 @@
import * as T from '@radix-ui/react-tabs'; import * as T from '@radix-ui/react-tabs';
import classnames from 'classnames'; import classnames from 'classnames';
import { memo } from 'react';
import type { ReactNode } from 'react'; import type { ReactNode } from 'react';
import { Button } from '../Button'; import { Button } from '../Button';
import type { DropdownMenuRadioItem, DropdownMenuRadioProps } from '../Dropdown'; import type { DropdownMenuRadioItem, DropdownMenuRadioProps } from '../Dropdown';
@@ -29,7 +30,7 @@ interface Props {
children: ReactNode; children: ReactNode;
} }
export function Tabs({ export const Tabs = memo(function Tabs({
value, value,
onChangeValue, onChangeValue,
label, label,
@@ -115,7 +116,7 @@ export function Tabs({
{children} {children}
</T.Root> </T.Root>
); );
} });
interface TabContentProps { interface TabContentProps {
value: string; value: string;
@@ -123,7 +124,11 @@ interface TabContentProps {
className?: string; className?: string;
} }
export function TabContent({ value, children, className }: TabContentProps) { export const TabContent = memo(function TabContent({
value,
children,
className,
}: TabContentProps) {
return ( return (
<T.Content <T.Content
forceMount forceMount
@@ -133,4 +138,4 @@ export function TabContent({ value, children, className }: TabContentProps) {
{children} {children}
</T.Content> </T.Content>
); );
} });

View File

@@ -0,0 +1,18 @@
import { useMutation } from '@tanstack/react-query';
import { invoke } from '@tauri-apps/api';
import { useNavigate } from 'react-router-dom';
import type { Workspace } from '../lib/models';
export function useCreateWorkspace({ navigateAfter }: { navigateAfter: boolean }) {
const navigate = useNavigate();
return useMutation<string, unknown, Pick<Workspace, 'name'>>({
mutationFn: (patch) => {
return invoke('create_workspace', patch);
},
onSuccess: async (workspaceId) => {
if (navigateAfter) {
navigate(`/workspaces/${workspaceId}`);
}
},
});
}

View File

@@ -1,7 +1,7 @@
import { useResponses } from './useResponses'; import { useResponses } from './useResponses';
export function useIsResponseLoading(): boolean { export function useIsResponseLoading(requestId: string | null): boolean {
const responses = useResponses(); const responses = useResponses(requestId);
const response = responses[responses.length - 1]; const response = responses[responses.length - 1];
if (!response) return false; if (!response) return false;
return !(response.body || response.status || response.error); return !(response.body || response.status || response.error);

View File

@@ -2,22 +2,20 @@ import { useQuery } from '@tanstack/react-query';
import { invoke } from '@tauri-apps/api'; import { invoke } from '@tauri-apps/api';
import type { HttpResponse } from '../lib/models'; import type { HttpResponse } from '../lib/models';
import { convertDates } from '../lib/models'; import { convertDates } from '../lib/models';
import { useActiveRequest } from './useActiveRequest';
export function responsesQueryKey(requestId: string) { export function responsesQueryKey(requestId: string) {
return ['http_responses', { requestId }]; return ['http_responses', { requestId }];
} }
export function useResponses() { export function useResponses(requestId: string | null) {
const activeRequest = useActiveRequest();
return ( return (
useQuery<HttpResponse[]>({ useQuery<HttpResponse[]>({
enabled: activeRequest != null, enabled: requestId !== null,
initialData: [], initialData: [],
queryKey: responsesQueryKey(activeRequest?.id ?? 'n/a'), queryKey: responsesQueryKey(requestId ?? 'n/a'),
queryFn: async () => { queryFn: async () => {
const responses = (await invoke('responses', { const responses = (await invoke('responses', {
requestId: activeRequest?.id, requestId,
})) as HttpResponse[]; })) as HttpResponse[];
return responses.map(convertDates); return responses.map(convertDates);
}, },

View File

@@ -3,9 +3,13 @@ import type { Workspace } from '../lib/models';
import { convertDates } from '../lib/models'; import { convertDates } from '../lib/models';
import { useQuery } from '@tanstack/react-query'; import { useQuery } from '@tanstack/react-query';
export function workspacesQueryKey() {
return ['workspaces'];
}
export function useWorkspaces() { export function useWorkspaces() {
return ( return (
useQuery(['workspaces'], async () => { useQuery(workspacesQueryKey(), async () => {
const workspaces = (await invoke('workspaces')) as Workspace[]; const workspaces = (await invoke('workspaces')) as Workspace[];
return workspaces.map(convertDates); return workspaces.map(convertDates);
}).data ?? [] }).data ?? []