feat: impl file rename
This commit is contained in:
2
packages/convex/_generated/api.d.ts
vendored
2
packages/convex/_generated/api.d.ts
vendored
@@ -17,6 +17,7 @@ import type * as files from "../files.js";
|
|||||||
import type * as functions from "../functions.js";
|
import type * as functions from "../functions.js";
|
||||||
import type * as model_directories from "../model/directories.js";
|
import type * as model_directories from "../model/directories.js";
|
||||||
import type * as model_error from "../model/error.js";
|
import type * as model_error from "../model/error.js";
|
||||||
|
import type * as model_files from "../model/files.js";
|
||||||
import type * as model_user from "../model/user.js";
|
import type * as model_user from "../model/user.js";
|
||||||
import type * as users from "../users.js";
|
import type * as users from "../users.js";
|
||||||
|
|
||||||
@@ -33,6 +34,7 @@ declare const fullApi: ApiFromModules<{
|
|||||||
functions: typeof functions;
|
functions: typeof functions;
|
||||||
"model/directories": typeof model_directories;
|
"model/directories": typeof model_directories;
|
||||||
"model/error": typeof model_error;
|
"model/error": typeof model_error;
|
||||||
|
"model/files": typeof model_files;
|
||||||
"model/user": typeof model_user;
|
"model/user": typeof model_user;
|
||||||
users: typeof users;
|
users: typeof users;
|
||||||
}>;
|
}>;
|
||||||
|
@@ -3,6 +3,7 @@ import { v } from "convex/values"
|
|||||||
import { authenticatedMutation, authenticatedQuery } from "./functions"
|
import { authenticatedMutation, authenticatedQuery } from "./functions"
|
||||||
import type { DirectoryItem } from "./model/directories"
|
import type { DirectoryItem } from "./model/directories"
|
||||||
import * as Directories from "./model/directories"
|
import * as Directories from "./model/directories"
|
||||||
|
import * as Files from "./model/files"
|
||||||
|
|
||||||
export const generateUploadUrl = authenticatedMutation({
|
export const generateUploadUrl = authenticatedMutation({
|
||||||
handler: async (ctx) => {
|
handler: async (ctx) => {
|
||||||
@@ -72,6 +73,17 @@ export const saveFile = authenticatedMutation({
|
|||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
|
export const renameFile = authenticatedMutation({
|
||||||
|
args: {
|
||||||
|
directoryId: v.optional(v.id("directories")),
|
||||||
|
itemId: v.id("files"),
|
||||||
|
newName: v.string(),
|
||||||
|
},
|
||||||
|
handler: async (ctx, { directoryId, itemId, newName }) => {
|
||||||
|
await Files.renameFile(ctx, { directoryId, itemId, newName })
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
export const moveToTrash = authenticatedMutation({
|
export const moveToTrash = authenticatedMutation({
|
||||||
args: {
|
args: {
|
||||||
kind: v.union(v.literal("file"), v.literal("directory")),
|
kind: v.union(v.literal("file"), v.literal("directory")),
|
||||||
|
36
packages/convex/model/files.ts
Normal file
36
packages/convex/model/files.ts
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
import type { Id } from "../_generated/dataModel"
|
||||||
|
import type { AuthenticatedMutationCtx } from "../functions"
|
||||||
|
import * as Err from "./error"
|
||||||
|
|
||||||
|
export async function renameFile(
|
||||||
|
ctx: AuthenticatedMutationCtx,
|
||||||
|
{
|
||||||
|
directoryId,
|
||||||
|
itemId,
|
||||||
|
newName,
|
||||||
|
}: {
|
||||||
|
directoryId?: Id<"directories">
|
||||||
|
itemId: Id<"files">
|
||||||
|
newName: string
|
||||||
|
},
|
||||||
|
) {
|
||||||
|
const existing = await ctx.db
|
||||||
|
.query("files")
|
||||||
|
.withIndex("uniqueFileInDirectory", (q) =>
|
||||||
|
q
|
||||||
|
.eq("userId", ctx.user._id)
|
||||||
|
.eq("directoryId", directoryId)
|
||||||
|
.eq("name", newName)
|
||||||
|
.eq("deletedAt", undefined),
|
||||||
|
)
|
||||||
|
.first()
|
||||||
|
|
||||||
|
if (existing) {
|
||||||
|
throw Err.create(
|
||||||
|
Err.Code.FileExists,
|
||||||
|
`File with name ${newName} already exists in ${directoryId ? `directory ${directoryId}` : "root"}`,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
await ctx.db.patch(itemId, { name: newName })
|
||||||
|
}
|
33
packages/web/convex/_generated/api.d.ts
vendored
33
packages/web/convex/_generated/api.d.ts
vendored
@@ -1,33 +0,0 @@
|
|||||||
/* eslint-disable */
|
|
||||||
/**
|
|
||||||
* Generated `api` utility.
|
|
||||||
*
|
|
||||||
* THIS CODE IS AUTOMATICALLY GENERATED.
|
|
||||||
*
|
|
||||||
* To regenerate, run `npx convex dev`.
|
|
||||||
* @module
|
|
||||||
*/
|
|
||||||
|
|
||||||
import type {
|
|
||||||
ApiFromModules,
|
|
||||||
FilterApi,
|
|
||||||
FunctionReference,
|
|
||||||
} from "convex/server";
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A utility for referencing Convex functions in your app's API.
|
|
||||||
*
|
|
||||||
* Usage:
|
|
||||||
* ```js
|
|
||||||
* const myFunctionReference = api.myModule.myFunction;
|
|
||||||
* ```
|
|
||||||
*/
|
|
||||||
declare const fullApi: ApiFromModules<{}>;
|
|
||||||
export declare const api: FilterApi<
|
|
||||||
typeof fullApi,
|
|
||||||
FunctionReference<any, "public">
|
|
||||||
>;
|
|
||||||
export declare const internal: FilterApi<
|
|
||||||
typeof fullApi,
|
|
||||||
FunctionReference<any, "internal">
|
|
||||||
>;
|
|
@@ -1,22 +0,0 @@
|
|||||||
/* eslint-disable */
|
|
||||||
/**
|
|
||||||
* Generated `api` utility.
|
|
||||||
*
|
|
||||||
* THIS CODE IS AUTOMATICALLY GENERATED.
|
|
||||||
*
|
|
||||||
* To regenerate, run `npx convex dev`.
|
|
||||||
* @module
|
|
||||||
*/
|
|
||||||
|
|
||||||
import { anyApi } from "convex/server";
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A utility for referencing Convex functions in your app's API.
|
|
||||||
*
|
|
||||||
* Usage:
|
|
||||||
* ```js
|
|
||||||
* const myFunctionReference = api.myModule.myFunction;
|
|
||||||
* ```
|
|
||||||
*/
|
|
||||||
export const api = anyApi;
|
|
||||||
export const internal = anyApi;
|
|
58
packages/web/convex/_generated/dataModel.d.ts
vendored
58
packages/web/convex/_generated/dataModel.d.ts
vendored
@@ -1,58 +0,0 @@
|
|||||||
/* eslint-disable */
|
|
||||||
/**
|
|
||||||
* Generated data model types.
|
|
||||||
*
|
|
||||||
* THIS CODE IS AUTOMATICALLY GENERATED.
|
|
||||||
*
|
|
||||||
* To regenerate, run `npx convex dev`.
|
|
||||||
* @module
|
|
||||||
*/
|
|
||||||
|
|
||||||
import { AnyDataModel } from "convex/server";
|
|
||||||
import type { GenericId } from "convex/values";
|
|
||||||
|
|
||||||
/**
|
|
||||||
* No `schema.ts` file found!
|
|
||||||
*
|
|
||||||
* This generated code has permissive types like `Doc = any` because
|
|
||||||
* Convex doesn't know your schema. If you'd like more type safety, see
|
|
||||||
* https://docs.convex.dev/using/schemas for instructions on how to add a
|
|
||||||
* schema file.
|
|
||||||
*
|
|
||||||
* After you change a schema, rerun codegen with `npx convex dev`.
|
|
||||||
*/
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The names of all of your Convex tables.
|
|
||||||
*/
|
|
||||||
export type TableNames = string;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The type of a document stored in Convex.
|
|
||||||
*/
|
|
||||||
export type Doc = any;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* An identifier for a document in Convex.
|
|
||||||
*
|
|
||||||
* Convex documents are uniquely identified by their `Id`, which is accessible
|
|
||||||
* on the `_id` field. To learn more, see [Document IDs](https://docs.convex.dev/using/document-ids).
|
|
||||||
*
|
|
||||||
* Documents can be loaded using `db.get(id)` in query and mutation functions.
|
|
||||||
*
|
|
||||||
* IDs are just strings at runtime, but this type can be used to distinguish them from other
|
|
||||||
* strings when type checking.
|
|
||||||
*/
|
|
||||||
export type Id<TableName extends TableNames = TableNames> =
|
|
||||||
GenericId<TableName>;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A type describing your Convex data model.
|
|
||||||
*
|
|
||||||
* This type includes information about what tables you have, the type of
|
|
||||||
* documents stored in those tables, and the indexes defined on them.
|
|
||||||
*
|
|
||||||
* This type is used to parameterize methods like `queryGeneric` and
|
|
||||||
* `mutationGeneric` to make them type-safe.
|
|
||||||
*/
|
|
||||||
export type DataModel = AnyDataModel;
|
|
142
packages/web/convex/_generated/server.d.ts
vendored
142
packages/web/convex/_generated/server.d.ts
vendored
@@ -1,142 +0,0 @@
|
|||||||
/* eslint-disable */
|
|
||||||
/**
|
|
||||||
* Generated utilities for implementing server-side Convex query and mutation functions.
|
|
||||||
*
|
|
||||||
* THIS CODE IS AUTOMATICALLY GENERATED.
|
|
||||||
*
|
|
||||||
* To regenerate, run `npx convex dev`.
|
|
||||||
* @module
|
|
||||||
*/
|
|
||||||
|
|
||||||
import {
|
|
||||||
ActionBuilder,
|
|
||||||
HttpActionBuilder,
|
|
||||||
MutationBuilder,
|
|
||||||
QueryBuilder,
|
|
||||||
GenericActionCtx,
|
|
||||||
GenericMutationCtx,
|
|
||||||
GenericQueryCtx,
|
|
||||||
GenericDatabaseReader,
|
|
||||||
GenericDatabaseWriter,
|
|
||||||
} from "convex/server";
|
|
||||||
import type { DataModel } from "./dataModel.js";
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Define a query in this Convex app's public API.
|
|
||||||
*
|
|
||||||
* This function will be allowed to read your Convex database and will be accessible from the client.
|
|
||||||
*
|
|
||||||
* @param func - The query function. It receives a {@link QueryCtx} as its first argument.
|
|
||||||
* @returns The wrapped query. Include this as an `export` to name it and make it accessible.
|
|
||||||
*/
|
|
||||||
export declare const query: QueryBuilder<DataModel, "public">;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Define a query that is only accessible from other Convex functions (but not from the client).
|
|
||||||
*
|
|
||||||
* This function will be allowed to read from your Convex database. It will not be accessible from the client.
|
|
||||||
*
|
|
||||||
* @param func - The query function. It receives a {@link QueryCtx} as its first argument.
|
|
||||||
* @returns The wrapped query. Include this as an `export` to name it and make it accessible.
|
|
||||||
*/
|
|
||||||
export declare const internalQuery: QueryBuilder<DataModel, "internal">;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Define a mutation in this Convex app's public API.
|
|
||||||
*
|
|
||||||
* This function will be allowed to modify your Convex database and will be accessible from the client.
|
|
||||||
*
|
|
||||||
* @param func - The mutation function. It receives a {@link MutationCtx} as its first argument.
|
|
||||||
* @returns The wrapped mutation. Include this as an `export` to name it and make it accessible.
|
|
||||||
*/
|
|
||||||
export declare const mutation: MutationBuilder<DataModel, "public">;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Define a mutation that is only accessible from other Convex functions (but not from the client).
|
|
||||||
*
|
|
||||||
* This function will be allowed to modify your Convex database. It will not be accessible from the client.
|
|
||||||
*
|
|
||||||
* @param func - The mutation function. It receives a {@link MutationCtx} as its first argument.
|
|
||||||
* @returns The wrapped mutation. Include this as an `export` to name it and make it accessible.
|
|
||||||
*/
|
|
||||||
export declare const internalMutation: MutationBuilder<DataModel, "internal">;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Define an action in this Convex app's public API.
|
|
||||||
*
|
|
||||||
* An action is a function which can execute any JavaScript code, including non-deterministic
|
|
||||||
* code and code with side-effects, like calling third-party services.
|
|
||||||
* They can be run in Convex's JavaScript environment or in Node.js using the "use node" directive.
|
|
||||||
* They can interact with the database indirectly by calling queries and mutations using the {@link ActionCtx}.
|
|
||||||
*
|
|
||||||
* @param func - The action. It receives an {@link ActionCtx} as its first argument.
|
|
||||||
* @returns The wrapped action. Include this as an `export` to name it and make it accessible.
|
|
||||||
*/
|
|
||||||
export declare const action: ActionBuilder<DataModel, "public">;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Define an action that is only accessible from other Convex functions (but not from the client).
|
|
||||||
*
|
|
||||||
* @param func - The function. It receives an {@link ActionCtx} as its first argument.
|
|
||||||
* @returns The wrapped function. Include this as an `export` to name it and make it accessible.
|
|
||||||
*/
|
|
||||||
export declare const internalAction: ActionBuilder<DataModel, "internal">;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Define an HTTP action.
|
|
||||||
*
|
|
||||||
* This function will be used to respond to HTTP requests received by a Convex
|
|
||||||
* deployment if the requests matches the path and method where this action
|
|
||||||
* is routed. Be sure to route your action in `convex/http.js`.
|
|
||||||
*
|
|
||||||
* @param func - The function. It receives an {@link ActionCtx} as its first argument.
|
|
||||||
* @returns The wrapped function. Import this function from `convex/http.js` and route it to hook it up.
|
|
||||||
*/
|
|
||||||
export declare const httpAction: HttpActionBuilder;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A set of services for use within Convex query functions.
|
|
||||||
*
|
|
||||||
* The query context is passed as the first argument to any Convex query
|
|
||||||
* function run on the server.
|
|
||||||
*
|
|
||||||
* This differs from the {@link MutationCtx} because all of the services are
|
|
||||||
* read-only.
|
|
||||||
*/
|
|
||||||
export type QueryCtx = GenericQueryCtx<DataModel>;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A set of services for use within Convex mutation functions.
|
|
||||||
*
|
|
||||||
* The mutation context is passed as the first argument to any Convex mutation
|
|
||||||
* function run on the server.
|
|
||||||
*/
|
|
||||||
export type MutationCtx = GenericMutationCtx<DataModel>;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A set of services for use within Convex action functions.
|
|
||||||
*
|
|
||||||
* The action context is passed as the first argument to any Convex action
|
|
||||||
* function run on the server.
|
|
||||||
*/
|
|
||||||
export type ActionCtx = GenericActionCtx<DataModel>;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* An interface to read from the database within Convex query functions.
|
|
||||||
*
|
|
||||||
* The two entry points are {@link DatabaseReader.get}, which fetches a single
|
|
||||||
* document by its {@link Id}, or {@link DatabaseReader.query}, which starts
|
|
||||||
* building a query.
|
|
||||||
*/
|
|
||||||
export type DatabaseReader = GenericDatabaseReader<DataModel>;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* An interface to read from and write to the database within Convex mutation
|
|
||||||
* functions.
|
|
||||||
*
|
|
||||||
* Convex guarantees that all writes within a single mutation are
|
|
||||||
* executed atomically, so you never have to worry about partial writes leaving
|
|
||||||
* your data in an inconsistent state. See [the Convex Guide](https://docs.convex.dev/understanding/convex-fundamentals/functions#atomicity-and-optimistic-concurrency-control)
|
|
||||||
* for the guarantees Convex provides your functions.
|
|
||||||
*/
|
|
||||||
export type DatabaseWriter = GenericDatabaseWriter<DataModel>;
|
|
@@ -1,89 +0,0 @@
|
|||||||
/* eslint-disable */
|
|
||||||
/**
|
|
||||||
* Generated utilities for implementing server-side Convex query and mutation functions.
|
|
||||||
*
|
|
||||||
* THIS CODE IS AUTOMATICALLY GENERATED.
|
|
||||||
*
|
|
||||||
* To regenerate, run `npx convex dev`.
|
|
||||||
* @module
|
|
||||||
*/
|
|
||||||
|
|
||||||
import {
|
|
||||||
actionGeneric,
|
|
||||||
httpActionGeneric,
|
|
||||||
queryGeneric,
|
|
||||||
mutationGeneric,
|
|
||||||
internalActionGeneric,
|
|
||||||
internalMutationGeneric,
|
|
||||||
internalQueryGeneric,
|
|
||||||
} from "convex/server";
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Define a query in this Convex app's public API.
|
|
||||||
*
|
|
||||||
* This function will be allowed to read your Convex database and will be accessible from the client.
|
|
||||||
*
|
|
||||||
* @param func - The query function. It receives a {@link QueryCtx} as its first argument.
|
|
||||||
* @returns The wrapped query. Include this as an `export` to name it and make it accessible.
|
|
||||||
*/
|
|
||||||
export const query = queryGeneric;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Define a query that is only accessible from other Convex functions (but not from the client).
|
|
||||||
*
|
|
||||||
* This function will be allowed to read from your Convex database. It will not be accessible from the client.
|
|
||||||
*
|
|
||||||
* @param func - The query function. It receives a {@link QueryCtx} as its first argument.
|
|
||||||
* @returns The wrapped query. Include this as an `export` to name it and make it accessible.
|
|
||||||
*/
|
|
||||||
export const internalQuery = internalQueryGeneric;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Define a mutation in this Convex app's public API.
|
|
||||||
*
|
|
||||||
* This function will be allowed to modify your Convex database and will be accessible from the client.
|
|
||||||
*
|
|
||||||
* @param func - The mutation function. It receives a {@link MutationCtx} as its first argument.
|
|
||||||
* @returns The wrapped mutation. Include this as an `export` to name it and make it accessible.
|
|
||||||
*/
|
|
||||||
export const mutation = mutationGeneric;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Define a mutation that is only accessible from other Convex functions (but not from the client).
|
|
||||||
*
|
|
||||||
* This function will be allowed to modify your Convex database. It will not be accessible from the client.
|
|
||||||
*
|
|
||||||
* @param func - The mutation function. It receives a {@link MutationCtx} as its first argument.
|
|
||||||
* @returns The wrapped mutation. Include this as an `export` to name it and make it accessible.
|
|
||||||
*/
|
|
||||||
export const internalMutation = internalMutationGeneric;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Define an action in this Convex app's public API.
|
|
||||||
*
|
|
||||||
* An action is a function which can execute any JavaScript code, including non-deterministic
|
|
||||||
* code and code with side-effects, like calling third-party services.
|
|
||||||
* They can be run in Convex's JavaScript environment or in Node.js using the "use node" directive.
|
|
||||||
* They can interact with the database indirectly by calling queries and mutations using the {@link ActionCtx}.
|
|
||||||
*
|
|
||||||
* @param func - The action. It receives an {@link ActionCtx} as its first argument.
|
|
||||||
* @returns The wrapped action. Include this as an `export` to name it and make it accessible.
|
|
||||||
*/
|
|
||||||
export const action = actionGeneric;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Define an action that is only accessible from other Convex functions (but not from the client).
|
|
||||||
*
|
|
||||||
* @param func - The function. It receives an {@link ActionCtx} as its first argument.
|
|
||||||
* @returns The wrapped function. Include this as an `export` to name it and make it accessible.
|
|
||||||
*/
|
|
||||||
export const internalAction = internalActionGeneric;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Define a Convex HTTP action.
|
|
||||||
*
|
|
||||||
* @param func - The function. It receives an {@link ActionCtx} as its first argument, and a `Request` object
|
|
||||||
* as its second.
|
|
||||||
* @returns The wrapped endpoint function. Route a URL path to this function in `convex/http.js`.
|
|
||||||
*/
|
|
||||||
export const httpAction = httpActionGeneric;
|
|
@@ -1,59 +1,72 @@
|
|||||||
import * as React from "react"
|
|
||||||
import { Slot } from "@radix-ui/react-slot"
|
import { Slot } from "@radix-ui/react-slot"
|
||||||
import { cva, type VariantProps } from "class-variance-authority"
|
import { cva, type VariantProps } from "class-variance-authority"
|
||||||
|
import type * as React from "react"
|
||||||
|
|
||||||
import { cn } from "@/lib/utils"
|
import { cn } from "@/lib/utils"
|
||||||
|
import { LoadingSpinner } from "./loading-spinner"
|
||||||
|
|
||||||
const buttonVariants = cva(
|
const buttonVariants = cva(
|
||||||
"inline-flex items-center justify-center gap-2 whitespace-nowrap rounded-md text-sm font-medium transition-all disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg:not([class*='size-'])]:size-4 shrink-0 [&_svg]:shrink-0 outline-none focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px] aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive",
|
"inline-flex items-center justify-center gap-2 whitespace-nowrap rounded-md text-sm font-medium transition-all disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg:not([class*='size-'])]:size-4 shrink-0 [&_svg]:shrink-0 outline-none focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px] aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive",
|
||||||
{
|
{
|
||||||
variants: {
|
variants: {
|
||||||
variant: {
|
variant: {
|
||||||
default:
|
default:
|
||||||
"bg-primary text-primary-foreground shadow-xs hover:bg-primary/90",
|
"bg-primary text-primary-foreground shadow-xs hover:bg-primary/90",
|
||||||
destructive:
|
destructive:
|
||||||
"bg-destructive text-white shadow-xs hover:bg-destructive/90 focus-visible:ring-destructive/20 dark:focus-visible:ring-destructive/40 dark:bg-destructive/60",
|
"bg-destructive text-white shadow-xs hover:bg-destructive/90 focus-visible:ring-destructive/20 dark:focus-visible:ring-destructive/40 dark:bg-destructive/60",
|
||||||
outline:
|
outline:
|
||||||
"border bg-background shadow-xs hover:bg-accent hover:text-accent-foreground dark:bg-input/30 dark:border-input dark:hover:bg-input/50",
|
"border bg-background shadow-xs hover:bg-accent hover:text-accent-foreground dark:bg-input/30 dark:border-input dark:hover:bg-input/50",
|
||||||
secondary:
|
secondary:
|
||||||
"bg-secondary text-secondary-foreground shadow-xs hover:bg-secondary/80",
|
"bg-secondary text-secondary-foreground shadow-xs hover:bg-secondary/80",
|
||||||
ghost:
|
ghost: "hover:bg-accent hover:text-accent-foreground dark:hover:bg-accent/50",
|
||||||
"hover:bg-accent hover:text-accent-foreground dark:hover:bg-accent/50",
|
link: "text-primary underline-offset-4 hover:underline",
|
||||||
link: "text-primary underline-offset-4 hover:underline",
|
},
|
||||||
},
|
size: {
|
||||||
size: {
|
default: "h-9 px-4 py-2 has-[>svg]:px-3",
|
||||||
default: "h-9 px-4 py-2 has-[>svg]:px-3",
|
sm: "h-8 rounded-md gap-1.5 px-3 has-[>svg]:px-2.5",
|
||||||
sm: "h-8 rounded-md gap-1.5 px-3 has-[>svg]:px-2.5",
|
lg: "h-10 rounded-md px-6 has-[>svg]:px-4",
|
||||||
lg: "h-10 rounded-md px-6 has-[>svg]:px-4",
|
icon: "size-9",
|
||||||
icon: "size-9",
|
},
|
||||||
},
|
},
|
||||||
},
|
defaultVariants: {
|
||||||
defaultVariants: {
|
variant: "default",
|
||||||
variant: "default",
|
size: "default",
|
||||||
size: "default",
|
},
|
||||||
},
|
},
|
||||||
}
|
|
||||||
)
|
)
|
||||||
|
|
||||||
function Button({
|
function Button({
|
||||||
className,
|
className,
|
||||||
variant,
|
variant,
|
||||||
size,
|
size,
|
||||||
asChild = false,
|
asChild = false,
|
||||||
...props
|
loading = false,
|
||||||
|
children,
|
||||||
|
...props
|
||||||
}: React.ComponentProps<"button"> &
|
}: React.ComponentProps<"button"> &
|
||||||
VariantProps<typeof buttonVariants> & {
|
VariantProps<typeof buttonVariants> & {
|
||||||
asChild?: boolean
|
loading?: boolean
|
||||||
}) {
|
asChild?: boolean
|
||||||
const Comp = asChild ? Slot : "button"
|
}) {
|
||||||
|
const Comp = asChild ? Slot : "button"
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Comp
|
<Comp
|
||||||
data-slot="button"
|
data-slot="button"
|
||||||
className={cn(buttonVariants({ variant, size, className }))}
|
className={cn(buttonVariants({ variant, size, className }))}
|
||||||
{...props}
|
disabled={loading || props.disabled}
|
||||||
/>
|
{...props}
|
||||||
)
|
>
|
||||||
|
{asChild ? (
|
||||||
|
children
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
{loading ? <LoadingSpinner /> : null}
|
||||||
|
{children}
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</Comp>
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
export { Button, buttonVariants }
|
export { Button, buttonVariants }
|
||||||
|
@@ -106,7 +106,7 @@ function ContextMenuContent({
|
|||||||
<ContextMenuPrimitive.Content
|
<ContextMenuPrimitive.Content
|
||||||
data-slot="context-menu-content"
|
data-slot="context-menu-content"
|
||||||
className={cn(
|
className={cn(
|
||||||
"bg-popover text-popover-foreground data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 z-50 max-h-(--radix-context-menu-content-available-height) min-w-[8rem] origin-(--radix-context-menu-content-transform-origin) overflow-x-hidden overflow-y-auto rounded-md border p-1 shadow-md",
|
"bg-popover text-popover-foreground data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 z-50 max-h-(--radix-context-menu-content-available-height) min-w-[8rem] origin-(--radix-context-menu-content-transform-origin) overflow-x-hidden overflow-y-auto rounded-md border p-1 shadow-md",
|
||||||
className,
|
className,
|
||||||
)}
|
)}
|
||||||
{...props}
|
{...props}
|
||||||
|
141
packages/web/src/components/ui/dialog.tsx
Normal file
141
packages/web/src/components/ui/dialog.tsx
Normal file
@@ -0,0 +1,141 @@
|
|||||||
|
import * as React from "react"
|
||||||
|
import * as DialogPrimitive from "@radix-ui/react-dialog"
|
||||||
|
import { XIcon } from "lucide-react"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
function Dialog({
|
||||||
|
...props
|
||||||
|
}: React.ComponentProps<typeof DialogPrimitive.Root>) {
|
||||||
|
return <DialogPrimitive.Root data-slot="dialog" {...props} />
|
||||||
|
}
|
||||||
|
|
||||||
|
function DialogTrigger({
|
||||||
|
...props
|
||||||
|
}: React.ComponentProps<typeof DialogPrimitive.Trigger>) {
|
||||||
|
return <DialogPrimitive.Trigger data-slot="dialog-trigger" {...props} />
|
||||||
|
}
|
||||||
|
|
||||||
|
function DialogPortal({
|
||||||
|
...props
|
||||||
|
}: React.ComponentProps<typeof DialogPrimitive.Portal>) {
|
||||||
|
return <DialogPrimitive.Portal data-slot="dialog-portal" {...props} />
|
||||||
|
}
|
||||||
|
|
||||||
|
function DialogClose({
|
||||||
|
...props
|
||||||
|
}: React.ComponentProps<typeof DialogPrimitive.Close>) {
|
||||||
|
return <DialogPrimitive.Close data-slot="dialog-close" {...props} />
|
||||||
|
}
|
||||||
|
|
||||||
|
function DialogOverlay({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: React.ComponentProps<typeof DialogPrimitive.Overlay>) {
|
||||||
|
return (
|
||||||
|
<DialogPrimitive.Overlay
|
||||||
|
data-slot="dialog-overlay"
|
||||||
|
className={cn(
|
||||||
|
"data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 fixed inset-0 z-50 bg-black/50",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
function DialogContent({
|
||||||
|
className,
|
||||||
|
children,
|
||||||
|
showCloseButton = true,
|
||||||
|
...props
|
||||||
|
}: React.ComponentProps<typeof DialogPrimitive.Content> & {
|
||||||
|
showCloseButton?: boolean
|
||||||
|
}) {
|
||||||
|
return (
|
||||||
|
<DialogPortal data-slot="dialog-portal">
|
||||||
|
<DialogOverlay />
|
||||||
|
<DialogPrimitive.Content
|
||||||
|
data-slot="dialog-content"
|
||||||
|
className={cn(
|
||||||
|
"bg-background data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 fixed top-[50%] left-[50%] z-50 grid w-full max-w-[calc(100%-2rem)] translate-x-[-50%] translate-y-[-50%] gap-4 rounded-lg border p-6 shadow-lg duration-200 sm:max-w-lg",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
{showCloseButton && (
|
||||||
|
<DialogPrimitive.Close
|
||||||
|
data-slot="dialog-close"
|
||||||
|
className="ring-offset-background focus:ring-ring data-[state=open]:bg-accent data-[state=open]:text-muted-foreground absolute top-4 right-4 rounded-xs opacity-70 transition-opacity hover:opacity-100 focus:ring-2 focus:ring-offset-2 focus:outline-hidden disabled:pointer-events-none [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4"
|
||||||
|
>
|
||||||
|
<XIcon />
|
||||||
|
<span className="sr-only">Close</span>
|
||||||
|
</DialogPrimitive.Close>
|
||||||
|
)}
|
||||||
|
</DialogPrimitive.Content>
|
||||||
|
</DialogPortal>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
function DialogHeader({ className, ...props }: React.ComponentProps<"div">) {
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
data-slot="dialog-header"
|
||||||
|
className={cn("flex flex-col gap-2 text-center sm:text-left", className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
function DialogFooter({ className, ...props }: React.ComponentProps<"div">) {
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
data-slot="dialog-footer"
|
||||||
|
className={cn(
|
||||||
|
"flex flex-col-reverse gap-2 sm:flex-row sm:justify-end",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
function DialogTitle({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: React.ComponentProps<typeof DialogPrimitive.Title>) {
|
||||||
|
return (
|
||||||
|
<DialogPrimitive.Title
|
||||||
|
data-slot="dialog-title"
|
||||||
|
className={cn("text-lg leading-none font-semibold", className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
function DialogDescription({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: React.ComponentProps<typeof DialogPrimitive.Description>) {
|
||||||
|
return (
|
||||||
|
<DialogPrimitive.Description
|
||||||
|
data-slot="dialog-description"
|
||||||
|
className={cn("text-muted-foreground text-sm", className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export {
|
||||||
|
Dialog,
|
||||||
|
DialogClose,
|
||||||
|
DialogContent,
|
||||||
|
DialogDescription,
|
||||||
|
DialogFooter,
|
||||||
|
DialogHeader,
|
||||||
|
DialogOverlay,
|
||||||
|
DialogPortal,
|
||||||
|
DialogTitle,
|
||||||
|
DialogTrigger,
|
||||||
|
}
|
@@ -38,6 +38,7 @@ import { withDefaultOnError } from "../lib/error"
|
|||||||
import { cn } from "../lib/utils"
|
import { cn } from "../lib/utils"
|
||||||
import {
|
import {
|
||||||
contextMenuTargeItemAtom,
|
contextMenuTargeItemAtom,
|
||||||
|
itemBeingRenamedAtom,
|
||||||
newItemKindAtom,
|
newItemKindAtom,
|
||||||
optimisticDeletedItemsAtom,
|
optimisticDeletedItemsAtom,
|
||||||
} from "./state"
|
} from "./state"
|
||||||
@@ -132,6 +133,7 @@ export function FileTableContextMenu({
|
|||||||
const target = useAtomValue(contextMenuTargeItemAtom)
|
const target = useAtomValue(contextMenuTargeItemAtom)
|
||||||
const setOptimisticDeletedItems = useSetAtom(optimisticDeletedItemsAtom)
|
const setOptimisticDeletedItems = useSetAtom(optimisticDeletedItemsAtom)
|
||||||
const moveToTrashMutation = useContextMutation(api.files.moveToTrash)
|
const moveToTrashMutation = useContextMutation(api.files.moveToTrash)
|
||||||
|
const setItemBeingRenamed = useSetAtom(itemBeingRenamedAtom)
|
||||||
const { mutate: moveToTrash } = useMutation({
|
const { mutate: moveToTrash } = useMutation({
|
||||||
mutationFn: moveToTrashMutation,
|
mutationFn: moveToTrashMutation,
|
||||||
onMutate: ({ itemId }) => {
|
onMutate: ({ itemId }) => {
|
||||||
@@ -150,8 +152,11 @@ export function FileTableContextMenu({
|
|||||||
const handleRename = () => {
|
const handleRename = () => {
|
||||||
const selectedItem = store.get(contextMenuTargeItemAtom)
|
const selectedItem = store.get(contextMenuTargeItemAtom)
|
||||||
if (selectedItem) {
|
if (selectedItem) {
|
||||||
console.log("Renaming:", selectedItem.doc.name)
|
setItemBeingRenamed({
|
||||||
// TODO: Implement rename functionality
|
kind: selectedItem.kind,
|
||||||
|
originalItem: selectedItem,
|
||||||
|
name: selectedItem.doc.name,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -169,7 +174,7 @@ export function FileTableContextMenu({
|
|||||||
<ContextMenu>
|
<ContextMenu>
|
||||||
<ContextMenuTrigger asChild>{children}</ContextMenuTrigger>
|
<ContextMenuTrigger asChild>{children}</ContextMenuTrigger>
|
||||||
{target && (
|
{target && (
|
||||||
<ContextMenuContent key={target?.doc._id}>
|
<ContextMenuContent>
|
||||||
<ContextMenuItem onClick={handleRename}>
|
<ContextMenuItem onClick={handleRename}>
|
||||||
<TextCursorInputIcon />
|
<TextCursorInputIcon />
|
||||||
Rename
|
Rename
|
||||||
|
@@ -30,6 +30,7 @@ import {
|
|||||||
} from "../components/ui/breadcrumb"
|
} from "../components/ui/breadcrumb"
|
||||||
import { Button } from "../components/ui/button"
|
import { Button } from "../components/ui/button"
|
||||||
import { FileTable } from "./file-table"
|
import { FileTable } from "./file-table"
|
||||||
|
import { RenameFileDialog } from "./rename-file-dialog"
|
||||||
import { newItemKindAtom } from "./state"
|
import { newItemKindAtom } from "./state"
|
||||||
|
|
||||||
export function FilesPage({ path }: { path: string }) {
|
export function FilesPage({ path }: { path: string }) {
|
||||||
@@ -45,6 +46,7 @@ export function FilesPage({ path }: { path: string }) {
|
|||||||
<div className="w-full">
|
<div className="w-full">
|
||||||
<FileTable path={path} />
|
<FileTable path={path} />
|
||||||
</div>
|
</div>
|
||||||
|
<RenameFileDialog />
|
||||||
</>
|
</>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
111
packages/web/src/files/rename-file-dialog.tsx
Normal file
111
packages/web/src/files/rename-file-dialog.tsx
Normal file
@@ -0,0 +1,111 @@
|
|||||||
|
import { api } from "@fileone/convex/_generated/api"
|
||||||
|
import { useMutation } from "@tanstack/react-query"
|
||||||
|
import { useMutation as useContextMutation } from "convex/react"
|
||||||
|
import { atom, useAtom, useStore } from "jotai"
|
||||||
|
import { useId } from "react"
|
||||||
|
import { toast } from "sonner"
|
||||||
|
import { Button } from "@/components/ui/button"
|
||||||
|
import {
|
||||||
|
Dialog,
|
||||||
|
DialogClose,
|
||||||
|
DialogContent,
|
||||||
|
DialogFooter,
|
||||||
|
DialogHeader,
|
||||||
|
DialogTitle,
|
||||||
|
} from "@/components/ui/dialog"
|
||||||
|
import { Input } from "@/components/ui/input"
|
||||||
|
import { itemBeingRenamedAtom } from "./state"
|
||||||
|
|
||||||
|
const fielNameAtom = atom(
|
||||||
|
(get) => get(itemBeingRenamedAtom)?.name,
|
||||||
|
(get, set, newName: string) => {
|
||||||
|
const current = get(itemBeingRenamedAtom)
|
||||||
|
if (current) {
|
||||||
|
set(itemBeingRenamedAtom, {
|
||||||
|
...current,
|
||||||
|
name: newName,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
export function RenameFileDialog() {
|
||||||
|
const [itemBeingRenamed, setItemBeingRenamed] =
|
||||||
|
useAtom(itemBeingRenamedAtom)
|
||||||
|
const store = useStore()
|
||||||
|
const formId = useId()
|
||||||
|
|
||||||
|
const { mutate: renameFile, isPending: isRenaming } = useMutation({
|
||||||
|
mutationFn: useContextMutation(api.files.renameFile),
|
||||||
|
onSuccess: () => {
|
||||||
|
setItemBeingRenamed(null)
|
||||||
|
toast.success("File renamed successfully")
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
const onSubmit = (event: React.FormEvent<HTMLFormElement>) => {
|
||||||
|
event.preventDefault()
|
||||||
|
|
||||||
|
const itemBeingRenamed = store.get(itemBeingRenamedAtom)
|
||||||
|
if (itemBeingRenamed) {
|
||||||
|
const formData = new FormData(event.currentTarget)
|
||||||
|
const newName = formData.get("itemName") as string
|
||||||
|
|
||||||
|
if (newName) {
|
||||||
|
switch (itemBeingRenamed.originalItem.kind) {
|
||||||
|
case "file":
|
||||||
|
renameFile({
|
||||||
|
directoryId:
|
||||||
|
itemBeingRenamed.originalItem.doc.directoryId,
|
||||||
|
itemId: itemBeingRenamed.originalItem.doc._id,
|
||||||
|
newName,
|
||||||
|
})
|
||||||
|
break
|
||||||
|
default:
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Dialog
|
||||||
|
open={itemBeingRenamed !== null}
|
||||||
|
onOpenChange={(open) =>
|
||||||
|
setItemBeingRenamed(open ? itemBeingRenamed : null)
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<DialogContent className="sm:max-w-md">
|
||||||
|
<DialogHeader>
|
||||||
|
<DialogTitle>Rename File</DialogTitle>
|
||||||
|
</DialogHeader>
|
||||||
|
|
||||||
|
<form id={formId} onSubmit={onSubmit}>
|
||||||
|
<RenameFileInput />
|
||||||
|
</form>
|
||||||
|
|
||||||
|
<DialogFooter>
|
||||||
|
<DialogClose asChild>
|
||||||
|
<Button loading={isRenaming} variant="outline">
|
||||||
|
<span>Cancel</span>
|
||||||
|
</Button>
|
||||||
|
</DialogClose>
|
||||||
|
<Button loading={isRenaming} type="submit" form={formId}>
|
||||||
|
<span>Rename</span>
|
||||||
|
</Button>
|
||||||
|
</DialogFooter>
|
||||||
|
</DialogContent>
|
||||||
|
</Dialog>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
function RenameFileInput() {
|
||||||
|
const [fileName, setFileName] = useAtom(fielNameAtom)
|
||||||
|
return (
|
||||||
|
<Input
|
||||||
|
value={fileName}
|
||||||
|
name="itemName"
|
||||||
|
onChange={(e) => setFileName(e.target.value)}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
@@ -14,3 +14,9 @@ export const optimisticDeletedItemsAtom = atom(
|
|||||||
export const selectedFileRowsAtom = atom<RowSelectionState>({})
|
export const selectedFileRowsAtom = atom<RowSelectionState>({})
|
||||||
|
|
||||||
export const newItemKindAtom = atom<DirectoryItemKind | null>(null)
|
export const newItemKindAtom = atom<DirectoryItemKind | null>(null)
|
||||||
|
|
||||||
|
export const itemBeingRenamedAtom = atom<{
|
||||||
|
kind: DirectoryItemKind
|
||||||
|
originalItem: DirectoryItem
|
||||||
|
name: string
|
||||||
|
} | null>(null)
|
||||||
|
Reference in New Issue
Block a user