-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathslicer.html
101 lines (85 loc) · 2.82 KB
/
slicer.html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>slicer</title>
</head>
<body>
<a href="index.html">← max bo</a>
<br />
<h1>slicer</h1>
inspired by <a href="https://www.hodinkee.com/articles/olympics-2024-mens-100m-photo-finish">How Omega Captured The Men's 100m Final Photo Finish (And The Watch Noah Lyles Wore While Winning)</a>
<br />
<br />
<input type="file" id="videoUploader" accept="video/*">
<br />
<br />
<video id="myVideo" width="1080" controls>
<source id="videoSource" src="video/sprinters2.mp4" type="video/mp4">
Your browser does not support the video tag.
</video>
<br />
<input type="range" id="slicePosition" min="0" max="1080" value="536" step="1" width="1080">
<br />
Align the slider with the finish line, then play the video to start slicing.
<br />
<canvas style="border: 1px auto black;" id="myCanvas" width="1080" height="800"></canvas>
<script>
const SLICE_WIDTH = 1;
let t = 0;
const video = document.getElementById('myVideo');
const videoSource = document.getElementById('videoSource');
const videoUploader = document.getElementById('videoUploader');
const canvas = document.getElementById('myCanvas');
const slicePosition = document.getElementById('slicePosition');
const ctx = canvas.getContext('2d');
// set range width to video width
slicePosition.max = video.width;
// ...and its style as well
slicePosition.style.width = video.width + 'px';
// Handle video upload
videoUploader.addEventListener('change', (event) => {
const file = event.target.files[0];
if (file) {
const fileURL = URL.createObjectURL(file);
videoSource.src = fileURL;
video.load();
}
});
video.addEventListener('play', () => {
t = 0;
requestAnimationFrame(onFrame);
});
function onFrame() {
t++;
if (!video.paused && !video.ended) {
drawFrame();
drawFrameTick();
requestAnimationFrame(onFrame);
}
}
function drawFrame() {
// Draw the current frame onto the canvas
const aspectRatio = video.videoWidth / video.videoHeight;
const sx = parseInt(slicePosition.value) * (video.videoWidth / video.width);
const sy = 0
const sw = SLICE_WIDTH;
const sh = video.videoHeight;
const dx = t * SLICE_WIDTH;
const dy = 0;
const dw = SLICE_WIDTH;
const dh = 1080 * (1 / aspectRatio);
ctx.drawImage(video, sx, sy, sw, sh, dx, dy, dw, dh);
}
function drawFrameTick() {
if (t % 60 !== 0) {
return;
}
// Draw a tick mark at the current slice position
ctx.font = '20px Arial';
ctx.fillText(t, t * SLICE_WIDTH - 40, 20, 1080);
}
</script>
</body>
</html>