Files
kiss-translator/src/subtitle/Menus.js
2025-11-11 23:36:06 +08:00

180 lines
4.3 KiB
JavaScript

import { useCallback, useEffect, useMemo, useState } from "react";
import { MSG_MENUS_PROGRESSED, MSG_MENUS_UPDATEFORM } from "../config";
function Label({ children }) {
return (
<div
style={{
overflow: "hidden",
textOverflow: "ellipsis",
whiteSpace: "nowrap",
}}
>
{children}
</div>
);
}
function MenuItem({ children, onClick, disabled = false }) {
const [hover, setHover] = useState(false);
return (
<div
style={{
display: "flex",
justifyContent: "space-between",
alignItems: "center",
padding: "0px 8px",
opacity: hover ? 1 : 0.8,
background: `rgba(255, 255, 255, ${hover ? 0.1 : 0})`,
cursor: disabled ? "default" : "pointer",
transition: "background 0.2s, opacity 0.2s",
borderRadius: 5,
}}
onMouseEnter={() => setHover(true)}
onMouseLeave={() => setHover(false)}
onClick={onClick}
>
{children}
</div>
);
}
function Switch({ label, name, value, onChange, disabled }) {
const handleClick = useCallback(() => {
if (disabled) return;
onChange({ name, value: !value });
}, [disabled, onChange, name, value]);
return (
<MenuItem onClick={handleClick} disabled={disabled}>
<Label>{label}</Label>
<div
style={{
width: 40,
height: 24,
borderRadius: 12,
background: value ? "rgba(32,156,238,.8)" : "rgba(255,255,255,.3)",
position: "relative",
}}
>
<div
style={{
width: 20,
height: 20,
borderRadius: 10,
position: "absolute",
left: 2,
top: 2,
background: "rgba(255,255,255,.9)",
transform: `translateX(${value ? 16 : 0}px)`,
}}
></div>
</div>
</MenuItem>
);
}
function Button({ label, onClick, disabled }) {
const handleClick = useCallback(() => {
if (disabled) return;
onClick();
}, [disabled, onClick]);
return (
<MenuItem onClick={handleClick} disabled={disabled}>
<Label>{label}</Label>
</MenuItem>
);
}
export function Menus({
i18n,
initData,
updateSetting,
downloadSubtitle,
hasSegApi,
eventName,
}) {
const [formData, setFormData] = useState(initData);
const [progressed, setProgressed] = useState(0);
const handleChange = useCallback(
({ name, value }) => {
setFormData((pre) => ({ ...pre, [name]: value }));
updateSetting({ name, value });
},
[updateSetting]
);
useEffect(() => {
const handler = (e) => {
const { action, data } = e.detail || {};
if (action === MSG_MENUS_PROGRESSED) {
setProgressed(data);
} else if (action === MSG_MENUS_UPDATEFORM) {
setFormData((pre) => ({ ...pre, ...data }));
}
};
window.addEventListener(eventName, handler);
return () => window.removeEventListener(eventName, handler);
}, [eventName]);
const status = useMemo(() => {
if (progressed === 0) return i18n("waiting_subtitles");
if (progressed === 100) return i18n("download_subtitles");
return i18n("processing_subtitles");
}, [progressed, i18n]);
const { isAISegment, skipAd, isBilingual, showOrigin } = formData;
return (
<div
style={{
position: "absolute",
left: 0,
bottom: 100,
background: "rgba(0,0,0,.6)",
width: 200,
lineHeight: "40px",
fontSize: 16,
padding: 8,
borderRadius: 5,
}}
>
<Switch
onChange={handleChange}
name="isAISegment"
value={isAISegment}
label={i18n("ai_segmentation")}
disabled={!hasSegApi}
/>
<Switch
onChange={handleChange}
name="isBilingual"
value={isBilingual}
label={i18n("is_bilingual_view")}
/>
<Switch
onChange={handleChange}
name="showOrigin"
value={showOrigin}
label={i18n("show_origin_subtitle")}
/>
<Switch
onChange={handleChange}
name="skipAd"
value={skipAd}
label={i18n("is_skip_ad")}
/>
<Button
label={`${status} [${progressed}%] `}
onClick={downloadSubtitle}
disabled={progressed !== 100}
/>
</div>
);
}