chore: format player skip index and segments

This commit is contained in:
2026-05-28 11:29:33 +02:00
committed by Milas Holsting
parent c2650aae07
commit 95a434cd04
2 changed files with 33 additions and 33 deletions

View File

@@ -1,34 +1,34 @@
import { state } from '../state';
import { displayTimeFromAbsolute, absoluteTimeFromDisplay } from '../timeline';
import { showControls } from '../controls';
import { saveProgress } from '../progress';
import { safeLocalStorage } from '../storage';
import { state } from "../state";
import { displayTimeFromAbsolute, absoluteTimeFromDisplay } from "../timeline";
import { showControls } from "../controls";
import { saveProgress } from "../progress";
import { safeLocalStorage } from "../storage";
// button label based on segment type
const skipLabel = (type: string): string => (type === 'ed' ? 'Skip outro' : 'Skip intro');
const skipLabel = (type: string): string => (type === "ed" ? "Skip outro" : "Skip intro");
/**
* Updates skip button visibility and auto-skip logic.
* Called on timeupdate. Shows button when in active segment.
*/
export const updateSkipButton = (currentTime: number): void => {
const btn = state.container.querySelector('[data-skip]') as HTMLButtonElement | null;
const btn = state.container.querySelector("[data-skip]") as HTMLButtonElement | null;
const displayTime = displayTimeFromAbsolute(currentTime);
// find segment that contains current time (with delay buffer)
const segment = state.activeSegments.find(s => {
const segment = state.activeSegments.find((s) => {
const delay = Math.min(1, Math.max(0.25, (s.end - s.start) * 0.02));
return displayTime >= s.start + delay && displayTime < s.end;
});
if (!segment) {
state.activeSkipSegment = null;
btn?.classList.add('hidden');
btn?.classList.add("hidden");
return;
}
// auto-skip: jump to end if enabled
const autoSkip = safeLocalStorage.getItem('mal:autoskip-enabled') === 'true';
const autoSkip = safeLocalStorage.getItem("mal:autoskip-enabled") === "true";
if (autoSkip && displayTime >= segment.start && displayTime < segment.end) {
state.video.currentTime = absoluteTimeFromDisplay(segment.end + 0.01);
void saveProgress();
@@ -40,7 +40,7 @@ export const updateSkipButton = (currentTime: number): void => {
if (btn) {
btn.textContent = skipLabel(segment.type);
btn.title = skipLabel(segment.type);
btn.classList.remove('hidden');
btn.classList.remove("hidden");
}
};
@@ -48,21 +48,21 @@ export const updateSkipButton = (currentTime: number): void => {
* Syncs autoskip checkbox with localStorage.
*/
export const updateAutoSkipButton = (): void => {
const btn = document.querySelector('[data-autoskip]') as HTMLInputElement | null;
const btn = document.querySelector("[data-autoskip]") as HTMLInputElement | null;
if (!btn) return;
btn.checked = safeLocalStorage.getItem('mal:autoskip-enabled') === 'true';
btn.checked = safeLocalStorage.getItem("mal:autoskip-enabled") === "true";
};
/**
* Binds autoskip toggle change handler.
*/
export const setupSkip = (): void => {
document.addEventListener('change', e => {
document.addEventListener("change", (e) => {
const target = e.target as HTMLElement;
if (target.hasAttribute('data-autoskip')) {
if (target.hasAttribute("data-autoskip")) {
safeLocalStorage.setItem(
'mal:autoskip-enabled',
(target as HTMLInputElement).checked ? 'true' : 'false'
"mal:autoskip-enabled",
(target as HTMLInputElement).checked ? "true" : "false",
);
showControls();
}

View File

@@ -1,4 +1,4 @@
import { state } from '../state';
import { state } from "../state";
// filter bounds for valid segments
const MIN_SEGMENT_DURATION = 20; // at least 20s
@@ -17,17 +17,17 @@ export const resolveActiveSegments = (): void => {
return;
}
const normalizeType = (t: string): 'op' | 'ed' | null => {
const v = (t || '').toLowerCase();
if (v === 'op' || v === 'opening' || v === 'intro') return 'op';
if (v === 'ed' || v === 'ending' || v === 'outro') return 'ed';
const normalizeType = (t: string): "op" | "ed" | null => {
const v = (t || "").toLowerCase();
if (v === "op" || v === "opening" || v === "intro") return "op";
if (v === "ed" || v === "ending" || v === "outro") return "ed";
return null;
};
state.activeSegments = state.parsedSegments.filter(s => {
state.activeSegments = state.parsedSegments.filter((s) => {
const t = normalizeType(s.type);
if (!t) return false;
const isOverride = (s.source || '').toLowerCase() === 'override';
const isOverride = (s.source || "").toLowerCase() === "override";
const len = s.end - s.start;
// duration filter
@@ -39,11 +39,11 @@ export const resolveActiveSegments = (): void => {
if (isOverride) return true;
// intro: starts early, before 50% of video
if (t === 'op') {
if (t === "op") {
return s.start <= MAX_INTRO_START && s.start <= bounds * 0.5;
}
// outro: starts in second half of video
if (t === 'ed') {
if (t === "ed") {
return s.start >= bounds * MIN_OUTRO_START_RATIO;
}
return false;
@@ -54,21 +54,21 @@ export const resolveActiveSegments = (): void => {
* Renders segment markers on the timeline progress bar.
*/
export const renderSegments = (): void => {
const track = state.container.querySelector('[data-segments]') as HTMLElement | null;
const track = state.container.querySelector("[data-segments]") as HTMLElement | null;
if (!track) return;
track.innerHTML = '';
track.innerHTML = "";
const bounds = state.video.duration;
if (bounds <= 0) return;
// create clearly visible colored bars for each segment
state.activeSegments.forEach(s => {
const bar = document.createElement('div');
state.activeSegments.forEach((s) => {
const bar = document.createElement("div");
// use distinct colors so segments are readable over buffered/progress fills
bar.className = 'absolute top-0 h-full opacity-95';
bar.className = "absolute top-0 h-full opacity-95";
// distinct colors for OP/ED, rendered above buffered/progress fills
const t = (s.type || '').toLowerCase();
bar.style.backgroundColor = t === 'ed' ? '#60a5fa' : '#f5c542';
const t = (s.type || "").toLowerCase();
bar.style.backgroundColor = t === "ed" ? "#60a5fa" : "#f5c542";
bar.style.left = `${(s.start / bounds) * 100}%`;
bar.style.width = `${((s.end - s.start) / bounds) * 100}%`;
track.appendChild(bar);