short-url/components/ShortUrlForm.tsx

107 lines
3.5 KiB
TypeScript
Raw Normal View History

2024-11-26 12:45:06 +08:00
'use client'
import { useState } from 'react'
export default function ShortUrlForm() {
const [longUrl, setLongUrl] = useState('')
const [shortUrl, setShortUrl] = useState('')
const [expiresIn, setExpiresIn] = useState('3600') // 默认1小时
const [isLoading, setIsLoading] = useState(false)
const [error, setError] = useState('')
const generateShortUrl = async (e: React.FormEvent) => {
e.preventDefault()
setIsLoading(true)
setError('')
try {
const response = await fetch('/api/shorten', {
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify({ url: longUrl, expiresIn: parseInt(expiresIn) }),
})
if (!response.ok) {
throw new Error('Failed to create short URL')
}
const data = await response.json()
setShortUrl(data.shortUrl)
} catch (err) {
setError('An error occurred while creating the short URL')
console.error(err)
} finally {
setIsLoading(false)
}
}
return (
<div className="bg-white rounded-lg shadow-md p-6">
<form onSubmit={generateShortUrl} className="space-y-4">
<div>
<label htmlFor="longUrl" className="block text-sm font-medium text-gray-700">
Enter your long URL
</label>
<input
type="url"
id="longUrl"
value={longUrl}
onChange={(e) => setLongUrl(e.target.value)}
required
className="mt-1 block w-full rounded-md border-gray-300 shadow-sm focus:border-indigo-300 focus:ring focus:ring-indigo-200 focus:ring-opacity-50"
placeholder="https://example.com/very/long/url"
/>
</div>
<div>
<label htmlFor="expiresIn" className="block text-sm font-medium text-gray-700">
Link expiration time
</label>
<select
id="expiresIn"
value={expiresIn}
onChange={(e) => setExpiresIn(e.target.value)}
className="mt-1 block w-full rounded-md border-gray-300 shadow-sm focus:border-indigo-300 focus:ring focus:ring-indigo-200 focus:ring-opacity-50"
>
<option value="3600">1 hour</option>
<option value="86400">1 day</option>
<option value="604800">1 week</option>
<option value="2592000">1 month</option>
<option value="31536000">1 year</option>
<option value="0">Never (permanent)</option>
</select>
</div>
<button
type="submit"
disabled={isLoading}
className="w-full px-4 py-2 text-white bg-blue-600 rounded-md hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-blue-500 focus:ring-opacity-50 transition duration-300 disabled:bg-blue-300"
>
{isLoading ? 'Creating...' : 'Shorten URL'}
</button>
</form>
{error && (
<div className="mt-4 p-2 bg-red-100 border border-red-400 text-red-700 rounded">
{error}
</div>
)}
{shortUrl && (
<div className="mt-6">
<h2 className="text-lg font-semibold text-gray-700">Your shortened URL:</h2>
<div className="mt-2 p-4 bg-gray-100 rounded-md">
<a
href={shortUrl}
target="_blank"
rel="noopener noreferrer"
className="text-blue-600 hover:underline break-all"
>
{shortUrl}
</a>
</div>
</div>
)}
</div>
)
}