* Fix vite build (tauri build still broken) * Fix yarn builds with a couple of shortcuts * Fix file creation * Fix documentDir behavior * Got stream with default file * Clean up * Clean up * Use 'unstable'; fix devtools callsite The API changed a bit here, which forces us to use the unstable crate feature. The call to open devtools is also new; it's now on the webview not window. Signed-off-by: Paul R. Tagliamonte <paul@kittycad.io> * Bring back read_dir_recursive from v1 * Fix dates * More fixes, incl. conf files * cargo fmt * Add Updater plugin * Fix types * Fix isTauri detection and updater bootup * Schemas * Clean up * Disable devtools * Attempt at fixing builds * WIP Ubuntu dep * WIP Ubuntu dep * WIP keys in debug * Enable updater only on release builds * Reenable webtools on debug * No linux bundles * Typo * Attemp at fixing --bundles none * Manual tauri debug build * Empty commit to trigger the CI * Fix settings * Empty commit to trigger the CI * Merge branch 'main' into pierremtb/issue1349 * Add allow-create perm * tauri-driver no cap * Empty commit to trigger the CI * Clean up * Clean up * Migrate to tauri v2 Fixes #1349 * Fix fmt * Merge branch 'main' into pierremtb/issue1349 * Force BUILD_RELEASE: true * Bump tauri to new beta * Merge branch 'main' into pierremtb/issue1349 * Fix linux tests * Fix types * Add --verbose to tauri-action * Move --verbose to front * Back to tauri-driver@0.1.3 and single \ for win * Back to latest driver, and windows wip * Disable release conf temporarily * Rollback to 2.0.0-beta.2 * Rollback to 2.0.0-beta.1 * Move bundle to root for src-tauri/tauri.release.conf.json * All packages to latest (add http and shell to package.json) * Testing latest commit for tauri-action * Remove tauri action * Add cat * WIP * Update ci.yml * Disable release conf * Disable rust cache * Add tauri-action back for release builds * A snapshot a day keeps the bugs away! 📷🐛 (OS: ubuntu) * Update .codespellrc * A snapshot a day keeps the bugs away! 📷🐛 (OS: ubuntu) * Trigger CI * A snapshot a day keeps the bugs away! 📷🐛 (OS: ubuntu) * A snapshot a day keeps the bugs away! 📷🐛 (OS: ubuntu) * Fix type * Clean up * More clean up * Fix path concatenation with join * Attempt at fixing linux tests * Config clean up * Downgrade to tauri-driver@0.1.3 * Looks like tauri v2 is actually doing better with linux package names ah! * Change Linux apt packages * Increase wdio connectionRetryTimeout * Revert connectionRetryTimeout and bump tauri packages * Back to latest tauri-driver * Disable linux e2e tests * A snapshot a day keeps the bugs away! 📷🐛 (OS: ubuntu) * Trigger CI * Clean up * Update snapshots * A snapshot a day keeps the bugs away! 📷🐛 (OS: ubuntu) * Trigger CI * Remove @sentry/react * A snapshot a day keeps the bugs away! 📷🐛 (OS: ubuntu) * Rename migrated.json to desktop.json * A snapshot a day keeps the bugs away! 📷🐛 (OS: ubuntu) * Trigger CI * Change wasm url to http on Windows --------- Signed-off-by: Paul R. Tagliamonte <paul@kittycad.io> Co-authored-by: Paul R. Tagliamonte <paul@kittycad.io> Co-authored-by: github-actions[bot] <github-actions[bot]@users.noreply.github.com> Co-authored-by: Adam Chalmers <adam.chalmers@zoo.dev>
241 lines
8.2 KiB
TypeScript
241 lines
8.2 KiB
TypeScript
import { FormEvent, useEffect, useRef, useState } from 'react'
|
|
import { type ProjectWithEntryPointMetadata } from 'lib/types'
|
|
import { paths } from 'lib/paths'
|
|
import { Link } from 'react-router-dom'
|
|
import { ActionButton } from './ActionButton'
|
|
import {
|
|
faCheck,
|
|
faPenAlt,
|
|
faTrashAlt,
|
|
faX,
|
|
} from '@fortawesome/free-solid-svg-icons'
|
|
import { getPartsCount, readProject } from '../lib/tauriFS'
|
|
import { FILE_EXT } from 'lib/constants'
|
|
import { Dialog } from '@headlessui/react'
|
|
import { useHotkeys } from 'react-hotkeys-hook'
|
|
import Tooltip from './Tooltip'
|
|
|
|
function ProjectCard({
|
|
project,
|
|
handleRenameProject,
|
|
handleDeleteProject,
|
|
...props
|
|
}: {
|
|
project: ProjectWithEntryPointMetadata
|
|
handleRenameProject: (
|
|
e: FormEvent<HTMLFormElement>,
|
|
f: ProjectWithEntryPointMetadata
|
|
) => Promise<void>
|
|
handleDeleteProject: (f: ProjectWithEntryPointMetadata) => Promise<void>
|
|
}) {
|
|
useHotkeys('esc', () => setIsEditing(false))
|
|
const [isEditing, setIsEditing] = useState(false)
|
|
const [isConfirmingDelete, setIsConfirmingDelete] = useState(false)
|
|
const [numberOfParts, setNumberOfParts] = useState(1)
|
|
const [numberOfFolders, setNumberOfFolders] = useState(0)
|
|
|
|
let inputRef = useRef<HTMLInputElement>(null)
|
|
|
|
function handleSave(e: FormEvent<HTMLFormElement>) {
|
|
e.preventDefault()
|
|
void handleRenameProject(e, project).then(() => setIsEditing(false))
|
|
}
|
|
|
|
function getDisplayedTime(date: Date) {
|
|
const startOfToday = new Date()
|
|
startOfToday.setHours(0, 0, 0, 0)
|
|
return date.getTime() < startOfToday.getTime()
|
|
? date.toLocaleDateString()
|
|
: date.toLocaleTimeString()
|
|
}
|
|
|
|
useEffect(() => {
|
|
async function getNumberOfParts() {
|
|
const { kclFileCount, kclDirCount } = getPartsCount(
|
|
await readProject(project.path)
|
|
)
|
|
setNumberOfParts(kclFileCount)
|
|
setNumberOfFolders(kclDirCount)
|
|
}
|
|
void getNumberOfParts()
|
|
}, [project.path])
|
|
|
|
useEffect(() => {
|
|
if (inputRef.current) {
|
|
inputRef.current.focus()
|
|
inputRef.current.select()
|
|
}
|
|
}, [inputRef.current])
|
|
|
|
return (
|
|
<li
|
|
{...props}
|
|
className="group relative min-h-[5em] p-1 rounded-sm border border-chalkboard-20 dark:border-chalkboard-80 hover:!border-primary"
|
|
>
|
|
{isEditing ? (
|
|
<form onSubmit={handleSave} className="flex gap-2 items-center">
|
|
<input
|
|
className="dark:bg-chalkboard-80 dark:border-chalkboard-40 min-w-0 p-1 focus:outline-none"
|
|
type="text"
|
|
id="newProjectName"
|
|
name="newProjectName"
|
|
autoCorrect="off"
|
|
autoCapitalize="off"
|
|
defaultValue={project.name}
|
|
ref={inputRef}
|
|
/>
|
|
<div className="flex gap-1 items-center">
|
|
<ActionButton
|
|
Element="button"
|
|
type="submit"
|
|
icon={{
|
|
icon: faCheck,
|
|
size: 'sm',
|
|
className: 'p-1',
|
|
bgClassName: '!bg-transparent',
|
|
}}
|
|
className="!p-0"
|
|
>
|
|
<Tooltip position="left" delay={1000}>
|
|
Rename project
|
|
</Tooltip>
|
|
</ActionButton>
|
|
<ActionButton
|
|
Element="button"
|
|
icon={{
|
|
icon: faX,
|
|
size: 'sm',
|
|
iconClassName: 'dark:!text-chalkboard-20',
|
|
bgClassName: '!bg-transparent',
|
|
className: 'p-1',
|
|
}}
|
|
className="!p-0"
|
|
onClick={() => setIsEditing(false)}
|
|
>
|
|
<Tooltip position="left" delay={1000}>
|
|
Cancel
|
|
</Tooltip>
|
|
</ActionButton>
|
|
</div>
|
|
</form>
|
|
) : (
|
|
<>
|
|
<div className="p-1 flex flex-col h-full gap-2">
|
|
<Link
|
|
className="flex-1 !no-underline !text-chalkboard-110 dark:!text-chalkboard-10 after:content-[''] after:absolute after:inset-0"
|
|
to={`${paths.FILE}/${encodeURIComponent(project.path)}`}
|
|
data-testid="project-link"
|
|
>
|
|
{project.name?.replace(FILE_EXT, '')}
|
|
</Link>
|
|
<span className="text-chalkboard-60 text-xs">
|
|
{numberOfParts} part{numberOfParts === 1 ? '' : 's'}{' '}
|
|
{numberOfFolders > 0 &&
|
|
`/ ${numberOfFolders} folder${
|
|
numberOfFolders === 1 ? '' : 's'
|
|
}`}
|
|
</span>
|
|
<span className="text-chalkboard-60 text-xs">
|
|
Edited{' '}
|
|
{project.entrypointMetadata.mtime
|
|
? getDisplayedTime(project.entrypointMetadata.mtime)
|
|
: 'never'}
|
|
</span>
|
|
<div className="absolute z-10 bottom-2 right-2 flex gap-1 items-center opacity-0 group-hover:opacity-100 group-focus-within:opacity-100">
|
|
<ActionButton
|
|
Element="button"
|
|
icon={{
|
|
icon: faPenAlt,
|
|
className: 'p-1',
|
|
iconClassName: 'dark:!text-chalkboard-20',
|
|
bgClassName: '!bg-transparent',
|
|
size: 'xs',
|
|
}}
|
|
onClick={(e) => {
|
|
e.stopPropagation()
|
|
e.nativeEvent.stopPropagation()
|
|
setIsEditing(true)
|
|
}}
|
|
className="!p-0"
|
|
>
|
|
<Tooltip position="left" delay={1000}>
|
|
Rename project
|
|
</Tooltip>
|
|
</ActionButton>
|
|
<ActionButton
|
|
Element="button"
|
|
icon={{
|
|
icon: faTrashAlt,
|
|
className: 'p-1',
|
|
size: 'xs',
|
|
bgClassName: '!bg-transparent',
|
|
iconClassName: '!text-destroy-70',
|
|
}}
|
|
className="!p-0 hover:border-destroy-40 dark:hover:border-destroy-40"
|
|
onClick={(e) => {
|
|
e.stopPropagation()
|
|
e.nativeEvent.stopPropagation()
|
|
setIsConfirmingDelete(true)
|
|
}}
|
|
>
|
|
<Tooltip position="left" delay={1000}>
|
|
Delete project
|
|
</Tooltip>
|
|
</ActionButton>
|
|
</div>
|
|
</div>
|
|
<Dialog
|
|
open={isConfirmingDelete}
|
|
onClose={() => setIsConfirmingDelete(false)}
|
|
className="relative z-50"
|
|
>
|
|
<div className="fixed inset-0 bg-chalkboard-110/80 grid place-content-center">
|
|
<Dialog.Panel className="rounded p-4 bg-chalkboard-10 dark:bg-chalkboard-100 border border-destroy-80 max-w-2xl">
|
|
<Dialog.Title as="h2" className="text-2xl font-bold mb-4">
|
|
Delete File
|
|
</Dialog.Title>
|
|
<Dialog.Description>
|
|
This will permanently delete "{project.name || 'this file'}".
|
|
</Dialog.Description>
|
|
|
|
<p className="my-4">
|
|
Are you sure you want to delete "{project.name || 'this file'}
|
|
"? This action cannot be undone.
|
|
</p>
|
|
|
|
<div className="flex justify-between">
|
|
<ActionButton
|
|
Element="button"
|
|
onClick={async () => {
|
|
await handleDeleteProject(project)
|
|
setIsConfirmingDelete(false)
|
|
}}
|
|
icon={{
|
|
icon: faTrashAlt,
|
|
bgClassName: 'bg-destroy-80',
|
|
className: 'p-1',
|
|
size: 'sm',
|
|
iconClassName: '!text-destroy-70 dark:!text-destroy-40',
|
|
}}
|
|
className="hover:border-destroy-40 dark:hover:border-destroy-40"
|
|
>
|
|
Delete
|
|
</ActionButton>
|
|
<ActionButton
|
|
Element="button"
|
|
onClick={() => setIsConfirmingDelete(false)}
|
|
>
|
|
Cancel
|
|
</ActionButton>
|
|
</div>
|
|
</Dialog.Panel>
|
|
</div>
|
|
</Dialog>
|
|
</>
|
|
)}
|
|
</li>
|
|
)
|
|
}
|
|
|
|
export default ProjectCard
|