feat没问题

This commit is contained in:
许永平
2025-07-09 17:44:16 +08:00
parent 4367d5a5f3
commit 37dabcfdf6
3 changed files with 164 additions and 4 deletions

View File

@@ -8,6 +8,7 @@ interface ButtonProps {
onClick?: (e?: React.MouseEvent) => void;
disabled?: boolean;
type?: 'button' | 'submit' | 'reset';
loading?: boolean;
}
export function Button({
@@ -17,7 +18,8 @@ export function Button({
className = '',
onClick,
disabled = false,
type = 'button'
type = 'button',
loading = false
}: ButtonProps) {
const baseClasses = 'inline-flex items-center justify-center rounded-md text-sm font-medium transition-colors focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:opacity-50 disabled:pointer-events-none';
@@ -43,9 +45,15 @@ export function Button({
<button
className={classes}
onClick={onClick}
disabled={disabled}
disabled={disabled || loading}
type={type}
>
{loading && (
<svg className="animate-spin -ml-1 mr-2 h-4 w-4 text-white" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24">
<circle className="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" strokeWidth="4"></circle>
<path className="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"></path>
</svg>
)}
{children}
</button>
);

View File

@@ -8,6 +8,12 @@ interface InputProps {
className?: string;
readOnly?: boolean;
id?: string;
name?: string;
type?: string;
required?: boolean;
min?: number;
max?: number;
step?: number;
}
export function Input({
@@ -17,17 +23,28 @@ export function Input({
placeholder,
className = '',
readOnly = false,
id
id,
name,
type = 'text',
required = false,
min,
max,
step,
}: InputProps) {
return (
<input
id={id}
type="text"
name={name}
type={type}
value={value}
onChange={onChange}
onKeyDown={onKeyDown}
placeholder={placeholder}
readOnly={readOnly}
required={required}
min={min}
max={max}
step={step}
className={`flex h-10 w-full rounded-md border border-gray-300 bg-white px-3 py-2 text-sm ring-offset-white file:border-0 file:bg-transparent file:text-sm file:font-medium placeholder:text-gray-500 focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-blue-500 focus-visible:ring-offset-2 disabled:cursor-not-allowed disabled:opacity-50 ${className}`}
/>
);

View File

@@ -0,0 +1,135 @@
import React, { useState } from 'react';
import { useNavigate } from 'react-router-dom';
import { Button } from '@/components/ui/button';
import { Input } from '@/components/ui/input';
import { useToast } from '@/components/ui/toast';
import { createMomentsSyncTask } from '@/api/momentsSync';
export default function NewMomentsSyncTask() {
const navigate = useNavigate();
const { toast } = useToast();
const [loading, setLoading] = useState(false);
const [form, setForm] = useState({
name: '',
deviceIds: '', // 逗号分隔
contentLib: '',
syncMode: 'auto',
syncInterval: '30',
maxSyncPerDay: '100',
timeStart: '08:00',
timeEnd: '22:00',
targetTags: '', // 逗号分隔
contentTypes: '', // 逗号分隔
filterKeywords: '', // 逗号分隔
});
const handleChange = (e: React.ChangeEvent<HTMLInputElement | HTMLSelectElement>) => {
const { name, value } = e.target;
setForm(prev => ({
...prev,
[name]: value,
}));
};
const handleSubmit = async (e: React.FormEvent) => {
e.preventDefault();
if (!form.name.trim()) {
toast({ title: '请输入任务名称', variant: 'destructive' });
return;
}
if (!form.deviceIds.trim()) {
toast({ title: '请输入推送设备', variant: 'destructive' });
return;
}
if (!form.contentLib.trim()) {
toast({ title: '请输入内容库', variant: 'destructive' });
return;
}
setLoading(true);
try {
await createMomentsSyncTask({
name: form.name,
devices: form.deviceIds.split(',').map(s => s.trim()).filter(Boolean),
contentLib: form.contentLib,
syncMode: form.syncMode as 'auto' | 'manual',
interval: Number(form.syncInterval),
maxSync: Number(form.maxSyncPerDay),
startTime: form.timeStart,
endTime: form.timeEnd,
targetTags: form.targetTags.split(',').map(s => s.trim()).filter(Boolean),
contentTypes: form.contentTypes.split(',').map(s => s.trim()).filter(Boolean) as ('text' | 'image' | 'video' | 'link')[],
filterKeywords: form.filterKeywords.split(',').map(s => s.trim()).filter(Boolean),
});
toast({ title: '创建成功' });
navigate('/workspace/moments-sync');
} catch (error) {
toast({ title: '创建失败', variant: 'destructive' });
} finally {
setLoading(false);
}
};
return (
<div className="flex-1 bg-gray-50 min-h-screen pb-20">
<div className="max-w-xl mx-auto bg-white rounded shadow p-6 mt-8">
<h2 className="text-lg font-bold mb-6"></h2>
<form onSubmit={handleSubmit} className="space-y-5">
<div>
<label className="block mb-1 font-medium"></label>
<Input value={form.name} onChange={handleChange} placeholder="请输入任务名称" name="name" required />
</div>
<div>
<label className="block mb-1 font-medium">ID</label>
<Input value={form.deviceIds} onChange={handleChange} placeholder="如dev1,dev2" name="deviceIds" required />
</div>
<div>
<label className="block mb-1 font-medium"></label>
<Input value={form.contentLib} onChange={handleChange} placeholder="请输入内容库名称" name="contentLib" required />
</div>
<div>
<label className="block mb-1 font-medium"></label>
<select name="syncMode" value={form.syncMode} onChange={handleChange} className="w-full border rounded px-3 py-2">
<option value="auto"></option>
<option value="manual"></option>
</select>
</div>
<div className="flex space-x-4">
<div className="flex-1">
<label className="block mb-1 font-medium">()</label>
<Input value={form.syncInterval} onChange={handleChange} name="syncInterval" type="number" min={1} required />
</div>
<div className="flex-1">
<label className="block mb-1 font-medium"></label>
<Input value={form.maxSyncPerDay} onChange={handleChange} name="maxSyncPerDay" type="number" min={1} required />
</div>
</div>
<div className="flex space-x-4">
<div className="flex-1">
<label className="block mb-1 font-medium"></label>
<Input value={form.timeStart} onChange={handleChange} name="timeStart" type="time" required />
</div>
<div className="flex-1">
<label className="block mb-1 font-medium"></label>
<Input value={form.timeEnd} onChange={handleChange} name="timeEnd" type="time" required />
</div>
</div>
<div>
<label className="block mb-1 font-medium"></label>
<Input value={form.targetTags} onChange={handleChange} placeholder="如:重要客户,活跃用户" name="targetTags" />
</div>
<div>
<label className="block mb-1 font-medium">text,image,video</label>
<Input value={form.contentTypes} onChange={handleChange} placeholder="如text,image,video" name="contentTypes" />
</div>
<div>
<label className="block mb-1 font-medium"></label>
<Input value={form.filterKeywords} onChange={handleChange} placeholder="如:产品,服务,优惠" name="filterKeywords" />
</div>
<div className="pt-2">
<Button type="submit" loading={loading} className="w-full">{loading ? '提交中...' : '创建任务'}</Button>
</div>
</form>
</div>
</div>
);
}