Compare commits
4 Commits
feature/re
...
1.5.4
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
b2dc0a3cb3 | ||
|
|
4beb7053f7 | ||
|
|
d6b6aff44e | ||
|
|
0ab3358e28 |
13
README.md
@@ -19,6 +19,8 @@
|
||||
- [Job Execution Logging](#job-execution-logging)
|
||||
- [Managing Scripts](#managing-scripts)
|
||||
- [Technologies Used](#technologies-used)
|
||||
- [Contributing](#contributing)
|
||||
- [License](#license)
|
||||
|
||||
---
|
||||
|
||||
@@ -91,6 +93,7 @@ services:
|
||||
- "40123:3000"
|
||||
environment:
|
||||
- NODE_ENV=production
|
||||
- DOCKER=true
|
||||
- NEXT_PUBLIC_CLOCK_UPDATE_INTERVAL=30000
|
||||
- AUTH_PASSWORD=very_strong_password
|
||||
- HOST_CRONTAB_USER=root
|
||||
@@ -332,6 +335,16 @@ I would like to thank the following members for raising issues and help test/deb
|
||||
</tbody>
|
||||
</table>
|
||||
|
||||
<a id="license"></a>
|
||||
|
||||
## License
|
||||
|
||||
This project is licensed under the MIT License.
|
||||
|
||||
## Support
|
||||
|
||||
For issues and questions, please open an issue on the GitHub repository.
|
||||
|
||||
## Star History
|
||||
|
||||
[](https://www.star-history.com/#fccview/cronmaster&Date)
|
||||
|
||||
@@ -9,16 +9,16 @@ import {
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
import { Switch } from "@/app/_components/GlobalComponents/UIElements/Switch";
|
||||
import {
|
||||
ClockIcon,
|
||||
PlusIcon,
|
||||
Clock,
|
||||
Plus,
|
||||
Archive,
|
||||
CaretDownIcon,
|
||||
CodeIcon,
|
||||
ChatTextIcon,
|
||||
GearIcon,
|
||||
CircleNotchIcon,
|
||||
FunnelIcon,
|
||||
} from "@phosphor-icons/react";
|
||||
ChevronDown,
|
||||
Code,
|
||||
MessageSquare,
|
||||
Settings,
|
||||
Loader2,
|
||||
Filter,
|
||||
} from "lucide-react";
|
||||
import { CronJob } from "@/app/_utils/cronjob-utils";
|
||||
import { Script } from "@/app/_utils/scripts-utils";
|
||||
import { UserFilter } from "@/app/_components/FeatureComponents/User/UserFilter";
|
||||
@@ -237,7 +237,7 @@ export const CronJobList = ({ cronJobs, scripts }: CronJobListProps) => {
|
||||
<div className="flex flex-col sm:flex-row sm:items-center justify-between gap-4">
|
||||
<div className="flex items-center gap-3">
|
||||
<div className="p-2 bg-primary/10 rounded-lg">
|
||||
<ClockIcon className="h-5 w-5 text-primary" />
|
||||
<Clock className="h-5 w-5 text-primary" />
|
||||
</div>
|
||||
<div>
|
||||
<CardTitle className="text-xl brand-gradient">
|
||||
@@ -261,7 +261,7 @@ export const CronJobList = ({ cronJobs, scripts }: CronJobListProps) => {
|
||||
className="btn-outline"
|
||||
title={t("cronjobs.filters")}
|
||||
>
|
||||
<FunnelIcon className="h-4 w-4" />
|
||||
<Filter className="h-4 w-4" />
|
||||
</Button>
|
||||
<Button
|
||||
onClick={() => setIsBackupModalOpen(true)}
|
||||
@@ -276,7 +276,7 @@ export const CronJobList = ({ cronJobs, scripts }: CronJobListProps) => {
|
||||
onClick={() => setIsNewCronModalOpen(true)}
|
||||
className="btn-primary glow-primary"
|
||||
>
|
||||
<PlusIcon className="h-4 w-4 mr-2" />
|
||||
<Plus className="h-4 w-4 mr-2" />
|
||||
{t("cronjobs.newTask")}
|
||||
</Button>
|
||||
</div>
|
||||
@@ -301,7 +301,7 @@ export const CronJobList = ({ cronJobs, scripts }: CronJobListProps) => {
|
||||
onNewTaskClick={() => setIsNewCronModalOpen(true)}
|
||||
/>
|
||||
) : (
|
||||
<div className="space-y-4 max-h-[55vh] min-h-[55vh] overflow-y-auto tui-scrollbar pr-2">
|
||||
<div className="space-y-3 max-h-[55vh] min-h-[55vh] overflow-y-auto">
|
||||
{loadedSettings ? (
|
||||
filteredJobs.map((job) =>
|
||||
minimalMode ? (
|
||||
@@ -347,7 +347,7 @@ export const CronJobList = ({ cronJobs, scripts }: CronJobListProps) => {
|
||||
)
|
||||
) : (
|
||||
<div className="flex items-center justify-center h-full min-h-[55vh]">
|
||||
<CircleNotchIcon className="h-8 w-8 animate-spin text-primary" />
|
||||
<Loader2 className="h-8 w-8 animate-spin text-primary" />
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
"use client";
|
||||
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
import { ClockIcon, PlusIcon } from "@phosphor-icons/react";
|
||||
import { Clock, Plus } from "lucide-react";
|
||||
|
||||
interface CronJobEmptyStateProps {
|
||||
selectedUser: string | null;
|
||||
@@ -15,7 +15,7 @@ export const CronJobEmptyState = ({
|
||||
return (
|
||||
<div className="text-center py-16">
|
||||
<div className="mx-auto w-20 h-20 bg-gradient-to-br from-primary/20 to-blue-500/20 rounded-full flex items-center justify-center mb-6">
|
||||
<ClockIcon className="h-10 w-10 text-primary" />
|
||||
<Clock className="h-10 w-10 text-primary" />
|
||||
</div>
|
||||
<h3 className="text-xl font-semibold mb-3 brand-gradient">
|
||||
{selectedUser
|
||||
@@ -32,7 +32,7 @@ export const CronJobEmptyState = ({
|
||||
className="btn-primary glow-primary"
|
||||
size="lg"
|
||||
>
|
||||
<PlusIcon className="h-5 w-5 mr-2" />
|
||||
<Plus className="h-5 w-5 mr-2" />
|
||||
Create Your First Task
|
||||
</Button>
|
||||
</div>
|
||||
|
||||
@@ -4,24 +4,24 @@ import { useState, useEffect } from "react";
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
import { DropdownMenu } from "@/app/_components/GlobalComponents/UIElements/DropdownMenu";
|
||||
import {
|
||||
TrashIcon,
|
||||
PencilSimpleIcon,
|
||||
FilesIcon,
|
||||
UserIcon,
|
||||
PlayIcon,
|
||||
PauseIcon,
|
||||
CodeIcon,
|
||||
InfoIcon,
|
||||
FileArrowDownIcon,
|
||||
FileXIcon,
|
||||
FileTextIcon,
|
||||
WarningCircleIcon,
|
||||
CheckCircleIcon,
|
||||
WarningIcon,
|
||||
DownloadIcon,
|
||||
HashIcon,
|
||||
CheckIcon,
|
||||
} from "@phosphor-icons/react";
|
||||
Trash2,
|
||||
Edit,
|
||||
Files,
|
||||
User,
|
||||
Play,
|
||||
Pause,
|
||||
Code,
|
||||
Info,
|
||||
FileOutput,
|
||||
FileX,
|
||||
FileText,
|
||||
AlertCircle,
|
||||
CheckCircle,
|
||||
AlertTriangle,
|
||||
Download,
|
||||
Hash,
|
||||
Check,
|
||||
} from "lucide-react";
|
||||
import { CronJob } from "@/app/_utils/cronjob-utils";
|
||||
import { JobError } from "@/app/_utils/error-utils";
|
||||
import { ErrorBadge } from "@/app/_components/GlobalComponents/Badges/ErrorBadge";
|
||||
@@ -92,7 +92,7 @@ export const CronJobItem = ({
|
||||
const dropdownMenuItems = [
|
||||
{
|
||||
label: t("cronjobs.editCronJob"),
|
||||
icon: <PencilSimpleIcon className="h-3 w-3" />,
|
||||
icon: <Edit className="h-3 w-3" />,
|
||||
onClick: () => onEdit(job),
|
||||
},
|
||||
{
|
||||
@@ -100,45 +100,45 @@ export const CronJobItem = ({
|
||||
? t("cronjobs.disableLogging")
|
||||
: t("cronjobs.enableLogging"),
|
||||
icon: job.logsEnabled ? (
|
||||
<FileXIcon className="h-3 w-3" />
|
||||
<FileX className="h-3 w-3" />
|
||||
) : (
|
||||
<FileArrowDownIcon className="h-3 w-3" />
|
||||
<FileOutput className="h-3 w-3" />
|
||||
),
|
||||
onClick: () => onToggleLogging(job.id),
|
||||
},
|
||||
...(job.logsEnabled
|
||||
? [
|
||||
{
|
||||
label: t("cronjobs.viewLogs"),
|
||||
icon: <FileTextIcon className="h-3 w-3" />,
|
||||
onClick: () => onViewLogs(job),
|
||||
},
|
||||
]
|
||||
{
|
||||
label: t("cronjobs.viewLogs"),
|
||||
icon: <FileText className="h-3 w-3" />,
|
||||
onClick: () => onViewLogs(job),
|
||||
},
|
||||
]
|
||||
: []),
|
||||
{
|
||||
label: job.paused
|
||||
? t("cronjobs.resumeCronJob")
|
||||
: t("cronjobs.pauseCronJob"),
|
||||
icon: job.paused ? (
|
||||
<PlayIcon className="h-3 w-3" />
|
||||
<Play className="h-3 w-3" />
|
||||
) : (
|
||||
<PauseIcon className="h-3 w-3" />
|
||||
<Pause className="h-3 w-3" />
|
||||
),
|
||||
onClick: () => (job.paused ? onResume(job.id) : onPause(job.id)),
|
||||
},
|
||||
{
|
||||
label: t("cronjobs.cloneCronJob"),
|
||||
icon: <FilesIcon className="h-3 w-3" />,
|
||||
icon: <Files className="h-3 w-3" />,
|
||||
onClick: () => onClone(job),
|
||||
},
|
||||
{
|
||||
label: t("cronjobs.backupJob"),
|
||||
icon: <DownloadIcon className="h-3 w-3" />,
|
||||
icon: <Download className="h-3 w-3" />,
|
||||
onClick: () => onBackup(job.id),
|
||||
},
|
||||
{
|
||||
label: t("cronjobs.deleteCronJob"),
|
||||
icon: <TrashIcon className="h-3 w-3" />,
|
||||
icon: <Trash2 className="h-3 w-3" />,
|
||||
onClick: () => onDelete(job),
|
||||
variant: "destructive" as const,
|
||||
disabled: deletingId === job.id,
|
||||
@@ -148,21 +148,22 @@ export const CronJobItem = ({
|
||||
return (
|
||||
<div
|
||||
key={job.id}
|
||||
className={`tui-card p-4 terminal-font transition-colors ${isDropdownOpen ? "relative z-10" : ""
|
||||
}`}
|
||||
className={`glass-card p-4 border border-border/50 rounded-lg hover:bg-accent/30 transition-colors ${
|
||||
isDropdownOpen ? "relative z-10" : ""
|
||||
}`}
|
||||
>
|
||||
<div className="flex flex-col sm:flex-row sm:items-start gap-4">
|
||||
<div className="flex-1 min-w-0">
|
||||
<div className="flex items-center gap-3 mb-2">
|
||||
{(scheduleDisplayMode === "cron" ||
|
||||
scheduleDisplayMode === "both") && (
|
||||
<code className="text-sm bg-background0 text-status-warning px-2 py-1 terminal-font ascii-border">
|
||||
{job.schedule}
|
||||
</code>
|
||||
)}
|
||||
<code className="text-sm bg-purple-500/10 text-purple-600 dark:text-purple-400 px-2 py-1 rounded font-mono border border-purple-500/20">
|
||||
{job.schedule}
|
||||
</code>
|
||||
)}
|
||||
{scheduleDisplayMode === "human" && cronExplanation?.isValid && (
|
||||
<div className="flex items-start gap-1.5 ascii-border bg-background2 px-2 py-0.5">
|
||||
<InfoIcon className="h-3 w-3 mt-0.5 flex-shrink-0" />
|
||||
<div className="flex items-start gap-1.5 border-b border-primary/30 bg-primary/10 rounded text-primary px-2 py-0.5">
|
||||
<Info className="h-3 w-3 text-primary mt-0.5 flex-shrink-0" />
|
||||
<p className="text-sm italic">
|
||||
{cronExplanation.humanReadable}
|
||||
</p>
|
||||
@@ -171,7 +172,7 @@ export const CronJobItem = ({
|
||||
<div className="flex-1 min-w-0">
|
||||
<div className="flex items-center gap-2 min-w-0 w-full">
|
||||
{commandCopied === job.id && (
|
||||
<CheckIcon className="h-3 w-3 text-status-success" />
|
||||
<Check className="h-3 w-3 text-green-600" />
|
||||
)}
|
||||
<pre
|
||||
onClick={(e) => {
|
||||
@@ -180,7 +181,7 @@ export const CronJobItem = ({
|
||||
setCommandCopied(job.id);
|
||||
setTimeout(() => setCommandCopied(null), 3000);
|
||||
}}
|
||||
className="w-full cursor-pointer overflow-x-auto text-sm font-medium terminal-font bg-background1 px-2 py-1 ascii-border hide-scrollbar"
|
||||
className="w-full cursor-pointer overflow-x-auto text-sm font-medium text-foreground bg-muted/30 px-2 py-1 rounded border border-border/30 hide-scrollbar"
|
||||
>
|
||||
{unwrapCommand(displayCommand)}
|
||||
</pre>
|
||||
@@ -190,8 +191,8 @@ export const CronJobItem = ({
|
||||
|
||||
<div className="flex items-center gap-2 pb-2 pt-4">
|
||||
{scheduleDisplayMode === "both" && cronExplanation?.isValid && (
|
||||
<div className="flex items-start gap-1.5 ascii-border bg-background2 px-2 py-0.5">
|
||||
<InfoIcon className="h-3 w-3 mt-0.5 flex-shrink-0" />
|
||||
<div className="flex items-start gap-1.5 border-b border-primary/30 bg-primary/10 rounded text-primary px-2 py-0.5">
|
||||
<Info className="h-3 w-3 text-primary mt-0.5 flex-shrink-0" />
|
||||
<p className="text-xs italic">
|
||||
{cronExplanation.humanReadable}
|
||||
</p>
|
||||
@@ -200,7 +201,7 @@ export const CronJobItem = ({
|
||||
|
||||
{job.comment && (
|
||||
<p
|
||||
className="text-xs italic truncate"
|
||||
className="text-xs text-muted-foreground italic truncate"
|
||||
title={job.comment}
|
||||
>
|
||||
{job.comment}
|
||||
@@ -209,13 +210,13 @@ export const CronJobItem = ({
|
||||
</div>
|
||||
|
||||
<div className="flex flex-wrap items-center gap-2 py-3">
|
||||
<div className="flex items-center gap-1 text-xs bg-background0 px-2 py-0.5 ascii-border cursor-pointer hover:bg-background2 transition-colors relative terminal-font">
|
||||
<UserIcon className="h-3 w-3" />
|
||||
<div className="flex items-center gap-1 text-xs bg-muted/50 text-muted-foreground px-2 py-0.5 rounded border border-border/30 cursor-pointer hover:bg-muted/70 transition-colors relative">
|
||||
<User className="h-3 w-3" />
|
||||
<span>{job.user}</span>
|
||||
</div>
|
||||
|
||||
<div
|
||||
className="flex items-center gap-1 text-xs bg-background0 px-2 py-0.5 ascii-border cursor-pointer hover:bg-background2 transition-colors relative terminal-font"
|
||||
className="flex items-center gap-1 text-xs bg-muted/50 text-muted-foreground px-2 py-0.5 rounded border border-border/30 cursor-pointer hover:bg-muted/70 transition-colors relative"
|
||||
title="Click to copy Job UUID"
|
||||
onClick={async () => {
|
||||
const success = await copyToClipboard(job.id);
|
||||
@@ -226,22 +227,22 @@ export const CronJobItem = ({
|
||||
}}
|
||||
>
|
||||
{showCopyConfirmation ? (
|
||||
<CheckIcon className="h-3 w-3 text-status-success" />
|
||||
<Check className="h-3 w-3 text-green-600" />
|
||||
) : (
|
||||
<HashIcon className="h-3 w-3" />
|
||||
<Hash className="h-3 w-3" />
|
||||
)}
|
||||
<span className="font-mono">{job.id}</span>
|
||||
</div>
|
||||
|
||||
{job.paused && (
|
||||
<span className="text-xs bg-background2 px-2 py-0.5 ascii-border terminal-font">
|
||||
<span className="text-status-warning">{t("cronjobs.paused")}</span>
|
||||
<span className="text-xs bg-yellow-500/10 text-yellow-600 dark:text-yellow-400 px-2 py-0.5 rounded border border-yellow-500/20">
|
||||
{t("cronjobs.paused")}
|
||||
</span>
|
||||
)}
|
||||
|
||||
{job.logsEnabled && (
|
||||
<span className="text-xs bg-background0 px-2 py-0.5 ascii-border terminal-font">
|
||||
<span className="text-status-info">{t("cronjobs.logged")}</span>
|
||||
<span className="text-xs bg-blue-500/10 text-blue-600 dark:text-blue-400 px-2 py-0.5 rounded border border-blue-500/20">
|
||||
{t("cronjobs.logged")}
|
||||
</span>
|
||||
)}
|
||||
|
||||
@@ -251,11 +252,11 @@ export const CronJobItem = ({
|
||||
e.stopPropagation();
|
||||
onViewLogs(job);
|
||||
}}
|
||||
className="flex items-center gap-1 text-xs bg-background0 px-2 py-0.5 ascii-border hover:bg-background1 transition-colors cursor-pointer terminal-font"
|
||||
className="flex items-center gap-1 text-xs bg-red-500/10 text-red-600 dark:text-red-400 px-2 py-0.5 rounded border border-red-500/30 hover:bg-red-500/20 transition-colors cursor-pointer"
|
||||
title="Latest execution failed - Click to view error log"
|
||||
>
|
||||
<WarningCircleIcon className="h-3 w-3 text-status-error" />
|
||||
<span className="text-status-error">
|
||||
<AlertCircle className="h-3 w-3" />
|
||||
<span>
|
||||
{t("cronjobs.failed", {
|
||||
exitCode: job.logError?.exitCode?.toString() ?? "",
|
||||
})}
|
||||
@@ -271,12 +272,12 @@ export const CronJobItem = ({
|
||||
e.stopPropagation();
|
||||
onViewLogs(job);
|
||||
}}
|
||||
className="flex items-center gap-1 text-xs bg-background0 px-2 py-0.5 ascii-border hover:bg-background1 transition-colors cursor-pointer terminal-font"
|
||||
className="flex items-center gap-1 text-xs bg-yellow-500/10 text-yellow-600 dark:text-yellow-400 px-2 py-0.5 rounded border border-yellow-500/30 hover:bg-yellow-500/20 transition-colors cursor-pointer"
|
||||
title="Latest execution succeeded, but has historical failures - Click to view logs"
|
||||
>
|
||||
<CheckCircleIcon className="h-3 w-3 text-status-success" />
|
||||
<span className="text-status-warning">{t("cronjobs.healthy")}</span>
|
||||
<WarningIcon className="h-3 w-3 text-status-warning" />
|
||||
<CheckCircle className="h-3 w-3" />
|
||||
<span>{t("cronjobs.healthy")}</span>
|
||||
<AlertTriangle className="h-3 w-3" />
|
||||
</button>
|
||||
)}
|
||||
|
||||
@@ -284,9 +285,9 @@ export const CronJobItem = ({
|
||||
!job.logError?.hasError &&
|
||||
!job.logError?.hasHistoricalFailures &&
|
||||
job.logError?.latestExitCode === 0 && (
|
||||
<div className="flex items-center gap-1 text-xs bg-background0 px-2 py-0.5 ascii-border terminal-font">
|
||||
<CheckCircleIcon className="h-3 w-3 text-status-success" />
|
||||
<span className="text-status-success">{t("cronjobs.healthy")}</span>
|
||||
<div className="flex items-center gap-1 text-xs bg-green-500/10 text-green-600 dark:text-green-400 px-2 py-0.5 rounded border border-green-500/30">
|
||||
<CheckCircle className="h-3 w-3" />
|
||||
<span>{t("cronjobs.healthy")}</span>
|
||||
</div>
|
||||
)}
|
||||
|
||||
@@ -314,7 +315,7 @@ export const CronJobItem = ({
|
||||
{runningJobId === job.id ? (
|
||||
<div className="h-3 w-3 animate-spin rounded-full border-2 border-current border-t-transparent" />
|
||||
) : (
|
||||
<CodeIcon className="h-3 w-3" />
|
||||
<Code className="h-3 w-3" />
|
||||
)}
|
||||
</Button>
|
||||
|
||||
@@ -333,9 +334,9 @@ export const CronJobItem = ({
|
||||
aria-label={t("cronjobs.pauseCronJob")}
|
||||
>
|
||||
{job.paused ? (
|
||||
<PlayIcon className="h-3 w-3" />
|
||||
<Play className="h-3 w-3" />
|
||||
) : (
|
||||
<PauseIcon className="h-3 w-3" />
|
||||
<Pause className="h-3 w-3" />
|
||||
)}
|
||||
</Button>
|
||||
|
||||
@@ -362,9 +363,9 @@ export const CronJobItem = ({
|
||||
}
|
||||
>
|
||||
{job.logsEnabled ? (
|
||||
<FileTextIcon className="h-3 w-3" />
|
||||
<FileText className="h-3 w-3" />
|
||||
) : (
|
||||
<FileArrowDownIcon className="h-3 w-3" />
|
||||
<FileOutput className="h-3 w-3" />
|
||||
)}
|
||||
</Button>
|
||||
</div>
|
||||
|
||||
@@ -4,19 +4,19 @@ import { useState, useEffect } from "react";
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
import { DropdownMenu } from "@/app/_components/GlobalComponents/UIElements/DropdownMenu";
|
||||
import {
|
||||
TrashIcon,
|
||||
PencilSimpleIcon,
|
||||
FilesIcon,
|
||||
PlayIcon,
|
||||
PauseIcon,
|
||||
CodeIcon,
|
||||
InfoIcon,
|
||||
DownloadIcon,
|
||||
CheckIcon,
|
||||
FileXIcon,
|
||||
FileTextIcon,
|
||||
FileArrowDownIcon,
|
||||
} from "@phosphor-icons/react";
|
||||
Trash2,
|
||||
Edit,
|
||||
Files,
|
||||
Play,
|
||||
Pause,
|
||||
Code,
|
||||
Info,
|
||||
Download,
|
||||
Check,
|
||||
FileX,
|
||||
FileText,
|
||||
FileOutput,
|
||||
} from "lucide-react";
|
||||
import { CronJob } from "@/app/_utils/cronjob-utils";
|
||||
import { JobError } from "@/app/_utils/error-utils";
|
||||
import {
|
||||
@@ -83,7 +83,7 @@ export const MinimalCronJobItem = ({
|
||||
const dropdownMenuItems = [
|
||||
{
|
||||
label: t("cronjobs.editCronJob"),
|
||||
icon: <PencilSimpleIcon className="h-3 w-3" />,
|
||||
icon: <Edit className="h-3 w-3" />,
|
||||
onClick: () => onEdit(job),
|
||||
},
|
||||
{
|
||||
@@ -91,9 +91,9 @@ export const MinimalCronJobItem = ({
|
||||
? t("cronjobs.disableLogging")
|
||||
: t("cronjobs.enableLogging"),
|
||||
icon: job.logsEnabled ? (
|
||||
<FileXIcon className="h-3 w-3" />
|
||||
<FileX className="h-3 w-3" />
|
||||
) : (
|
||||
<CodeIcon className="h-3 w-3" />
|
||||
<Code className="h-3 w-3" />
|
||||
),
|
||||
onClick: () => onToggleLogging(job.id),
|
||||
},
|
||||
@@ -101,7 +101,7 @@ export const MinimalCronJobItem = ({
|
||||
? [
|
||||
{
|
||||
label: t("cronjobs.viewLogs"),
|
||||
icon: <CodeIcon className="h-3 w-3" />,
|
||||
icon: <Code className="h-3 w-3" />,
|
||||
onClick: () => onViewLogs(job),
|
||||
},
|
||||
]
|
||||
@@ -111,25 +111,25 @@ export const MinimalCronJobItem = ({
|
||||
? t("cronjobs.resumeCronJob")
|
||||
: t("cronjobs.pauseCronJob"),
|
||||
icon: job.paused ? (
|
||||
<PlayIcon className="h-3 w-3" />
|
||||
<Play className="h-3 w-3" />
|
||||
) : (
|
||||
<PauseIcon className="h-3 w-3" />
|
||||
<Pause className="h-3 w-3" />
|
||||
),
|
||||
onClick: () => (job.paused ? onResume(job.id) : onPause(job.id)),
|
||||
},
|
||||
{
|
||||
label: t("cronjobs.cloneCronJob"),
|
||||
icon: <FilesIcon className="h-3 w-3" />,
|
||||
icon: <Files className="h-3 w-3" />,
|
||||
onClick: () => onClone(job),
|
||||
},
|
||||
{
|
||||
label: t("cronjobs.backupJob"),
|
||||
icon: <DownloadIcon className="h-3 w-3" />,
|
||||
icon: <Download className="h-3 w-3" />,
|
||||
onClick: () => onBackup(job.id),
|
||||
},
|
||||
{
|
||||
label: t("cronjobs.deleteCronJob"),
|
||||
icon: <TrashIcon className="h-3 w-3" />,
|
||||
icon: <Trash2 className="h-3 w-3" />,
|
||||
onClick: () => onDelete(job),
|
||||
variant: "destructive" as const,
|
||||
disabled: deletingId === job.id,
|
||||
@@ -139,19 +139,19 @@ export const MinimalCronJobItem = ({
|
||||
return (
|
||||
<div
|
||||
key={job.id}
|
||||
className={`tui-card p-3 terminal-font transition-colors ${isDropdownOpen ? "relative z-10" : ""
|
||||
className={`glass-card p-3 border border-border/50 rounded-lg hover:bg-accent/30 transition-colors ${isDropdownOpen ? "relative z-10" : ""
|
||||
}`}
|
||||
>
|
||||
<div className="flex items-center gap-3">
|
||||
<div className="flex items-center gap-1 flex-shrink-0">
|
||||
{scheduleDisplayMode === "cron" && (
|
||||
<code className="text-xs bg-background0 text-status-warning px-1.5 py-0.5 terminal-font ascii-border">
|
||||
<code className="text-xs bg-purple-500/10 text-purple-600 dark:text-purple-400 px-1.5 py-0.5 rounded font-mono border border-purple-500/20">
|
||||
{job.schedule}
|
||||
</code>
|
||||
)}
|
||||
{scheduleDisplayMode === "human" && cronExplanation?.isValid && (
|
||||
<div className="flex items-center gap-1 ascii-border bg-background2 px-1.5 py-0.5">
|
||||
<InfoIcon className="h-3 w-3 flex-shrink-0" />
|
||||
<div className="flex items-center gap-1 border-b border-primary/30 bg-primary/10 rounded text-primary px-1.5 py-0.5">
|
||||
<Info className="h-3 w-3 text-primary flex-shrink-0" />
|
||||
<span className="text-xs italic truncate max-w-32">
|
||||
{cronExplanation.humanReadable}
|
||||
</span>
|
||||
@@ -159,15 +159,15 @@ export const MinimalCronJobItem = ({
|
||||
)}
|
||||
{scheduleDisplayMode === "both" && (
|
||||
<div className="flex items-center gap-1">
|
||||
<code className="text-xs bg-background0 text-status-warning px-1 py-0.5 terminal-font ascii-border">
|
||||
<code className="text-xs bg-purple-500/10 text-purple-600 dark:text-purple-400 px-1 py-0.5 rounded font-mono border border-purple-500/20">
|
||||
{job.schedule}
|
||||
</code>
|
||||
{cronExplanation?.isValid && (
|
||||
<div
|
||||
className="flex items-center gap-1 ascii-border bg-background0 px-1 py-0.5 cursor-help"
|
||||
className="flex items-center gap-1 border-b border-primary/30 bg-primary/10 rounded text-primary px-1 py-0.5 cursor-help"
|
||||
title={cronExplanation.humanReadable}
|
||||
>
|
||||
<InfoIcon className="h-2.5 w-2.5 flex-shrink-0" />
|
||||
<Info className="h-2.5 w-2.5 text-primary flex-shrink-0" />
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
@@ -177,7 +177,7 @@ export const MinimalCronJobItem = ({
|
||||
<div className="flex-1 min-w-0">
|
||||
<div className="flex items-center gap-2">
|
||||
{commandCopied === job.id && (
|
||||
<CheckIcon className="h-3 w-3 text-status-success flex-shrink-0" />
|
||||
<Check className="h-3 w-3 text-green-600 flex-shrink-0" />
|
||||
)}
|
||||
<pre
|
||||
onClick={(e) => {
|
||||
@@ -186,7 +186,7 @@ export const MinimalCronJobItem = ({
|
||||
setCommandCopied(job.id);
|
||||
setTimeout(() => setCommandCopied(null), 3000);
|
||||
}}
|
||||
className="flex-1 cursor-pointer overflow-hidden text-sm font-medium terminal-font bg-background1 px-2 py-1 ascii-border truncate"
|
||||
className="flex-1 cursor-pointer overflow-hidden text-sm font-medium text-foreground bg-muted/30 px-2 py-1 rounded border border-border/30 truncate"
|
||||
title={unwrapCommand(job.command)}
|
||||
>
|
||||
{unwrapCommand(displayCommand)}
|
||||
@@ -197,25 +197,25 @@ export const MinimalCronJobItem = ({
|
||||
<div className="flex items-center gap-1 flex-shrink-0">
|
||||
{job.logsEnabled && (
|
||||
<div
|
||||
className="w-2 h-2 bg-status-info ascii-border"
|
||||
className="w-2 h-2 bg-blue-500 rounded-full"
|
||||
title={t("cronjobs.logged")}
|
||||
/>
|
||||
)}
|
||||
{job.paused && (
|
||||
<div
|
||||
className="w-2 h-2 bg-status-warning ascii-border"
|
||||
className="w-2 h-2 bg-yellow-500 rounded-full"
|
||||
title={t("cronjobs.paused")}
|
||||
/>
|
||||
)}
|
||||
{!job.logError?.hasError && job.logsEnabled && (
|
||||
<div
|
||||
className="w-2 h-2 bg-status-success ascii-border"
|
||||
className="w-2 h-2 bg-green-500 rounded-full"
|
||||
title={t("cronjobs.healthy")}
|
||||
/>
|
||||
)}
|
||||
{job.logsEnabled && job.logError?.hasError && (
|
||||
<div
|
||||
className="w-2 h-2 bg-status-error ascii-border cursor-pointer"
|
||||
className="w-2 h-2 bg-red-500 rounded-full cursor-pointer"
|
||||
title="Latest execution failed - Click to view error log"
|
||||
onClick={(e) => {
|
||||
e.stopPropagation();
|
||||
@@ -225,32 +225,31 @@ export const MinimalCronJobItem = ({
|
||||
)}
|
||||
{!job.logsEnabled && errors.length > 0 && (
|
||||
<div
|
||||
className="w-2 h-2 bg-status-warning ascii-border cursor-pointer"
|
||||
className="w-2 h-2 bg-orange-500 rounded-full cursor-pointer"
|
||||
title={`${errors.length} error(s)`}
|
||||
onClick={(e) => onErrorClick(errors[0])}
|
||||
/>
|
||||
)}
|
||||
</div>
|
||||
|
||||
<div className="flex items-center gap-2 flex-shrink-0">
|
||||
<div className="flex items-center gap-1 flex-shrink-0">
|
||||
<Button
|
||||
variant="outline"
|
||||
variant="ghost"
|
||||
size="sm"
|
||||
onClick={() => onRun(job.id)}
|
||||
disabled={runningJobId === job.id || job.paused}
|
||||
className="btn-outline h-8 px-3"
|
||||
className="h-6 w-6 p-0"
|
||||
title={t("cronjobs.runCronManually")}
|
||||
aria-label={t("cronjobs.runCronManually")}
|
||||
>
|
||||
{runningJobId === job.id ? (
|
||||
<div className="h-3 w-3 animate-spin rounded-full border-2 border-current border-t-transparent" />
|
||||
) : (
|
||||
<CodeIcon className="h-3 w-3" />
|
||||
<Code className="h-3 w-3" />
|
||||
)}
|
||||
</Button>
|
||||
|
||||
<Button
|
||||
variant="outline"
|
||||
variant="ghost"
|
||||
size="sm"
|
||||
onClick={() => {
|
||||
if (job.paused) {
|
||||
@@ -259,19 +258,18 @@ export const MinimalCronJobItem = ({
|
||||
onPause(job.id);
|
||||
}
|
||||
}}
|
||||
className="btn-outline h-8 px-3"
|
||||
className="h-6 w-6 p-0"
|
||||
title={t("cronjobs.pauseCronJob")}
|
||||
aria-label={t("cronjobs.pauseCronJob")}
|
||||
>
|
||||
{job.paused ? (
|
||||
<PlayIcon className="h-3 w-3" />
|
||||
<Play className="h-3 w-3" />
|
||||
) : (
|
||||
<PauseIcon className="h-3 w-3" />
|
||||
<Pause className="h-3 w-3" />
|
||||
)}
|
||||
</Button>
|
||||
|
||||
<Button
|
||||
variant="outline"
|
||||
variant="ghost"
|
||||
size="sm"
|
||||
onClick={() => {
|
||||
if (job.logsEnabled) {
|
||||
@@ -280,22 +278,17 @@ export const MinimalCronJobItem = ({
|
||||
onToggleLogging(job.id);
|
||||
}
|
||||
}}
|
||||
className="btn-outline h-8 px-3"
|
||||
className="h-6 w-6 p-0"
|
||||
title={
|
||||
job.logsEnabled
|
||||
? t("cronjobs.viewLogs")
|
||||
: t("cronjobs.enableLogging")
|
||||
}
|
||||
aria-label={
|
||||
job.logsEnabled
|
||||
? t("cronjobs.viewLogs")
|
||||
: t("cronjobs.enableLogging")
|
||||
}
|
||||
>
|
||||
{job.logsEnabled ? (
|
||||
<FileTextIcon className="h-3 w-3" />
|
||||
<FileText className="h-3 w-3" />
|
||||
) : (
|
||||
<FileArrowDownIcon className="h-3 w-3" />
|
||||
<FileOutput className="h-3 w-3" />
|
||||
)}
|
||||
</Button>
|
||||
|
||||
|
||||
@@ -1,431 +0,0 @@
|
||||
"use client";
|
||||
|
||||
import { useEffect, useRef, useState, useCallback } from "react";
|
||||
import { useTranslations } from "next-intl";
|
||||
import { ArrowUpIcon, ArrowDownIcon, ArrowLeftIcon, ArrowRightIcon, ArrowClockwiseIcon, PlayIcon, PauseIcon } from "@phosphor-icons/react";
|
||||
|
||||
interface Position {
|
||||
x: number;
|
||||
y: number;
|
||||
}
|
||||
|
||||
type Direction = "UP" | "DOWN" | "LEFT" | "RIGHT";
|
||||
|
||||
const GRID_SIZE = 20;
|
||||
const INITIAL_SNAKE: Position[] = [
|
||||
{ x: 10, y: 10 },
|
||||
{ x: 9, y: 10 },
|
||||
{ x: 8, y: 10 },
|
||||
];
|
||||
const INITIAL_DIRECTION: Direction = "RIGHT";
|
||||
const GAME_SPEED = 150;
|
||||
|
||||
export const SnakeGame = () => {
|
||||
const t = useTranslations("notFound");
|
||||
const canvasRef = useRef<HTMLCanvasElement>(null);
|
||||
const containerRef = useRef<HTMLDivElement>(null);
|
||||
const [snake, setSnake] = useState<Position[]>(INITIAL_SNAKE);
|
||||
const [direction, setDirection] = useState<Direction>(INITIAL_DIRECTION);
|
||||
const [food, setFood] = useState<Position>({ x: 15, y: 15 });
|
||||
const [gameOver, setGameOver] = useState(false);
|
||||
const [gameStarted, setGameStarted] = useState(false);
|
||||
const [score, setScore] = useState(0);
|
||||
const [highScore, setHighScore] = useState(0);
|
||||
const [isPaused, setIsPaused] = useState(false);
|
||||
const [colors, setColors] = useState({ snake: "#00ff00", food: "#ff0000", grid: "#333333" });
|
||||
const [cellSize, setCellSize] = useState(20);
|
||||
|
||||
const directionRef = useRef<Direction>(INITIAL_DIRECTION);
|
||||
const gameLoopRef = useRef<NodeJS.Timeout | null>(null);
|
||||
|
||||
useEffect(() => {
|
||||
const savedHighScore = localStorage.getItem("snakeHighScore");
|
||||
if (savedHighScore) {
|
||||
setHighScore(parseInt(savedHighScore));
|
||||
}
|
||||
|
||||
const updateColors = () => {
|
||||
const theme = document.documentElement.getAttribute("data-webtui-theme");
|
||||
if (theme === "catppuccin-mocha") {
|
||||
setColors({
|
||||
snake: "#9ca0b0",
|
||||
food: "#f38ba8",
|
||||
grid: "#313244",
|
||||
});
|
||||
} else {
|
||||
setColors({
|
||||
snake: "#313244",
|
||||
food: "#d20f39",
|
||||
grid: "#9ca0b0",
|
||||
});
|
||||
}
|
||||
};
|
||||
|
||||
updateColors();
|
||||
|
||||
const observer = new MutationObserver(updateColors);
|
||||
observer.observe(document.documentElement, {
|
||||
attributes: true,
|
||||
attributeFilter: ["data-webtui-theme"],
|
||||
});
|
||||
|
||||
const updateCellSize = () => {
|
||||
if (containerRef.current) {
|
||||
const containerWidth = containerRef.current.offsetWidth;
|
||||
const maxCanvasSize = Math.min(containerWidth - 32, 400);
|
||||
const newCellSize = Math.floor(maxCanvasSize / GRID_SIZE);
|
||||
setCellSize(newCellSize);
|
||||
}
|
||||
};
|
||||
|
||||
updateCellSize();
|
||||
window.addEventListener("resize", updateCellSize);
|
||||
|
||||
return () => {
|
||||
observer.disconnect();
|
||||
window.removeEventListener("resize", updateCellSize);
|
||||
};
|
||||
}, []);
|
||||
|
||||
const generateFood = useCallback((): Position => {
|
||||
let newFood: Position;
|
||||
do {
|
||||
newFood = {
|
||||
x: Math.floor(Math.random() * GRID_SIZE),
|
||||
y: Math.floor(Math.random() * GRID_SIZE),
|
||||
};
|
||||
} while (snake.some((segment) => segment.x === newFood.x && segment.y === newFood.y));
|
||||
return newFood;
|
||||
}, [snake]);
|
||||
|
||||
const resetGame = useCallback(() => {
|
||||
setSnake(INITIAL_SNAKE);
|
||||
setDirection(INITIAL_DIRECTION);
|
||||
directionRef.current = INITIAL_DIRECTION;
|
||||
setFood(generateFood());
|
||||
setGameOver(false);
|
||||
setGameStarted(true);
|
||||
setScore(0);
|
||||
setIsPaused(false);
|
||||
}, [generateFood]);
|
||||
|
||||
const draw = useCallback(() => {
|
||||
const canvas = canvasRef.current;
|
||||
if (!canvas) return;
|
||||
|
||||
const ctx = canvas.getContext("2d");
|
||||
if (!ctx) return;
|
||||
|
||||
const theme = document.documentElement.getAttribute("data-webtui-theme");
|
||||
const bgColor = theme === "catppuccin-mocha" ? "#1e1e2e" : "#eff1f5";
|
||||
|
||||
ctx.fillStyle = bgColor;
|
||||
ctx.fillRect(0, 0, canvas.width, canvas.height);
|
||||
|
||||
ctx.strokeStyle = colors.grid;
|
||||
ctx.lineWidth = 1;
|
||||
for (let i = 0; i <= GRID_SIZE; i++) {
|
||||
ctx.beginPath();
|
||||
ctx.moveTo(i * cellSize, 0);
|
||||
ctx.lineTo(i * cellSize, GRID_SIZE * cellSize);
|
||||
ctx.stroke();
|
||||
|
||||
ctx.beginPath();
|
||||
ctx.moveTo(0, i * cellSize);
|
||||
ctx.lineTo(GRID_SIZE * cellSize, i * cellSize);
|
||||
ctx.stroke();
|
||||
}
|
||||
|
||||
snake.forEach((segment) => {
|
||||
ctx.fillStyle = colors.snake;
|
||||
ctx.fillRect(
|
||||
segment.x * cellSize + 1,
|
||||
segment.y * cellSize + 1,
|
||||
cellSize - 2,
|
||||
cellSize - 2
|
||||
);
|
||||
});
|
||||
|
||||
ctx.fillStyle = colors.food;
|
||||
ctx.fillRect(
|
||||
food.x * cellSize + 1,
|
||||
food.y * cellSize + 1,
|
||||
cellSize - 2,
|
||||
cellSize - 2
|
||||
);
|
||||
}, [snake, food, colors, cellSize]);
|
||||
|
||||
useEffect(() => {
|
||||
draw();
|
||||
}, [draw]);
|
||||
|
||||
const moveSnake = useCallback(() => {
|
||||
if (gameOver || !gameStarted || isPaused) return;
|
||||
|
||||
setSnake((prevSnake) => {
|
||||
const head = prevSnake[0];
|
||||
const newHead: Position = { ...head };
|
||||
|
||||
switch (directionRef.current) {
|
||||
case "UP":
|
||||
newHead.y -= 1;
|
||||
break;
|
||||
case "DOWN":
|
||||
newHead.y += 1;
|
||||
break;
|
||||
case "LEFT":
|
||||
newHead.x -= 1;
|
||||
break;
|
||||
case "RIGHT":
|
||||
newHead.x += 1;
|
||||
break;
|
||||
}
|
||||
|
||||
if (
|
||||
newHead.x < 0 ||
|
||||
newHead.x >= GRID_SIZE ||
|
||||
newHead.y < 0 ||
|
||||
newHead.y >= GRID_SIZE ||
|
||||
prevSnake.some((segment) => segment.x === newHead.x && segment.y === newHead.y)
|
||||
) {
|
||||
setGameOver(true);
|
||||
setGameStarted(false);
|
||||
return prevSnake;
|
||||
}
|
||||
|
||||
const newSnake = [newHead, ...prevSnake];
|
||||
|
||||
if (newHead.x === food.x && newHead.y === food.y) {
|
||||
setScore((prev) => {
|
||||
const newScore = prev + 10;
|
||||
if (newScore > highScore) {
|
||||
setHighScore(newScore);
|
||||
localStorage.setItem("snakeHighScore", newScore.toString());
|
||||
}
|
||||
return newScore;
|
||||
});
|
||||
setFood(generateFood());
|
||||
} else {
|
||||
newSnake.pop();
|
||||
}
|
||||
|
||||
return newSnake;
|
||||
});
|
||||
}, [gameOver, gameStarted, isPaused, food, highScore, generateFood]);
|
||||
|
||||
useEffect(() => {
|
||||
if (gameStarted && !gameOver && !isPaused) {
|
||||
gameLoopRef.current = setInterval(moveSnake, GAME_SPEED);
|
||||
}
|
||||
|
||||
return () => {
|
||||
if (gameLoopRef.current) {
|
||||
clearInterval(gameLoopRef.current);
|
||||
}
|
||||
};
|
||||
}, [gameStarted, gameOver, isPaused, moveSnake]);
|
||||
|
||||
useEffect(() => {
|
||||
const handleKeyPress = (e: KeyboardEvent) => {
|
||||
if (e.code === "Space") {
|
||||
e.preventDefault();
|
||||
if (gameOver) {
|
||||
resetGame();
|
||||
} else if (!gameStarted) {
|
||||
setGameStarted(true);
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
if (e.code === "KeyP") {
|
||||
e.preventDefault();
|
||||
if (gameStarted && !gameOver) {
|
||||
setIsPaused((prev) => !prev);
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
if (!gameStarted || gameOver || isPaused) return;
|
||||
|
||||
let newDirection: Direction | null = null;
|
||||
|
||||
switch (e.key) {
|
||||
case "ArrowUp":
|
||||
if (directionRef.current !== "DOWN") {
|
||||
newDirection = "UP";
|
||||
}
|
||||
break;
|
||||
case "ArrowDown":
|
||||
if (directionRef.current !== "UP") {
|
||||
newDirection = "DOWN";
|
||||
}
|
||||
break;
|
||||
case "ArrowLeft":
|
||||
if (directionRef.current !== "RIGHT") {
|
||||
newDirection = "LEFT";
|
||||
}
|
||||
break;
|
||||
case "ArrowRight":
|
||||
if (directionRef.current !== "LEFT") {
|
||||
newDirection = "RIGHT";
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
if (newDirection) {
|
||||
e.preventDefault();
|
||||
directionRef.current = newDirection;
|
||||
setDirection(newDirection);
|
||||
}
|
||||
};
|
||||
|
||||
window.addEventListener("keydown", handleKeyPress);
|
||||
return () => window.removeEventListener("keydown", handleKeyPress);
|
||||
}, [gameOver, gameStarted, isPaused, resetGame]);
|
||||
|
||||
const handleTouchMove = (dir: Direction) => {
|
||||
if (!gameStarted) {
|
||||
setGameStarted(true);
|
||||
directionRef.current = dir;
|
||||
setDirection(dir);
|
||||
return;
|
||||
}
|
||||
if (gameOver || isPaused) return;
|
||||
|
||||
let canMove = false;
|
||||
|
||||
switch (dir) {
|
||||
case "UP":
|
||||
canMove = directionRef.current !== "DOWN";
|
||||
break;
|
||||
case "DOWN":
|
||||
canMove = directionRef.current !== "UP";
|
||||
break;
|
||||
case "LEFT":
|
||||
canMove = directionRef.current !== "RIGHT";
|
||||
break;
|
||||
case "RIGHT":
|
||||
canMove = directionRef.current !== "LEFT";
|
||||
break;
|
||||
}
|
||||
|
||||
if (canMove) {
|
||||
directionRef.current = dir;
|
||||
setDirection(dir);
|
||||
}
|
||||
};
|
||||
|
||||
const handleCanvasClick = () => {
|
||||
if (gameOver) {
|
||||
resetGame();
|
||||
} else if (!gameStarted) {
|
||||
setGameStarted(true);
|
||||
}
|
||||
};
|
||||
|
||||
return (
|
||||
<div ref={containerRef} className="flex flex-col items-center gap-4 w-full">
|
||||
<div className="tui-card p-4 w-full max-w-md">
|
||||
<div className="flex justify-between items-center mb-4 terminal-font">
|
||||
<div className="text-sm">
|
||||
<span className="text-status-success">{t("score")}:</span> {score}
|
||||
</div>
|
||||
<div className="text-sm">
|
||||
<span className="text-status-info">{t("highScore")}:</span> {highScore}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div className="relative">
|
||||
<canvas
|
||||
ref={canvasRef}
|
||||
width={GRID_SIZE * cellSize}
|
||||
height={GRID_SIZE * cellSize}
|
||||
className="ascii-border mx-auto cursor-pointer bg-background0"
|
||||
onClick={handleCanvasClick}
|
||||
/>
|
||||
|
||||
{!gameStarted && !gameOver && (
|
||||
<div
|
||||
className="absolute inset-0 flex items-center justify-center bg-background0 bg-opacity-90 ascii-border cursor-pointer"
|
||||
onClick={handleCanvasClick}
|
||||
>
|
||||
<div className="text-center terminal-font">
|
||||
<p className="text-lg font-bold uppercase mb-2">{t("pressToStart")}</p>
|
||||
<p className="text-xs text-foreground0 opacity-70">{t("pauseGame")}</p>
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
|
||||
{gameOver && (
|
||||
<div
|
||||
className="absolute inset-0 flex items-center justify-center bg-background0 bg-opacity-90 ascii-border cursor-pointer"
|
||||
onClick={handleCanvasClick}
|
||||
>
|
||||
<div className="text-center terminal-font">
|
||||
<p className="text-2xl font-bold uppercase text-status-error mb-2">
|
||||
{t("gameOver")}
|
||||
</p>
|
||||
<p className="text-lg mb-1">
|
||||
{t("score")}: {score}
|
||||
</p>
|
||||
<p className="text-sm text-foreground0 opacity-70">{t("pressToRestart")}</p>
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
|
||||
{isPaused && (
|
||||
<div className="absolute inset-0 flex items-center justify-center bg-background0 bg-opacity-90 ascii-border">
|
||||
<div className="text-center terminal-font">
|
||||
<p className="text-2xl font-bold uppercase text-status-warning">
|
||||
{t("paused")}
|
||||
</p>
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
|
||||
<div className="mt-4 grid grid-cols-3 gap-2 md:hidden">
|
||||
<div></div>
|
||||
<button
|
||||
onClick={() => handleTouchMove("UP")}
|
||||
className="ascii-border bg-background1 p-3 active:bg-background2 terminal-font flex items-center justify-center"
|
||||
>
|
||||
<ArrowUpIcon size={20} weight="bold" />
|
||||
</button>
|
||||
<div></div>
|
||||
<button
|
||||
onClick={() => handleTouchMove("LEFT")}
|
||||
className="ascii-border bg-background1 p-3 active:bg-background2 terminal-font flex items-center justify-center"
|
||||
>
|
||||
<ArrowLeftIcon size={20} weight="bold" />
|
||||
</button>
|
||||
<button
|
||||
onClick={() => (gameStarted && !gameOver ? setIsPaused(!isPaused) : resetGame())}
|
||||
className="ascii-border bg-background1 p-3 active:bg-background2 terminal-font flex items-center justify-center"
|
||||
>
|
||||
{gameOver ? <ArrowClockwiseIcon size={20} weight="bold" /> : isPaused ? <PlayIcon size={20} weight="fill" /> : <PauseIcon size={20} weight="fill" />}
|
||||
</button>
|
||||
<button
|
||||
onClick={() => handleTouchMove("RIGHT")}
|
||||
className="ascii-border bg-background1 p-3 active:bg-background2 terminal-font flex items-center justify-center"
|
||||
>
|
||||
<ArrowRightIcon size={20} weight="bold" />
|
||||
</button>
|
||||
<div></div>
|
||||
<button
|
||||
onClick={() => handleTouchMove("DOWN")}
|
||||
className="ascii-border bg-background1 p-3 active:bg-background2 terminal-font flex items-center justify-center"
|
||||
>
|
||||
<ArrowDownIcon size={20} weight="bold" />
|
||||
</button>
|
||||
<div></div>
|
||||
</div>
|
||||
|
||||
<div className="mt-4 terminal-font text-xs text-center text-foreground0 opacity-70">
|
||||
<p className="hidden md:block">{t("useArrowKeys")}</p>
|
||||
<p className="md:hidden">{t("tapToMove")}</p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
@@ -2,15 +2,15 @@ import { cn } from "@/app/_utils/global-utils";
|
||||
import { HTMLAttributes, forwardRef, useState, useEffect } from "react";
|
||||
import React from "react";
|
||||
import {
|
||||
CaretLeftIcon,
|
||||
CaretRightIcon,
|
||||
HardDrivesIcon,
|
||||
ListIcon,
|
||||
XIcon,
|
||||
CpuIcon,
|
||||
HardDriveIcon,
|
||||
WifiHighIcon,
|
||||
} from "@phosphor-icons/react";
|
||||
ChevronLeft,
|
||||
ChevronRight,
|
||||
Server,
|
||||
Menu,
|
||||
X,
|
||||
Cpu,
|
||||
HardDrive,
|
||||
Wifi,
|
||||
} from "lucide-react";
|
||||
import { useTranslations } from "next-intl";
|
||||
|
||||
export interface SidebarProps extends HTMLAttributes<HTMLDivElement> {
|
||||
@@ -54,18 +54,18 @@ export const Sidebar = forwardRef<HTMLDivElement, SidebarProps>(
|
||||
<>
|
||||
<button
|
||||
onClick={() => setIsMobileOpen(!isMobileOpen)}
|
||||
className="fixed bottom-4 right-4 z-50 lg:hidden p-2 bg-background0 ascii-border transition-colors terminal-font"
|
||||
className="fixed bottom-4 right-4 z-50 lg:hidden p-2 bg-background/80 backdrop-blur-md border border-border/50 rounded-lg hover:bg-accent transition-colors"
|
||||
>
|
||||
{isMobileOpen ? (
|
||||
<XIcon className="h-5 w-5" />
|
||||
<X className="h-5 w-5" />
|
||||
) : (
|
||||
<ListIcon className="h-5 w-5" />
|
||||
<Menu className="h-5 w-5" />
|
||||
)}
|
||||
</button>
|
||||
|
||||
<div
|
||||
className={cn(
|
||||
"fixed inset-0 bg-background0 z-20 lg:hidden transition-opacity duration-300",
|
||||
"fixed inset-0 bg-black/50 z-20 lg:hidden transition-opacity duration-300",
|
||||
isMobileOpen ? "opacity-100" : "opacity-0 pointer-events-none"
|
||||
)}
|
||||
onClick={() => setIsMobileOpen(false)}
|
||||
@@ -74,7 +74,7 @@ export const Sidebar = forwardRef<HTMLDivElement, SidebarProps>(
|
||||
<div
|
||||
ref={ref}
|
||||
className={cn(
|
||||
"bg-background0 ascii-border transition-all duration-300 ease-in-out terminal-font",
|
||||
"bg-background/95 backdrop-blur-md border-r border-border/50 transition-all duration-300 ease-in-out glass-card",
|
||||
isMobileOpen
|
||||
? "fixed left-0 top-0 h-full w-80 z-30 translate-x-0"
|
||||
: "fixed left-0 top-0 h-full w-80 z-30 -translate-x-full lg:translate-x-0",
|
||||
@@ -92,27 +92,27 @@ export const Sidebar = forwardRef<HTMLDivElement, SidebarProps>(
|
||||
>
|
||||
<button
|
||||
onClick={() => setIsCollapsed(!isCollapsed)}
|
||||
className="absolute -right-3 top-[21.5vh] w-6 h-6 bg-background0 ascii-border items-center justify-center transition-colors z-40 hidden lg:flex"
|
||||
className="absolute -right-3 top-[21.5vh] w-6 h-6 bg-background border border-border rounded-full items-center justify-center hover:bg-accent transition-colors z-40 hidden lg:flex"
|
||||
>
|
||||
{isCollapsed ? (
|
||||
<CaretRightIcon className="h-3 w-3" />
|
||||
<ChevronRight className="h-3 w-3" />
|
||||
) : (
|
||||
<CaretLeftIcon className="h-3 w-3" />
|
||||
<ChevronLeft className="h-3 w-3" />
|
||||
)}
|
||||
</button>
|
||||
|
||||
<div className="p-4 ascii-border !border-t-0 border-l-0 !border-r-0 bg-background0">
|
||||
<div className="p-4 border-b border-border/50 bg-background/95 backdrop-blur-md">
|
||||
<div
|
||||
className={cn(
|
||||
"flex items-center gap-3",
|
||||
isCollapsed && "lg:justify-center"
|
||||
)}
|
||||
>
|
||||
<div className="p-2 bg-background0 ascii-border flex-shrink-0">
|
||||
<HardDrivesIcon className="h-4 w-4" />
|
||||
<div className="p-2 bg-gradient-to-br from-cyan-500/20 to-blue-500/20 rounded-lg flex-shrink-0">
|
||||
<Server className="h-4 w-4 text-cyan-500" />
|
||||
</div>
|
||||
{(!isCollapsed || !isCollapsed) && (
|
||||
<h2 className="text-sm font-semibold truncate terminal-font">
|
||||
<h2 className="text-sm font-semibold text-foreground truncate">
|
||||
{t("sidebar.systemOverview")}
|
||||
</h2>
|
||||
)}
|
||||
@@ -121,7 +121,7 @@ export const Sidebar = forwardRef<HTMLDivElement, SidebarProps>(
|
||||
|
||||
<div
|
||||
className={cn(
|
||||
"overflow-y-auto tui-scrollbar",
|
||||
"overflow-y-auto custom-scrollbar",
|
||||
isCollapsed ? "lg:p-2" : "p-4",
|
||||
"h-full lg:h-[calc(100vh-88px-80px)]"
|
||||
)}
|
||||
@@ -131,22 +131,22 @@ export const Sidebar = forwardRef<HTMLDivElement, SidebarProps>(
|
||||
<div className="flex flex-col items-center space-y-4">
|
||||
{quickStats ? (
|
||||
<>
|
||||
<div className="w-12 h-12 bg-background0 ascii-border flex flex-col items-center justify-center p-1">
|
||||
<CpuIcon className="h-3 w-3 mb-1" />
|
||||
<div className="w-12 h-12 bg-card/50 border border-border/30 rounded-lg flex flex-col items-center justify-center p-1">
|
||||
<Cpu className="h-3 w-3 text-pink-500 mb-1" />
|
||||
<span className="text-xs font-bold text-foreground">
|
||||
{quickStats.cpu}%
|
||||
</span>
|
||||
</div>
|
||||
|
||||
<div className="w-12 h-12 bg-background0 ascii-border flex flex-col items-center justify-center p-1">
|
||||
<HardDriveIcon className="h-3 w-3 mb-1" />
|
||||
<div className="w-12 h-12 bg-card/50 border border-border/30 rounded-lg flex flex-col items-center justify-center p-1">
|
||||
<HardDrive className="h-3 w-3 text-cyan-500 mb-1" />
|
||||
<span className="text-xs font-bold text-foreground">
|
||||
{quickStats.memory}%
|
||||
</span>
|
||||
</div>
|
||||
|
||||
<div className="w-12 h-12 bg-background0 ascii-border flex flex-col items-center justify-center p-1">
|
||||
<WifiHighIcon className="h-3 w-3 mb-1" />
|
||||
<div className="w-12 h-12 bg-card/50 border border-border/30 rounded-lg flex flex-col items-center justify-center p-1">
|
||||
<Wifi className="h-3 w-3 text-teal-500 mb-1" />
|
||||
<div className="flex flex-col items-center">
|
||||
<span className="text-xs font-bold text-foreground leading-none">
|
||||
{quickStats.network}
|
||||
@@ -163,9 +163,9 @@ export const Sidebar = forwardRef<HTMLDivElement, SidebarProps>(
|
||||
return (
|
||||
<div
|
||||
key={index}
|
||||
className="w-8 h-8 bg-background2 ascii-border flex items-center justify-center"
|
||||
className="w-8 h-8 bg-card/50 border border-border/30 rounded-lg flex items-center justify-center"
|
||||
>
|
||||
<HardDrivesIcon className="h-4 w-4 text-muted-foreground" />
|
||||
<Server className="h-4 w-4 text-muted-foreground" />
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
@@ -5,7 +5,7 @@ import { CronJobList } from "@/app/_components/FeatureComponents/Cronjobs/CronJo
|
||||
import { ScriptsManager } from "@/app/_components/FeatureComponents/Scripts/ScriptsManager";
|
||||
import { CronJob } from "@/app/_utils/cronjob-utils";
|
||||
import { Script } from "@/app/_utils/scripts-utils";
|
||||
import { ClockIcon, FileTextIcon } from "@phosphor-icons/react";
|
||||
import { Clock, FileText } from "lucide-react";
|
||||
import { useTranslations } from "next-intl";
|
||||
|
||||
interface TabbedInterfaceProps {
|
||||
@@ -24,31 +24,33 @@ export const TabbedInterface = ({
|
||||
|
||||
return (
|
||||
<div className="space-y-6">
|
||||
<div className="tui-card p-1 terminal-font">
|
||||
<div className="flex gap-2">
|
||||
<div className="bg-background/80 backdrop-blur-md border border-border/50 rounded-lg p-1 glass-card">
|
||||
<div className="flex">
|
||||
<button
|
||||
onClick={() => setActiveTab("cronjobs")}
|
||||
className={`flex items-center gap-2 px-4 py-2 text-sm font-medium flex-1 justify-center terminal-font ${activeTab === "cronjobs"
|
||||
? "bg-background0 ascii-border"
|
||||
: "hover:ascii-border"
|
||||
}`}
|
||||
className={`flex items-center gap-2 px-4 py-2 text-sm font-medium transition-all duration-200 rounded-md flex-1 justify-center ${
|
||||
activeTab === "cronjobs"
|
||||
? "bg-primary/10 text-primary shadow-sm border border-primary/20"
|
||||
: "text-muted-foreground hover:text-foreground hover:bg-muted/50"
|
||||
}`}
|
||||
>
|
||||
<ClockIcon className="h-4 w-4" />
|
||||
<Clock className="h-4 w-4" />
|
||||
{t("cronjobs.cronJobs")}
|
||||
<span className="ml-1 text-xs bg-background0 px-2 py-0.5 ascii-border font-medium">
|
||||
<span className="ml-1 text-xs bg-primary/20 text-primary px-2 py-0.5 rounded-full font-medium">
|
||||
{cronJobs.length}
|
||||
</span>
|
||||
</button>
|
||||
<button
|
||||
onClick={() => setActiveTab("scripts")}
|
||||
className={`flex items-center gap-2 px-4 py-2 text-sm font-medium flex-1 justify-center terminal-font ${activeTab === "scripts"
|
||||
? "bg-background0 ascii-border"
|
||||
: "hover:ascii-border"
|
||||
}`}
|
||||
className={`flex items-center gap-2 px-4 py-2 text-sm font-medium transition-all duration-200 rounded-md flex-1 justify-center ${
|
||||
activeTab === "scripts"
|
||||
? "bg-primary/10 text-primary shadow-sm border border-primary/20"
|
||||
: "text-muted-foreground hover:text-foreground hover:bg-muted/50"
|
||||
}`}
|
||||
>
|
||||
<FileTextIcon className="h-4 w-4" />
|
||||
<FileText className="h-4 w-4" />
|
||||
{t("scripts.scripts")}
|
||||
<span className="ml-1 text-xs bg-background0 px-2 py-0.5 ascii-border font-medium">
|
||||
<span className="ml-1 text-xs bg-primary/20 text-primary px-2 py-0.5 rounded-full font-medium">
|
||||
{scripts.length}
|
||||
</span>
|
||||
</button>
|
||||
|
||||
@@ -12,7 +12,7 @@ import {
|
||||
CardDescription,
|
||||
CardContent,
|
||||
} from "@/app/_components/GlobalComponents/Cards/Card";
|
||||
import { LockIcon, EyeIcon, EyeSlashIcon, ShieldIcon, WarningIcon, CircleNotchIcon } from "@phosphor-icons/react";
|
||||
import { Lock, Eye, EyeOff, Shield, AlertTriangle, Loader2 } from "lucide-react";
|
||||
|
||||
interface LoginFormProps {
|
||||
hasPassword?: boolean;
|
||||
@@ -88,7 +88,7 @@ export const LoginForm = ({
|
||||
<Card className="w-full max-w-md shadow-xl">
|
||||
<CardContent className="pt-6">
|
||||
<div className="flex flex-col items-center justify-center py-8 space-y-4">
|
||||
<CircleNotchIcon className="w-12 h-12 text-primary animate-spin" />
|
||||
<Loader2 className="w-12 h-12 text-primary animate-spin" />
|
||||
<div className="text-center">
|
||||
<p className="text-lg font-medium">{t("login.redirectingToOIDC")}</p>
|
||||
<p className="text-sm text-muted-foreground mt-1">
|
||||
@@ -105,15 +105,15 @@ export const LoginForm = ({
|
||||
<Card className="w-full max-w-md shadow-xl">
|
||||
<CardHeader className="text-center">
|
||||
<div className="mx-auto w-16 h-16 bg-primary/10 rounded-full flex items-center justify-center mb-4">
|
||||
<LockIcon className="w-8 h-8 text-primary" />
|
||||
<Lock className="w-8 h-8 text-primary" />
|
||||
</div>
|
||||
<CardTitle>{t("login.welcomeTitle")}</CardTitle>
|
||||
<CardDescription>
|
||||
{hasPassword && hasOIDC
|
||||
? t("login.signInWithPasswordOrSSO")
|
||||
: hasOIDC
|
||||
? t("login.signInWithSSO")
|
||||
: t("login.enterPasswordToContinue")}
|
||||
? t("login.signInWithSSO")
|
||||
: t("login.enterPasswordToContinue")}
|
||||
</CardDescription>
|
||||
</CardHeader>
|
||||
|
||||
@@ -121,7 +121,7 @@ export const LoginForm = ({
|
||||
{!hasPassword && !hasOIDC && (
|
||||
<div className="mb-4 p-3 bg-amber-500/10 border border-amber-500/20 rounded-md">
|
||||
<div className="flex items-start space-x-2">
|
||||
<WarningIcon className="w-4 h-4 text-amber-500 mt-0.5 flex-shrink-0" />
|
||||
<AlertTriangle className="w-4 h-4 text-amber-500 mt-0.5 flex-shrink-0" />
|
||||
<div className="text-sm text-amber-700 dark:text-amber-400">
|
||||
<div className="font-medium">
|
||||
{t("login.authenticationNotConfigured")}
|
||||
@@ -152,9 +152,9 @@ export const LoginForm = ({
|
||||
disabled={isLoading}
|
||||
>
|
||||
{showPassword ? (
|
||||
<EyeSlashIcon className="w-4 h-4" />
|
||||
<EyeOff className="w-4 h-4" />
|
||||
) : (
|
||||
<EyeIcon className="w-4 h-4" />
|
||||
<Eye className="w-4 h-4" />
|
||||
)}
|
||||
</button>
|
||||
</div>
|
||||
@@ -175,7 +175,7 @@ export const LoginForm = ({
|
||||
<span className="w-full border-t" />
|
||||
</div>
|
||||
<div className="relative flex justify-center text-xs uppercase">
|
||||
<span className="bg-background0 px-2 text-muted-foreground">
|
||||
<span className="bg-background px-2 text-muted-foreground">
|
||||
{t("login.orContinueWith")}
|
||||
</span>
|
||||
</div>
|
||||
@@ -190,7 +190,7 @@ export const LoginForm = ({
|
||||
onClick={handleOIDCLogin}
|
||||
disabled={isLoading}
|
||||
>
|
||||
<ShieldIcon className="w-4 h-4 mr-2" />
|
||||
<Shield className="w-4 h-4 mr-2" />
|
||||
{isLoading ? t("login.redirecting") : t("login.signInWithSSO")}
|
||||
</Button>
|
||||
)}
|
||||
@@ -203,7 +203,7 @@ export const LoginForm = ({
|
||||
</div>
|
||||
|
||||
{version && (
|
||||
<div className="mt-6 pt-4 border-t border-border">
|
||||
<div className="mt-6 pt-4 border-t border-border/50">
|
||||
<div className="text-center text-xs text-muted-foreground">
|
||||
Cr*nMaster {t("common.version", { version })}
|
||||
</div>
|
||||
|
||||
@@ -3,7 +3,7 @@
|
||||
import { useState } from "react";
|
||||
import { useRouter } from "next/navigation";
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
import { SignOutIcon } from "@phosphor-icons/react";
|
||||
import { LogOut } from "lucide-react";
|
||||
|
||||
export const LogoutButton = () => {
|
||||
const [isLoading, setIsLoading] = useState(false);
|
||||
@@ -35,7 +35,7 @@ export const LogoutButton = () => {
|
||||
disabled={isLoading}
|
||||
title="Logout"
|
||||
>
|
||||
<SignOutIcon className="h-[1.2rem] w-[1.2rem]" />
|
||||
<LogOut className="h-[1.2rem] w-[1.2rem]" />
|
||||
<span className="sr-only">Logout</span>
|
||||
</Button>
|
||||
);
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
"use client";
|
||||
|
||||
import { useState } from "react";
|
||||
import { CopyIcon } from "@phosphor-icons/react";
|
||||
import { Copy } from "lucide-react";
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
import { Modal } from "@/app/_components/GlobalComponents/UIElements/Modal";
|
||||
import { Input } from "@/app/_components/GlobalComponents/FormElements/Input";
|
||||
@@ -89,7 +89,7 @@ export const CloneScriptModal = ({
|
||||
</>
|
||||
) : (
|
||||
<>
|
||||
<CopyIcon className="h-4 w-4 mr-2" />
|
||||
<Copy className="h-4 w-4 mr-2" />
|
||||
Clone Script
|
||||
</>
|
||||
)}
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
"use client";
|
||||
|
||||
import { useState } from "react";
|
||||
import { CopyIcon } from "@phosphor-icons/react";
|
||||
import { Copy } from "lucide-react";
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
import { Modal } from "@/app/_components/GlobalComponents/UIElements/Modal";
|
||||
import { Input } from "@/app/_components/GlobalComponents/FormElements/Input";
|
||||
@@ -89,7 +89,7 @@ export const CloneTaskModal = ({
|
||||
</>
|
||||
) : (
|
||||
<>
|
||||
<CopyIcon className="h-4 w-4 mr-2" />
|
||||
<Copy className="h-4 w-4 mr-2" />
|
||||
Clone Cron Job
|
||||
</>
|
||||
)}
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
"use client";
|
||||
|
||||
import { PlusIcon } from "@phosphor-icons/react";
|
||||
import { Plus } from "lucide-react";
|
||||
import { ScriptModal } from "@/app/_components/FeatureComponents/Modals/ScriptModal";
|
||||
|
||||
interface CreateScriptModalProps {
|
||||
@@ -35,7 +35,7 @@ export const CreateScriptModal = ({
|
||||
onSubmit={onSubmit}
|
||||
title="Create New Script"
|
||||
submitButtonText="Create Script"
|
||||
submitButtonIcon={<PlusIcon className="h-4 w-4 mr-2" />}
|
||||
submitButtonIcon={<Plus className="h-4 w-4 mr-2" />}
|
||||
form={form}
|
||||
onFormChange={onFormChange}
|
||||
isDraft={isDraft}
|
||||
|
||||
@@ -4,11 +4,10 @@ import { useState, useEffect } from "react";
|
||||
import { Modal } from "@/app/_components/GlobalComponents/UIElements/Modal";
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
import { Input } from "@/app/_components/GlobalComponents/FormElements/Input";
|
||||
import { Switch } from "@/app/_components/GlobalComponents/UIElements/Switch";
|
||||
import { CronExpressionHelper } from "@/app/_components/FeatureComponents/Scripts/CronExpressionHelper";
|
||||
import { SelectScriptModal } from "@/app/_components/FeatureComponents/Modals/SelectScriptModal";
|
||||
import { UserSwitcher } from "@/app/_components/FeatureComponents/User/UserSwitcher";
|
||||
import { PlusIcon, TerminalIcon, FileTextIcon, XIcon, FileArrowDownIcon } from "@phosphor-icons/react";
|
||||
import { Plus, Terminal, FileText, X, FileOutput } from "lucide-react";
|
||||
import { getScriptContent } from "@/app/_server/actions/scripts";
|
||||
import { getHostScriptPath } from "@/app/_server/actions/scripts";
|
||||
import { useTranslations } from "next-intl";
|
||||
@@ -101,7 +100,7 @@ export const CreateTaskModal = ({
|
||||
</label>
|
||||
<UserSwitcher
|
||||
selectedUser={form.user}
|
||||
onUserChange={(user: string) => onFormChange({ user })}
|
||||
onUserChange={(user) => onFormChange({ user })}
|
||||
/>
|
||||
</div>
|
||||
|
||||
@@ -125,13 +124,13 @@ export const CreateTaskModal = ({
|
||||
<button
|
||||
type="button"
|
||||
onClick={handleCustomCommand}
|
||||
className={`p-4 rounded-lg transition-all ${!form.selectedScriptId
|
||||
? "border-border border-2"
|
||||
: "border-border border"
|
||||
className={`p-4 rounded-lg border-2 transition-all ${!form.selectedScriptId
|
||||
? "border-primary bg-primary/5 text-primary"
|
||||
: "border-border bg-muted/30 text-muted-foreground hover:border-border/60"
|
||||
}`}
|
||||
>
|
||||
<div className="flex items-center gap-3">
|
||||
<TerminalIcon className="h-5 w-5" />
|
||||
<Terminal className="h-5 w-5" />
|
||||
<div className="text-left">
|
||||
<div className="font-medium">
|
||||
{t("cronjobs.customCommand")}
|
||||
@@ -146,13 +145,13 @@ export const CreateTaskModal = ({
|
||||
<button
|
||||
type="button"
|
||||
onClick={() => setIsSelectScriptModalOpen(true)}
|
||||
className={`p-4 rounded-lg transition-all ${form.selectedScriptId
|
||||
? "border-border border-2"
|
||||
: "border-border border"
|
||||
className={`p-4 rounded-lg border-2 transition-all ${form.selectedScriptId
|
||||
? "border-primary bg-primary/5 text-primary"
|
||||
: "border-border bg-muted/30 text-muted-foreground hover:border-border/60"
|
||||
}`}
|
||||
>
|
||||
<div className="flex items-center gap-3">
|
||||
<FileTextIcon className="h-5 w-5" />
|
||||
<FileText className="h-5 w-5" />
|
||||
<div className="text-left">
|
||||
<div className="font-medium">
|
||||
{t("scripts.savedScript")}
|
||||
@@ -171,7 +170,7 @@ export const CreateTaskModal = ({
|
||||
<div className="flex items-start justify-between">
|
||||
<div className="flex-1">
|
||||
<div className="flex items-center gap-2 mb-1">
|
||||
<FileTextIcon className="h-4 w-4 text-primary" />
|
||||
<FileText className="h-4 w-4 text-primary" />
|
||||
<h4 className="font-medium text-foreground">
|
||||
{selectedScript.name}
|
||||
</h4>
|
||||
@@ -179,7 +178,7 @@ export const CreateTaskModal = ({
|
||||
<p className="text-sm text-muted-foreground mb-2">
|
||||
{selectedScript.description}
|
||||
</p>
|
||||
<div className="bg-muted/30 p-2 rounded border border-border">
|
||||
<div className="bg-muted/30 p-2 rounded border border-border/30">
|
||||
<code className="text-xs font-mono text-foreground break-all">
|
||||
{form.command}
|
||||
</code>
|
||||
@@ -202,7 +201,7 @@ export const CreateTaskModal = ({
|
||||
onClick={handleClearScript}
|
||||
className="h-8 w-8 p-0 text-muted-foreground hover:text-foreground"
|
||||
>
|
||||
<XIcon className="h-4 w-4" />
|
||||
<X className="h-4 w-4" />
|
||||
</Button>
|
||||
</div>
|
||||
</div>
|
||||
@@ -223,12 +222,12 @@ export const CreateTaskModal = ({
|
||||
? "/app/scripts/script_name.sh"
|
||||
: "/usr/bin/command"
|
||||
}
|
||||
className="w-full h-24 p-2 border border-border rounded bg-background0 text-foreground font-mono text-sm resize-none focus:outline-none focus:ring-1 focus:ring-primary/20"
|
||||
className="w-full h-24 p-2 border border-border rounded bg-background text-foreground font-mono text-sm resize-none focus:outline-none focus:ring-1 focus:ring-primary/20"
|
||||
required
|
||||
readOnly={!!form.selectedScriptId}
|
||||
/>
|
||||
<div className="absolute right-3 top-2">
|
||||
<TerminalIcon className="h-4 w-4 text-muted-foreground" />
|
||||
<Terminal className="h-4 w-4 text-muted-foreground" />
|
||||
</div>
|
||||
</div>
|
||||
{form.selectedScriptId && (
|
||||
@@ -250,28 +249,29 @@ export const CreateTaskModal = ({
|
||||
value={form.comment}
|
||||
onChange={(e) => onFormChange({ comment: e.target.value })}
|
||||
placeholder={t("cronjobs.whatDoesThisTaskDo")}
|
||||
className="bg-muted/30 border-border focus:border-primary/50"
|
||||
className="bg-muted/30 border-border/50 focus:border-primary/50"
|
||||
/>
|
||||
</div>
|
||||
|
||||
|
||||
<div className="border border-border bg-muted/10 rounded-lg p-4">
|
||||
<div
|
||||
className="flex items-start gap-3 cursor-pointer"
|
||||
onClick={() => onFormChange({ logsEnabled: !form.logsEnabled })}
|
||||
>
|
||||
<Switch
|
||||
<div className="border border-border/30 bg-muted/10 rounded-lg p-4">
|
||||
<div className="flex items-start gap-3">
|
||||
<input
|
||||
type="checkbox"
|
||||
id="logsEnabled"
|
||||
checked={form.logsEnabled}
|
||||
onCheckedChange={(checked) =>
|
||||
onFormChange({ logsEnabled: checked })
|
||||
onChange={(e) =>
|
||||
onFormChange({ logsEnabled: e.target.checked })
|
||||
}
|
||||
className="mt-1"
|
||||
className="mt-1 h-4 w-4 rounded border-border text-primary focus:ring-primary/20 cursor-pointer"
|
||||
/>
|
||||
<div className="flex-1">
|
||||
<div className="flex items-center gap-2 text-sm font-medium text-foreground">
|
||||
<FileArrowDownIcon className="h-4 w-4 text-primary" />
|
||||
<label
|
||||
htmlFor="logsEnabled"
|
||||
className="flex items-center gap-2 text-sm font-medium text-foreground cursor-pointer"
|
||||
>
|
||||
<FileOutput className="h-4 w-4 text-primary" />
|
||||
{t("cronjobs.enableLogging")}
|
||||
</div>
|
||||
</label>
|
||||
<p className="text-xs text-muted-foreground mt-1">
|
||||
{t("cronjobs.loggingDescription")}
|
||||
</p>
|
||||
@@ -279,8 +279,7 @@ export const CreateTaskModal = ({
|
||||
</div>
|
||||
</div>
|
||||
|
||||
|
||||
<div className="flex justify-end gap-2 pt-3 border-t border-border">
|
||||
<div className="flex justify-end gap-2 pt-3 border-t border-border/50">
|
||||
<Button
|
||||
type="button"
|
||||
variant="outline"
|
||||
@@ -290,7 +289,7 @@ export const CreateTaskModal = ({
|
||||
{t("common.cancel")}
|
||||
</Button>
|
||||
<Button type="submit" className="btn-primary glow-primary">
|
||||
<PlusIcon className="h-4 w-4 mr-2" />
|
||||
<Plus className="h-4 w-4 mr-2" />
|
||||
{t("cronjobs.createTask")}
|
||||
</Button>
|
||||
</div>
|
||||
|
||||
@@ -2,7 +2,7 @@
|
||||
|
||||
import { Modal } from "@/app/_components/GlobalComponents/UIElements/Modal";
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
import { FileTextIcon, WarningCircleIcon, TrashIcon } from "@phosphor-icons/react";
|
||||
import { FileText, AlertCircle, Trash2 } from "lucide-react";
|
||||
import { Script } from "@/app/_utils/scripts-utils";
|
||||
|
||||
interface DeleteScriptModalProps {
|
||||
@@ -25,10 +25,10 @@ export const DeleteScriptModal = ({
|
||||
return (
|
||||
<Modal isOpen={isOpen} onClose={onClose} title="Delete Script" size="sm">
|
||||
<div className="space-y-3">
|
||||
<div className="bg-muted/30 rounded p-2 border border-border">
|
||||
<div className="bg-muted/30 rounded p-2 border border-border/50">
|
||||
<div className="space-y-1">
|
||||
<div className="flex items-center gap-2">
|
||||
<FileTextIcon className="h-3 w-3 text-muted-foreground" />
|
||||
<FileText className="h-3 w-3 text-muted-foreground" />
|
||||
<span className="text-xs font-medium text-foreground">
|
||||
{script.name}
|
||||
</span>
|
||||
@@ -36,7 +36,7 @@ export const DeleteScriptModal = ({
|
||||
|
||||
{script.description && (
|
||||
<div className="flex items-start gap-2">
|
||||
<FileTextIcon className="h-3 w-3 text-muted-foreground mt-0.5 flex-shrink-0" />
|
||||
<FileText className="h-3 w-3 text-muted-foreground mt-0.5 flex-shrink-0" />
|
||||
<p className="text-xs text-muted-foreground break-words italic">
|
||||
{script.description}
|
||||
</p>
|
||||
@@ -44,8 +44,8 @@ export const DeleteScriptModal = ({
|
||||
)}
|
||||
|
||||
<div className="flex items-start gap-2">
|
||||
<FileTextIcon className="h-3 w-3 text-muted-foreground mt-0.5 flex-shrink-0" />
|
||||
<code className="text-xs font-mono bg-muted/30 px-1 py-0.5 rounded border border-border">
|
||||
<FileText className="h-3 w-3 text-muted-foreground mt-0.5 flex-shrink-0" />
|
||||
<code className="text-xs font-mono bg-muted/30 px-1 py-0.5 rounded border border-border/30">
|
||||
{script.filename}
|
||||
</code>
|
||||
</div>
|
||||
@@ -54,7 +54,7 @@ export const DeleteScriptModal = ({
|
||||
|
||||
<div className="bg-destructive/5 border border-destructive/20 rounded p-2">
|
||||
<div className="flex items-start gap-2">
|
||||
<WarningCircleIcon className="h-4 w-4 text-destructive mt-0.5 flex-shrink-0" />
|
||||
<AlertCircle className="h-4 w-4 text-destructive mt-0.5 flex-shrink-0" />
|
||||
<div>
|
||||
<p className="text-xs font-medium text-destructive mb-0.5">
|
||||
This action cannot be undone
|
||||
@@ -66,7 +66,7 @@ export const DeleteScriptModal = ({
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div className="flex justify-end gap-2 pt-2 border-t border-border">
|
||||
<div className="flex justify-end gap-2 pt-2 border-t border-border/50">
|
||||
<Button
|
||||
variant="outline"
|
||||
onClick={onClose}
|
||||
@@ -78,6 +78,7 @@ export const DeleteScriptModal = ({
|
||||
<Button
|
||||
variant="destructive"
|
||||
onClick={onConfirm}
|
||||
className="btn-destructive"
|
||||
disabled={isDeleting}
|
||||
>
|
||||
{isDeleting ? (
|
||||
@@ -87,7 +88,7 @@ export const DeleteScriptModal = ({
|
||||
</>
|
||||
) : (
|
||||
<>
|
||||
<TrashIcon className="h-4 w-4 mr-2" />
|
||||
<Trash2 className="h-4 w-4 mr-2" />
|
||||
Delete Script
|
||||
</>
|
||||
)}
|
||||
|
||||
@@ -4,11 +4,11 @@ import { Modal } from "@/app/_components/GlobalComponents/UIElements/Modal";
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
import {
|
||||
Calendar,
|
||||
TerminalIcon,
|
||||
ChatTextIcon,
|
||||
WarningCircleIcon,
|
||||
TrashIcon,
|
||||
} from "@phosphor-icons/react";
|
||||
Terminal,
|
||||
MessageSquare,
|
||||
AlertCircle,
|
||||
Trash2,
|
||||
} from "lucide-react";
|
||||
import { CronJob } from "@/app/_utils/cronjob-utils";
|
||||
|
||||
interface DeleteTaskModalProps {
|
||||
@@ -34,7 +34,7 @@ export const DeleteTaskModal = ({
|
||||
size="sm"
|
||||
>
|
||||
<div className="space-y-3">
|
||||
<div className="bg-muted/30 rounded p-2 border border-border">
|
||||
<div className="bg-muted/30 rounded p-2 border border-border/50">
|
||||
<div className="space-y-1">
|
||||
<div className="flex items-center gap-2">
|
||||
<Calendar className="h-3 w-3 text-muted-foreground" />
|
||||
@@ -44,15 +44,15 @@ export const DeleteTaskModal = ({
|
||||
</div>
|
||||
|
||||
<div className="flex items-start gap-2">
|
||||
<TerminalIcon className="h-3 w-3 text-muted-foreground mt-0.5 flex-shrink-0" />
|
||||
<pre className="max-w-full overflow-x-auto text-xs font-medium text-foreground break-words bg-muted/30 px-1 py-0.5 rounded border border-border flex-1 hide-scrollbar">
|
||||
<Terminal className="h-3 w-3 text-muted-foreground mt-0.5 flex-shrink-0" />
|
||||
<pre className="max-w-full overflow-x-auto text-xs font-medium text-foreground break-words bg-muted/30 px-1 py-0.5 rounded border border-border/30 flex-1 hide-scrollbar">
|
||||
{job.command}
|
||||
</pre>
|
||||
</div>
|
||||
|
||||
{job.comment && (
|
||||
<div className="flex items-start gap-2">
|
||||
<ChatTextIcon className="h-3 w-3 text-muted-foreground mt-0.5 flex-shrink-0" />
|
||||
<MessageSquare className="h-3 w-3 text-muted-foreground mt-0.5 flex-shrink-0" />
|
||||
<p className="text-xs text-muted-foreground break-words italic">
|
||||
{job.comment}
|
||||
</p>
|
||||
@@ -63,7 +63,7 @@ export const DeleteTaskModal = ({
|
||||
|
||||
<div className="bg-destructive/5 border border-destructive/20 rounded p-2">
|
||||
<div className="flex items-start gap-2">
|
||||
<WarningCircleIcon className="h-4 w-4 text-destructive mt-0.5 flex-shrink-0" />
|
||||
<AlertCircle className="h-4 w-4 text-destructive mt-0.5 flex-shrink-0" />
|
||||
<div>
|
||||
<p className="text-xs font-medium text-destructive mb-0.5">
|
||||
This action cannot be undone
|
||||
@@ -75,15 +75,16 @@ export const DeleteTaskModal = ({
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div className="flex justify-end gap-2 pt-2 border-t border-border">
|
||||
<div className="flex justify-end gap-2 pt-2 border-t border-border/50">
|
||||
<Button variant="outline" onClick={onClose} className="btn-outline">
|
||||
Cancel
|
||||
</Button>
|
||||
<Button
|
||||
variant="destructive"
|
||||
onClick={onConfirm}
|
||||
className="btn-destructive"
|
||||
>
|
||||
<TrashIcon className="h-4 w-4 mr-2" />
|
||||
<Trash2 className="h-4 w-4 mr-2" />
|
||||
Delete Task
|
||||
</Button>
|
||||
</div>
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
"use client";
|
||||
|
||||
import { PencilSimpleIcon } from "@phosphor-icons/react";
|
||||
import { Edit } from "lucide-react";
|
||||
import { Script } from "@/app/_utils/scripts-utils";
|
||||
import { ScriptModal } from "@/app/_components/FeatureComponents/Modals/ScriptModal";
|
||||
|
||||
@@ -36,7 +36,7 @@ export const EditScriptModal = ({
|
||||
onSubmit={onSubmit}
|
||||
title="Edit Script"
|
||||
submitButtonText="Update Script"
|
||||
submitButtonIcon={<PencilSimpleIcon className="h-4 w-4 mr-2" />}
|
||||
submitButtonIcon={<Edit className="h-4 w-4 mr-2" />}
|
||||
form={form}
|
||||
onFormChange={onFormChange}
|
||||
additionalFormData={{ id: script.id }}
|
||||
|
||||
@@ -3,9 +3,8 @@
|
||||
import { Modal } from "@/app/_components/GlobalComponents/UIElements/Modal";
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
import { Input } from "@/app/_components/GlobalComponents/FormElements/Input";
|
||||
import { Switch } from "@/app/_components/GlobalComponents/UIElements/Switch";
|
||||
import { CronExpressionHelper } from "@/app/_components/FeatureComponents/Scripts/CronExpressionHelper";
|
||||
import { PencilSimpleIcon, TerminalIcon, FileArrowDownIcon } from "@phosphor-icons/react";
|
||||
import { Edit, Terminal, FileOutput } from "lucide-react";
|
||||
import { useTranslations } from "next-intl";
|
||||
|
||||
interface EditTaskModalProps {
|
||||
@@ -60,11 +59,11 @@ export const EditTaskModal = ({
|
||||
value={form.command}
|
||||
onChange={(e) => onFormChange({ command: e.target.value })}
|
||||
placeholder="/usr/bin/command"
|
||||
className="font-mono bg-muted/30 border-border focus:border-primary/50"
|
||||
className="font-mono bg-muted/30 border-border/50 focus:border-primary/50"
|
||||
required
|
||||
/>
|
||||
<div className="absolute right-3 top-1/2 -translate-y-1/2">
|
||||
<TerminalIcon className="h-4 w-4 text-muted-foreground" />
|
||||
<Terminal className="h-4 w-4 text-muted-foreground" />
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
@@ -81,26 +80,27 @@ export const EditTaskModal = ({
|
||||
value={form.comment}
|
||||
onChange={(e) => onFormChange({ comment: e.target.value })}
|
||||
placeholder={t("cronjobs.whatDoesThisTaskDo")}
|
||||
className="bg-muted/30 border-border focus:border-primary/50"
|
||||
className="bg-muted/30 border-border/50 focus:border-primary/50"
|
||||
/>
|
||||
</div>
|
||||
|
||||
|
||||
<div className="border border-border bg-muted/10 rounded-lg p-4">
|
||||
<div
|
||||
className="flex items-start gap-3 cursor-pointer"
|
||||
onClick={() => onFormChange({ logsEnabled: !form.logsEnabled })}
|
||||
>
|
||||
<Switch
|
||||
<div className="border border-border/30 bg-muted/10 rounded-lg p-4">
|
||||
<div className="flex items-start gap-3">
|
||||
<input
|
||||
type="checkbox"
|
||||
id="logsEnabled"
|
||||
checked={form.logsEnabled}
|
||||
onCheckedChange={(checked) => onFormChange({ logsEnabled: checked })}
|
||||
className="mt-1"
|
||||
onChange={(e) => onFormChange({ logsEnabled: e.target.checked })}
|
||||
className="mt-1 h-4 w-4 rounded border-border text-primary focus:ring-primary/20 cursor-pointer"
|
||||
/>
|
||||
<div className="flex-1">
|
||||
<div className="flex items-center gap-2 text-sm font-medium text-foreground">
|
||||
<FileArrowDownIcon className="h-4 w-4 text-primary" />
|
||||
<label
|
||||
htmlFor="logsEnabled"
|
||||
className="flex items-center gap-2 text-sm font-medium text-foreground cursor-pointer"
|
||||
>
|
||||
<FileOutput className="h-4 w-4 text-primary" />
|
||||
{t("cronjobs.enableLogging")}
|
||||
</div>
|
||||
</label>
|
||||
<p className="text-xs text-muted-foreground mt-1">
|
||||
{t("cronjobs.loggingDescription")}
|
||||
</p>
|
||||
@@ -108,8 +108,7 @@ export const EditTaskModal = ({
|
||||
</div>
|
||||
</div>
|
||||
|
||||
|
||||
<div className="flex justify-end gap-2 pt-3 border-t border-border">
|
||||
<div className="flex justify-end gap-2 pt-3 border-t border-border/50">
|
||||
<Button
|
||||
type="button"
|
||||
variant="outline"
|
||||
@@ -119,7 +118,7 @@ export const EditTaskModal = ({
|
||||
Cancel
|
||||
</Button>
|
||||
<Button type="submit" className="btn-primary glow-primary">
|
||||
<PencilSimpleIcon className="h-4 w-4 mr-2" />
|
||||
<Edit className="h-4 w-4 mr-2" />
|
||||
Update Task
|
||||
</Button>
|
||||
</div>
|
||||
|
||||
@@ -2,7 +2,7 @@
|
||||
|
||||
import { Modal } from "@/app/_components/GlobalComponents/UIElements/Modal";
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
import { WarningCircleIcon, CopyIcon, XIcon } from "@phosphor-icons/react";
|
||||
import { AlertCircle, Copy, X } from "lucide-react";
|
||||
import { showToast } from "@/app/_components/GlobalComponents/UIElements/Toast";
|
||||
|
||||
interface ErrorDetails {
|
||||
@@ -54,7 +54,7 @@ Timestamp: ${error.timestamp}
|
||||
<div className="space-y-4">
|
||||
<div className="bg-destructive/5 border border-destructive/20 rounded-lg p-4">
|
||||
<div className="flex items-start gap-3">
|
||||
<WarningCircleIcon className="h-5 w-5 text-destructive mt-0.5 flex-shrink-0" />
|
||||
<AlertCircle className="h-5 w-5 text-destructive mt-0.5 flex-shrink-0" />
|
||||
<div className="flex-1">
|
||||
<h3 className="font-medium text-destructive mb-1">
|
||||
{error.title}
|
||||
@@ -69,7 +69,7 @@ Timestamp: ${error.timestamp}
|
||||
<h4 className="text-sm font-medium text-foreground mb-2">
|
||||
Details
|
||||
</h4>
|
||||
<div className="bg-muted/30 p-3 rounded border border-border">
|
||||
<div className="bg-muted/30 p-3 rounded border border-border/30">
|
||||
<pre className="text-sm font-mono text-foreground whitespace-pre-wrap break-words">
|
||||
{error.details}
|
||||
</pre>
|
||||
@@ -82,7 +82,7 @@ Timestamp: ${error.timestamp}
|
||||
<h4 className="text-sm font-medium text-foreground mb-2">
|
||||
Command
|
||||
</h4>
|
||||
<div className="bg-muted/30 p-3 rounded border border-border">
|
||||
<div className="bg-muted/30 p-3 rounded border border-border/30">
|
||||
<code className="text-sm font-mono text-foreground break-all">
|
||||
{error.command}
|
||||
</code>
|
||||
@@ -93,7 +93,7 @@ Timestamp: ${error.timestamp}
|
||||
{error.output && (
|
||||
<div>
|
||||
<h4 className="text-sm font-medium text-foreground mb-2">Output</h4>
|
||||
<div className="bg-muted/30 p-3 rounded border border-border max-h-32 overflow-y-auto tui-scrollbar">
|
||||
<div className="bg-muted/30 p-3 rounded border border-border/30 max-h-32 overflow-y-auto">
|
||||
<pre className="text-sm font-mono text-foreground whitespace-pre-wrap">
|
||||
{error.output}
|
||||
</pre>
|
||||
@@ -106,7 +106,7 @@ Timestamp: ${error.timestamp}
|
||||
<h4 className="text-sm font-medium text-foreground mb-2">
|
||||
Error Output
|
||||
</h4>
|
||||
<div className="bg-destructive/5 p-3 rounded border border-destructive/20 max-h-32 overflow-y-auto tui-scrollbar">
|
||||
<div className="bg-destructive/5 p-3 rounded border border-destructive/20 max-h-32 overflow-y-auto">
|
||||
<pre className="text-sm font-mono text-destructive whitespace-pre-wrap">
|
||||
{error.stderr}
|
||||
</pre>
|
||||
@@ -118,14 +118,14 @@ Timestamp: ${error.timestamp}
|
||||
Timestamp: {error.timestamp}
|
||||
</div>
|
||||
|
||||
<div className="flex justify-end gap-2 pt-3 border-t border-border">
|
||||
<div className="flex justify-end gap-2 pt-3 border-t border-border/50">
|
||||
<Button
|
||||
variant="outline"
|
||||
onClick={handleCopyDetails}
|
||||
className="btn-outline"
|
||||
>
|
||||
<CopyIcon className="h-4 w-4 mr-2" />
|
||||
CopyIcon Details
|
||||
<Copy className="h-4 w-4 mr-2" />
|
||||
Copy Details
|
||||
</Button>
|
||||
<Button onClick={onClose} className="btn-primary">
|
||||
Close
|
||||
|
||||
@@ -3,7 +3,7 @@
|
||||
import { useState, useEffect } from "react";
|
||||
import { Modal } from "@/app/_components/GlobalComponents/UIElements/Modal";
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
import { CaretDownIcon, CodeIcon, ChatTextIcon } from "@phosphor-icons/react";
|
||||
import { ChevronDown, Code, MessageSquare } from "lucide-react";
|
||||
import { UserFilter } from "@/app/_components/FeatureComponents/User/UserFilter";
|
||||
import { useTranslations } from "next-intl";
|
||||
|
||||
@@ -72,15 +72,15 @@ export const FiltersModal = ({
|
||||
>
|
||||
<div className="flex items-center">
|
||||
{localScheduleMode === "cron" && (
|
||||
<CodeIcon className="h-4 w-4 mr-2" />
|
||||
<Code className="h-4 w-4 mr-2" />
|
||||
)}
|
||||
{localScheduleMode === "human" && (
|
||||
<ChatTextIcon className="h-4 w-4 mr-2" />
|
||||
<MessageSquare className="h-4 w-4 mr-2" />
|
||||
)}
|
||||
{localScheduleMode === "both" && (
|
||||
<>
|
||||
<CodeIcon className="h-4 w-4 mr-1" />
|
||||
<ChatTextIcon className="h-4 w-4 mr-2" />
|
||||
<Code className="h-4 w-4 mr-1" />
|
||||
<MessageSquare className="h-4 w-4 mr-2" />
|
||||
</>
|
||||
)}
|
||||
<span>
|
||||
@@ -90,22 +90,23 @@ export const FiltersModal = ({
|
||||
{localScheduleMode === "both" && t("cronjobs.both")}
|
||||
</span>
|
||||
</div>
|
||||
<CaretDownIcon className="h-4 w-4 ml-2" />
|
||||
<ChevronDown className="h-4 w-4 ml-2" />
|
||||
</Button>
|
||||
|
||||
{isScheduleDropdownOpen && (
|
||||
<div className="absolute top-full left-0 right-0 mt-1 bg-background0 border border-border rounded-md shadow-lg z-50 min-w-[140px]">
|
||||
<div className="absolute top-full left-0 right-0 mt-1 bg-background border border-border rounded-md shadow-lg z-50 min-w-[140px]">
|
||||
<button
|
||||
onClick={() => {
|
||||
setLocalScheduleMode("cron");
|
||||
setIsScheduleDropdownOpen(false);
|
||||
}}
|
||||
className={`w-full text-left px-3 py-2 text-sm hover:bg-accent transition-colors flex items-center gap-2 ${localScheduleMode === "cron"
|
||||
? "bg-accent text-accent-foreground"
|
||||
: ""
|
||||
}`}
|
||||
className={`w-full text-left px-3 py-2 text-sm hover:bg-accent transition-colors flex items-center gap-2 ${
|
||||
localScheduleMode === "cron"
|
||||
? "bg-accent text-accent-foreground"
|
||||
: ""
|
||||
}`}
|
||||
>
|
||||
<CodeIcon className="h-3 w-3" />
|
||||
<Code className="h-3 w-3" />
|
||||
{t("cronjobs.cronSyntax")}
|
||||
</button>
|
||||
<button
|
||||
@@ -113,12 +114,13 @@ export const FiltersModal = ({
|
||||
setLocalScheduleMode("human");
|
||||
setIsScheduleDropdownOpen(false);
|
||||
}}
|
||||
className={`w-full text-left px-3 py-2 text-sm hover:bg-accent transition-colors flex items-center gap-2 ${localScheduleMode === "human"
|
||||
? "bg-accent text-accent-foreground"
|
||||
: ""
|
||||
}`}
|
||||
className={`w-full text-left px-3 py-2 text-sm hover:bg-accent transition-colors flex items-center gap-2 ${
|
||||
localScheduleMode === "human"
|
||||
? "bg-accent text-accent-foreground"
|
||||
: ""
|
||||
}`}
|
||||
>
|
||||
<ChatTextIcon className="h-3 w-3" />
|
||||
<MessageSquare className="h-3 w-3" />
|
||||
{t("cronjobs.humanReadable")}
|
||||
</button>
|
||||
<button
|
||||
@@ -126,13 +128,14 @@ export const FiltersModal = ({
|
||||
setLocalScheduleMode("both");
|
||||
setIsScheduleDropdownOpen(false);
|
||||
}}
|
||||
className={`w-full text-left px-3 py-2 text-sm hover:bg-accent transition-colors flex items-center gap-2 ${localScheduleMode === "both"
|
||||
? "bg-accent text-accent-foreground"
|
||||
: ""
|
||||
}`}
|
||||
className={`w-full text-left px-3 py-2 text-sm hover:bg-accent transition-colors flex items-center gap-2 ${
|
||||
localScheduleMode === "both"
|
||||
? "bg-accent text-accent-foreground"
|
||||
: ""
|
||||
}`}
|
||||
>
|
||||
<CodeIcon className="h-3 w-3" />
|
||||
<ChatTextIcon className="h-3 w-3" />
|
||||
<Code className="h-3 w-3" />
|
||||
<MessageSquare className="h-3 w-3" />
|
||||
{t("cronjobs.both")}
|
||||
</button>
|
||||
</div>
|
||||
@@ -141,7 +144,7 @@ export const FiltersModal = ({
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div className="flex justify-end gap-2 pt-4">
|
||||
<div className="flex justify-end gap-2 pt-4 border-t border-border">
|
||||
<Button variant="outline" onClick={onClose}>
|
||||
{t("common.cancel")}
|
||||
</Button>
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
"use client";
|
||||
|
||||
import { useState, useEffect, useRef, useCallback } from "react";
|
||||
import { CircleNotchIcon, CheckCircleIcon, XCircleIcon, WarningIcon, ArrowsInIcon, ArrowsOutIcon } from "@phosphor-icons/react";
|
||||
import { Loader2, CheckCircle2, XCircle, AlertTriangle, Minimize2, Maximize2 } from "lucide-react";
|
||||
import { Modal } from "@/app/_components/GlobalComponents/UIElements/Modal";
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
import { useSSEContext } from "@/app/_contexts/SSEContext";
|
||||
@@ -228,20 +228,20 @@ export const LiveLogModal = ({
|
||||
<div className="flex items-center gap-3">
|
||||
<span>{t("cronjobs.liveJobExecution")}{jobComment && `: ${jobComment}`}</span>
|
||||
{status === "running" && (
|
||||
<span className="flex items-center gap-1 text-sm text-status-info">
|
||||
<CircleNotchIcon className="w-4 h-4 animate-spin" />
|
||||
<span className="flex items-center gap-1 text-sm text-blue-500">
|
||||
<Loader2 className="w-4 h-4 animate-spin" />
|
||||
{t("cronjobs.running")}
|
||||
</span>
|
||||
)}
|
||||
{status === "completed" && (
|
||||
<span className="flex items-center gap-1 text-sm text-status-success">
|
||||
<CheckCircleIcon className="w-4 h-4" />
|
||||
<span className="flex items-center gap-1 text-sm text-green-500">
|
||||
<CheckCircle2 className="w-4 h-4" />
|
||||
{t("cronjobs.completed", { exitCode: exitCode ?? 0 })}
|
||||
</span>
|
||||
)}
|
||||
{status === "failed" && (
|
||||
<span className="flex items-center gap-1 text-sm text-status-error">
|
||||
<XCircleIcon className="w-4 h-4" />
|
||||
<span className="flex items-center gap-1 text-sm text-red-500">
|
||||
<XCircle className="w-4 h-4" />
|
||||
{t("cronjobs.jobFailed", { exitCode: exitCode ?? 1 })}
|
||||
</span>
|
||||
)}
|
||||
@@ -268,7 +268,7 @@ export const LiveLogModal = ({
|
||||
id="maxLines"
|
||||
value={maxLines}
|
||||
onChange={(e) => setMaxLines(parseInt(e.target.value, 10))}
|
||||
className="bg-background0 border border-border rounded px-2 py-1 text-sm"
|
||||
className="bg-background border border-border rounded px-2 py-1 text-sm"
|
||||
>
|
||||
<option value="100">{t("cronjobs.nLines", { count: "100" })}</option>
|
||||
<option value="500">{t("cronjobs.nLines", { count: "500" })}</option>
|
||||
@@ -316,8 +316,8 @@ export const LiveLogModal = ({
|
||||
)}
|
||||
</div>
|
||||
{truncated && !showFullLog && (
|
||||
<div className="text-sm text-status-warning flex items-center gap-1 terminal-font">
|
||||
<WarningIcon className="h-4 w-4" />
|
||||
<div className="text-sm text-orange-500 flex items-center gap-1">
|
||||
<AlertTriangle className="h-4 w-4" />
|
||||
{t("cronjobs.showingLastOf", {
|
||||
lineCount: lineCount.toLocaleString(),
|
||||
totalLines: totalLines.toLocaleString()
|
||||
@@ -327,8 +327,8 @@ export const LiveLogModal = ({
|
||||
</div>
|
||||
|
||||
{showSizeWarning && (
|
||||
<div className="bg-background2 ascii-border p-3 flex items-start gap-3 terminal-font">
|
||||
<WarningIcon className="h-4 w-4 text-status-warning mt-0.5 flex-shrink-0" />
|
||||
<div className="bg-orange-500/10 border border-orange-500/30 rounded-lg p-3 flex items-start gap-3">
|
||||
<AlertTriangle className="h-4 w-4 text-orange-500 mt-0.5 flex-shrink-0" />
|
||||
<div className="flex-1 min-w-0">
|
||||
<p className="text-sm text-foreground">
|
||||
<span className="font-medium">{t("cronjobs.largeLogFileDetected")}</span> ({formatFileSize(fileSize)})
|
||||
@@ -340,16 +340,16 @@ export const LiveLogModal = ({
|
||||
variant="ghost"
|
||||
size="sm"
|
||||
onClick={toggleTailMode}
|
||||
className="text-status-warning hover:text-status-warning hover:bg-background2 h-auto py-1 px-2 text-xs"
|
||||
className="text-orange-500 hover:text-orange-400 hover:bg-orange-500/10 h-auto py-1 px-2 text-xs"
|
||||
title={tailMode ? t("cronjobs.showAllLines") : t("cronjobs.enableTailMode")}
|
||||
>
|
||||
{tailMode ? <ArrowsOutIcon className="h-3 w-3" /> : <ArrowsInIcon className="h-3 w-3" />}
|
||||
{tailMode ? <Maximize2 className="h-3 w-3" /> : <Minimize2 className="h-3 w-3" />}
|
||||
</Button>
|
||||
</div>
|
||||
)}
|
||||
|
||||
<div className="bg-black/90 dark:bg-black/60 p-4 max-h-[60vh] overflow-auto terminal-font ascii-border">
|
||||
<pre className="text-xs font-mono text-status-success whitespace-pre-wrap break-words">
|
||||
<div className="bg-black/90 dark:bg-black/60 rounded-lg p-4 max-h-[60vh] overflow-auto">
|
||||
<pre className="text-xs font-mono text-green-400 whitespace-pre-wrap break-words">
|
||||
{logContent || t("cronjobs.waitingForJobToStart")}
|
||||
<div ref={logEndRef} />
|
||||
</pre>
|
||||
|
||||
@@ -3,7 +3,7 @@
|
||||
import { useState, useEffect } from "react";
|
||||
import { Modal } from "@/app/_components/GlobalComponents/UIElements/Modal";
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
import { FileTextIcon, TrashIcon, EyeIcon, XIcon, ArrowsClockwiseIcon, WarningCircleIcon, CheckCircleIcon } from "@phosphor-icons/react";
|
||||
import { FileText, Trash2, Eye, X, RefreshCw, AlertCircle, CheckCircle } from "lucide-react";
|
||||
import { useTranslations } from "next-intl";
|
||||
import {
|
||||
getJobLogs,
|
||||
@@ -173,7 +173,7 @@ export const LogsModal = ({
|
||||
className="btn-primary glow-primary"
|
||||
size="sm"
|
||||
>
|
||||
<ArrowsClockwiseIcon
|
||||
<RefreshCw
|
||||
className={`w-4 h-4 mr-2 ${isLoadingLogs ? "animate-spin" : ""
|
||||
}`}
|
||||
/>
|
||||
@@ -182,10 +182,10 @@ export const LogsModal = ({
|
||||
{logs.length > 0 && (
|
||||
<Button
|
||||
onClick={handleDeleteAllLogs}
|
||||
variant="destructive"
|
||||
className="btn-destructive glow-primary"
|
||||
size="sm"
|
||||
>
|
||||
<TrashIcon className="w-4 h-4 mr-2" />
|
||||
<Trash2 className="w-4 h-4 mr-2" />
|
||||
{t("cronjobs.deleteAll")}
|
||||
</Button>
|
||||
)}
|
||||
@@ -208,11 +208,11 @@ export const LogsModal = ({
|
||||
logs.map((log) => (
|
||||
<div
|
||||
key={log.filename}
|
||||
className={`p-3 ascii-border cursor-pointer transition-colors terminal-font ${selectedLog === log.filename
|
||||
? "border-primary bg-background2"
|
||||
className={`p-3 rounded border cursor-pointer transition-colors ${selectedLog === log.filename
|
||||
? "border-primary bg-primary/10"
|
||||
: log.hasError
|
||||
? "border-red-600 hover:border-red-600"
|
||||
: "ascii-border hover:border-primary"
|
||||
? "border-red-500/50 hover:border-red-500"
|
||||
: "border-border hover:border-primary/50"
|
||||
}`}
|
||||
onClick={() => handleViewLog(log.filename)}
|
||||
>
|
||||
@@ -220,11 +220,11 @@ export const LogsModal = ({
|
||||
<div className="flex-1 min-w-0">
|
||||
<div className="flex items-center gap-2 mb-1">
|
||||
{log.hasError ? (
|
||||
<WarningCircleIcon className="w-4 h-4 flex-shrink-0 text-status-error" />
|
||||
<AlertCircle className="w-4 h-4 flex-shrink-0 text-red-500" />
|
||||
) : log.exitCode === 0 ? (
|
||||
<CheckCircleIcon className="w-4 h-4 flex-shrink-0 text-status-success" />
|
||||
<CheckCircle className="w-4 h-4 flex-shrink-0 text-green-500" />
|
||||
) : (
|
||||
<FileTextIcon className="w-4 h-4 flex-shrink-0" />
|
||||
<FileText className="w-4 h-4 flex-shrink-0" />
|
||||
)}
|
||||
<span className="text-sm font-medium truncate">
|
||||
{formatTimestamp(log.timestamp)}
|
||||
@@ -236,9 +236,9 @@ export const LogsModal = ({
|
||||
</p>
|
||||
{log.exitCode !== undefined && (
|
||||
<span
|
||||
className={`text-xs px-1.5 py-0.5 ${log.hasError
|
||||
? "bg-background2 text-status-error"
|
||||
: "bg-background2 text-status-success"
|
||||
className={`text-xs px-1.5 py-0.5 rounded ${log.hasError
|
||||
? "bg-red-500/10 text-red-600 dark:text-red-400"
|
||||
: "bg-green-500/10 text-green-600 dark:text-green-400"
|
||||
}`}
|
||||
>
|
||||
Exit: {log.exitCode}
|
||||
@@ -251,10 +251,10 @@ export const LogsModal = ({
|
||||
e.stopPropagation();
|
||||
handleDeleteLog(log.filename);
|
||||
}}
|
||||
variant="destructive"
|
||||
className="btn-destructive glow-primary p-1 h-auto"
|
||||
size="sm"
|
||||
>
|
||||
<TrashIcon className="w-3 h-3" />
|
||||
<Trash2 className="w-3 h-3" />
|
||||
</Button>
|
||||
</div>
|
||||
</div>
|
||||
@@ -271,13 +271,13 @@ export const LogsModal = ({
|
||||
{t("common.loading")}...
|
||||
</div>
|
||||
) : selectedLog ? (
|
||||
<pre className="h-full overflow-auto bg-background0 tui-scrollbar p-4 ascii-border text-xs font-mono whitespace-pre-wrap terminal-font">
|
||||
<pre className="h-full overflow-auto bg-muted/50 p-4 rounded border border-border text-xs font-mono whitespace-pre-wrap">
|
||||
{logContent}
|
||||
</pre>
|
||||
) : (
|
||||
<div className="h-full flex items-center justify-center text-muted-foreground">
|
||||
<div className="text-center">
|
||||
<EyeIcon className="w-12 h-12 mx-auto mb-2 opacity-50" />
|
||||
<Eye className="w-12 h-12 mx-auto mb-2 opacity-50" />
|
||||
<p>{t("cronjobs.selectLogToView")}</p>
|
||||
</div>
|
||||
</div>
|
||||
@@ -288,7 +288,7 @@ export const LogsModal = ({
|
||||
|
||||
<div className="mt-4 pt-4 border-t border-border flex justify-end">
|
||||
<Button onClick={onClose} className="btn-primary glow-primary">
|
||||
<XIcon className="w-4 h-4 mr-2" />
|
||||
<X className="w-4 h-4 mr-2" />
|
||||
{t("common.close")}
|
||||
</Button>
|
||||
</div>
|
||||
|
||||
@@ -4,14 +4,14 @@ import { useState, useEffect } from "react";
|
||||
import { Modal } from "@/app/_components/GlobalComponents/UIElements/Modal";
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
import {
|
||||
UploadIcon,
|
||||
TrashIcon,
|
||||
CalendarIcon,
|
||||
UserIcon,
|
||||
DownloadIcon,
|
||||
ArrowsClockwiseIcon,
|
||||
CheckIcon,
|
||||
} from "@phosphor-icons/react";
|
||||
Upload,
|
||||
Trash2,
|
||||
Calendar,
|
||||
User,
|
||||
Download,
|
||||
RefreshCw,
|
||||
Check,
|
||||
} from "lucide-react";
|
||||
import { useTranslations } from "next-intl";
|
||||
import { CronJob } from "@/app/_utils/cronjob-utils";
|
||||
import { unwrapCommand } from "@/app/_utils/wrapper-utils-client";
|
||||
@@ -92,7 +92,7 @@ export const RestoreBackupModal = ({
|
||||
onClick={onBackupAll}
|
||||
className="btn-outline flex-1"
|
||||
>
|
||||
<DownloadIcon className="h-4 w-4 mr-2" />
|
||||
<Download className="h-4 w-4 mr-2" />
|
||||
{t("cronjobs.backupAll")}
|
||||
</Button>
|
||||
{backups.length > 0 && (
|
||||
@@ -101,7 +101,7 @@ export const RestoreBackupModal = ({
|
||||
onClick={handleRestoreAll}
|
||||
className="btn-primary flex-1"
|
||||
>
|
||||
<UploadIcon className="h-4 w-4 mr-2" />
|
||||
<Upload className="h-4 w-4 mr-2" />
|
||||
{t("cronjobs.restoreAll")}
|
||||
</Button>
|
||||
)}
|
||||
@@ -111,7 +111,7 @@ export const RestoreBackupModal = ({
|
||||
className="btn-outline"
|
||||
title={t("common.refresh")}
|
||||
>
|
||||
<ArrowsClockwiseIcon className="h-4 w-4" />
|
||||
<RefreshCw className="h-4 w-4" />
|
||||
</Button>
|
||||
</div>
|
||||
|
||||
@@ -120,15 +120,15 @@ export const RestoreBackupModal = ({
|
||||
<p>{t("cronjobs.noBackupsFound")}</p>
|
||||
</div>
|
||||
) : (
|
||||
<div className="space-y-3 max-h-[500px] overflow-y-auto tui-scrollbar pr-2 pb-2">
|
||||
<div className="space-y-2 max-h-[500px] overflow-y-auto">
|
||||
{backups.map((backup) => (
|
||||
<div
|
||||
key={backup.filename}
|
||||
className="tui-card p-3 terminal-font"
|
||||
className="glass-card p-3 border border-border/50 rounded-lg hover:bg-accent/30 transition-colors"
|
||||
>
|
||||
<div className="flex items-center gap-3">
|
||||
<div className="flex-shrink-0">
|
||||
<code className="text-xs bg-background0 text-status-warning px-1.5 py-0.5 terminal-font ascii-border">
|
||||
<code className="text-xs bg-purple-500/10 text-purple-600 dark:text-purple-400 px-1.5 py-0.5 rounded font-mono border border-purple-500/20">
|
||||
{backup.job.schedule}
|
||||
</code>
|
||||
</div>
|
||||
@@ -136,7 +136,7 @@ export const RestoreBackupModal = ({
|
||||
<div className="flex-1 min-w-0">
|
||||
<div className="flex items-center gap-2">
|
||||
{commandCopied === backup.filename && (
|
||||
<CheckIcon className="h-3 w-3 text-status-success flex-shrink-0" />
|
||||
<Check className="h-3 w-3 text-green-600 flex-shrink-0" />
|
||||
)}
|
||||
<pre
|
||||
onClick={(e) => {
|
||||
@@ -145,7 +145,7 @@ export const RestoreBackupModal = ({
|
||||
setCommandCopied(backup.filename);
|
||||
setTimeout(() => setCommandCopied(null), 3000);
|
||||
}}
|
||||
className="flex-1 cursor-pointer overflow-hidden text-sm font-medium terminal-font bg-background1 px-2 py-1 ascii-border truncate"
|
||||
className="flex-1 cursor-pointer overflow-hidden text-sm font-medium text-foreground bg-muted/30 px-2 py-1 rounded border border-border/30 truncate"
|
||||
title={unwrapCommand(backup.job.command)}
|
||||
>
|
||||
{unwrapCommand(backup.job.command)}
|
||||
@@ -155,47 +155,47 @@ export const RestoreBackupModal = ({
|
||||
|
||||
<div className="flex items-center gap-3 text-xs text-muted-foreground flex-shrink-0">
|
||||
<div className="flex items-center gap-1">
|
||||
<UserIcon className="h-3 w-3" />
|
||||
<User className="h-3 w-3" />
|
||||
<span>{backup.job.user}</span>
|
||||
</div>
|
||||
<div className="flex items-center gap-1">
|
||||
<CalendarIcon className="h-3 w-3" />
|
||||
<Calendar className="h-3 w-3" />
|
||||
<span>{formatDate(backup.backedUpAt)}</span>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div className="flex items-center gap-2 flex-shrink-0">
|
||||
<div className="flex items-center gap-1 flex-shrink-0">
|
||||
<Button
|
||||
variant="outline"
|
||||
variant="ghost"
|
||||
size="sm"
|
||||
onClick={() => {
|
||||
onRestore(backup.filename);
|
||||
onClose();
|
||||
}}
|
||||
className="btn-outline h-8 px-3"
|
||||
className="h-7 w-7 p-0"
|
||||
title={t("cronjobs.restoreThisBackup")}
|
||||
>
|
||||
<UploadIcon className="h-3 w-3" />
|
||||
<Upload className="h-3 w-3" />
|
||||
</Button>
|
||||
<Button
|
||||
variant="destructive"
|
||||
variant="ghost"
|
||||
size="sm"
|
||||
onClick={() => handleDelete(backup.filename)}
|
||||
disabled={deletingFilename === backup.filename}
|
||||
className="h-8 px-3"
|
||||
className="h-7 w-7 p-0 text-destructive hover:text-destructive"
|
||||
title={t("cronjobs.deleteBackup")}
|
||||
>
|
||||
{deletingFilename === backup.filename ? (
|
||||
<div className="h-3 w-3 animate-spin rounded-full border-2 border-current border-t-transparent" />
|
||||
) : (
|
||||
<TrashIcon className="h-3 w-3" />
|
||||
<Trash2 className="h-3 w-3" />
|
||||
)}
|
||||
</Button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{backup.job.comment && (
|
||||
<p className="text-xs text-muted-foreground italic mt-2">
|
||||
<p className="text-xs text-muted-foreground italic mt-2 ml-0">
|
||||
{backup.job.comment}
|
||||
</p>
|
||||
)}
|
||||
@@ -204,7 +204,7 @@ export const RestoreBackupModal = ({
|
||||
</div>
|
||||
)}
|
||||
|
||||
<div className="flex justify-between gap-2 pt-4 border-t border-border">
|
||||
<div className="flex justify-between gap-2 pt-4 border-t border-border/50">
|
||||
<p className="text-sm text-muted-foreground">
|
||||
{t("cronjobs.availableBackups")}: {backups.length}
|
||||
</p>
|
||||
|
||||
@@ -6,7 +6,7 @@ import { Input } from "@/app/_components/GlobalComponents/FormElements/Input";
|
||||
import { BashEditor } from "@/app/_components/FeatureComponents/Scripts/BashEditor";
|
||||
import { BashSnippetHelper } from "@/app/_components/FeatureComponents/Scripts/BashSnippetHelper";
|
||||
import { showToast } from "@/app/_components/GlobalComponents/UIElements/Toast";
|
||||
import { FileTextIcon, CodeIcon, InfoIcon, TrashIcon } from "@phosphor-icons/react";
|
||||
import { FileText, Code, Info, Trash2 } from "lucide-react";
|
||||
import { useTranslations } from "next-intl";
|
||||
|
||||
interface ScriptModalProps {
|
||||
@@ -80,11 +80,11 @@ export const ScriptModal = ({
|
||||
|
||||
return (
|
||||
<Modal isOpen={isOpen} onClose={onClose} title={title} size="2xl">
|
||||
<form onSubmit={handleSubmit} className="space-y-6 terminal-font">
|
||||
<form onSubmit={handleSubmit} className="space-y-6">
|
||||
<div className="grid grid-cols-1 md:grid-cols-2 gap-4">
|
||||
<div>
|
||||
<label className="block text-sm font-medium mb-2">
|
||||
Script Name <span className="text-status-error">*</span>
|
||||
<label className="block text-sm font-medium text-foreground mb-2">
|
||||
Script Name <span className="text-red-500">*</span>
|
||||
</label>
|
||||
<Input
|
||||
value={form.name}
|
||||
@@ -93,15 +93,15 @@ export const ScriptModal = ({
|
||||
required
|
||||
className={
|
||||
!form.name.trim()
|
||||
? "border-status-error focus:border-status-error"
|
||||
? "border-red-300 focus:border-red-500 focus:ring-red-500"
|
||||
: ""
|
||||
}
|
||||
/>
|
||||
</div>
|
||||
<div>
|
||||
<label className="block text-sm font-medium mb-2">
|
||||
<label className="block text-sm font-medium text-foreground mb-2">
|
||||
Description{" "}
|
||||
<span className="text-xs opacity-60">(optional)</span>
|
||||
<span className="text-muted-foreground text-xs">(optional)</span>
|
||||
</label>
|
||||
<Input
|
||||
value={form.description}
|
||||
@@ -112,24 +112,24 @@ export const ScriptModal = ({
|
||||
</div>
|
||||
|
||||
<div className="grid grid-cols-1 lg:grid-cols-3 gap-6 h-[500px]">
|
||||
<div className="lg:col-span-1 bg-background0 ascii-border p-4 flex flex-col h-full overflow-hidden">
|
||||
<div className="lg:col-span-1 bg-muted/20 rounded-lg p-4 flex flex-col h-full overflow-hidden">
|
||||
<div className="flex items-center gap-2 mb-4 flex-shrink-0">
|
||||
<CodeIcon className="h-4 w-4" />
|
||||
<h3 className="text-sm font-medium">Snippets</h3>
|
||||
<Code className="h-4 w-4 text-primary" />
|
||||
<h3 className="text-sm font-medium text-foreground">Snippets</h3>
|
||||
</div>
|
||||
<div className="flex-1 overflow-y-auto min-h-0 !pr-0 tui-scrollbar">
|
||||
<div className="flex-1 overflow-y-auto min-h-0 !pr-0">
|
||||
<BashSnippetHelper onInsertSnippet={handleInsertSnippet} />
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div className="lg:col-span-2 flex flex-col h-full overflow-hidden">
|
||||
<div className="flex items-center gap-2 mb-4 flex-shrink-0">
|
||||
<FileTextIcon className="h-4 w-4" />
|
||||
<h3 className="text-sm font-medium">
|
||||
Script Content <span className="text-status-error">*</span>
|
||||
<FileText className="h-4 w-4 text-primary" />
|
||||
<h3 className="text-sm font-medium text-foreground">
|
||||
Script Content <span className="text-red-500">*</span>
|
||||
</h3>
|
||||
{isDraft && (
|
||||
<span className="ml-auto px-2 py-0.5 text-xs font-medium bg-background0 text-status-info ascii-border">
|
||||
<span className="ml-auto px-2 py-0.5 text-xs font-medium bg-blue-500/10 text-blue-500 border border-blue-500/30 rounded-full">
|
||||
{t("scripts.draft")}
|
||||
</span>
|
||||
)}
|
||||
@@ -145,16 +145,16 @@ export const ScriptModal = ({
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div className="flex justify-between items-center gap-3 pt-4 border-border border-t">
|
||||
<div className="flex justify-between items-center gap-3 pt-4 border-t border-border/30">
|
||||
<div>
|
||||
{isDraft && onClearDraft && (
|
||||
<Button
|
||||
type="button"
|
||||
variant="ghost"
|
||||
onClick={onClearDraft}
|
||||
className="opacity-60 hover:opacity-100"
|
||||
className="text-muted-foreground hover:text-foreground"
|
||||
>
|
||||
<TrashIcon className="h-4 w-4 mr-2" />
|
||||
<Trash2 className="h-4 w-4 mr-2" />
|
||||
{t("scripts.clearDraft")}
|
||||
</Button>
|
||||
)}
|
||||
|
||||
@@ -4,7 +4,7 @@ import { useEffect, useState } from "react";
|
||||
import { Modal } from "@/app/_components/GlobalComponents/UIElements/Modal";
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
import { Input } from "@/app/_components/GlobalComponents/FormElements/Input";
|
||||
import { FileTextIcon, MagnifyingGlassIcon, CheckIcon, TerminalIcon } from "@phosphor-icons/react";
|
||||
import { FileText, Search, Check, Terminal } from "lucide-react";
|
||||
import { Script } from "@/app/_utils/scripts-utils";
|
||||
import { getScriptContent } from "@/app/_server/actions/scripts";
|
||||
import { getHostScriptPath } from "@/app/_server/actions/scripts";
|
||||
@@ -84,7 +84,7 @@ export const SelectScriptModal = ({
|
||||
>
|
||||
<div className="space-y-4">
|
||||
<div className="relative">
|
||||
<MagnifyingGlassIcon className="absolute left-3 top-1/2 transform -translate-y-1/2 h-4 w-4 text-muted-foreground" />
|
||||
<Search className="absolute left-3 top-1/2 transform -translate-y-1/2 h-4 w-4 text-muted-foreground" />
|
||||
<Input
|
||||
value={searchQuery}
|
||||
onChange={(e) => setSearchQuery(e.target.value)}
|
||||
@@ -119,12 +119,12 @@ export const SelectScriptModal = ({
|
||||
<div className="flex items-start justify-between">
|
||||
<div className="flex-1 min-w-0">
|
||||
<div className="flex items-center gap-2 mb-1">
|
||||
<FileTextIcon className="h-4 w-4 text-primary flex-shrink-0" />
|
||||
<FileText className="h-4 w-4 text-primary flex-shrink-0" />
|
||||
<h4 className="font-medium text-foreground truncate">
|
||||
{script.name}
|
||||
</h4>
|
||||
{selectedScriptId === script.id && (
|
||||
<CheckIcon className="h-4 w-4 text-status-success flex-shrink-0" />
|
||||
<Check className="h-4 w-4 text-green-500 flex-shrink-0" />
|
||||
)}
|
||||
</div>
|
||||
<p className="text-sm text-muted-foreground line-clamp-2">
|
||||
@@ -148,7 +148,7 @@ export const SelectScriptModal = ({
|
||||
{t("scripts.scriptPreview")}
|
||||
</h3>
|
||||
</div>
|
||||
<div className="p-4 h-full max-h-80 overflow-y-auto tui-scrollbar">
|
||||
<div className="p-4 h-full max-h-80 overflow-y-auto">
|
||||
{previewScript ? (
|
||||
<div className="space-y-4">
|
||||
<div>
|
||||
@@ -162,12 +162,12 @@ export const SelectScriptModal = ({
|
||||
|
||||
<div>
|
||||
<div className="flex items-center gap-2 mb-2">
|
||||
<TerminalIcon className="h-4 w-4 text-primary" />
|
||||
<Terminal className="h-4 w-4 text-primary" />
|
||||
<span className="text-sm font-medium text-foreground">
|
||||
{t("scripts.commandPreview")}
|
||||
</span>
|
||||
</div>
|
||||
<div className="bg-muted/30 p-3 rounded border border-border">
|
||||
<div className="bg-muted/30 p-3 rounded border border-border/30">
|
||||
<code className="text-sm font-mono text-foreground break-all">
|
||||
{hostScriptPath}
|
||||
</code>
|
||||
@@ -178,7 +178,7 @@ export const SelectScriptModal = ({
|
||||
<span className="text-sm font-medium text-foreground">
|
||||
{t("scripts.scriptContent")}
|
||||
</span>
|
||||
<div className="bg-muted/30 p-3 rounded border border-border mt-2 max-h-32 overflow-auto tui-scrollbar">
|
||||
<div className="bg-muted/30 p-3 rounded border border-border/30 mt-2 max-h-32 overflow-auto">
|
||||
<pre className="text-xs font-mono text-foreground whitespace-pre-wrap">
|
||||
{previewContent}
|
||||
</pre>
|
||||
@@ -187,7 +187,7 @@ export const SelectScriptModal = ({
|
||||
</div>
|
||||
) : (
|
||||
<div className="text-center text-muted-foreground py-8">
|
||||
<FileTextIcon className="h-12 w-12 mx-auto mb-4 opacity-50" />
|
||||
<FileText className="h-12 w-12 mx-auto mb-4 opacity-50" />
|
||||
<p>{t("scripts.selectScriptToPreview")}</p>
|
||||
</div>
|
||||
)}
|
||||
@@ -195,7 +195,7 @@ export const SelectScriptModal = ({
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div className="flex justify-end gap-2 pt-3 border-t border-border">
|
||||
<div className="flex justify-end gap-2 pt-3 border-t border-border/50">
|
||||
<Button
|
||||
type="button"
|
||||
variant="outline"
|
||||
@@ -210,7 +210,7 @@ export const SelectScriptModal = ({
|
||||
disabled={!previewScript}
|
||||
className="btn-primary glow-primary"
|
||||
>
|
||||
<CheckIcon className="h-4 w-4 mr-2" />
|
||||
<Check className="h-4 w-4 mr-2" />
|
||||
{t("scripts.selectScript")}
|
||||
</Button>
|
||||
</div>
|
||||
|
||||
@@ -42,14 +42,14 @@ export const PWAInstallPrompt = (): JSX.Element | null => {
|
||||
if (choice.outcome === "accepted") {
|
||||
setDeferred(null);
|
||||
}
|
||||
} catch (_err) { }
|
||||
} catch (_err) {}
|
||||
}, [deferred]);
|
||||
|
||||
if (isInstalled || !deferred) return null;
|
||||
|
||||
return (
|
||||
<button
|
||||
className="px-3 py-1 rounded-md border border-border bg-background/80 hover:bg-background/60"
|
||||
className="px-3 py-1 rounded-md border border-border/50 bg-background/80 hover:bg-background/60"
|
||||
onClick={onInstall}
|
||||
>
|
||||
Install App
|
||||
|
||||
@@ -5,10 +5,9 @@ import { EditorView, keymap } from "@codemirror/view";
|
||||
import { EditorState, Transaction } from "@codemirror/state";
|
||||
import { shell } from "@codemirror/legacy-modes/mode/shell";
|
||||
import { StreamLanguage } from "@codemirror/language";
|
||||
import { catppuccinMocha, catppuccinLatte } from './catppuccin-theme';
|
||||
import { useTheme } from 'next-themes';
|
||||
import { oneDark } from "@codemirror/theme-one-dark";
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
import { TerminalIcon, CopyIcon, CheckIcon } from "@phosphor-icons/react";
|
||||
import { Terminal, Copy, Check } from "lucide-react";
|
||||
|
||||
interface BashEditorProps {
|
||||
value: string;
|
||||
@@ -28,7 +27,6 @@ export const BashEditor = ({
|
||||
const [copied, setCopied] = useState(false);
|
||||
const editorRef = useRef<HTMLDivElement>(null);
|
||||
const editorViewRef = useRef<EditorView | null>(null);
|
||||
const { theme } = useTheme();
|
||||
|
||||
const insertFourSpaces = ({
|
||||
state,
|
||||
@@ -101,54 +99,13 @@ export const BashEditor = ({
|
||||
useEffect(() => {
|
||||
if (!editorRef.current) return;
|
||||
|
||||
const isDark = theme === 'catppuccin-mocha';
|
||||
const bashLanguage = StreamLanguage.define(shell);
|
||||
|
||||
const getThemeColors = () => {
|
||||
const root = document.documentElement;
|
||||
const style = getComputedStyle(root);
|
||||
|
||||
return {
|
||||
background: style.getPropertyValue('--base').trim() || (isDark ? '#1e1e2e' : '#eff1f5'),
|
||||
foreground: style.getPropertyValue('--text').trim() || (isDark ? '#cdd6f4' : '#4c4f69'),
|
||||
border: style.getPropertyValue('--box-border-color').trim() || (isDark ? '#313244' : '#9ca0b0'),
|
||||
surface: style.getPropertyValue('--surface0').trim() || (isDark ? '#313244' : '#ccd0da'),
|
||||
};
|
||||
};
|
||||
|
||||
const colors = getThemeColors();
|
||||
|
||||
const customTheme = EditorView.theme({
|
||||
"&": {
|
||||
backgroundColor: colors.background,
|
||||
color: colors.foreground,
|
||||
border: `1px solid ${colors.border}`,
|
||||
borderRadius: '0',
|
||||
},
|
||||
".cm-content": {
|
||||
caretColor: colors.foreground,
|
||||
padding: '12px'
|
||||
},
|
||||
".cm-gutters": {
|
||||
backgroundColor: colors.surface,
|
||||
color: colors.foreground,
|
||||
borderRight: `1px solid ${colors.border}`,
|
||||
opacity: '0.6',
|
||||
},
|
||||
".cm-activeLineGutter": {
|
||||
backgroundColor: colors.surface,
|
||||
opacity: '1',
|
||||
},
|
||||
".cm-scroller": {
|
||||
fontFamily: 'JetBrains Mono, Fira CodeIcon, monospace',
|
||||
},
|
||||
}, { dark: isDark });
|
||||
|
||||
const state = EditorState.create({
|
||||
doc: value || placeholder,
|
||||
extensions: [
|
||||
bashLanguage,
|
||||
customTheme,
|
||||
oneDark,
|
||||
keymap.of([
|
||||
{ key: "Tab", run: insertFourSpaces },
|
||||
{ key: "Shift-Tab", run: removeFourSpaces },
|
||||
@@ -162,7 +119,7 @@ export const BashEditor = ({
|
||||
"&": {
|
||||
fontSize: "14px",
|
||||
fontFamily:
|
||||
'JetBrains Mono, Fira CodeIcon, ui-monospace, SFMono-Regular, "SF Mono", Consolas, "Liberation Mono", Menlo, monospace',
|
||||
'ui-monospace, SFMono-Regular, "SF Mono", Consolas, "Liberation Mono", Menlo, monospace',
|
||||
height: "100%",
|
||||
maxHeight: "100%",
|
||||
},
|
||||
@@ -175,7 +132,7 @@ export const BashEditor = ({
|
||||
},
|
||||
".cm-scroller": {
|
||||
fontFamily:
|
||||
'JetBrains Mono, Fira CodeIcon, ui-monospace, SFMono-Regular, "SF Mono", Consolas, "Liberation Mono", Menlo, monospace',
|
||||
'ui-monospace, SFMono-Regular, "SF Mono", Consolas, "Liberation Mono", Menlo, monospace',
|
||||
height: "100%",
|
||||
maxHeight: "100%",
|
||||
},
|
||||
@@ -193,7 +150,7 @@ export const BashEditor = ({
|
||||
return () => {
|
||||
view.destroy();
|
||||
};
|
||||
}, [theme]);
|
||||
}, []);
|
||||
|
||||
useEffect(() => {
|
||||
if (editorViewRef.current) {
|
||||
@@ -224,7 +181,7 @@ export const BashEditor = ({
|
||||
{label && (
|
||||
<div className="flex items-center justify-between mb-2">
|
||||
<div className="flex items-center gap-2">
|
||||
<TerminalIcon className="h-4 w-4" />
|
||||
<Terminal className="h-4 w-4 text-cyan-500" />
|
||||
<span className="text-sm font-medium">{label}</span>
|
||||
</div>
|
||||
<Button
|
||||
@@ -235,16 +192,16 @@ export const BashEditor = ({
|
||||
className="btn-outline h-7 px-2"
|
||||
>
|
||||
{copied ? (
|
||||
<CheckIcon className="h-3 w-3 mr-1" />
|
||||
<Check className="h-3 w-3 mr-1" />
|
||||
) : (
|
||||
<CopyIcon className="h-3 w-3 mr-1" />
|
||||
<Copy className="h-3 w-3 mr-1" />
|
||||
)}
|
||||
{copied ? "Copied!" : "CopyIcon"}
|
||||
{copied ? "Copied!" : "Copy"}
|
||||
</Button>
|
||||
</div>
|
||||
)}
|
||||
<div className="overflow-hidden h-full">
|
||||
<div ref={editorRef} className="h-full" />
|
||||
<div className="border border-border overflow-hidden h-full">
|
||||
<div ref={editorRef} className="h-full rounded-lg" />
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
|
||||
@@ -4,15 +4,15 @@ import { useState, useEffect } from "react";
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
import { Input } from "@/app/_components/GlobalComponents/FormElements/Input";
|
||||
import {
|
||||
MagnifyingGlassIcon,
|
||||
FileTextIcon,
|
||||
Search,
|
||||
FileText,
|
||||
FolderOpen,
|
||||
CodeIcon,
|
||||
GearIcon,
|
||||
Code,
|
||||
Settings,
|
||||
Database,
|
||||
CopyIcon,
|
||||
CheckIcon,
|
||||
} from "@phosphor-icons/react";
|
||||
Copy,
|
||||
Check,
|
||||
} from "lucide-react";
|
||||
import {
|
||||
fetchSnippets,
|
||||
fetchSnippetCategories,
|
||||
@@ -25,13 +25,13 @@ interface BashSnippetHelperProps {
|
||||
}
|
||||
|
||||
const categoryIcons = {
|
||||
"File Operations": FileTextIcon,
|
||||
Loops: CodeIcon,
|
||||
Conditionals: CodeIcon,
|
||||
"System Operations": GearIcon,
|
||||
"File Operations": FileText,
|
||||
Loops: Code,
|
||||
Conditionals: Code,
|
||||
"System Operations": Settings,
|
||||
"Database Operations": Database,
|
||||
"UserIcon Examples": FolderOpen,
|
||||
"Custom Scripts": CodeIcon,
|
||||
"User Examples": FolderOpen,
|
||||
"Custom Scripts": Code,
|
||||
};
|
||||
|
||||
export const BashSnippetHelper = ({
|
||||
@@ -109,7 +109,7 @@ export const BashSnippetHelper = ({
|
||||
return (
|
||||
<div className="space-y-3">
|
||||
<div className="text-center py-8">
|
||||
<CodeIcon className="h-8 w-8 text-muted-foreground mx-auto mb-2 animate-spin" />
|
||||
<Code className="h-8 w-8 text-muted-foreground mx-auto mb-2 animate-spin" />
|
||||
<p className="text-sm text-muted-foreground">Loading snippets...</p>
|
||||
</div>
|
||||
</div>
|
||||
@@ -119,7 +119,7 @@ export const BashSnippetHelper = ({
|
||||
return (
|
||||
<div className="space-y-3">
|
||||
<div className="relative">
|
||||
<MagnifyingGlassIcon className="absolute left-3 top-1/2 -translate-y-1/2 h-4 w-4 text-muted-foreground" />
|
||||
<Search className="absolute left-3 top-1/2 -translate-y-1/2 h-4 w-4 text-muted-foreground" />
|
||||
<Input
|
||||
value={searchQuery}
|
||||
onChange={(e) => setSearchQuery(e.target.value)}
|
||||
@@ -129,7 +129,7 @@ export const BashSnippetHelper = ({
|
||||
</div>
|
||||
|
||||
{!searchQuery && (
|
||||
<div className="overflow-x-auto tui-scrollbar">
|
||||
<div className="overflow-x-auto">
|
||||
<div className="flex gap-1 pb-2 min-w-max">
|
||||
<Button
|
||||
type="button"
|
||||
@@ -142,7 +142,7 @@ export const BashSnippetHelper = ({
|
||||
</Button>
|
||||
{categories.map((category) => {
|
||||
const Icon =
|
||||
categoryIcons[category as keyof typeof categoryIcons] || CodeIcon;
|
||||
categoryIcons[category as keyof typeof categoryIcons] || Code;
|
||||
return (
|
||||
<Button
|
||||
key={category}
|
||||
@@ -163,15 +163,15 @@ export const BashSnippetHelper = ({
|
||||
</div>
|
||||
)}
|
||||
|
||||
<div className="space-y-2 overflow-y-auto !pr-0 tui-scrollbar">
|
||||
<div className="space-y-2 overflow-y-auto !pr-0 custom-scrollbar">
|
||||
{filteredSnippets.map((snippet) => {
|
||||
const Icon =
|
||||
categoryIcons[snippet.category as keyof typeof categoryIcons] ||
|
||||
CodeIcon;
|
||||
Code;
|
||||
return (
|
||||
<div
|
||||
key={snippet.id}
|
||||
className="bg-muted/30 rounded-lg border border-border p-3 hover:bg-accent/30 transition-colors"
|
||||
className="bg-muted/30 rounded-lg border border-border/50 p-3 hover:bg-accent/30 transition-colors"
|
||||
>
|
||||
<div className="flex-1 min-w-0">
|
||||
<div className="flex items-center gap-2 mb-1">
|
||||
@@ -181,7 +181,7 @@ export const BashSnippetHelper = ({
|
||||
</h4>
|
||||
{snippet.source === "user" && (
|
||||
<span className="inline-block px-1.5 py-0.5 text-xs bg-green-100 text-green-700 rounded border border-green-200">
|
||||
UserIcon
|
||||
User
|
||||
</span>
|
||||
)}
|
||||
</div>
|
||||
@@ -192,7 +192,7 @@ export const BashSnippetHelper = ({
|
||||
{snippet.tags.slice(0, 3).map((tag) => (
|
||||
<span
|
||||
key={tag}
|
||||
className="inline-block px-2 py-1 text-xs bg-primary/10 text-primary border border-border"
|
||||
className="inline-block px-2 py-1 text-xs bg-primary/10 text-primary rounded border border-primary/20"
|
||||
>
|
||||
{tag}
|
||||
</span>
|
||||
@@ -209,11 +209,12 @@ export const BashSnippetHelper = ({
|
||||
variant="outline"
|
||||
size="sm"
|
||||
onClick={() => handleCopy(snippet)}
|
||||
className="h-6 w-8 p-0 text-xs"
|
||||
>
|
||||
{copiedId === snippet.id ? (
|
||||
<CheckIcon className="h-3 w-3" />
|
||||
<Check className="h-3 w-3" />
|
||||
) : (
|
||||
<CopyIcon className="h-3 w-3" />
|
||||
<Copy className="h-3 w-3" />
|
||||
)}
|
||||
</Button>
|
||||
<Button
|
||||
@@ -221,7 +222,7 @@ export const BashSnippetHelper = ({
|
||||
variant="default"
|
||||
size="sm"
|
||||
onClick={() => handleInsert(snippet)}
|
||||
className="flex-1"
|
||||
className="h-6 px-3 text-xs flex-1"
|
||||
>
|
||||
Insert
|
||||
</Button>
|
||||
@@ -233,7 +234,7 @@ export const BashSnippetHelper = ({
|
||||
|
||||
{filteredSnippets.length === 0 && (
|
||||
<div className="text-center py-8">
|
||||
<CodeIcon className="h-8 w-8 text-muted-foreground mx-auto mb-2" />
|
||||
<Code className="h-8 w-8 text-muted-foreground mx-auto mb-2" />
|
||||
<p className="text-sm text-muted-foreground">
|
||||
{searchQuery
|
||||
? `No snippets found for "${searchQuery}"`
|
||||
|
||||
@@ -9,15 +9,15 @@ import {
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
import { Input } from "@/app/_components/GlobalComponents/FormElements/Input";
|
||||
import {
|
||||
ClockIcon,
|
||||
InfoIcon,
|
||||
CheckCircleIcon,
|
||||
WarningCircleIcon,
|
||||
Clock,
|
||||
Info,
|
||||
CheckCircle,
|
||||
AlertCircle,
|
||||
Calendar,
|
||||
CaretDownIcon,
|
||||
CaretUpIcon,
|
||||
MagnifyingGlassIcon,
|
||||
} from "@phosphor-icons/react";
|
||||
ChevronDown,
|
||||
ChevronUp,
|
||||
Search,
|
||||
} from "lucide-react";
|
||||
import { useLocale } from "next-intl";
|
||||
|
||||
interface CronExpressionHelperProps {
|
||||
@@ -87,20 +87,20 @@ export const CronExpressionHelper = ({
|
||||
/>
|
||||
<div className="absolute right-3 top-1/2 -translate-y-1/2">
|
||||
{explanation?.isValid ? (
|
||||
<CheckCircleIcon className="h-4 w-4 text-status-success" />
|
||||
<CheckCircle className="h-4 w-4 text-green-500" />
|
||||
) : value ? (
|
||||
<WarningCircleIcon className="h-4 w-4 text-status-error" />
|
||||
<AlertCircle className="h-4 w-4 text-red-500" />
|
||||
) : (
|
||||
<ClockIcon className="h-4 w-4 text-muted-foreground" />
|
||||
<Clock className="h-4 w-4 text-muted-foreground" />
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{explanation && (
|
||||
<div className="bg-background0 p-2 text-status-warning ascii-border terminal-font">
|
||||
<div className="bg-muted/30 rounded p-2 border border-border/30">
|
||||
<div className="space-y-1">
|
||||
<div className="flex items-start gap-2">
|
||||
<InfoIcon className="h-3 w-3 text-primary mt-0.5 flex-shrink-0" />
|
||||
<Info className="h-3 w-3 text-primary mt-0.5 flex-shrink-0" />
|
||||
<div className="flex-1">
|
||||
<p className="text-xs italic text-muted-foreground">
|
||||
{explanation.isValid
|
||||
@@ -108,7 +108,7 @@ export const CronExpressionHelper = ({
|
||||
: "Invalid Expression"}
|
||||
</p>
|
||||
{explanation.error && (
|
||||
<p className="text-xs text-status-error mt-0.5">
|
||||
<p className="text-xs text-red-600 dark:text-red-400 mt-0.5">
|
||||
{explanation.error}
|
||||
</p>
|
||||
)}
|
||||
@@ -117,7 +117,7 @@ export const CronExpressionHelper = ({
|
||||
|
||||
{explanation.isValid && explanation.nextRuns.length > 0 && (
|
||||
<div className="flex items-start gap-2">
|
||||
<Calendar className="h-3 w-3 text-status-info mt-0.5 flex-shrink-0" />
|
||||
<Calendar className="h-3 w-3 text-blue-500 mt-0.5 flex-shrink-0" />
|
||||
<div className="flex-1">
|
||||
<p className="text-xs text-muted-foreground mb-1">
|
||||
Next executions:
|
||||
@@ -137,7 +137,7 @@ export const CronExpressionHelper = ({
|
||||
)}
|
||||
|
||||
{showPatterns && (
|
||||
<div className="bg-background0 ascii-border terminal-font">
|
||||
<div className="bg-muted/30 rounded-lg border border-border/50">
|
||||
<button
|
||||
type="button"
|
||||
onClick={(e) => {
|
||||
@@ -145,24 +145,24 @@ export const CronExpressionHelper = ({
|
||||
e.stopPropagation();
|
||||
setShowPatternsPanel(!showPatternsPanel);
|
||||
}}
|
||||
className="w-full text-left p-3 hover:bg-background0 transition-colors"
|
||||
className="w-full text-left p-3 hover:bg-accent/30 transition-colors rounded-t-lg"
|
||||
>
|
||||
<div className="flex items-center justify-between">
|
||||
<span className="text-sm font-medium">Quick Patterns</span>
|
||||
<div className="p-1">
|
||||
{showPatternsPanel ? (
|
||||
<CaretUpIcon className="h-4 w-4" />
|
||||
<ChevronUp className="h-4 w-4" />
|
||||
) : (
|
||||
<CaretDownIcon className="h-4 w-4" />
|
||||
<ChevronDown className="h-4 w-4" />
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
</button>
|
||||
|
||||
{showPatternsPanel && (
|
||||
<div className="p-3 border-t border-border">
|
||||
<div className="p-3 border-t border-border/50">
|
||||
<div className="relative mb-3">
|
||||
<MagnifyingGlassIcon className="absolute left-3 top-1/2 -translate-y-1/2 h-4 w-4 text-muted-foreground" />
|
||||
<Search className="absolute left-3 top-1/2 -translate-y-1/2 h-4 w-4 text-muted-foreground" />
|
||||
<Input
|
||||
value={patternSearch}
|
||||
onChange={(e) => setPatternSearch(e.target.value)}
|
||||
@@ -171,7 +171,7 @@ export const CronExpressionHelper = ({
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div className="space-y-3 max-h-64 overflow-y-auto tui-scrollbar">
|
||||
<div className="space-y-3 max-h-64 overflow-y-auto custom-scrollbar">
|
||||
{filteredPatterns.map((category) => (
|
||||
<div key={category.category} className="space-y-2">
|
||||
<h4 className="font-medium text-foreground text-sm">
|
||||
|
||||
@@ -4,14 +4,15 @@ import { useState, useEffect } from "react";
|
||||
import { Card, CardContent, CardHeader, CardTitle } from "@/app/_components/GlobalComponents/Cards/Card";
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
import {
|
||||
FileTextIcon,
|
||||
PlusIcon,
|
||||
PencilSimpleIcon,
|
||||
TrashIcon,
|
||||
CopyIcon,
|
||||
CheckCircleIcon,
|
||||
FilesIcon,
|
||||
} from "@phosphor-icons/react";
|
||||
FileText,
|
||||
Plus,
|
||||
Edit,
|
||||
Trash2,
|
||||
Copy,
|
||||
Copy as CopyIcon,
|
||||
CheckCircle,
|
||||
Files,
|
||||
} from "lucide-react";
|
||||
import { Script } from "@/app/_utils/scripts-utils";
|
||||
import {
|
||||
createScript,
|
||||
@@ -206,8 +207,8 @@ export const ScriptsManager = ({
|
||||
<CardHeader>
|
||||
<div className="flex flex-col sm:flex-row sm:items-center justify-between gap-4">
|
||||
<div className="flex items-center gap-3">
|
||||
<div className="p-2 bg-background0 ascii-border">
|
||||
<FileTextIcon className="h-5 w-5 text-primary" />
|
||||
<div className="p-2 bg-primary/10 rounded-lg">
|
||||
<FileText className="h-5 w-5 text-primary" />
|
||||
</div>
|
||||
<div>
|
||||
<CardTitle className="text-xl brand-gradient">
|
||||
@@ -222,16 +223,16 @@ export const ScriptsManager = ({
|
||||
onClick={() => setIsCreateModalOpen(true)}
|
||||
className="btn-primary glow-primary"
|
||||
>
|
||||
<PlusIcon className="h-4 w-4 mr-2" />
|
||||
<Plus className="h-4 w-4 mr-2" />
|
||||
{t("scripts.newScript")}
|
||||
</Button>
|
||||
</div>
|
||||
</CardHeader>
|
||||
<CardContent>
|
||||
{scripts.length === 0 ? (
|
||||
<div className="text-center py-16 terminal-font">
|
||||
<div className="mx-auto w-20 h-20 bg-background2 ascii-border flex items-center justify-center mb-6">
|
||||
<FileTextIcon className="h-10 w-10 text-primary" />
|
||||
<div className="text-center py-16">
|
||||
<div className="mx-auto w-20 h-20 bg-gradient-to-br from-primary/20 to-blue-500/20 rounded-full flex items-center justify-center mb-6">
|
||||
<FileText className="h-10 w-10 text-primary" />
|
||||
</div>
|
||||
<h3 className="text-xl font-semibold mb-3 brand-gradient">
|
||||
{t("scripts.noScriptsYet")}
|
||||
@@ -244,7 +245,7 @@ export const ScriptsManager = ({
|
||||
className="btn-primary glow-primary"
|
||||
size="lg"
|
||||
>
|
||||
<PlusIcon className="h-5 w-5 mr-2" />
|
||||
<Plus className="h-5 w-5 mr-2" />
|
||||
{t("scripts.createYourFirstScript")}
|
||||
</Button>
|
||||
</div>
|
||||
@@ -253,7 +254,7 @@ export const ScriptsManager = ({
|
||||
{scripts.map((script) => (
|
||||
<div
|
||||
key={script.id}
|
||||
className="glass-card p-4 ascii-border hover:bg-accent/30 transition-colors terminal-font"
|
||||
className="glass-card p-4 border border-border/50 rounded-lg hover:bg-accent/30 transition-colors"
|
||||
>
|
||||
<div className="flex items-start justify-between gap-4">
|
||||
<div className="flex-1 min-w-0">
|
||||
@@ -281,11 +282,11 @@ export const ScriptsManager = ({
|
||||
size="sm"
|
||||
onClick={() => handleCopy(script)}
|
||||
className="btn-outline h-8 px-3"
|
||||
title="CopyIcon script content to clipboard"
|
||||
aria-label="CopyIcon script content to clipboard"
|
||||
title="Copy script content to clipboard"
|
||||
aria-label="Copy script content to clipboard"
|
||||
>
|
||||
{copiedId === script.id ? (
|
||||
<CheckCircleIcon className="h-3 w-3 text-status-success" />
|
||||
<CheckCircle className="h-3 w-3 text-green-500" />
|
||||
) : (
|
||||
<CopyIcon className="h-3 w-3" />
|
||||
)}
|
||||
@@ -301,7 +302,7 @@ export const ScriptsManager = ({
|
||||
title="Clone script"
|
||||
aria-label="Clone script"
|
||||
>
|
||||
<FilesIcon className="h-3 w-3" />
|
||||
<Files className="h-3 w-3" />
|
||||
</Button>
|
||||
<Button
|
||||
variant="outline"
|
||||
@@ -322,7 +323,7 @@ export const ScriptsManager = ({
|
||||
title="Edit script"
|
||||
aria-label="Edit script"
|
||||
>
|
||||
<PencilSimpleIcon className="h-3 w-3" />
|
||||
<Edit className="h-3 w-3" />
|
||||
</Button>
|
||||
<Button
|
||||
variant="destructive"
|
||||
@@ -331,11 +332,11 @@ export const ScriptsManager = ({
|
||||
setSelectedScript(script);
|
||||
setIsDeleteModalOpen(true);
|
||||
}}
|
||||
className="h-8 px-3"
|
||||
className="btn-destructive h-8 px-3"
|
||||
title="Delete script"
|
||||
aria-label="Delete script"
|
||||
>
|
||||
<TrashIcon className="h-3 w-3" />
|
||||
<Trash2 className="h-3 w-3" />
|
||||
</Button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
@@ -1,36 +0,0 @@
|
||||
import { Extension } from '@codemirror/state';
|
||||
import { EditorView } from '@codemirror/view';
|
||||
|
||||
export const catppuccinMocha: Extension = EditorView.theme({
|
||||
'&': {
|
||||
backgroundColor: '#1e1e2e',
|
||||
color: '#cdd6f4',
|
||||
fontSize: '14px',
|
||||
fontFamily: 'JetBrains Mono, monospace',
|
||||
border: '1px solid #45475a',
|
||||
borderRadius: '0',
|
||||
},
|
||||
'.cm-content': { caretColor: '#f5e0dc', padding: '12px' },
|
||||
'.cm-gutters': {
|
||||
backgroundColor: '#181825',
|
||||
color: '#6c7086',
|
||||
borderRight: '1px solid #45475a',
|
||||
},
|
||||
}, { dark: true });
|
||||
|
||||
export const catppuccinLatte: Extension = EditorView.theme({
|
||||
'&': {
|
||||
backgroundColor: '#eff1f5',
|
||||
color: '#4c4f69',
|
||||
fontSize: '14px',
|
||||
fontFamily: 'JetBrains Mono, monospace',
|
||||
border: '1px solid #9ca0b0',
|
||||
borderRadius: '0',
|
||||
},
|
||||
'.cm-content': { caretColor: '#dc8a78', padding: '12px' },
|
||||
'.cm-gutters': {
|
||||
backgroundColor: '#e6e9ef',
|
||||
color: '#8c8fa1',
|
||||
borderRight: '1px solid #9ca0b0',
|
||||
},
|
||||
}, { dark: false });
|
||||
@@ -1,6 +1,6 @@
|
||||
import { cn } from "@/app/_utils/global-utils";
|
||||
import { HTMLAttributes, forwardRef } from "react";
|
||||
import { LightningIcon } from "@phosphor-icons/react";
|
||||
import { Zap } from "lucide-react";
|
||||
import { StatusBadge } from "@/app/_components/GlobalComponents/Badges/StatusBadge";
|
||||
|
||||
export interface PerformanceMetric {
|
||||
@@ -20,14 +20,14 @@ export const PerformanceSummary = forwardRef<HTMLDivElement, PerformanceSummaryP
|
||||
<div
|
||||
ref={ref}
|
||||
className={cn(
|
||||
"p-3 bg-background0 ascii-border glass-card terminal-font",
|
||||
"p-3 bg-gradient-to-r from-purple-500/5 to-pink-500/5 border border-purple-500/20 rounded-lg glass-card",
|
||||
className
|
||||
)}
|
||||
{...props}
|
||||
>
|
||||
<div className="flex items-center gap-2 mb-3">
|
||||
<LightningIcon className="h-4 w-4" />
|
||||
<span className="text-sm font-medium">
|
||||
<Zap className="h-4 w-4 text-purple-500" />
|
||||
<span className="text-sm font-medium text-purple-600 dark:text-purple-400">
|
||||
Performance Summary
|
||||
</span>
|
||||
</div>
|
||||
|
||||
@@ -4,7 +4,7 @@ import { MetricCard } from "@/app/_components/GlobalComponents/Cards/MetricCard"
|
||||
import { SystemStatus } from "@/app/_components/FeatureComponents/System/SystemStatus";
|
||||
import { PerformanceSummary } from "@/app/_components/FeatureComponents/System/PerformanceSummary";
|
||||
import { Sidebar } from "@/app/_components/FeatureComponents/Layout/Sidebar";
|
||||
import { ClockIcon, HardDriveIcon, CpuIcon, MonitorIcon, WifiHighIcon } from "@phosphor-icons/react";
|
||||
import { Clock, HardDrive, Cpu, Monitor, Wifi } from "lucide-react";
|
||||
|
||||
interface SystemInfoType {
|
||||
hostname: string;
|
||||
@@ -170,48 +170,53 @@ export const SystemInfoCard = ({
|
||||
|
||||
const basicInfoItems = [
|
||||
{
|
||||
icon: ClockIcon,
|
||||
icon: Clock,
|
||||
label: t("sidebar.uptime"),
|
||||
value: systemInfo.uptime,
|
||||
color: "text-orange-500",
|
||||
},
|
||||
];
|
||||
|
||||
const performanceItems = [
|
||||
{
|
||||
icon: HardDriveIcon,
|
||||
icon: HardDrive,
|
||||
label: t("sidebar.memory"),
|
||||
value: `${systemInfo.memory.used} / ${systemInfo.memory.total}`,
|
||||
detail: `${systemInfo.memory.free} free`,
|
||||
status: systemInfo.memory.status,
|
||||
color: "text-cyan-500",
|
||||
showProgress: true,
|
||||
progressValue: systemInfo.memory.usage,
|
||||
},
|
||||
{
|
||||
icon: CpuIcon,
|
||||
icon: Cpu,
|
||||
label: t("sidebar.cpu"),
|
||||
value: systemInfo.cpu.model,
|
||||
detail: `${systemInfo.cpu.cores} cores`,
|
||||
status: systemInfo.cpu.status,
|
||||
color: "text-pink-500",
|
||||
showProgress: true,
|
||||
progressValue: systemInfo.cpu.usage,
|
||||
},
|
||||
{
|
||||
icon: MonitorIcon,
|
||||
icon: Monitor,
|
||||
label: t("sidebar.gpu"),
|
||||
value: systemInfo.gpu.model,
|
||||
detail: systemInfo.gpu.memory
|
||||
? `${systemInfo.gpu.memory} VRAM`
|
||||
: systemInfo.gpu.status,
|
||||
status: systemInfo.gpu.status,
|
||||
color: "text-indigo-500",
|
||||
},
|
||||
...(systemInfo.network
|
||||
? [
|
||||
{
|
||||
icon: WifiHighIcon,
|
||||
icon: Wifi,
|
||||
label: t("sidebar.network"),
|
||||
value: `${systemInfo.network.latency}ms`,
|
||||
detail: `${systemInfo.network.latency}ms latency • ${systemInfo.network.speed}`,
|
||||
status: systemInfo.network.status,
|
||||
color: "text-teal-500",
|
||||
},
|
||||
]
|
||||
: []),
|
||||
@@ -259,6 +264,7 @@ export const SystemInfoCard = ({
|
||||
icon={item.icon}
|
||||
label={item.label}
|
||||
value={item.value}
|
||||
color={item.color}
|
||||
variant="basic"
|
||||
/>
|
||||
))}
|
||||
@@ -269,7 +275,7 @@ export const SystemInfoCard = ({
|
||||
<h3 className="text-xs font-semibold text-foreground mb-2 uppercase tracking-wide">
|
||||
{t("sidebar.performanceMetrics")}
|
||||
</h3>
|
||||
<div className="space-y-4">
|
||||
<div className="space-y-2">
|
||||
{performanceItems.map((item) => (
|
||||
<MetricCard
|
||||
key={item.label}
|
||||
@@ -278,6 +284,7 @@ export const SystemInfoCard = ({
|
||||
value={item.value}
|
||||
detail={item.detail}
|
||||
status={item.status}
|
||||
color={item.color}
|
||||
variant="performance"
|
||||
showProgress={item.showProgress}
|
||||
progressValue={item.progressValue}
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
import { cn } from "@/app/_utils/global-utils";
|
||||
import { HTMLAttributes, forwardRef } from "react";
|
||||
import { PulseIcon } from "@phosphor-icons/react";
|
||||
import { Activity } from "lucide-react";
|
||||
import { useTranslations } from "next-intl";
|
||||
|
||||
export interface SystemStatusProps extends HTMLAttributes<HTMLDivElement> {
|
||||
@@ -22,27 +22,27 @@ export const SystemStatus = forwardRef<HTMLDivElement, SystemStatusProps>(
|
||||
switch (lowerStatus) {
|
||||
case "operational":
|
||||
return {
|
||||
bgColor: "bg-background0",
|
||||
borderColor: "ascii-border",
|
||||
dotColor: "bg-status-success",
|
||||
bgColor: "bg-emerald-500/10",
|
||||
borderColor: "border-emerald-500/20",
|
||||
dotColor: "bg-emerald-500",
|
||||
};
|
||||
case "warning":
|
||||
return {
|
||||
bgColor: "bg-background0",
|
||||
borderColor: "ascii-border",
|
||||
dotColor: "bg-status-warning",
|
||||
bgColor: "bg-yellow-500/10",
|
||||
borderColor: "border-yellow-500/20",
|
||||
dotColor: "bg-yellow-500",
|
||||
};
|
||||
case "critical":
|
||||
return {
|
||||
bgColor: "bg-background0",
|
||||
borderColor: "ascii-border",
|
||||
dotColor: "bg-status-error",
|
||||
bgColor: "bg-destructive/10",
|
||||
borderColor: "border-destructive/20",
|
||||
dotColor: "bg-destructive",
|
||||
};
|
||||
default:
|
||||
return {
|
||||
bgColor: "bg-background0",
|
||||
borderColor: "ascii-border",
|
||||
dotColor: "bg-status-success",
|
||||
bgColor: "bg-muted",
|
||||
borderColor: "border-border",
|
||||
dotColor: "bg-muted-foreground",
|
||||
};
|
||||
}
|
||||
};
|
||||
@@ -53,7 +53,7 @@ export const SystemStatus = forwardRef<HTMLDivElement, SystemStatusProps>(
|
||||
<div
|
||||
ref={ref}
|
||||
className={cn(
|
||||
"p-4 glass-card terminal-font",
|
||||
"p-4 border border-border/50 rounded-lg glass-card",
|
||||
config.bgColor,
|
||||
config.borderColor,
|
||||
className
|
||||
@@ -61,10 +61,10 @@ export const SystemStatus = forwardRef<HTMLDivElement, SystemStatusProps>(
|
||||
{...props}
|
||||
>
|
||||
<div className="flex items-center gap-3">
|
||||
<div className={cn("w-3 h-3", config.dotColor)} />
|
||||
<div className={cn("w-3 h-3 rounded-full", config.dotColor)} />
|
||||
<div className="flex-1">
|
||||
<div className="flex items-center gap-2">
|
||||
<PulseIcon className="h-4 w-4 text-muted-foreground" />
|
||||
<Activity className="h-4 w-4 text-muted-foreground" />
|
||||
<span className="text-sm font-medium text-foreground">
|
||||
{t("system.systemStatus")}: {status}
|
||||
</span>
|
||||
|
||||
@@ -2,7 +2,7 @@
|
||||
|
||||
import { useState, useEffect } from "react";
|
||||
import { useTranslations } from "next-intl";
|
||||
import { WarningIcon, XIcon } from "@phosphor-icons/react";
|
||||
import { AlertTriangle, X } from "lucide-react";
|
||||
|
||||
export const WrapperScriptWarning = () => {
|
||||
const [isVisible, setIsVisible] = useState(false);
|
||||
@@ -46,7 +46,7 @@ export const WrapperScriptWarning = () => {
|
||||
<div className="bg-amber-500/10 border border-amber-500/20 rounded-lg p-4 mb-4">
|
||||
<div className="flex items-start justify-between">
|
||||
<div className="flex items-start space-x-3">
|
||||
<WarningIcon className="w-5 h-5 text-amber-500 mt-0.5 flex-shrink-0" />
|
||||
<AlertTriangle className="w-5 h-5 text-amber-500 mt-0.5 flex-shrink-0" />
|
||||
<div className="flex-1">
|
||||
<h3 className="text-sm font-medium text-amber-800 dark:text-amber-400">
|
||||
{t("warnings.wrapperScriptModified")}
|
||||
@@ -61,7 +61,7 @@ export const WrapperScriptWarning = () => {
|
||||
className="text-amber-600 dark:text-amber-400 hover:text-amber-800 dark:hover:text-amber-300 transition-colors ml-4"
|
||||
aria-label="Dismiss warning"
|
||||
>
|
||||
<XIcon className="w-4 h-4" />
|
||||
<X className="w-4 h-4" />
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
@@ -1,7 +1,9 @@
|
||||
'use client'
|
||||
|
||||
import { Moon, Sun } from 'lucide-react';
|
||||
import { useTheme } from 'next-themes';
|
||||
import { useEffect, useState } from 'react';
|
||||
import { Button } from '@/app/_components/GlobalComponents/UIElements/Button';
|
||||
|
||||
export const ThemeToggle = () => {
|
||||
const [mounted, setMounted] = useState(false);
|
||||
@@ -11,16 +13,19 @@ export const ThemeToggle = () => {
|
||||
setMounted(true);
|
||||
}, []);
|
||||
|
||||
if (!mounted) return null;
|
||||
|
||||
const isDark = theme === 'dark';
|
||||
if (!mounted) {
|
||||
return null;
|
||||
}
|
||||
|
||||
return (
|
||||
<button
|
||||
onClick={() => setTheme(isDark ? 'light' : 'dark')}
|
||||
className="px-3 py-2 ascii-border terminal-font text-sm bg-background0"
|
||||
<Button
|
||||
variant="ghost"
|
||||
size="icon"
|
||||
onClick={() => setTheme(theme === 'light' ? 'dark' : 'light')}
|
||||
>
|
||||
{isDark ? 'LIGHT' : 'DARK'}
|
||||
</button>
|
||||
<Sun className="h-[1.2rem] w-[1.2rem] rotate-0 scale-100 transition-all dark:-rotate-90 dark:scale-0" />
|
||||
<Moon className="absolute h-[1.2rem] w-[1.2rem] rotate-90 scale-0 transition-all dark:rotate-0 dark:scale-100" />
|
||||
<span className="sr-only">Toggle theme</span>
|
||||
</Button>
|
||||
);
|
||||
};
|
||||
|
||||
@@ -2,7 +2,7 @@
|
||||
|
||||
import { useState, useEffect } from "react";
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
import { CaretDownIcon, UserIcon, XIcon } from "@phosphor-icons/react";
|
||||
import { ChevronDown, User, X } from "lucide-react";
|
||||
import { fetchAvailableUsers } from "@/app/_server/actions/cronjobs";
|
||||
import { useTranslations } from "next-intl";
|
||||
|
||||
@@ -42,7 +42,7 @@ export const UserFilter = ({
|
||||
<div
|
||||
className={`flex items-center gap-2 px-3 py-2 bg-muted/50 rounded-md ${className}`}
|
||||
>
|
||||
<UserIcon className="h-4 w-4 text-muted-foreground" />
|
||||
<User className="h-4 w-4 text-muted-foreground" />
|
||||
<span className="text-sm text-muted-foreground">Loading users...</span>
|
||||
</div>
|
||||
);
|
||||
@@ -57,14 +57,14 @@ export const UserFilter = ({
|
||||
className="flex-1 justify-between"
|
||||
>
|
||||
<div className="flex items-center gap-2">
|
||||
<UserIcon className="h-4 w-4" />
|
||||
<User className="h-4 w-4" />
|
||||
<span className="text-sm">
|
||||
{selectedUser
|
||||
? `${t("common.userWithUsername", { user: selectedUser })}`
|
||||
: t("common.allUsers")}
|
||||
</span>
|
||||
</div>
|
||||
<CaretDownIcon className="h-4 w-4" />
|
||||
<ChevronDown className="h-4 w-4" />
|
||||
</Button>
|
||||
{selectedUser && (
|
||||
<Button
|
||||
@@ -73,20 +73,21 @@ export const UserFilter = ({
|
||||
onClick={() => onUserChange(null)}
|
||||
className="p-2 h-8 w-8 flex-shrink-0"
|
||||
>
|
||||
<XIcon className="h-3 w-3" />
|
||||
<X className="h-3 w-3" />
|
||||
</Button>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{isOpen && (
|
||||
<div className="absolute top-full left-0 right-0 mt-1 bg-background0 border border-border rounded-md shadow-lg z-50 max-h-48 overflow-y-auto tui-scrollbar">
|
||||
<div className="absolute top-full left-0 right-0 mt-1 bg-background border border-border rounded-md shadow-lg z-50 max-h-48 overflow-y-auto">
|
||||
<button
|
||||
onClick={() => {
|
||||
onUserChange(null);
|
||||
setIsOpen(false);
|
||||
}}
|
||||
className={`w-full text-left px-3 py-2 text-sm hover:bg-accent transition-colors ${!selectedUser ? "bg-accent text-accent-foreground" : ""
|
||||
}`}
|
||||
className={`w-full text-left px-3 py-2 text-sm hover:bg-accent transition-colors ${
|
||||
!selectedUser ? "bg-accent text-accent-foreground" : ""
|
||||
}`}
|
||||
>
|
||||
{t("common.allUsers")}
|
||||
</button>
|
||||
@@ -97,8 +98,9 @@ export const UserFilter = ({
|
||||
onUserChange(user);
|
||||
setIsOpen(false);
|
||||
}}
|
||||
className={`w-full text-left px-3 py-2 text-sm hover:bg-accent transition-colors ${selectedUser === user ? "bg-accent text-accent-foreground" : ""
|
||||
}`}
|
||||
className={`w-full text-left px-3 py-2 text-sm hover:bg-accent transition-colors ${
|
||||
selectedUser === user ? "bg-accent text-accent-foreground" : ""
|
||||
}`}
|
||||
>
|
||||
{user}
|
||||
</button>
|
||||
|
||||
@@ -2,7 +2,7 @@
|
||||
|
||||
import { useState, useEffect } from "react";
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
import { CaretDownIcon, UserIcon } from "@phosphor-icons/react";
|
||||
import { ChevronDown, User } from "lucide-react";
|
||||
import { fetchAvailableUsers } from "@/app/_server/actions/cronjobs";
|
||||
|
||||
interface UserSwitcherProps {
|
||||
@@ -43,7 +43,7 @@ export const UserSwitcher = ({
|
||||
<div
|
||||
className={`flex items-center gap-2 px-3 py-2 bg-muted/50 rounded-md ${className}`}
|
||||
>
|
||||
<UserIcon className="h-4 w-4 text-muted-foreground" />
|
||||
<User className="h-4 w-4 text-muted-foreground" />
|
||||
<span className="text-sm text-muted-foreground">Loading users...</span>
|
||||
</div>
|
||||
);
|
||||
@@ -62,14 +62,14 @@ export const UserSwitcher = ({
|
||||
className="w-full justify-between"
|
||||
>
|
||||
<div className="flex items-center gap-2">
|
||||
<UserIcon className="h-4 w-4" />
|
||||
<User className="h-4 w-4" />
|
||||
<span className="text-sm">{selectedUser || "Select user"}</span>
|
||||
</div>
|
||||
<CaretDownIcon className="h-4 w-4" />
|
||||
<ChevronDown className="h-4 w-4" />
|
||||
</Button>
|
||||
|
||||
{isOpen && (
|
||||
<div className="absolute top-full left-0 right-0 mt-1 bg-background0 border border-border rounded-md shadow-lg z-50 max-h-48 overflow-y-auto tui-scrollbar">
|
||||
<div className="absolute top-full left-0 right-0 mt-1 bg-background border border-border rounded-md shadow-lg z-50 max-h-48 overflow-y-auto">
|
||||
{users.map((user) => (
|
||||
<button
|
||||
type="button"
|
||||
@@ -80,8 +80,9 @@ export const UserSwitcher = ({
|
||||
onUserChange(user);
|
||||
setIsOpen(false);
|
||||
}}
|
||||
className={`w-full text-left px-3 py-2 text-sm hover:bg-accent transition-colors ${selectedUser === user ? "bg-accent text-accent-foreground" : ""
|
||||
}`}
|
||||
className={`w-full text-left px-3 py-2 text-sm hover:bg-accent transition-colors ${
|
||||
selectedUser === user ? "bg-accent text-accent-foreground" : ""
|
||||
}`}
|
||||
>
|
||||
{user}
|
||||
</button>
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
"use client";
|
||||
|
||||
import { WarningCircleIcon, XIcon } from "@phosphor-icons/react";
|
||||
import { AlertCircle, X } from "lucide-react";
|
||||
import { JobError, removeJobError } from "@/app/_utils/error-utils";
|
||||
|
||||
interface ErrorBadgeProps {
|
||||
@@ -30,7 +30,7 @@ export const ErrorBadge = ({
|
||||
className="flex items-center gap-1 px-2 py-1 bg-destructive/10 text-destructive border border-destructive/20 rounded text-xs hover:bg-destructive/20 transition-colors"
|
||||
title={error.message}
|
||||
>
|
||||
<WarningCircleIcon className="h-3 w-3" />
|
||||
<AlertCircle className="h-3 w-3" />
|
||||
<span className="hidden sm:inline">Error</span>
|
||||
</button>
|
||||
<button
|
||||
@@ -38,7 +38,7 @@ export const ErrorBadge = ({
|
||||
className="p-1 text-destructive hover:bg-destructive/10 rounded transition-colors"
|
||||
title="Dismiss error"
|
||||
>
|
||||
<XIcon className="h-3 w-3" />
|
||||
<X className="h-3 w-3" />
|
||||
</button>
|
||||
</div>
|
||||
))}
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
import { cn } from "@/app/_utils/global-utils";
|
||||
import { HTMLAttributes, forwardRef } from "react";
|
||||
import { CheckCircleIcon, WarningIcon, XCircleIcon, PulseIcon } from "@phosphor-icons/react";
|
||||
import { CheckCircle, AlertTriangle, XCircle, Activity } from "lucide-react";
|
||||
import { useTranslations } from "next-intl";
|
||||
|
||||
export interface StatusBadgeProps extends HTMLAttributes<HTMLDivElement> {
|
||||
@@ -31,41 +31,46 @@ export const StatusBadge = forwardRef<HTMLDivElement, StatusBadgeProps>(
|
||||
case "operational":
|
||||
case "stable":
|
||||
return {
|
||||
color: "text-status-success",
|
||||
bgColor: "bg-background0",
|
||||
icon: CheckCircleIcon,
|
||||
color: "text-emerald-500",
|
||||
bgColor: "bg-emerald-500/10",
|
||||
borderColor: "border-emerald-500/20",
|
||||
icon: CheckCircle,
|
||||
label: t("system.optimal"),
|
||||
};
|
||||
case "moderate":
|
||||
case "warning":
|
||||
return {
|
||||
color: "text-status-warning",
|
||||
bgColor: "bg-background0",
|
||||
icon: WarningIcon,
|
||||
color: "text-yellow-500",
|
||||
bgColor: "bg-yellow-500/10",
|
||||
borderColor: "border-yellow-500/20",
|
||||
icon: AlertTriangle,
|
||||
label: t("system.warning"),
|
||||
};
|
||||
case "high":
|
||||
case "slow":
|
||||
return {
|
||||
color: "text-status-warning",
|
||||
bgColor: "bg-background0",
|
||||
icon: WarningIcon,
|
||||
color: "text-orange-500",
|
||||
bgColor: "bg-orange-500/10",
|
||||
borderColor: "border-orange-500/20",
|
||||
icon: AlertTriangle,
|
||||
label: t("system.high"),
|
||||
};
|
||||
case "critical":
|
||||
case "poor":
|
||||
case "offline":
|
||||
return {
|
||||
color: "text-status-error",
|
||||
bgColor: "bg-background0",
|
||||
icon: XCircleIcon,
|
||||
color: "text-destructive",
|
||||
bgColor: "bg-destructive/10",
|
||||
borderColor: "border-destructive/20",
|
||||
icon: XCircle,
|
||||
label: t("system.critical"),
|
||||
};
|
||||
default:
|
||||
return {
|
||||
color: "",
|
||||
bgColor: "bg-background0",
|
||||
icon: PulseIcon,
|
||||
color: "text-muted-foreground",
|
||||
bgColor: "bg-muted",
|
||||
borderColor: "border-border",
|
||||
icon: Activity,
|
||||
label: t("system.unknown"),
|
||||
};
|
||||
}
|
||||
@@ -78,8 +83,9 @@ export const StatusBadge = forwardRef<HTMLDivElement, StatusBadgeProps>(
|
||||
<div
|
||||
ref={ref}
|
||||
className={cn(
|
||||
"inline-flex items-center gap-1.5 ascii-border px-2 py-1 terminal-font",
|
||||
"inline-flex items-center gap-1.5 rounded-full border px-2 py-1",
|
||||
config.bgColor,
|
||||
config.borderColor,
|
||||
{
|
||||
"text-xs": size === "sm",
|
||||
"text-sm": size === "md",
|
||||
|
||||
@@ -1,43 +1,72 @@
|
||||
import { cn } from '@/app/_utils/global-utils';
|
||||
import { HTMLAttributes, forwardRef } from 'react';
|
||||
|
||||
export const Card = forwardRef<HTMLDivElement, HTMLAttributes<HTMLDivElement>>(
|
||||
({ className = '', ...props }, ref) => (
|
||||
<div ref={ref} className={`tui-card ${className}`} {...props} />
|
||||
({ className, ...props }, ref) => (
|
||||
<div
|
||||
ref={ref}
|
||||
className={cn(
|
||||
'rounded-lg border bg-card text-card-foreground shadow-sm',
|
||||
className
|
||||
)}
|
||||
{...props}
|
||||
/>
|
||||
)
|
||||
);
|
||||
Card.displayName = 'Card';
|
||||
|
||||
export const CardHeader = forwardRef<HTMLDivElement, HTMLAttributes<HTMLDivElement>>(
|
||||
({ className = '', ...props }, ref) => (
|
||||
<div ref={ref} className={`p-4 border-b border-foreground1 ${className}`} {...props} />
|
||||
const CardHeader = forwardRef<HTMLDivElement, HTMLAttributes<HTMLDivElement>>(
|
||||
({ className, ...props }, ref) => (
|
||||
<div
|
||||
ref={ref}
|
||||
className={cn('flex flex-col space-y-1.5 p-6', className)}
|
||||
{...props}
|
||||
/>
|
||||
)
|
||||
);
|
||||
CardHeader.displayName = 'CardHeader';
|
||||
|
||||
export const CardTitle = forwardRef<HTMLHeadingElement, HTMLAttributes<HTMLHeadingElement>>(
|
||||
({ className = '', ...props }, ref) => (
|
||||
<h3 ref={ref} className={`terminal-font font-bold uppercase ${className}`} {...props} />
|
||||
const CardTitle = forwardRef<HTMLParagraphElement, HTMLAttributes<HTMLHeadingElement>>(
|
||||
({ className, ...props }, ref) => (
|
||||
<h3
|
||||
ref={ref}
|
||||
className={cn(
|
||||
'text-2xl font-semibold leading-none tracking-tight',
|
||||
className
|
||||
)}
|
||||
{...props}
|
||||
/>
|
||||
)
|
||||
);
|
||||
CardTitle.displayName = 'CardTitle';
|
||||
|
||||
export const CardDescription = forwardRef<HTMLParagraphElement, HTMLAttributes<HTMLParagraphElement>>(
|
||||
({ className = '', ...props }, ref) => (
|
||||
<p ref={ref} className={`terminal-font text-sm ${className}`} {...props} />
|
||||
const CardDescription = forwardRef<HTMLParagraphElement, HTMLAttributes<HTMLParagraphElement>>(
|
||||
({ className, ...props }, ref) => (
|
||||
<p
|
||||
ref={ref}
|
||||
className={cn('text-sm text-muted-foreground', className)}
|
||||
{...props}
|
||||
/>
|
||||
)
|
||||
);
|
||||
CardDescription.displayName = 'CardDescription';
|
||||
|
||||
export const CardContent = forwardRef<HTMLDivElement, HTMLAttributes<HTMLDivElement>>(
|
||||
({ className = '', ...props }, ref) => (
|
||||
<div ref={ref} className={`p-4 ${className}`} {...props} />
|
||||
const CardContent = forwardRef<HTMLDivElement, HTMLAttributes<HTMLDivElement>>(
|
||||
({ className, ...props }, ref) => (
|
||||
<div ref={ref} className={cn('p-4 lg:p-6 pt-0', className)} {...props} />
|
||||
)
|
||||
);
|
||||
CardContent.displayName = 'CardContent';
|
||||
|
||||
export const CardFooter = forwardRef<HTMLDivElement, HTMLAttributes<HTMLDivElement>>(
|
||||
({ className = '', ...props }, ref) => (
|
||||
<div ref={ref} className={`flex items-center p-4 border-t border-foreground1 ${className}`} {...props} />
|
||||
const CardFooter = forwardRef<HTMLDivElement, HTMLAttributes<HTMLDivElement>>(
|
||||
({ className, ...props }, ref) => (
|
||||
<div
|
||||
ref={ref}
|
||||
className={cn('flex items-center p-6 pt-0', className)}
|
||||
{...props}
|
||||
/>
|
||||
)
|
||||
);
|
||||
CardFooter.displayName = 'CardFooter';
|
||||
|
||||
export { CardHeader, CardFooter, CardTitle, CardDescription, CardContent };
|
||||
|
||||
@@ -1,12 +1,12 @@
|
||||
import { cn } from "@/app/_utils/global-utils";
|
||||
import { HTMLAttributes, forwardRef, ComponentType } from "react";
|
||||
import { IconProps } from "@phosphor-icons/react";
|
||||
import { HTMLAttributes, forwardRef } from "react";
|
||||
import { LucideIcon } from "lucide-react";
|
||||
import { StatusBadge } from "@/app/_components/GlobalComponents/Badges/StatusBadge";
|
||||
import { ProgressBar } from "@/app/_components/GlobalComponents/UIElements/ProgressBar";
|
||||
import { TruncatedText } from "@/app/_components/GlobalComponents/UIElements/TruncatedText";
|
||||
|
||||
export interface MetricCardProps extends HTMLAttributes<HTMLDivElement> {
|
||||
icon: ComponentType<IconProps>;
|
||||
icon: LucideIcon;
|
||||
label: string;
|
||||
value: string;
|
||||
detail?: string;
|
||||
@@ -27,7 +27,7 @@ export const MetricCard = forwardRef<HTMLDivElement, MetricCardProps>(
|
||||
value,
|
||||
detail,
|
||||
status,
|
||||
color,
|
||||
color = "text-blue-500",
|
||||
variant = "basic",
|
||||
showProgress = false,
|
||||
progressValue = 0,
|
||||
@@ -40,14 +40,14 @@ export const MetricCard = forwardRef<HTMLDivElement, MetricCardProps>(
|
||||
<div
|
||||
ref={ref}
|
||||
className={cn(
|
||||
"flex items-start gap-3 p-3 tui-card-mini transition-colors duration-200 terminal-font",
|
||||
"flex items-start gap-3 p-3 border border-border/50 rounded-lg hover:bg-accent/50 transition-colors duration-200 glass-card-hover",
|
||||
className
|
||||
)}
|
||||
{...props}
|
||||
>
|
||||
<div
|
||||
className={cn(
|
||||
"p-2 ascii-border flex-shrink-0 bg-background0"
|
||||
"p-2 rounded-lg border border-border/50 flex-shrink-0 bg-card/50"
|
||||
)}
|
||||
>
|
||||
<Icon className={cn("h-4 w-4", color)} />
|
||||
@@ -55,7 +55,7 @@ export const MetricCard = forwardRef<HTMLDivElement, MetricCardProps>(
|
||||
|
||||
<div className="flex-1 min-w-0">
|
||||
<div className="flex items-center gap-2 mb-1">
|
||||
<p className="text-xs font-medium uppercase tracking-wide terminal-font">
|
||||
<p className="text-xs font-medium text-muted-foreground uppercase tracking-wide">
|
||||
{label}
|
||||
</p>
|
||||
{status && variant === "performance" && (
|
||||
@@ -67,12 +67,12 @@ export const MetricCard = forwardRef<HTMLDivElement, MetricCardProps>(
|
||||
<TruncatedText
|
||||
text={value}
|
||||
maxLength={40}
|
||||
className="text-sm font-medium terminal-font"
|
||||
className="text-sm font-medium text-foreground"
|
||||
/>
|
||||
</div>
|
||||
|
||||
{detail && (
|
||||
<p className="text-xs mb-2 terminal-font">{detail}</p>
|
||||
<p className="text-xs text-muted-foreground mb-2">{detail}</p>
|
||||
)}
|
||||
|
||||
{showProgress && (
|
||||
|
||||
@@ -1,12 +1,17 @@
|
||||
import { cn } from '@/app/_utils/global-utils';
|
||||
import { InputHTMLAttributes, forwardRef } from 'react';
|
||||
|
||||
export interface InputProps extends InputHTMLAttributes<HTMLInputElement> {}
|
||||
export interface InputProps extends InputHTMLAttributes<HTMLInputElement> { }
|
||||
|
||||
export const Input = forwardRef<HTMLInputElement, InputProps>(
|
||||
({ className = '', ...props }, ref) => {
|
||||
({ className, type, ...props }, ref) => {
|
||||
return (
|
||||
<input
|
||||
className={`terminal-font ascii-border px-3 py-2 bg-background0 w-full ${className}`}
|
||||
type={type}
|
||||
className={cn(
|
||||
'flex h-10 w-full rounded-md border border-input bg-background px-3 py-2 text-sm ring-offset-background file:border-0 file:bg-transparent file:text-sm file:font-medium placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:cursor-not-allowed disabled:opacity-50',
|
||||
className
|
||||
)}
|
||||
ref={ref}
|
||||
{...props}
|
||||
/>
|
||||
|
||||
@@ -1,46 +0,0 @@
|
||||
"use client";
|
||||
|
||||
import { AsteriskIcon, TerminalIcon } from "@phosphor-icons/react";
|
||||
|
||||
interface LogoProps {
|
||||
size?: number;
|
||||
showGlow?: boolean;
|
||||
}
|
||||
|
||||
export const Logo = ({ size = 48, showGlow = false }: LogoProps) => {
|
||||
const iconSize = size * 0.8;
|
||||
const asteriskSize = size * 0.4;
|
||||
const asteriskOffset = size * 0.08;
|
||||
|
||||
return (
|
||||
<div
|
||||
className="relative flex items-center justify-center flex-shrink-0"
|
||||
style={{ width: size, height: size }}
|
||||
>
|
||||
{showGlow && (
|
||||
<div
|
||||
className="absolute inset-0 bg-gradient-to-br from-primary/20 via-primary/10 to-transparent blur-xl"
|
||||
style={{ width: size, height: size }}
|
||||
/>
|
||||
)}
|
||||
|
||||
<div className="relative z-10 flex items-center justify-center w-full h-full">
|
||||
<TerminalIcon
|
||||
className="text-primary drop-shadow-[0_0_10px_rgba(var(--primary-rgb),0.4)]"
|
||||
weight="duotone"
|
||||
style={{ width: iconSize, height: iconSize }}
|
||||
/>
|
||||
<AsteriskIcon
|
||||
className="text-primary absolute drop-shadow-[0_0_8px_rgba(var(--primary-rgb),0.6)]"
|
||||
weight="bold"
|
||||
style={{
|
||||
width: asteriskSize,
|
||||
height: asteriskSize,
|
||||
top: -asteriskOffset,
|
||||
right: -asteriskOffset
|
||||
}}
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
@@ -1,3 +1,4 @@
|
||||
import { cn } from '@/app/_utils/global-utils';
|
||||
import { ButtonHTMLAttributes, forwardRef } from 'react';
|
||||
|
||||
export interface ButtonProps extends ButtonHTMLAttributes<HTMLButtonElement> {
|
||||
@@ -6,31 +7,30 @@ export interface ButtonProps extends ButtonHTMLAttributes<HTMLButtonElement> {
|
||||
}
|
||||
|
||||
export const Button = forwardRef<HTMLButtonElement, ButtonProps>(
|
||||
({ className = '', variant = 'default', size = 'default', children, ...props }, ref) => {
|
||||
const baseClasses = 'terminal-font ascii-border px-4 py-2 cursor-pointer inline-flex items-center justify-center disabled:opacity-50 disabled:cursor-not-allowed';
|
||||
const variantClasses = {
|
||||
default: 'bg-background1 hover:bg-background2',
|
||||
destructive: 'bg-status-error text-white hover:bg-status-error',
|
||||
outline: 'bg-background0 hover:bg-background1',
|
||||
secondary: 'bg-background2 hover:bg-background1',
|
||||
ghost: 'border-0 bg-transparent hover:bg-background1',
|
||||
link: 'border-0 underline bg-transparent',
|
||||
};
|
||||
const sizeClasses = {
|
||||
default: '',
|
||||
sm: 'px-2 py-1 text-sm',
|
||||
lg: 'px-6 py-3',
|
||||
icon: 'p-2',
|
||||
};
|
||||
|
||||
({ className, variant = 'default', size = 'default', ...props }, ref) => {
|
||||
return (
|
||||
<button
|
||||
className={`${baseClasses} ${variantClasses[variant]} ${sizeClasses[size]} ${className}`}
|
||||
className={cn(
|
||||
'inline-flex items-center justify-center whitespace-nowrap rounded-md text-sm font-medium ring-offset-background transition-colors focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:pointer-events-none disabled:opacity-50',
|
||||
{
|
||||
'bg-primary text-primary-foreground hover:bg-primary/90': variant === 'default',
|
||||
'bg-destructive text-destructive-foreground hover:bg-destructive/90': variant === 'destructive',
|
||||
'border border-input bg-background hover:bg-accent hover:text-accent-foreground': variant === 'outline',
|
||||
'bg-secondary text-secondary-foreground hover:bg-secondary/80': variant === 'secondary',
|
||||
'hover:bg-accent hover:text-accent-foreground': variant === 'ghost',
|
||||
'text-primary underline-offset-4 hover:underline': variant === 'link',
|
||||
},
|
||||
{
|
||||
'h-10 px-4 py-2': size === 'default',
|
||||
'h-9 rounded-md px-3': size === 'sm',
|
||||
'h-11 rounded-md px-8': size === 'lg',
|
||||
'h-10 w-10': size === 'icon',
|
||||
},
|
||||
className
|
||||
)}
|
||||
ref={ref}
|
||||
{...props}
|
||||
>
|
||||
{children}
|
||||
</button>
|
||||
/>
|
||||
);
|
||||
}
|
||||
);
|
||||
|
||||
@@ -2,7 +2,7 @@
|
||||
|
||||
import { useState, useRef, useEffect, ReactNode } from "react";
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
import { DotsThreeVerticalIcon } from "@phosphor-icons/react";
|
||||
import { MoreVertical } from "lucide-react";
|
||||
|
||||
const DROPDOWN_HEIGHT = 200;
|
||||
|
||||
@@ -25,7 +25,7 @@ interface DropdownMenuProps {
|
||||
export const DropdownMenu = ({
|
||||
items,
|
||||
triggerLabel,
|
||||
triggerIcon = <DotsThreeVerticalIcon className="h-3 w-3" />,
|
||||
triggerIcon = <MoreVertical className="h-3 w-3" />,
|
||||
triggerClassName = "btn-outline h-8 px-3",
|
||||
onOpenChange,
|
||||
}: DropdownMenuProps) => {
|
||||
@@ -98,8 +98,9 @@ export const DropdownMenu = ({
|
||||
|
||||
{isOpen && (
|
||||
<div
|
||||
className={`absolute right-0 w-56 ascii-border bg-background0 shadow-lg z-[9999] overflow-hidden terminal-font ${positionAbove ? "bottom-full mb-2" : "top-full mt-2"
|
||||
}`}
|
||||
className={`absolute right-0 w-56 rounded-lg border border-border/50 bg-background shadow-lg z-[9999] overflow-hidden ${
|
||||
positionAbove ? "bottom-full mb-2" : "top-full mt-2"
|
||||
}`}
|
||||
>
|
||||
<div className="py-1">
|
||||
{items.map((item, index) => (
|
||||
@@ -107,12 +108,13 @@ export const DropdownMenu = ({
|
||||
key={index}
|
||||
onClick={() => handleItemClick(item)}
|
||||
disabled={item.disabled}
|
||||
className={`w-full flex items-center gap-3 px-4 py-2 text-sm transition-colors ${item.disabled
|
||||
? "opacity-50 cursor-not-allowed"
|
||||
: item.variant === "destructive"
|
||||
? "text-status-error hover:bg-background1"
|
||||
: "hover:bg-background1"
|
||||
}`}
|
||||
className={`w-full flex items-center gap-3 px-4 py-2 text-sm transition-colors ${
|
||||
item.disabled
|
||||
? "opacity-50 cursor-not-allowed"
|
||||
: item.variant === "destructive"
|
||||
? "text-destructive hover:bg-destructive/10"
|
||||
: "text-foreground hover:bg-accent"
|
||||
}`}
|
||||
>
|
||||
{item.icon && (
|
||||
<span className="flex-shrink-0">{item.icon}</span>
|
||||
|
||||
@@ -1,8 +1,9 @@
|
||||
"use client";
|
||||
|
||||
import { useEffect, useRef } from "react";
|
||||
import { XIcon } from "@phosphor-icons/react";
|
||||
import { Button } from "./Button";
|
||||
import { X } from "lucide-react";
|
||||
import { cn } from "@/app/_utils/global-utils";
|
||||
import { Button } from "@/app/_components/GlobalComponents/UIElements/Button";
|
||||
|
||||
interface ModalProps {
|
||||
isOpen: boolean;
|
||||
@@ -25,51 +26,99 @@ export const Modal = ({
|
||||
preventCloseOnClickOutside = false,
|
||||
className = "",
|
||||
}: ModalProps) => {
|
||||
const dialogRef = useRef<HTMLDialogElement>(null);
|
||||
const modalRef = useRef<HTMLDivElement>(null);
|
||||
|
||||
useEffect(() => {
|
||||
const dialog = dialogRef.current;
|
||||
if (!dialog) return;
|
||||
const handleEscape = (e: KeyboardEvent) => {
|
||||
if (e.key === "Escape") {
|
||||
onClose();
|
||||
}
|
||||
};
|
||||
|
||||
if (isOpen) {
|
||||
dialog.showModal();
|
||||
document.addEventListener("keydown", handleEscape);
|
||||
document.body.style.overflow = "hidden";
|
||||
} else {
|
||||
dialog.close();
|
||||
document.body.style.overflow = "unset";
|
||||
}
|
||||
}, [isOpen]);
|
||||
|
||||
return () => {
|
||||
document.removeEventListener("keydown", handleEscape);
|
||||
document.body.style.overflow = "unset";
|
||||
};
|
||||
}, [isOpen, onClose]);
|
||||
|
||||
useEffect(() => {
|
||||
const handleClickOutside = (event: MouseEvent) => {
|
||||
if (
|
||||
modalRef.current &&
|
||||
!modalRef.current.contains(event.target as Node) &&
|
||||
!preventCloseOnClickOutside
|
||||
) {
|
||||
const target = event.target as Element;
|
||||
const isClickingOnModal = target.closest('[data-modal="true"]');
|
||||
const isClickingOnBackdrop =
|
||||
target.classList.contains("modal-backdrop");
|
||||
|
||||
if (isClickingOnBackdrop) {
|
||||
onClose();
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
if (isOpen) {
|
||||
document.addEventListener("mousedown", handleClickOutside);
|
||||
}
|
||||
|
||||
return () => {
|
||||
document.removeEventListener("mousedown", handleClickOutside);
|
||||
};
|
||||
}, [isOpen, onClose, preventCloseOnClickOutside]);
|
||||
|
||||
if (!isOpen) return null;
|
||||
|
||||
const sizeClasses = {
|
||||
sm: "w-[600px]",
|
||||
md: "w-[800px]",
|
||||
lg: "w-[1000px]",
|
||||
xl: "w-[1200px]",
|
||||
"2xl": "w-[1400px]",
|
||||
"3xl": "w-[90vw]",
|
||||
sm: "max-w-md",
|
||||
md: "max-w-lg",
|
||||
lg: "max-w-2xl",
|
||||
xl: "max-w-4xl",
|
||||
"2xl": "max-w-6xl",
|
||||
"3xl": "max-w-8xl",
|
||||
};
|
||||
|
||||
return (
|
||||
<dialog
|
||||
ref={dialogRef}
|
||||
className={`ascii-border terminal-font bg-background0 ${sizeClasses[size]} max-w-[95vw] ${className}`}
|
||||
onClick={(e) => {
|
||||
if (e.target === dialogRef.current && !preventCloseOnClickOutside) {
|
||||
onClose();
|
||||
}
|
||||
}}
|
||||
<div
|
||||
className="fixed inset-0 z-50 flex items-end justify-center sm:items-center p-0 sm:p-4"
|
||||
data-modal="true"
|
||||
>
|
||||
<div className="border-border border-b p-4 flex justify-between items-center bg-background0">
|
||||
<h2 className="terminal-font font-bold uppercase">{title}</h2>
|
||||
{showCloseButton && (
|
||||
<Button variant="ghost" size="icon" onClick={onClose}>
|
||||
<XIcon className="h-4 w-4" />
|
||||
</Button>
|
||||
<div className="absolute inset-0 bg-black/50 backdrop-blur-sm modal-backdrop" />
|
||||
|
||||
<div
|
||||
ref={modalRef}
|
||||
className={cn(
|
||||
"relative w-full bg-card border border-border shadow-lg",
|
||||
"max-h-[85vh]",
|
||||
"sm:rounded-lg sm:max-h-[90vh] sm:w-full",
|
||||
sizeClasses[size],
|
||||
className
|
||||
)}
|
||||
>
|
||||
<div className="flex items-center justify-between p-4 sm:p-6 border-b border-border sticky top-0 bg-card z-10">
|
||||
<h2 className="text-lg font-semibold text-foreground">{title}</h2>
|
||||
{showCloseButton && (
|
||||
<Button
|
||||
variant="ghost"
|
||||
size="icon"
|
||||
onClick={onClose}
|
||||
className="h-8 w-8"
|
||||
>
|
||||
<X className="h-4 w-4" />
|
||||
</Button>
|
||||
)}
|
||||
</div>
|
||||
|
||||
<div className="p-4 sm:p-6 overflow-y-auto max-h-[calc(80vh-100px)]">
|
||||
{children}
|
||||
</div>
|
||||
</div>
|
||||
<div className="p-4 max-h-[70vh] overflow-y-auto tui-scrollbar bg-background0">
|
||||
{children}
|
||||
</div>
|
||||
</dialog>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
@@ -25,29 +25,35 @@ export const ProgressBar = forwardRef<HTMLDivElement, ProgressBarProps>(
|
||||
const percentage = Math.min(Math.max((value / max) * 100, 0), 100);
|
||||
|
||||
const getColorClass = (percentage: number) => {
|
||||
if (percentage >= 90) return "bg-red-600";
|
||||
if (percentage >= 80) return "bg-yellow-600";
|
||||
if (percentage >= 70) return "bg-yellow-600";
|
||||
return "bg-green-600";
|
||||
if (percentage >= 90) return "bg-destructive";
|
||||
if (percentage >= 80) return "bg-orange-500";
|
||||
if (percentage >= 70) return "bg-yellow-500";
|
||||
return "bg-emerald-500";
|
||||
};
|
||||
|
||||
const getGradientClass = (percentage: number) => {
|
||||
return getColorClass(percentage);
|
||||
if (percentage >= 90)
|
||||
return "bg-gradient-to-r from-destructive to-red-600";
|
||||
if (percentage >= 80)
|
||||
return "bg-gradient-to-r from-orange-500 to-orange-600";
|
||||
if (percentage >= 70)
|
||||
return "bg-gradient-to-r from-yellow-500 to-yellow-600";
|
||||
return "bg-gradient-to-r from-emerald-500 to-emerald-600";
|
||||
};
|
||||
|
||||
return (
|
||||
<div ref={ref} className={cn("w-full terminal-font", className)} {...props}>
|
||||
<div ref={ref} className={cn("w-full", className)} {...props}>
|
||||
{showLabel && (
|
||||
<div className="flex justify-between items-center mb-1">
|
||||
<span className="text-xs">Usage</span>
|
||||
<span className="text-xs font-medium">
|
||||
<span className="text-xs text-muted-foreground">Usage</span>
|
||||
<span className="text-xs font-medium text-foreground">
|
||||
{Math.round(percentage)}%
|
||||
</span>
|
||||
</div>
|
||||
)}
|
||||
|
||||
<div
|
||||
className={cn("w-full bg-background2 ascii-border overflow-hidden", {
|
||||
className={cn("w-full bg-muted rounded-full overflow-hidden", {
|
||||
"h-1.5": size === "sm",
|
||||
"h-2": size === "md",
|
||||
"h-3": size === "lg",
|
||||
|
||||
@@ -1,54 +1,35 @@
|
||||
"use client";
|
||||
|
||||
import { cn } from "@/app/_utils/global-utils";
|
||||
|
||||
interface SwitchProps {
|
||||
checked: boolean;
|
||||
onCheckedChange: (checked: boolean) => void;
|
||||
className?: string;
|
||||
disabled?: boolean;
|
||||
id?: string;
|
||||
}
|
||||
|
||||
export const Switch = ({ checked, onCheckedChange, className = "", disabled = false, id }: SwitchProps) => {
|
||||
const handleClick = () => {
|
||||
if (!disabled) {
|
||||
onCheckedChange(!checked);
|
||||
}
|
||||
};
|
||||
|
||||
export const Switch = ({
|
||||
checked,
|
||||
onCheckedChange,
|
||||
className = "",
|
||||
disabled = false,
|
||||
}: SwitchProps) => {
|
||||
return (
|
||||
<div
|
||||
className={`inline-flex items-center ${disabled ? 'opacity-50 cursor-not-allowed' : 'cursor-pointer'} ${className}`}
|
||||
onClick={handleClick}
|
||||
role="checkbox"
|
||||
aria-checked={checked}
|
||||
aria-labelledby={id}
|
||||
tabIndex={disabled ? -1 : 0}
|
||||
onKeyDown={(e) => {
|
||||
if (!disabled && (e.key === ' ' || e.key === 'Enter')) {
|
||||
e.preventDefault();
|
||||
handleClick();
|
||||
}
|
||||
}}
|
||||
<label
|
||||
className={cn(
|
||||
"relative inline-flex items-center cursor-pointer",
|
||||
className
|
||||
)}
|
||||
>
|
||||
<div className="relative w-5 h-5 ascii-border bg-background0 transition-all focus-within:ring-2 focus-within:ring-primary/20 flex items-center justify-center group">
|
||||
{checked && (
|
||||
<svg
|
||||
className="w-3.5 h-3.5 text-primary transition-transform duration-200"
|
||||
viewBox="0 0 12 12"
|
||||
fill="none"
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
>
|
||||
<path
|
||||
d="M10 3L4.5 8.5L2 6"
|
||||
stroke="currentColor"
|
||||
strokeWidth="2"
|
||||
strokeLinecap="square"
|
||||
strokeLinejoin="miter"
|
||||
/>
|
||||
</svg>
|
||||
)}
|
||||
<div className="absolute inset-0 border border-primary/0 group-hover:border-primary/50 transition-colors pointer-events-none" />
|
||||
</div>
|
||||
</div>
|
||||
<input
|
||||
type="checkbox"
|
||||
className="sr-only peer"
|
||||
checked={checked}
|
||||
onChange={(e) => onCheckedChange(e.target.checked)}
|
||||
disabled={disabled}
|
||||
/>
|
||||
<div className="w-9 h-5 bg-muted peer-focus:outline-none peer-focus:ring-2 peer-focus:ring-primary/25 rounded-full peer peer-checked:after:translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:left-[2px] after:bg-white after:rounded-full after:h-4 after:w-4 after:transition-all peer-checked:bg-primary peer-disabled:opacity-50 peer-disabled:cursor-not-allowed"></div>
|
||||
</label>
|
||||
);
|
||||
};
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
"use client";
|
||||
|
||||
import { useEffect, useState } from "react";
|
||||
import { XIcon, CheckCircleIcon, WarningCircleIcon, InfoIcon, WarningIcon } from "@phosphor-icons/react";
|
||||
import { X, CheckCircle, AlertCircle, Info, AlertTriangle } from "lucide-react";
|
||||
import { cn } from "@/app/_utils/global-utils";
|
||||
import { ErrorDetailsModal } from "@/app/_components/FeatureComponents/Modals/ErrorDetailsModal";
|
||||
|
||||
@@ -30,17 +30,19 @@ interface ToastProps {
|
||||
}
|
||||
|
||||
const toastIcons = {
|
||||
success: CheckCircleIcon,
|
||||
error: WarningCircleIcon,
|
||||
info: InfoIcon,
|
||||
warning: WarningIcon,
|
||||
success: CheckCircle,
|
||||
error: AlertCircle,
|
||||
info: Info,
|
||||
warning: AlertTriangle,
|
||||
};
|
||||
|
||||
const toastStyles = {
|
||||
success: "ascii-border bg-background0 text-status-success",
|
||||
error: "ascii-border bg-background0 text-status-error",
|
||||
info: "ascii-border bg-background0 text-status-info",
|
||||
warning: "ascii-border bg-background0 text-status-warning",
|
||||
success:
|
||||
"border-green-500/20 bg-green-500/10 text-green-700 dark:text-green-400",
|
||||
error: "border-red-500/20 bg-red-500/10 text-red-700 dark:text-red-400",
|
||||
info: "border-blue-500/20 bg-blue-500/10 text-blue-700 dark:text-blue-400",
|
||||
warning:
|
||||
"border-yellow-500/20 bg-yellow-500/10 text-yellow-700 dark:text-yellow-400",
|
||||
};
|
||||
|
||||
export const Toast = ({ toast, onRemove, onErrorClick }: ToastProps) => {
|
||||
@@ -60,7 +62,7 @@ export const Toast = ({ toast, onRemove, onErrorClick }: ToastProps) => {
|
||||
return (
|
||||
<div
|
||||
className={cn(
|
||||
"flex items-start gap-3 p-4 terminal-font transition-all duration-300 ease-in-out",
|
||||
"flex items-start gap-3 p-4 rounded-lg border backdrop-blur-md transition-all duration-300 ease-in-out",
|
||||
toastStyles[toast.type],
|
||||
isVisible ? "translate-x-0 opacity-100" : "translate-x-full opacity-0"
|
||||
)}
|
||||
@@ -90,7 +92,7 @@ export const Toast = ({ toast, onRemove, onErrorClick }: ToastProps) => {
|
||||
}}
|
||||
className="flex-shrink-0 p-1 rounded-md hover:bg-black/10 transition-colors"
|
||||
>
|
||||
<XIcon className="h-4 w-4" />
|
||||
<X className="h-4 w-4" />
|
||||
</button>
|
||||
</div>
|
||||
);
|
||||
|
||||
@@ -4,21 +4,7 @@ import { ThemeProvider as NextThemesProvider } from 'next-themes';
|
||||
import { type ThemeProviderProps } from 'next-themes/dist/types';
|
||||
|
||||
export const ThemeProvider = ({ children, ...props }: ThemeProviderProps) => {
|
||||
return (
|
||||
<NextThemesProvider
|
||||
attribute="data-webtui-theme"
|
||||
defaultTheme="light"
|
||||
themes={['light', 'dark']}
|
||||
value={{
|
||||
light: 'catppuccin-latte',
|
||||
dark: 'catppuccin-mocha',
|
||||
}}
|
||||
disableTransitionOnChange
|
||||
{...props}
|
||||
>
|
||||
{children}
|
||||
</NextThemesProvider>
|
||||
);
|
||||
return <NextThemesProvider {...props}>{children}</NextThemesProvider>;
|
||||
}
|
||||
|
||||
|
||||
|
||||
@@ -11,7 +11,7 @@
|
||||
"close": "Close",
|
||||
"refresh": "Refresh",
|
||||
"loading": "Loading",
|
||||
"version": "Version {version}"
|
||||
"version": "{version}"
|
||||
},
|
||||
"cronjobs": {
|
||||
"cronJobs": "Cron Jobs",
|
||||
@@ -188,21 +188,5 @@
|
||||
"warnings": {
|
||||
"wrapperScriptModified": "Wrapper Script Modified",
|
||||
"wrapperScriptModifiedDescription": "Your cron-log-wrapper.sh script has been modified from the official version. This may affect logging functionality. Consider reverting to the official version or ensure your changes don't break the required format for log parsing."
|
||||
},
|
||||
"notFound": {
|
||||
"title": "404 - Page Not Found",
|
||||
"subtitle": "ERROR: The requested resource could not be located",
|
||||
"message": "The page you're looking for doesn't exist. Want to play snake instead?",
|
||||
"gameOver": "GAME OVER",
|
||||
"score": "Score",
|
||||
"highScore": "High Score",
|
||||
"pressToStart": "Press SPACE or tap to start",
|
||||
"pressToRestart": "Press SPACE or tap to restart",
|
||||
"controls": "Controls",
|
||||
"useArrowKeys": "Use arrow keys to move",
|
||||
"tapToMove": "Tap screen edges to move",
|
||||
"goHome": "Return to Dashboard",
|
||||
"pauseGame": "Press P to pause",
|
||||
"paused": "PAUSED"
|
||||
}
|
||||
}
|
||||
@@ -10,7 +10,7 @@
|
||||
"cancel": "Annulla",
|
||||
"refresh": "Aggiorna",
|
||||
"close": "Chiudi",
|
||||
"version": "Versione {version}"
|
||||
"version": "{version}"
|
||||
},
|
||||
"cronjobs": {
|
||||
"cronJobs": "Operazioni Cron",
|
||||
@@ -184,22 +184,5 @@
|
||||
"warnings": {
|
||||
"wrapperScriptModified": "Script Wrapper Modificato",
|
||||
"wrapperScriptModifiedDescription": "Il tuo script cron-log-wrapper.sh è stato modificato dalla versione ufficiale. Questo potrebbe influenzare la funzionalità di logging. Considera di ripristinare la versione ufficiale o assicurati che le tue modifiche non interrompano il formato richiesto per l'analisi dei log."
|
||||
},
|
||||
"notFound": {
|
||||
"title": "404 - Pagina Non Trovata",
|
||||
"subtitle": "ERRORE: La risorsa richiesta non è stata trovata",
|
||||
"message": "La pagina che stai cercando non esiste. Partitella a snake?",
|
||||
"playSnake": "Gioca a Snake mentre sei qui",
|
||||
"gameOver": "GAME OVER",
|
||||
"score": "Punteggio",
|
||||
"highScore": "Punteggio Massimo",
|
||||
"pressToStart": "Premi SPAZIO o tocca per iniziare",
|
||||
"pressToRestart": "Premi SPAZIO o tocca per ricominciare",
|
||||
"controls": "Controlli",
|
||||
"useArrowKeys": "Usa i tasti freccia per muoverti",
|
||||
"tapToMove": "Tocca i bordi dello schermo per muoverti",
|
||||
"goHome": "Torna alla Dashboard",
|
||||
"pauseGame": "Premi P per mettere in pausa",
|
||||
"paused": "IN PAUSA"
|
||||
}
|
||||
}
|
||||
412
app/globals.css
@@ -1,207 +1,325 @@
|
||||
/* eslint-disable */
|
||||
/* @ts-nocheck */
|
||||
|
||||
@import url('https://fonts.googleapis.com/css2?family=Inter:wght@300;400;500;600;700&display=swap');
|
||||
@import url('https://fonts.googleapis.com/css2?family=JetBrains+Mono:wght@300;400;500;600;700&display=swap');
|
||||
@import url('https://fonts.googleapis.com/css2?family=Fira+Code:wght@300;400;500;600;700&display=swap');
|
||||
|
||||
@tailwind base;
|
||||
@tailwind components;
|
||||
@tailwind utilities;
|
||||
|
||||
@layer base {
|
||||
[data-webtui-theme="catppuccin-latte"] {
|
||||
--box-border-color: #9ca0b0;
|
||||
--table-border-color: #9ca0b0;
|
||||
--separator-color: #9ca0b0;
|
||||
:root {
|
||||
--font-sans: 'Inter', system-ui, -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
|
||||
--font-mono: 'JetBrains Mono', 'Fira Code', 'Consolas', 'Monaco', 'Courier New', monospace;
|
||||
--background: 0 0% 100%;
|
||||
--foreground: 240 10% 3.9%;
|
||||
--card: 0 0% 100%;
|
||||
--card-foreground: 240 10% 3.9%;
|
||||
--popover: 0 0% 100%;
|
||||
--popover-foreground: 240 10% 3.9%;
|
||||
--primary: 280 100% 60%;
|
||||
--primary-foreground: 0 0% 98%;
|
||||
--secondary: 240 4.8% 95.9%;
|
||||
--secondary-foreground: 240 5.9% 10%;
|
||||
--muted: 240 4.8% 95.9%;
|
||||
--muted-foreground: 240 3.8% 46.1%;
|
||||
--accent: 240 4.8% 95.9%;
|
||||
--accent-foreground: 240 5.9% 10%;
|
||||
--destructive: 0 84.2% 60.2%;
|
||||
--destructive-foreground: 210 40% 98%;
|
||||
--border: 240 5.9% 90%;
|
||||
--input: 240 5.9% 90%;
|
||||
--ring: 280 100% 60%;
|
||||
--radius: 0.75rem;
|
||||
--chart-1: 280 100% 60%;
|
||||
--chart-2: 160 84% 39%;
|
||||
--chart-3: 30 100% 50%;
|
||||
--chart-4: 340 100% 50%;
|
||||
--chart-5: 200 100% 50%;
|
||||
}
|
||||
|
||||
[data-webtui-theme="catppuccin-mocha"] {
|
||||
--box-border-color: #313244;
|
||||
--table-border-color: #313244;
|
||||
--separator-color: #313244;
|
||||
}
|
||||
|
||||
* {
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
::-webkit-scrollbar {
|
||||
-webkit-appearance: none;
|
||||
}
|
||||
|
||||
body {
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
overflow-x: hidden;
|
||||
.dark {
|
||||
--background: 240 10% 8%;
|
||||
--foreground: 0 0% 98%;
|
||||
--card: 240 10% 12%;
|
||||
--card-foreground: 0 0% 98%;
|
||||
--popover: 240 10% 12%;
|
||||
--popover-foreground: 0 0% 98%;
|
||||
--primary: 280 100% 40%;
|
||||
--primary-foreground: 240 10% 8%;
|
||||
--secondary: 240 3.7% 18%;
|
||||
--secondary-foreground: 0 0% 98%;
|
||||
--muted: 240 3.7% 18%;
|
||||
--muted-foreground: 240 5% 64.9%;
|
||||
--accent: 240 3.7% 18%;
|
||||
--accent-foreground: 0 0% 98%;
|
||||
--destructive: 0 62.8% 30.6%;
|
||||
--destructive-foreground: 0 0% 98%;
|
||||
--border: 240 3.7% 25%;
|
||||
--input: 240 3.7% 18%;
|
||||
--ring: 280 100% 40%;
|
||||
--chart-1: 280 100% 40%;
|
||||
--chart-2: 160 84% 30%;
|
||||
--chart-3: 30 100% 35%;
|
||||
--chart-4: 340 100% 35%;
|
||||
--chart-5: 200 100% 35%;
|
||||
}
|
||||
}
|
||||
|
||||
@layer utilities {
|
||||
.ascii-border {
|
||||
border: 1px solid var(--box-border-color, var(--foreground2));
|
||||
border-radius: 0;
|
||||
}
|
||||
.hide-scrollbar::-webkit-scrollbar {
|
||||
display: none;
|
||||
}
|
||||
|
||||
.border-border {
|
||||
border-color: var(--box-border-color, var(--foreground2))!important;
|
||||
}
|
||||
.hide-scrollbar {
|
||||
scrollbar-width: none;
|
||||
}
|
||||
|
||||
.tui-scrollbar {
|
||||
scrollbar-width: auto !important;
|
||||
padding-right: 15px;
|
||||
}
|
||||
.overflow-y-auto {
|
||||
padding-right: 1em;
|
||||
}
|
||||
|
||||
.tui-scrollbar::-webkit-scrollbar {
|
||||
-webkit-appearance: none;
|
||||
width: 7px;
|
||||
height: 10px;
|
||||
background-color: var(--background1) !important;
|
||||
}
|
||||
.overflow-y-auto::-webkit-scrollbar {
|
||||
width: 4px;
|
||||
}
|
||||
.overflow-y-auto::-webkit-scrollbar-track {
|
||||
background: transparent;
|
||||
}
|
||||
.overflow-y-auto::-webkit-scrollbar-thumb {
|
||||
background-color: hsl(var(--primary) / 0.8);
|
||||
border-radius: 3px;
|
||||
}
|
||||
.overflow-y-auto::-webkit-scrollbar-thumb:hover {
|
||||
background-color: hsl(var(--primary) / 0.1);
|
||||
}
|
||||
|
||||
.tui-scrollbar::-webkit-scrollbar-thumb {
|
||||
background-color: var(--background0) !important;
|
||||
border: 1px solid var(--box-border-color, var(--foreground2));
|
||||
visibility: visible !important;
|
||||
opacity: 1 !important;
|
||||
min-height: 40px !important;
|
||||
height: 40px !important;
|
||||
@layer base {
|
||||
* {
|
||||
@apply border-border;
|
||||
}
|
||||
|
||||
.tui-scrollbar::-webkit-scrollbar-thumb:hover {
|
||||
background-color: var(--primary) !important;
|
||||
box-shadow: 0 0 8px var(--primary);
|
||||
body {
|
||||
@apply bg-background text-foreground;
|
||||
position: relative;
|
||||
overflow-x: hidden;
|
||||
font-family: var(--font-sans);
|
||||
font-feature-settings: "cv02", "cv03", "cv04", "cv11";
|
||||
font-variation-settings: normal;
|
||||
}
|
||||
|
||||
.tui-scrollbar::-webkit-scrollbar-track {
|
||||
background: var(--background1);
|
||||
border-left: 1px solid var(--box-border-color, var(--foreground2));
|
||||
|
||||
code, pre, .font-mono {
|
||||
font-family: var(--font-mono);
|
||||
font-feature-settings: "liga" 1, "calt" 1;
|
||||
}
|
||||
|
||||
.tui-scrollbar::-webkit-scrollbar-corner {
|
||||
background: var(--background1);
|
||||
|
||||
.brand-text {
|
||||
font-family: var(--font-mono);
|
||||
font-weight: 600;
|
||||
letter-spacing: -0.025em;
|
||||
}
|
||||
|
||||
.bg-background0 {
|
||||
background-color: var(--background0) !important;
|
||||
|
||||
h1, h2, h3, h4, h5, h6 {
|
||||
font-family: var(--font-sans);
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
.bg-background1 {
|
||||
background-color: var(--background1) !important;
|
||||
|
||||
button, input, textarea, select {
|
||||
font-family: var(--font-sans);
|
||||
}
|
||||
|
||||
.bg-background2 {
|
||||
background-color: var(--background2) !important;
|
||||
}
|
||||
|
||||
.border-foreground1 {
|
||||
border-color: var(--box-border-color, var(--foreground2)) !important;
|
||||
}
|
||||
|
||||
.text-foreground0 {
|
||||
color: var(--foreground0) !important;
|
||||
}
|
||||
|
||||
.sr-only {
|
||||
position: absolute;
|
||||
width: 1px;
|
||||
height: 1px;
|
||||
padding: 0;
|
||||
margin: -1px;
|
||||
overflow: hidden;
|
||||
clip: rect(0, 0, 0, 0);
|
||||
white-space: nowrap;
|
||||
border-width: 0;
|
||||
|
||||
.terminal-text {
|
||||
font-family: var(--font-mono);
|
||||
font-size: 0.875rem;
|
||||
line-height: 1.5;
|
||||
}
|
||||
}
|
||||
|
||||
@layer components {
|
||||
.tui-card {
|
||||
background: var(--background0) !important;
|
||||
border: 1px solid var(--box-border-color, var(--foreground2)) !important;
|
||||
box-shadow: 8px 4px 0 var(--box-border-color, var(--foreground2));
|
||||
border-radius: 0 !important;
|
||||
.hero-gradient {
|
||||
background: linear-gradient(135deg,
|
||||
hsl(280 100% 60% / 0.1) 0%,
|
||||
hsl(160 84% 39% / 0.05) 25%,
|
||||
hsl(30 100% 50% / 0.05) 50%,
|
||||
hsl(340 100% 50% / 0.05) 75%,
|
||||
hsl(200 100% 50% / 0.1) 100%);
|
||||
}
|
||||
|
||||
.tui-card-mini {
|
||||
background: var(--background0) !important;
|
||||
border: 1px solid var(--box-border-color, var(--foreground2)) !important;
|
||||
box-shadow: 2px 4px 0 var(--box-border-color, var(--foreground2));
|
||||
border-radius: 0 !important;
|
||||
.hero-gradient::before {
|
||||
content: '';
|
||||
position: absolute;
|
||||
top: 0;
|
||||
left: 0;
|
||||
right: 0;
|
||||
bottom: 0;
|
||||
background:
|
||||
radial-gradient(circle at 20% 80%, hsl(280 100% 60% / 0.15) 0%, transparent 50%),
|
||||
radial-gradient(circle at 80% 20%, hsl(160 84% 39% / 0.15) 0%, transparent 50%),
|
||||
radial-gradient(circle at 40% 40%, hsl(340 100% 50% / 0.1) 0%, transparent 50%);
|
||||
pointer-events: none;
|
||||
}
|
||||
|
||||
.terminal-log {
|
||||
background: var(--background0) !important;
|
||||
border: 1px solid var(--box-border-color, var(--foreground2)) !important;
|
||||
font-family: 'JetBrains Mono', monospace;
|
||||
font-size: 0.875rem;
|
||||
line-height: 1.5;
|
||||
padding: 1rem;
|
||||
color: var(--foreground0);
|
||||
.dark .hero-gradient::before {
|
||||
background:
|
||||
radial-gradient(circle at 20% 80%, hsl(280 100% 50% / 0.08) 0%, transparent 50%),
|
||||
radial-gradient(circle at 80% 20%, hsl(160 84% 35% / 0.08) 0%, transparent 50%),
|
||||
radial-gradient(circle at 40% 40%, hsl(340 100% 45% / 0.06) 0%, transparent 50%);
|
||||
}
|
||||
|
||||
.text-status-info {
|
||||
color: var(--mauve);
|
||||
.glass-card {
|
||||
@apply backdrop-blur-md bg-card/80 border border-border/50;
|
||||
}
|
||||
|
||||
.text-status-warning {
|
||||
color: var(--yellow);
|
||||
.dark .glass-card {
|
||||
@apply backdrop-blur-md bg-card/80 border border-border/70;
|
||||
}
|
||||
|
||||
.text-status-success {
|
||||
color: var(--green);
|
||||
.glass-card-hover {
|
||||
@apply glass-card;
|
||||
}
|
||||
|
||||
.text-status-error {
|
||||
color: var(--red);
|
||||
.brand-gradient {
|
||||
@apply bg-gradient-to-r from-purple-500 via-pink-500 to-orange-500 bg-clip-text text-transparent;
|
||||
}
|
||||
|
||||
.bg-status-info {
|
||||
background-color: var(--mauve);
|
||||
.dark .brand-gradient {
|
||||
@apply bg-gradient-to-r from-purple-600 via-pink-600 to-orange-600 bg-clip-text text-transparent;
|
||||
}
|
||||
|
||||
.bg-status-warning {
|
||||
background-color: var(--yellow);
|
||||
.brand-gradient-alt {
|
||||
@apply bg-gradient-to-r from-cyan-500 via-blue-500 to-purple-500 bg-clip-text text-transparent;
|
||||
}
|
||||
|
||||
.bg-status-success {
|
||||
background-color: var(--green);
|
||||
.dark .brand-gradient-alt {
|
||||
@apply bg-gradient-to-r from-cyan-600 via-blue-600 to-purple-600 bg-clip-text text-transparent;
|
||||
}
|
||||
|
||||
.bg-status-error {
|
||||
background-color: var(--red);
|
||||
.glow-primary {
|
||||
box-shadow: none;
|
||||
}
|
||||
|
||||
dialog {
|
||||
background: var(--background0) !important;
|
||||
margin: auto;
|
||||
padding: 0;
|
||||
width: 90vw;
|
||||
max-height: 90vh;
|
||||
.glow-primary:hover {
|
||||
box-shadow: none;
|
||||
}
|
||||
|
||||
dialog::backdrop {
|
||||
background: rgba(0, 0, 0, 0.8) !important;
|
||||
.glow-cyan {
|
||||
box-shadow: none;
|
||||
}
|
||||
|
||||
dialog[open] {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
.glow-orange {
|
||||
box-shadow: none;
|
||||
}
|
||||
|
||||
/* Sidebar layout adjustment */
|
||||
.no-sidebar main {
|
||||
margin-left: 0 !important;
|
||||
.glow-pink {
|
||||
box-shadow: none;
|
||||
}
|
||||
|
||||
body:not(.sidebar-collapsed) main {
|
||||
margin-left: 0;
|
||||
.status-error {
|
||||
@apply bg-red-500/20 text-red-700 dark:text-red-400 border-red-500/30;
|
||||
}
|
||||
|
||||
@media (min-width: 1024px) {
|
||||
body:not(.sidebar-collapsed) main {
|
||||
margin-left: 320px;
|
||||
}
|
||||
.custom-scrollbar {
|
||||
scrollbar-width: thin;
|
||||
scrollbar-color: hsl(var(--muted-foreground) / 0.3) transparent;
|
||||
}
|
||||
|
||||
body.sidebar-collapsed main {
|
||||
margin-left: 64px;
|
||||
}
|
||||
.custom-scrollbar::-webkit-scrollbar {
|
||||
width: 6px;
|
||||
}
|
||||
|
||||
.no-sidebar main {
|
||||
margin-left: 0 !important;
|
||||
}
|
||||
.custom-scrollbar::-webkit-scrollbar-track {
|
||||
background: transparent;
|
||||
}
|
||||
|
||||
.custom-scrollbar::-webkit-scrollbar-thumb {
|
||||
background-color: hsl(var(--muted-foreground) / 0.3);
|
||||
border-radius: 3px;
|
||||
}
|
||||
|
||||
.custom-scrollbar::-webkit-scrollbar-thumb:hover {
|
||||
background-color: hsl(var(--muted-foreground) / 0.5);
|
||||
}
|
||||
|
||||
.tooltip {
|
||||
@apply absolute z-50 px-3 py-2 text-sm text-white bg-gray-900 rounded-lg shadow-lg opacity-0 invisible transition-all duration-200;
|
||||
}
|
||||
|
||||
.tooltip.show {
|
||||
@apply opacity-100 visible;
|
||||
}
|
||||
|
||||
.text-responsive {
|
||||
@apply text-sm sm:text-base lg:text-lg;
|
||||
}
|
||||
|
||||
.text-responsive-lg {
|
||||
@apply text-base sm:text-lg lg:text-xl;
|
||||
}
|
||||
|
||||
.text-responsive-xl {
|
||||
@apply text-lg sm:text-xl lg:text-2xl;
|
||||
}
|
||||
|
||||
.btn-primary {
|
||||
@apply bg-gradient-to-r from-purple-500 to-pink-500 text-white hover:from-purple-600 hover:to-pink-600 transition-all;
|
||||
}
|
||||
|
||||
.dark .btn-primary {
|
||||
@apply bg-gradient-to-r from-purple-600 to-pink-600 text-white hover:from-purple-700 hover:to-pink-700 transition-all;
|
||||
}
|
||||
|
||||
.btn-secondary {
|
||||
@apply bg-gradient-to-r from-cyan-500 to-blue-500 text-white hover:from-cyan-600 hover:to-blue-600 transition-all;
|
||||
}
|
||||
|
||||
.dark .btn-secondary {
|
||||
@apply bg-gradient-to-r from-cyan-600 to-blue-600 text-white hover:from-cyan-700 hover:to-blue-700 transition-all;
|
||||
}
|
||||
|
||||
.btn-outline {
|
||||
@apply border border-border bg-background hover:bg-accent hover:text-accent-foreground transition-colors;
|
||||
}
|
||||
|
||||
.btn-ghost {
|
||||
@apply hover:bg-accent hover:text-accent-foreground transition-colors;
|
||||
}
|
||||
|
||||
.btn-destructive {
|
||||
@apply bg-gradient-to-r from-red-500 to-pink-500 text-white hover:from-red-600 hover:to-pink-600 transition-all;
|
||||
}
|
||||
|
||||
.dark .btn-destructive {
|
||||
@apply bg-gradient-to-r from-red-600 to-pink-600 text-white hover:from-red-700 hover:to-pink-700 transition-all;
|
||||
}
|
||||
|
||||
.neon-border {
|
||||
border: 1px solid transparent;
|
||||
background: linear-gradient(white, white) padding-box,
|
||||
linear-gradient(45deg, hsl(280 100% 60%), hsl(160 84% 39%), hsl(30 100% 50%)) border-box;
|
||||
}
|
||||
|
||||
.neon-border-dark {
|
||||
border: 1px solid transparent;
|
||||
background: linear-gradient(hsl(240 10% 3.9%), hsl(240 10% 3.9%)) padding-box,
|
||||
linear-gradient(45deg, hsl(280 100% 70%), hsl(160 84% 45%), hsl(30 100% 60%)) border-box;
|
||||
}
|
||||
}
|
||||
|
||||
@layer utilities {
|
||||
body.sidebar-collapsed main.lg\:ml-80 {
|
||||
margin-left: 4rem !important;
|
||||
}
|
||||
}
|
||||
|
||||
.dropdown-overflow-fix {
|
||||
overflow: visible;
|
||||
position: relative;
|
||||
}
|
||||
|
||||
.dropdown-overflow-fix > * {
|
||||
position: relative;
|
||||
}
|
||||
|
||||
.dropdown-overflow-fix .dropdown-container,
|
||||
.dropdown-overflow-fix [class*="dropdown"] {
|
||||
overflow: visible !important;
|
||||
}
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
import type { Metadata } from "next";
|
||||
import { JetBrains_Mono } from "next/font/google";
|
||||
import { JetBrains_Mono, Inter } from "next/font/google";
|
||||
import "@/app/globals.css";
|
||||
import { ThemeProvider } from "@/app/_providers/ThemeProvider";
|
||||
import { ServiceWorkerRegister } from "@/app/_components/FeatureComponents/PWA/ServiceWorkerRegister";
|
||||
@@ -13,6 +13,12 @@ const jetbrainsMono = JetBrains_Mono({
|
||||
display: "swap",
|
||||
});
|
||||
|
||||
const inter = Inter({
|
||||
subsets: ["latin"],
|
||||
variable: "--font-sans",
|
||||
display: "swap",
|
||||
});
|
||||
|
||||
export const metadata: Metadata = {
|
||||
title: "Cr*nMaster - Cron Management made easy",
|
||||
description:
|
||||
@@ -52,7 +58,7 @@ export default async function RootLayout({
|
||||
messages = await loadTranslationMessages(locale);
|
||||
|
||||
return (
|
||||
<html lang="en" suppressHydrationWarning data-webtui-theme="catppuccin-latte">
|
||||
<html lang="en" suppressHydrationWarning>
|
||||
<head>
|
||||
<meta name="application-name" content="Cr*nMaster" />
|
||||
<meta name="apple-mobile-web-app-capable" content="yes" />
|
||||
@@ -60,23 +66,15 @@ export default async function RootLayout({
|
||||
<meta name="apple-mobile-web-app-title" content="Cr*nMaster" />
|
||||
<meta name="mobile-web-app-capable" content="yes" />
|
||||
<link rel="apple-touch-icon" href="/logo.png" />
|
||||
<link rel="stylesheet" href="/webtui/base.css" />
|
||||
<link rel="stylesheet" href="/webtui/theme-catppuccin.css" />
|
||||
<script
|
||||
dangerouslySetInnerHTML={{
|
||||
__html: `
|
||||
(function() {
|
||||
const theme = localStorage.getItem('theme') || 'light';
|
||||
const webtui = theme === 'dark' ? 'catppuccin-mocha' : 'catppuccin-latte';
|
||||
document.documentElement.setAttribute('data-webtui-theme', webtui);
|
||||
})();
|
||||
`,
|
||||
}}
|
||||
/>
|
||||
</head>
|
||||
<body className={`${jetbrainsMono.variable} terminal-font`}>
|
||||
<body className={`${inter.variable} ${jetbrainsMono.variable} font-sans`}>
|
||||
<NextIntlClientProvider locale={locale} messages={messages}>
|
||||
<ThemeProvider>
|
||||
<ThemeProvider
|
||||
attribute="class"
|
||||
defaultTheme="dark"
|
||||
enableSystem
|
||||
disableTransitionOnChange
|
||||
>
|
||||
{children}
|
||||
</ThemeProvider>
|
||||
<ServiceWorkerRegister />
|
||||
|
||||
@@ -1,24 +1,12 @@
|
||||
"use client";
|
||||
import { Asterisk, Terminal } from "lucide-react";
|
||||
|
||||
import { AsteriskIcon, TerminalIcon } from "@phosphor-icons/react";
|
||||
|
||||
export default function Logo() {
|
||||
export default async function Logo() {
|
||||
return (
|
||||
<div className="min-h-screen bg-background0 flex items-center justify-center p-8">
|
||||
<div className="relative">
|
||||
<div className="w-[600px] h-[600px] ascii-border bg-background1 p-16 flex items-center justify-center">
|
||||
<div className="absolute w-[600px] h-[600px] bg-gradient-to-br from-primary/20 via-primary/10 to-transparent blur-3xl" />
|
||||
|
||||
<div className="relative z-10 flex items-center justify-center w-full h-full">
|
||||
<TerminalIcon
|
||||
className="h-80 w-80 text-primary drop-shadow-[0_0_40px_rgba(var(--primary-rgb),0.6)]"
|
||||
weight="duotone"
|
||||
/>
|
||||
<AsteriskIcon
|
||||
className="h-40 w-40 text-primary absolute -top-4 -right-4 drop-shadow-[0_0_30px_rgba(var(--primary-rgb),0.8)]"
|
||||
weight="bold"
|
||||
/>
|
||||
</div>
|
||||
<div className="m-auto mt-20 relative w-[600px] h-[600px]">
|
||||
<div className="p-3 bg-gradient-to-br from-purple-500 via-pink-500 to-orange-500 rounded-[200px] w-full h-full">
|
||||
<div className="relative">
|
||||
<Terminal className="h-[350px] w-[350px] text-white relative top-[120px] left-[120px]" />
|
||||
<Asterisk className="h-[200px] w-[200px] text-white absolute top-14 right-[90px]" />
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
@@ -1,117 +0,0 @@
|
||||
import Link from "next/link";
|
||||
import { getTranslations } from "@/app/_server/actions/translations";
|
||||
import { SnakeGame } from "@/app/_components/FeatureComponents/Games/SnakeGame";
|
||||
import { Logo } from "@/app/_components/GlobalComponents/Logo/Logo";
|
||||
import { SystemInfoCard } from "@/app/_components/FeatureComponents/System/SystemInfo";
|
||||
import { ThemeToggle } from "@/app/_components/FeatureComponents/Theme/ThemeToggle";
|
||||
import { LogoutButton } from "@/app/_components/FeatureComponents/LoginForm/LogoutButton";
|
||||
import { ToastContainer } from "@/app/_components/GlobalComponents/UIElements/Toast";
|
||||
import { PWAInstallPrompt } from "@/app/_components/FeatureComponents/PWA/PWAInstallPrompt";
|
||||
import { SSEProvider } from "@/app/_contexts/SSEContext";
|
||||
import { readFileSync } from "fs";
|
||||
import path from "path";
|
||||
|
||||
export default async function NotFound() {
|
||||
const t = await getTranslations();
|
||||
const liveUpdatesEnabled =
|
||||
(typeof process.env.LIVE_UPDATES === "boolean" &&
|
||||
process.env.LIVE_UPDATES === true) ||
|
||||
process.env.LIVE_UPDATES !== "false";
|
||||
|
||||
const packageJsonPath = path.join(process.cwd(), "package.json");
|
||||
const packageJson = JSON.parse(readFileSync(packageJsonPath, "utf-8"));
|
||||
const version = packageJson.version;
|
||||
|
||||
const initialSystemInfo = {
|
||||
hostname: "Loading...",
|
||||
platform: "Loading...",
|
||||
uptime: "Loading...",
|
||||
memory: {
|
||||
total: "0 B",
|
||||
used: "0 B",
|
||||
free: "0 B",
|
||||
usage: 0,
|
||||
status: "Loading",
|
||||
},
|
||||
cpu: {
|
||||
model: "Loading...",
|
||||
cores: 0,
|
||||
usage: 0,
|
||||
status: "Loading",
|
||||
},
|
||||
gpu: {
|
||||
model: "Loading...",
|
||||
status: "Loading",
|
||||
},
|
||||
disk: {
|
||||
total: "0 B",
|
||||
used: "0 B",
|
||||
free: "0 B",
|
||||
usage: 0,
|
||||
status: "Loading",
|
||||
},
|
||||
systemStatus: {
|
||||
overall: "Loading",
|
||||
details: "Fetching system information...",
|
||||
},
|
||||
};
|
||||
|
||||
const bodyClass = process.env.DISABLE_SYSTEM_STATS === "true" ? "no-sidebar" : "";
|
||||
|
||||
return (
|
||||
<SSEProvider liveUpdatesEnabled={liveUpdatesEnabled}>
|
||||
<div className={`min-h-screen bg-background0 ${bodyClass}`}>
|
||||
<header className="border-border border-b sticky top-0 z-20 bg-background0 lg:h-[90px]">
|
||||
<div className="container mx-auto px-4 py-4">
|
||||
<div className="flex items-center justify-between lg:justify-center">
|
||||
<div className="flex items-center gap-4">
|
||||
<Logo size={48} showGlow={true} />
|
||||
<div>
|
||||
<h1 className="text-xl sm:text-2xl lg:text-3xl font-bold terminal-font uppercase">
|
||||
Cr*nMaster
|
||||
</h1>
|
||||
<p className="text-xs terminal-font flex items-center gap-2">
|
||||
{t("common.version").replace("{version}", version)}
|
||||
</p>
|
||||
</div>
|
||||
</div>
|
||||
{process.env.AUTH_PASSWORD && (
|
||||
<div className="lg:absolute lg:right-10">
|
||||
<LogoutButton />
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
</header>
|
||||
|
||||
{process.env.DISABLE_SYSTEM_STATS !== "true" && (
|
||||
<SystemInfoCard systemInfo={initialSystemInfo} />
|
||||
)}
|
||||
|
||||
<main className="transition-all duration-300">
|
||||
<div className="px-4 py-8 lg:px-8">
|
||||
<div className="text-center mt-6 mb-12">
|
||||
<div className="text-6xl font-bold terminal-font text-status-error mb-2">404</div>
|
||||
<p className="terminal-font text-sm mb-4">{t("notFound.message")}</p>
|
||||
<Link
|
||||
href="/"
|
||||
className="ascii-border bg-background1 hover:bg-background2 px-4 py-2 terminal-font uppercase font-bold transition-colors text-sm inline-block"
|
||||
>
|
||||
{t("notFound.goHome")}
|
||||
</Link>
|
||||
</div>
|
||||
|
||||
<SnakeGame />
|
||||
</div>
|
||||
</main>
|
||||
|
||||
<ToastContainer />
|
||||
|
||||
<div className="flex items-center gap-2 fixed bottom-4 left-4 lg:right-4 lg:left-auto z-10 bg-background0 ascii-border p-1">
|
||||
<ThemeToggle />
|
||||
<PWAInstallPrompt />
|
||||
</div>
|
||||
</div>
|
||||
</SSEProvider>
|
||||
);
|
||||
}
|
||||
75
app/page.tsx
@@ -9,9 +9,6 @@ import { PWAInstallPrompt } from "@/app/_components/FeatureComponents/PWA/PWAIns
|
||||
import { WrapperScriptWarning } from "@/app/_components/FeatureComponents/System/WrapperScriptWarning";
|
||||
import { getTranslations } from "@/app/_server/actions/translations";
|
||||
import { SSEProvider } from "@/app/_contexts/SSEContext";
|
||||
import { Logo } from "@/app/_components/GlobalComponents/Logo/Logo";
|
||||
import { readFileSync } from "fs";
|
||||
import path from "path";
|
||||
|
||||
export const dynamic = "force-dynamic";
|
||||
export const maxDuration = 300;
|
||||
@@ -23,10 +20,6 @@ export default async function Home() {
|
||||
process.env.LIVE_UPDATES === true) ||
|
||||
process.env.LIVE_UPDATES !== "false";
|
||||
|
||||
const packageJsonPath = path.join(process.cwd(), "package.json");
|
||||
const packageJson = JSON.parse(readFileSync(packageJsonPath, "utf-8"));
|
||||
const version = packageJson.version;
|
||||
|
||||
const [cronJobs, scripts] = await Promise.all([
|
||||
getCronJobs(),
|
||||
fetchScripts(),
|
||||
@@ -66,48 +59,52 @@ export default async function Home() {
|
||||
},
|
||||
};
|
||||
|
||||
const bodyClass = process.env.DISABLE_SYSTEM_STATS === "true" ? "no-sidebar" : "";
|
||||
|
||||
return (
|
||||
<SSEProvider liveUpdatesEnabled={liveUpdatesEnabled}>
|
||||
<div className={`min-h-screen bg-background0 ${bodyClass}`}>
|
||||
<header className="border-border border-b sticky top-0 z-20 bg-background0 lg:h-[90px]">
|
||||
<div className="container mx-auto px-4 py-4">
|
||||
<div className="flex items-center justify-between lg:justify-center">
|
||||
<div className="flex items-center gap-4">
|
||||
<Logo size={48} showGlow={true} />
|
||||
<div>
|
||||
<h1 className="text-xl sm:text-2xl lg:text-3xl font-bold terminal-font uppercase">
|
||||
Cr*nMaster
|
||||
</h1>
|
||||
<p className="text-xs terminal-font flex items-center gap-2">
|
||||
{t("common.version").replace("{version}", version)}
|
||||
</p>
|
||||
<div className="min-h-screen relative">
|
||||
<div className="hero-gradient absolute inset-0 -z-10"></div>
|
||||
<div className="relative z-10">
|
||||
<header className="border-b border-border/50 bg-background/80 backdrop-blur-md sticky top-0 z-20 shadow-sm lg:h-[90px]">
|
||||
<div className="container mx-auto px-4 py-4">
|
||||
<div className="flex items-center justify-between lg:justify-center">
|
||||
<div className="flex items-center gap-4">
|
||||
<div className="relative">
|
||||
<img src="/logo.png" alt="logo" className="w-14 h-14" />
|
||||
<div className="absolute top-0 right-0 w-3 h-3 bg-emerald-500 rounded-full animate-pulse"></div>
|
||||
</div>
|
||||
<div>
|
||||
<h1 className="text-xl sm:text-2xl lg:text-3xl font-bold brand-gradient brand-text">
|
||||
Cr*nMaster
|
||||
</h1>
|
||||
<p className="text-xs text-muted-foreground font-mono tracking-wide">
|
||||
{t("common.cronManagementMadeEasy")}
|
||||
</p>
|
||||
</div>
|
||||
</div>
|
||||
{process.env.AUTH_PASSWORD && (
|
||||
<div className="lg:absolute lg:right-10">
|
||||
<LogoutButton />
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
{process.env.AUTH_PASSWORD && (
|
||||
<div className="lg:absolute lg:right-10">
|
||||
<LogoutButton />
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
</header>
|
||||
</header>
|
||||
|
||||
{process.env.DISABLE_SYSTEM_STATS !== "true" && (
|
||||
<SystemInfoCard systemInfo={initialSystemInfo} />
|
||||
)}
|
||||
{process.env.DISABLE_SYSTEM_STATS !== "true" && (
|
||||
<SystemInfoCard systemInfo={initialSystemInfo} />
|
||||
)}
|
||||
|
||||
<main className="transition-all duration-300">
|
||||
<div className="px-4 py-8 lg:px-8">
|
||||
<WrapperScriptWarning />
|
||||
<TabbedInterface cronJobs={cronJobs} scripts={scripts} />
|
||||
</div>
|
||||
</main>
|
||||
<main className={`${process.env.DISABLE_SYSTEM_STATS === "true" ? "lg:ml-0" : "lg:ml-80"} transition-all duration-300 ml-0 sidebar-collapsed:lg:ml-16`}>
|
||||
<div className="container mx-auto px-4 py-8 lg:px-8">
|
||||
<WrapperScriptWarning />
|
||||
<TabbedInterface cronJobs={cronJobs} scripts={scripts} />
|
||||
</div>
|
||||
</main>
|
||||
</div>
|
||||
|
||||
<ToastContainer />
|
||||
|
||||
<div className="flex items-center gap-2 fixed bottom-4 left-4 lg:right-4 lg:left-auto z-10 bg-background0 ascii-border p-1">
|
||||
<div className="flex items-center gap-2 fixed bottom-4 left-4 lg:right-4 lg:left-auto z-10 bg-background/80 backdrop-blur-md border border-border/50 rounded-lg p-1">
|
||||
<ThemeToggle />
|
||||
<PWAInstallPrompt />
|
||||
</div>
|
||||
|
||||
@@ -35,12 +35,14 @@ ports:
|
||||
```yaml
|
||||
environment:
|
||||
- NODE_ENV=production
|
||||
- DOCKER=true
|
||||
- NEXT_PUBLIC_CLOCK_UPDATE_INTERVAL=30000
|
||||
- AUTH_PASSWORD=very_strong_password
|
||||
- HOST_CRONTAB_USER=root
|
||||
```
|
||||
|
||||
- **NODE_ENV**: Set to `production` for production deployments
|
||||
- **DOCKER**: Must be `true` when running in Docker
|
||||
- **NEXT_PUBLIC_CLOCK_UPDATE_INTERVAL**: Clock update interval in milliseconds (default: 30000)
|
||||
- **AUTH_PASSWORD**: Strong password for authentication
|
||||
- **HOST_CRONTAB_USER**: User whose crontab to read (default: root, can be comma-separated for multiple users)
|
||||
@@ -153,6 +155,7 @@ services:
|
||||
- "40123:3000"
|
||||
environment:
|
||||
- NODE_ENV=production
|
||||
- DOCKER=true
|
||||
- NEXT_PUBLIC_CLOCK_UPDATE_INTERVAL=30000
|
||||
|
||||
# Localization
|
||||
|
||||
10
package.json
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "cronjob-manager",
|
||||
"version": "2.0.0",
|
||||
"version": "1.5.4",
|
||||
"private": true,
|
||||
"scripts": {
|
||||
"dev": "next dev",
|
||||
@@ -19,13 +19,10 @@
|
||||
"@codemirror/state": "^6.5.2",
|
||||
"@codemirror/theme-one-dark": "^6.1.3",
|
||||
"@codemirror/view": "^6.38.1",
|
||||
"@phosphor-icons/react": "^2.1.10",
|
||||
"@types/node": "^20",
|
||||
"@types/react": "^18",
|
||||
"@types/react-dom": "^18",
|
||||
"@types/react-syntax-highlighter": "^15.5.13",
|
||||
"@webtui/css": "^0.1.5",
|
||||
"@webtui/theme-catppuccin": "^0.0.3",
|
||||
"autoprefixer": "^10.0.1",
|
||||
"bcryptjs": "^2.4.3",
|
||||
"clsx": "^2.0.0",
|
||||
@@ -33,6 +30,7 @@
|
||||
"cron-parser": "^5.3.0",
|
||||
"cronstrue": "^3.2.0",
|
||||
"jose": "^6.1.1",
|
||||
"lucide-react": "^0.294.0",
|
||||
"minimatch": "^10.0.3",
|
||||
"next": "14.2.35",
|
||||
"next-intl": "^4.4.0",
|
||||
@@ -42,7 +40,6 @@
|
||||
"proper-lockfile": "^4.1.2",
|
||||
"react": "^18",
|
||||
"react-dom": "^18",
|
||||
"react-icons": "^5.5.0",
|
||||
"react-syntax-highlighter": "^15.6.1",
|
||||
"systeminformation": "^5.27.11",
|
||||
"tailwind-merge": "^2.0.0",
|
||||
@@ -53,7 +50,6 @@
|
||||
"@types/bcryptjs": "^2.4.6",
|
||||
"@types/minimatch": "^6.0.0",
|
||||
"eslint": "^8",
|
||||
"eslint-config-next": "14.0.4",
|
||||
"postcss-import": "^16.1.1"
|
||||
"eslint-config-next": "14.0.4"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,6 +1,5 @@
|
||||
module.exports = {
|
||||
plugins: {
|
||||
'postcss-import': {},
|
||||
tailwindcss: {},
|
||||
autoprefixer: {},
|
||||
},
|
||||
|
||||
|
Before Width: | Height: | Size: 52 KiB |
|
Before Width: | Height: | Size: 82 KiB After Width: | Height: | Size: 52 KiB |
|
Before Width: | Height: | Size: 33 KiB |
|
Before Width: | Height: | Size: 1.7 KiB After Width: | Height: | Size: 1.7 KiB |
BIN
public/logo.png
|
Before Width: | Height: | Size: 24 KiB After Width: | Height: | Size: 33 KiB |
@@ -1 +0,0 @@
|
||||
@layer base{:root{--background0: #fff;--background1: #ddd;--background2: #bbb;--background3: #999;--foreground0: #000;--foreground1: #444;--foreground2: #888;--font-size: 16px;--line-height: 1.3;--font-weight-bold: 700;--font-weight-normal: 400;--font-family: monospace;--box-border-color: var(--foreground0);--table-border-color: var(--box-border-color);--separator-color: var(--box-border-color);--separator-background: transparent}[data-webtui-theme=dark]{--background0: #000;--background1: #222;--background2: #444;--background3: #666;--foreground0: #fff;--foreground1: #ccc;--foreground2: #999}body,html{background-color:var(--background0);color:var(--foreground0);font-family:var(--font-family);font-size:var(--font-size);font-weight:var(--font-weight-normal);line-height:var(--line-height, 1.5);font-variant-ligatures:common-ligatures}*{box-sizing:border-box;margin:0;padding:0;outline:none}}
|
||||
34
yarn.lock
@@ -1289,11 +1289,6 @@
|
||||
"@parcel/watcher-win32-ia32" "2.5.1"
|
||||
"@parcel/watcher-win32-x64" "2.5.1"
|
||||
|
||||
"@phosphor-icons/react@^2.1.10":
|
||||
version "2.1.10"
|
||||
resolved "https://registry.yarnpkg.com/@phosphor-icons/react/-/react-2.1.10.tgz#3a97ec5b7a4b8d53afeb29125bc17e74ed2daf92"
|
||||
integrity sha512-vt8Tvq8GLjheAZZYa+YG/pW7HDbov8El/MANW8pOAz4eGxrwhnbfrQZq0Cp4q8zBEu8NIhHdnr+r8thnfRSNYA==
|
||||
|
||||
"@rollup/plugin-babel@^5.2.0":
|
||||
version "5.3.1"
|
||||
resolved "https://registry.yarnpkg.com/@rollup/plugin-babel/-/plugin-babel-5.3.1.tgz#04bc0608f4aa4b2e4b1aebf284344d0f68fda283"
|
||||
@@ -1705,16 +1700,6 @@
|
||||
resolved "https://registry.yarnpkg.com/@unrs/resolver-binding-win32-x64-msvc/-/resolver-binding-win32-x64-msvc-1.11.1.tgz#538b1e103bf8d9864e7b85cc96fa8d6fb6c40777"
|
||||
integrity sha512-lrW200hZdbfRtztbygyaq/6jP6AKE8qQN2KvPcJ+x7wiD038YtnYtZ82IMNJ69GJibV7bwL3y9FgK+5w/pYt6g==
|
||||
|
||||
"@webtui/css@^0.1.5":
|
||||
version "0.1.5"
|
||||
resolved "https://registry.yarnpkg.com/@webtui/css/-/css-0.1.5.tgz#2a1d2fe50200ee1dfbbe87448acc0298c3a0d5a6"
|
||||
integrity sha512-6+yk/XjW4JHaRUJpglYcvE9pcxay+PmoPFYeedHksCz0JHp3POW0LG2c6q1hSu2y7U4V0yHO7/eQBC3P0qCyIw==
|
||||
|
||||
"@webtui/theme-catppuccin@^0.0.3":
|
||||
version "0.0.3"
|
||||
resolved "https://registry.yarnpkg.com/@webtui/theme-catppuccin/-/theme-catppuccin-0.0.3.tgz#bc451e8334f697acd17aeb31b721c17012edd37a"
|
||||
integrity sha512-mn3qpyIDYzPm9DoDX7Rs/Ma/3YawbAZEu6/Cemj4kTodr4eBFuEOmkkd1RtChsXGH75fZ054MIzZoVwYvA5W2g==
|
||||
|
||||
acorn-jsx@^5.3.2:
|
||||
version "5.3.2"
|
||||
resolved "https://registry.yarnpkg.com/acorn-jsx/-/acorn-jsx-5.3.2.tgz#7ed5bb55908b3b2f1bc55c6af1653bada7f07937"
|
||||
@@ -3729,6 +3714,11 @@ lru-cache@^5.1.1:
|
||||
dependencies:
|
||||
yallist "^3.0.2"
|
||||
|
||||
lucide-react@^0.294.0:
|
||||
version "0.294.0"
|
||||
resolved "https://registry.yarnpkg.com/lucide-react/-/lucide-react-0.294.0.tgz#dc406e1e7e2f722cf93218fe5b31cf3c95778817"
|
||||
integrity sha512-V7o0/VECSGbLHn3/1O67FUgBwWB+hmzshrgDVRJQhMh8uj5D3HBuIvhuAmQTtlupILSplwIZg5FTc4tTKMA2SA==
|
||||
|
||||
luxon@^3.7.1:
|
||||
version "3.7.2"
|
||||
resolved "https://registry.yarnpkg.com/luxon/-/luxon-3.7.2.tgz#d697e48f478553cca187a0f8436aff468e3ba0ba"
|
||||
@@ -4170,15 +4160,6 @@ postcss-import@^15.1.0:
|
||||
read-cache "^1.0.0"
|
||||
resolve "^1.1.7"
|
||||
|
||||
postcss-import@^16.1.1:
|
||||
version "16.1.1"
|
||||
resolved "https://registry.yarnpkg.com/postcss-import/-/postcss-import-16.1.1.tgz#cfbe79e6c9232b0dbbe1c18f35308825cfe8ff2a"
|
||||
integrity sha512-2xVS1NCZAfjtVdvXiyegxzJ447GyqCeEI5V7ApgQVOWnros1p5lGNovJNapwPpMombyFBfqDwt7AD3n2l0KOfQ==
|
||||
dependencies:
|
||||
postcss-value-parser "^4.0.0"
|
||||
read-cache "^1.0.0"
|
||||
resolve "^1.1.7"
|
||||
|
||||
postcss-js@^4.0.1:
|
||||
version "4.1.0"
|
||||
resolved "https://registry.yarnpkg.com/postcss-js/-/postcss-js-4.1.0.tgz#003b63c6edde948766e40f3daf7e997ae43a5ce6"
|
||||
@@ -4301,11 +4282,6 @@ react-dom@^18:
|
||||
loose-envify "^1.1.0"
|
||||
scheduler "^0.23.2"
|
||||
|
||||
react-icons@^5.5.0:
|
||||
version "5.5.0"
|
||||
resolved "https://registry.yarnpkg.com/react-icons/-/react-icons-5.5.0.tgz#8aa25d3543ff84231685d3331164c00299cdfaf2"
|
||||
integrity sha512-MEFcXdkP3dLo8uumGI5xN3lDFNsRtrjbOEKDLD7yv76v4wpnEq2Lt2qeHaQOr34I/wPN3s3+N08WkQ+CW37Xiw==
|
||||
|
||||
react-is@^16.13.1:
|
||||
version "16.13.1"
|
||||
resolved "https://registry.yarnpkg.com/react-is/-/react-is-16.13.1.tgz#789729a4dc36de2999dc156dd6c1d9c18cea56a4"
|
||||
|
||||