mirror of
https://github.com/spacedriveapp/spacedrive.git
synced 2026-04-21 06:59:17 -04:00
* update contributing & environment-setup * fix pods deployment target to 13.0 * Track app screen changes with plausible * Don't track onboarding * remove platformType from usePlausibleEvent * submit custom plausible events
212 lines
6.0 KiB
TypeScript
212 lines
6.0 KiB
TypeScript
import { useQueryClient } from '@tanstack/react-query';
|
|
import { ArrowsClockwise, Clipboard, Eye, EyeSlash, Info } from 'phosphor-react';
|
|
import { useState } from 'react';
|
|
import {
|
|
Algorithm,
|
|
HASHING_ALGOS,
|
|
HashingAlgoSlug,
|
|
hashingAlgoSlugSchema,
|
|
useBridgeMutation,
|
|
usePlausibleEvent
|
|
} from '@sd/client';
|
|
import {
|
|
Button,
|
|
CheckBox,
|
|
Dialog,
|
|
Select,
|
|
SelectOption,
|
|
Tooltip,
|
|
UseDialogProps,
|
|
forms,
|
|
useDialog
|
|
} from '@sd/ui';
|
|
import { PasswordMeter } from '~/components/PasswordMeter';
|
|
import { generatePassword } from '~/util';
|
|
|
|
const { Input, z, useZodForm } = forms;
|
|
|
|
const schema = z.object({
|
|
name: z.string(),
|
|
password: z.string(),
|
|
password_validate: z.string(),
|
|
algorithm: z.string(),
|
|
hashing_algorithm: hashingAlgoSlugSchema,
|
|
share_telemetry: z.boolean()
|
|
});
|
|
|
|
export default (props: UseDialogProps) => {
|
|
const dialog = useDialog(props);
|
|
const submitPlausibleEvent = usePlausibleEvent();
|
|
|
|
const form = useZodForm({
|
|
schema,
|
|
defaultValues: {
|
|
password: '',
|
|
algorithm: 'XChaCha20Poly1305',
|
|
hashing_algorithm: 'Argon2id-s'
|
|
}
|
|
});
|
|
|
|
const [showMasterPassword1, setShowMasterPassword1] = useState(false);
|
|
const [showMasterPassword2, setShowMasterPassword2] = useState(false);
|
|
const MP1CurrentEyeIcon = showMasterPassword1 ? EyeSlash : Eye;
|
|
const MP2CurrentEyeIcon = showMasterPassword2 ? EyeSlash : Eye;
|
|
|
|
const queryClient = useQueryClient();
|
|
const createLibrary = useBridgeMutation('library.create', {
|
|
onSuccess: (library) => {
|
|
queryClient.setQueryData(['library.list'], (libraries: any) => [
|
|
...(libraries || []),
|
|
library
|
|
]);
|
|
|
|
submitPlausibleEvent({
|
|
event: {
|
|
type: 'libraryCreate'
|
|
}
|
|
});
|
|
},
|
|
onError: (err: any) => {
|
|
console.error(err);
|
|
}
|
|
});
|
|
|
|
const onSubmit = form.handleSubmit(async (data) => {
|
|
if (data.password !== data.password_validate) {
|
|
alert('Passwords are not the same');
|
|
} else {
|
|
await createLibrary.mutateAsync({
|
|
...data,
|
|
algorithm: data.algorithm as Algorithm,
|
|
hashing_algorithm: HASHING_ALGOS[data.hashing_algorithm],
|
|
auth: {
|
|
type: 'Password',
|
|
value: data.password
|
|
}
|
|
});
|
|
}
|
|
});
|
|
|
|
return (
|
|
<Dialog
|
|
form={form}
|
|
onSubmit={onSubmit}
|
|
dialog={dialog}
|
|
title="Create New Library"
|
|
description="Choose a name for your new library, you can configure this and more settings from the library settings later on."
|
|
submitDisabled={!form.formState.isValid}
|
|
ctaLabel="Create"
|
|
>
|
|
<Input
|
|
label="Library name"
|
|
placeholder="My Cool Library"
|
|
className="mt-2"
|
|
{...form.register('name', { required: true })}
|
|
/>
|
|
|
|
<div className="mt-3 mb-1 flex flex-row items-center">
|
|
<div className="space-x-2">
|
|
<CheckBox
|
|
className="bg-app-selected"
|
|
defaultChecked={true}
|
|
{...form.register('share_telemetry', { required: true })}
|
|
/>
|
|
</div>
|
|
<span className="mt-1 text-xs font-medium">Share anonymous usage</span>
|
|
<Tooltip label="Share completely anonymous telemetry data to help the developers improve the app">
|
|
<Info className="ml-1.5 h-4 w-4 text-ink-faint" />
|
|
</Tooltip>
|
|
</div>
|
|
|
|
{/* TODO: Proper UI for this. Maybe checkbox for encrypted or not and then reveal these fields. Select encrypted by default. */}
|
|
{/* <span className="text-sm">Make the secret key field empty to skip key setup.</span> */}
|
|
|
|
<div className="relative flex flex-col">
|
|
<p className="mt-2 mb-1 text-center text-[0.95rem] font-bold">Key Manager</p>
|
|
<div className="my-1 h-[2px] w-full bg-gray-500" />
|
|
|
|
<Input
|
|
label="Master password"
|
|
placeholder="Password"
|
|
type={showMasterPassword1 ? 'text' : 'password'}
|
|
className="mt-2"
|
|
{...form.register('password')}
|
|
right={
|
|
<div className="flex">
|
|
<Button
|
|
onClick={() => {
|
|
const password = generatePassword(32);
|
|
|
|
form.setValue('password', password);
|
|
form.setValue('password_validate', password);
|
|
|
|
setShowMasterPassword1(true);
|
|
setShowMasterPassword2(true);
|
|
}}
|
|
size="icon"
|
|
>
|
|
<ArrowsClockwise className="h-4 w-4" />
|
|
</Button>
|
|
<Button
|
|
onClick={() => {
|
|
navigator.clipboard.writeText(form.watch('password') as string);
|
|
}}
|
|
size="icon"
|
|
>
|
|
<Clipboard className="h-4 w-4" />
|
|
</Button>
|
|
<Button onClick={() => setShowMasterPassword1(!showMasterPassword1)} size="icon">
|
|
<MP1CurrentEyeIcon className="h-4 w-4" />
|
|
</Button>
|
|
</div>
|
|
}
|
|
/>
|
|
</div>
|
|
|
|
<Input
|
|
label="Master password (again)"
|
|
placeholder="Password"
|
|
type={showMasterPassword2 ? 'text' : 'password'}
|
|
className="mt-2"
|
|
right={
|
|
<Button onClick={() => setShowMasterPassword2(!showMasterPassword2)} size="icon">
|
|
<MP2CurrentEyeIcon className="h-4 w-4" />
|
|
</Button>
|
|
}
|
|
{...form.register('password_validate')}
|
|
/>
|
|
|
|
<div className="mt-4 mb-3 grid w-full grid-cols-2 gap-4">
|
|
<div className="flex flex-col">
|
|
<span className="text-sm font-bold">Encryption</span>
|
|
<Select
|
|
className="mt-2"
|
|
value={form.watch('algorithm')}
|
|
onChange={(e) => form.setValue('algorithm', e)}
|
|
>
|
|
<SelectOption value="XChaCha20Poly1305">XChaCha20-Poly1305</SelectOption>
|
|
<SelectOption value="Aes256Gcm">AES-256-GCM</SelectOption>
|
|
</Select>
|
|
</div>
|
|
<div className="flex flex-col">
|
|
<span className="text-sm font-bold">Hashing</span>
|
|
<Select
|
|
className="mt-2"
|
|
value={form.watch('hashing_algorithm')}
|
|
onChange={(e) => form.setValue('hashing_algorithm', e as HashingAlgoSlug)}
|
|
>
|
|
<SelectOption value="Argon2id-s">Argon2id (standard)</SelectOption>
|
|
<SelectOption value="Argon2id-h">Argon2id (hardened)</SelectOption>
|
|
<SelectOption value="Argon2id-p">Argon2id (paranoid)</SelectOption>
|
|
<SelectOption value="BalloonBlake3-s">BLAKE3-Balloon (standard)</SelectOption>
|
|
<SelectOption value="BalloonBlake3-h">BLAKE3-Balloon (hardened)</SelectOption>
|
|
<SelectOption value="BalloonBlake3-p">BLAKE3-Balloon (paranoid)</SelectOption>
|
|
</Select>
|
|
</div>
|
|
</div>
|
|
|
|
<PasswordMeter password={form.watch('password')} />
|
|
</Dialog>
|
|
);
|
|
};
|