Files
spacedrive/interface/app/$libraryId/settings/node/libraries/CreateDialog.tsx
Utku 63afacbc54 Mobile Analytics and some changes (#667)
* 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
2023-04-04 05:41:09 +00:00

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>
);
};