mirror of
https://github.com/neosubhamoy/neodlp.git
synced 2026-02-04 14:12:22 +05:30
feat: added support for multiple audio stream selection in combine mode
This commit is contained in:
114
src/components/custom/formatToggleGroup.tsx
Normal file
114
src/components/custom/formatToggleGroup.tsx
Normal file
@@ -0,0 +1,114 @@
|
||||
import * as React from "react";
|
||||
import * as ToggleGroupPrimitive from "@radix-ui/react-toggle-group";
|
||||
import { type VariantProps } from "class-variance-authority";
|
||||
import { cn } from "@/lib/utils";
|
||||
import { toggleVariants } from "@/components/ui/toggle";
|
||||
import { VideoFormat } from "@/types/video";
|
||||
import { determineFileType, formatBitrate, formatCodec, formatFileSize } from "@/utils";
|
||||
import { Music, Video, File } from "lucide-react";
|
||||
|
||||
const FormatToggleGroupContext = React.createContext<
|
||||
VariantProps<typeof toggleVariants> & { toggleType?: "single" | "multiple" }
|
||||
>({
|
||||
size: "default",
|
||||
variant: "default",
|
||||
toggleType: "multiple",
|
||||
});
|
||||
|
||||
type FormatToggleGroupProps =
|
||||
| (Omit<React.ComponentPropsWithoutRef<typeof ToggleGroupPrimitive.Root>, "type"> &
|
||||
VariantProps<typeof toggleVariants> & { type: "single", value?: string, onValueChange?: (value: string) => void })
|
||||
| (Omit<React.ComponentPropsWithoutRef<typeof ToggleGroupPrimitive.Root>, "type"> &
|
||||
VariantProps<typeof toggleVariants> & { type: "multiple", value?: string[], onValueChange?: (value: string[]) => void });
|
||||
|
||||
export const FormatToggleGroup = React.forwardRef<
|
||||
React.ComponentRef<typeof ToggleGroupPrimitive.Root>,
|
||||
FormatToggleGroupProps
|
||||
>(({ className, variant, size, children, type = "multiple", ...props }, ref) => {
|
||||
if (type === "single") {
|
||||
return (
|
||||
<ToggleGroupPrimitive.Root
|
||||
ref={ref}
|
||||
type="single"
|
||||
className={cn("flex flex-col gap-2", className)}
|
||||
{...(props as any)}
|
||||
>
|
||||
<FormatToggleGroupContext.Provider value={{ variant, size, toggleType: "single" }}>
|
||||
{children}
|
||||
</FormatToggleGroupContext.Provider>
|
||||
</ToggleGroupPrimitive.Root>
|
||||
);
|
||||
}
|
||||
|
||||
return (
|
||||
<ToggleGroupPrimitive.Root
|
||||
ref={ref}
|
||||
type="multiple"
|
||||
className={cn("flex flex-col gap-2", className)}
|
||||
{...(props as any)}
|
||||
>
|
||||
<FormatToggleGroupContext.Provider value={{ variant, size, toggleType: "multiple" }}>
|
||||
{children}
|
||||
</FormatToggleGroupContext.Provider>
|
||||
</ToggleGroupPrimitive.Root>
|
||||
);
|
||||
});
|
||||
FormatToggleGroup.displayName = "FormatToggleGroup";
|
||||
|
||||
export const FormatToggleGroupItem = React.forwardRef<
|
||||
React.ComponentRef<typeof ToggleGroupPrimitive.Item>,
|
||||
React.ComponentPropsWithoutRef<typeof ToggleGroupPrimitive.Item> &
|
||||
VariantProps<typeof toggleVariants> & {
|
||||
format: VideoFormat
|
||||
}
|
||||
>(({ className, children, variant, size, format, value, ...props }, ref) => {
|
||||
const determineFileTypeIcon = (format: VideoFormat) => {
|
||||
const fileFormat = determineFileType(/*format.video_ext, format.audio_ext,*/ format.vcodec, format.acodec)
|
||||
switch (fileFormat) {
|
||||
case 'video+audio':
|
||||
return (
|
||||
<span className="absolute flex items-center right-2 bottom-2">
|
||||
<Video className="w-3 h-3 mr-2" />
|
||||
<Music className="w-3 h-3" />
|
||||
</span>
|
||||
)
|
||||
case 'video':
|
||||
return (
|
||||
<Video className="w-3 h-3 absolute right-2 bottom-2" />
|
||||
)
|
||||
case 'audio':
|
||||
return (
|
||||
<Music className="w-3 h-3 absolute right-2 bottom-2" />
|
||||
)
|
||||
default:
|
||||
return (
|
||||
<File className="w-3 h-3 absolute right-2 bottom-2" />
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
return (
|
||||
<ToggleGroupPrimitive.Item
|
||||
ref={ref}
|
||||
className={cn(
|
||||
"relative flex w-full p-2 rounded-lg border-2 border-border bg-background px-3 py-2 shadow-sm transition-all",
|
||||
"hover:bg-muted/70 data-[state=on]:bg-primary/10",
|
||||
"data-[state=on]:border-primary",
|
||||
className
|
||||
)}
|
||||
value={value}
|
||||
{...props}
|
||||
>
|
||||
<div className="flex flex-col items-start text-start gap-1">
|
||||
<h5 className="text-sm">{format.format}</h5>
|
||||
<p className="text-muted-foreground text-xs">{format.filesize_approx ? formatFileSize(format.filesize_approx) : 'unknown'} {format.tbr ? formatBitrate(format.tbr) : 'unknown'}</p>
|
||||
<p className="text-muted-foreground text-xs">{format.ext ? format.ext.toUpperCase() : 'unknown'} {
|
||||
((format.vcodec && format.vcodec !== 'none') || (format.acodec && format.acodec !== 'none')) && (
|
||||
`(${format.vcodec && format.vcodec !== 'none' ? formatCodec(format.vcodec) : ''}${format.vcodec && format.vcodec !== 'none' && format.acodec && format.acodec !== 'none' ? ' ' : ''}${format.acodec && format.acodec !== 'none' ? formatCodec(format.acodec) : ''})`
|
||||
)}</p>
|
||||
{determineFileTypeIcon(format)}
|
||||
</div>
|
||||
</ToggleGroupPrimitive.Item>
|
||||
);
|
||||
});
|
||||
FormatToggleGroupItem.displayName = "FormatToggleGroupItem";
|
||||
Reference in New Issue
Block a user