Initial implementation of Import page, project cleanup

This commit is contained in:
2026-01-02 23:08:26 +05:00
parent 05814a100a
commit 885aab0d9d
8 changed files with 107 additions and 60 deletions

View File

@ -0,0 +1,43 @@
<template>
<Frame margin="none" class="px-3 py-4 flex items-center gap-2">
<div>
<AudioWaveform :size="32" />
</div>
<div class="w-full">
<div class="flex flex-row gap-4">
<p class="font-medium">
{{ title }}
</p>
<Pen />
</div>
<div class="flex flex-row">
<p class="text-sm text-muted-foreground">
{{ size }}
</p>
<Dot />
<p class="text-sm text-muted-foreground">
{{ format }}
</p>
</div>
</div>
<div>
<EllipsisVertical :size="32" />
</div>
</Frame>
</template>
<script setup lang="ts">
import Frame from '@/components/ui/frame/Frame.vue'
import { AudioWaveform, Dot, EllipsisVertical, Pen } from 'lucide-vue-next'
interface Props {
title: string
size: string
format: string
progress?: number
error?: string
}
withDefaults(defineProps<Props>(), {
})
</script>

View File

@ -6,11 +6,11 @@
<Separator orientation="vertical" /> <Separator orientation="vertical" />
<div class="w-fit flex-1 flex flex-col justify-between"> <div class="w-fit flex-1 flex flex-col justify-between">
<div class="w-full"> <div class="w-full">
<div class="flex items-center justify-between w-full"> <div class="flex items-start justify-between w-full">
<h4 class="scroll-m-20 text-xl font-semibold tracking-tight truncate max-w-[32ch]"> <h4 class="scroll-m-20 text-xl font-semibold tracking-tight truncate max-w-[32ch]">
{{ title }} {{ title }}
</h4> </h4>
<p class="leading-7" v-if="date"> <p class="max-w-[16ch]" v-if="date">
{{ date }} {{ date }}
</p> </p>
</div> </div>

View File

@ -7,23 +7,19 @@ import { cn } from "@/lib/utils"
import { buttonVariants } from "." import { buttonVariants } from "."
interface Props extends PrimitiveProps { interface Props extends PrimitiveProps {
variant?: ButtonVariants["variant"] variant?: ButtonVariants["variant"]
size?: ButtonVariants["size"] size?: ButtonVariants["size"]
class?: HTMLAttributes["class"] class?: HTMLAttributes["class"]
} }
const props = withDefaults(defineProps<Props>(), { const props = withDefaults(defineProps<Props>(), {
as: "button", as: "button",
}) })
</script> </script>
<template> <template>
<Primitive <Primitive data-slot="button" :as="as" :as-child="asChild"
data-slot="button" :class="cn(buttonVariants({ variant, size }), props.class)">
:as="as" <slot />
:as-child="asChild" </Primitive>
:class="cn(buttonVariants({ variant, size }), props.class)"
>
<slot />
</Primitive>
</template> </template>

View File

@ -4,36 +4,36 @@ import { cva } from "class-variance-authority"
export { default as Button } from "./Button.vue" export { default as Button } from "./Button.vue"
export const buttonVariants = cva( export 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",
"icon-sm": "size-8", "icon-sm": "size-8",
"icon-lg": "size-10", "icon-lg": "size-10",
}, },
},
defaultVariants: {
variant: "default",
size: "default",
},
}, },
defaultVariants: {
variant: "default",
size: "default",
},
},
) )
export type ButtonVariants = VariantProps<typeof buttonVariants> export type ButtonVariants = VariantProps<typeof buttonVariants>

View File

@ -3,7 +3,7 @@ import type { PrimitiveProps } from "reka-ui"
import type { HTMLAttributes } from "vue" import type { HTMLAttributes } from "vue"
import type { FrameVariants } from "../frame" import type { FrameVariants } from "../frame"
import { cn } from "@/lib/utils" import { cn } from "@/lib/utils"
import Frame from "@/components/ui/frame/Frame.vue" import Frame from "../frame/Frame.vue"
interface Props extends PrimitiveProps { interface Props extends PrimitiveProps {

View File

@ -6,7 +6,6 @@ export const axiosInstance = <T>(
options?: AxiosRequestConfig, options?: AxiosRequestConfig,
): Promise<AxiosResponse<T, any>> => { ): Promise<AxiosResponse<T, any>> => {
const baseURL = useRuntimeConfig().public.apiBaseUrl; const baseURL = useRuntimeConfig().public.apiBaseUrl;
console.log(baseURL)
const source = Axios.CancelToken.source(); const source = Axios.CancelToken.source();
const promise = AXIOS_INSTANCE({ const promise = AXIOS_INSTANCE({
...config, ...config,

View File

@ -46,6 +46,7 @@ const { mutate: reorderTracks } = useMutation({
return { previousTracks }; return { previousTracks };
}, },
onError: (err, { playlistId }, context) => { onError: (err, { playlistId }, context) => {
console.log(err)
if (context?.previousTracks) { if (context?.previousTracks) {
queryClient.setQueryData(getGetPlaylistTracksQueryKey(playlistId), context.previousTracks); queryClient.setQueryData(getGetPlaylistTracksQueryKey(playlistId), context.previousTracks);
} }

View File

@ -1,7 +1,8 @@
<script setup lang="ts"> <script setup lang="ts">
import { Outline } from '@/components/ui/outline'; import { Outline } from '@/components/ui/outline';
import { SidebarTrigger } from '@/components/ui/sidebar'; import { SidebarTrigger } from '@/components/ui/sidebar';
import Frame from '@/components/ui/frame/Frame.vue'; import { Download, Play } from 'lucide-vue-next';
import UploadEntry from '~/components/internal/import/uploadentry/UploadEntry.vue';
</script> </script>
<template> <template>
@ -12,25 +13,32 @@ import Frame from '@/components/ui/frame/Frame.vue';
<div class="flex gap-8 w-full items-center"> <div class="flex gap-8 w-full items-center">
<SidebarTrigger :size="5" /> <SidebarTrigger :size="5" />
<h2 class="scroll-m-20 text-3xl font-semibold tracking-tight transition-colors first:mt-0"> <h2 class="scroll-m-20 text-3xl font-semibold tracking-tight transition-colors first:mt-0">
Import Import tracks
</h2> </h2>
</div> </div>
</Outline> </Outline>
</template> </template>
<div class="w-full"> <div class="w-full flex flex-col p-8">
<Frame> <Outline class="rounded-xl bg-muted flex flex-col items-center justify-center gap-1">
Hello <Download />
</Frame> <h4 class="scroll-m-20 text-xl font-semibold tracking-tight">
</div> Drag and drop your audio files
<template #sidebar> </h4>
<Outline padding="none" class="h-full" side="left"> <p class="text-sm text-muted-foreground">
<Outline padding="dense" side="bottom"> or
<p class="leading-7 not-first:mt-6 font-semibold"> </p>
Metadata editor <UiButton variant="destructive">
</p> <Play />
</Outline> From Youtube
</UiButton>
</Outline> </Outline>
</template> <div>
<h3 class="scroll-m-20 text-2xl font-semibold tracking-tight">
Uploaded files
</h3>
<UploadEntry title="Test" size="3.8 MB" format="mp4" />
</div>
</div>
</NuxtLayout> </NuxtLayout>
</div> </div>
</template> </template>