mirror of
https://github.com/RayLabsHQ/gitea-mirror.git
synced 2025-12-10 21:46:45 +03:00
229 lines
7.5 KiB
TypeScript
229 lines
7.5 KiB
TypeScript
import React, { useState } from "react";
|
|
import { Button } from "@/components/ui/button";
|
|
import {
|
|
Card,
|
|
CardContent,
|
|
CardFooter,
|
|
CardHeader,
|
|
CardTitle,
|
|
} from "@/components/ui/card";
|
|
import {
|
|
Select,
|
|
SelectContent,
|
|
SelectItem,
|
|
SelectTrigger,
|
|
SelectValue,
|
|
} from "../ui/select";
|
|
import { giteaApi } from "@/lib/api";
|
|
import type { GiteaConfig, GiteaOrgVisibility } from "@/types/config";
|
|
import { toast } from "sonner";
|
|
|
|
interface GiteaConfigFormProps {
|
|
config: GiteaConfig;
|
|
setConfig: React.Dispatch<React.SetStateAction<GiteaConfig>>;
|
|
}
|
|
|
|
export function GiteaConfigForm({ config, setConfig }: GiteaConfigFormProps) {
|
|
const [isLoading, setIsLoading] = useState(false);
|
|
|
|
const handleChange = (
|
|
e: React.ChangeEvent<HTMLInputElement | HTMLSelectElement>
|
|
) => {
|
|
const { name, value } = e.target;
|
|
setConfig({
|
|
...config,
|
|
[name]: value,
|
|
});
|
|
};
|
|
|
|
const testConnection = async () => {
|
|
if (!config.url || !config.token) {
|
|
toast.error("Gitea URL and token are required to test the connection");
|
|
return;
|
|
}
|
|
|
|
setIsLoading(true);
|
|
|
|
try {
|
|
const result = await giteaApi.testConnection(config.url, config.token);
|
|
if (result.success) {
|
|
toast.success("Successfully connected to Gitea!");
|
|
} else {
|
|
toast.error(
|
|
"Failed to connect to Gitea. Please check your URL and token."
|
|
);
|
|
}
|
|
} catch (error) {
|
|
toast.error(
|
|
error instanceof Error ? error.message : "An unknown error occurred"
|
|
);
|
|
} finally {
|
|
setIsLoading(false);
|
|
}
|
|
};
|
|
|
|
return (
|
|
<Card className="w-full">
|
|
<CardHeader className="flex flex-row items-center justify-between gap-4">
|
|
<CardTitle className="text-lg font-semibold">
|
|
Gitea Configuration
|
|
</CardTitle>
|
|
<Button
|
|
type="button"
|
|
variant="outline"
|
|
onClick={testConnection}
|
|
disabled={isLoading || !config.url || !config.token}
|
|
>
|
|
{isLoading ? "Testing..." : "Test Connection"}
|
|
</Button>
|
|
</CardHeader>
|
|
|
|
<CardContent className="flex flex-col gap-y-6">
|
|
<div>
|
|
<label
|
|
htmlFor="gitea-username"
|
|
className="block text-sm font-medium mb-1.5"
|
|
>
|
|
Gitea Username
|
|
</label>
|
|
<input
|
|
id="gitea-username"
|
|
name="username"
|
|
type="text"
|
|
value={config.username}
|
|
onChange={handleChange}
|
|
className="w-full rounded-md border border-input bg-background px-3 py-2 text-sm shadow-sm transition-colors placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring"
|
|
placeholder="Your Gitea username"
|
|
required
|
|
/>
|
|
</div>
|
|
|
|
<div>
|
|
<label
|
|
htmlFor="gitea-url"
|
|
className="block text-sm font-medium mb-1.5"
|
|
>
|
|
Gitea URL
|
|
</label>
|
|
<input
|
|
id="gitea-url"
|
|
name="url"
|
|
type="url"
|
|
value={config.url}
|
|
onChange={handleChange}
|
|
className="w-full rounded-md border border-input bg-background px-3 py-2 text-sm shadow-sm transition-colors placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring"
|
|
placeholder="https://your-gitea-instance.com"
|
|
required
|
|
/>
|
|
</div>
|
|
|
|
<div>
|
|
<label
|
|
htmlFor="gitea-token"
|
|
className="block text-sm font-medium mb-1.5"
|
|
>
|
|
Gitea Token
|
|
</label>
|
|
<input
|
|
id="gitea-token"
|
|
name="token"
|
|
type="password"
|
|
value={config.token}
|
|
onChange={handleChange}
|
|
className="w-full rounded-md border border-input bg-background px-3 py-2 text-sm shadow-sm transition-colors placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring"
|
|
placeholder="Your Gitea access token"
|
|
required
|
|
/>
|
|
<p className="text-xs text-muted-foreground mt-1">
|
|
Create a token in your Gitea instance under Settings >
|
|
Applications.
|
|
</p>
|
|
</div>
|
|
|
|
<div>
|
|
<label
|
|
htmlFor="organization"
|
|
className="block text-sm font-medium mb-1.5"
|
|
>
|
|
Default Organization (Optional)
|
|
</label>
|
|
<input
|
|
id="organization"
|
|
name="organization"
|
|
type="text"
|
|
value={config.organization}
|
|
onChange={handleChange}
|
|
className="w-full rounded-md border border-input bg-background px-3 py-2 text-sm shadow-sm transition-colors placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring"
|
|
placeholder="Organization name"
|
|
/>
|
|
<p className="text-xs text-muted-foreground mt-1">
|
|
If specified, repositories will be mirrored to this organization.
|
|
</p>
|
|
</div>
|
|
|
|
<div className="grid grid-cols-1 md:grid-cols-2 gap-4">
|
|
<div>
|
|
<label
|
|
htmlFor="visibility"
|
|
className="block text-sm font-medium mb-1.5"
|
|
>
|
|
Organization Visibility
|
|
</label>
|
|
<Select
|
|
name="visibility"
|
|
value={config.visibility}
|
|
onValueChange={(value) =>
|
|
handleChange({
|
|
target: { name: "visibility", value },
|
|
} as React.ChangeEvent<HTMLInputElement>)
|
|
}
|
|
>
|
|
<SelectTrigger className="w-full border border-input dark:bg-background dark:hover:bg-background">
|
|
<SelectValue placeholder="Select visibility" />
|
|
</SelectTrigger>
|
|
<SelectContent className="bg-background text-foreground border border-input shadow-sm">
|
|
{(["public", "private", "limited"] as GiteaOrgVisibility[]).map(
|
|
(option) => (
|
|
<SelectItem
|
|
key={option}
|
|
value={option}
|
|
className="cursor-pointer text-sm px-3 py-2 hover:bg-accent focus:bg-accent focus:text-accent-foreground"
|
|
>
|
|
{option.charAt(0).toUpperCase() + option.slice(1)}
|
|
</SelectItem>
|
|
)
|
|
)}
|
|
</SelectContent>
|
|
</Select>
|
|
</div>
|
|
|
|
<div>
|
|
<label
|
|
htmlFor="starred-repos-org"
|
|
className="block text-sm font-medium mb-1.5"
|
|
>
|
|
Starred Repositories Organization
|
|
</label>
|
|
<input
|
|
id="starred-repos-org"
|
|
name="starredReposOrg"
|
|
type="text"
|
|
value={config.starredReposOrg}
|
|
onChange={handleChange}
|
|
className="w-full rounded-md border border-input bg-background px-3 py-2 text-sm shadow-sm transition-colors placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring"
|
|
placeholder="github"
|
|
/>
|
|
<p className="text-xs text-muted-foreground mt-1">
|
|
Organization for starred repositories (default: github)
|
|
</p>
|
|
</div>
|
|
</div>
|
|
</CardContent>
|
|
|
|
<CardFooter className="">
|
|
{/* Footer content can be added here if needed */}
|
|
</CardFooter>
|
|
</Card>
|
|
);
|
|
}
|