@kkcompany/player 2.25.0-canary.2 → 2.25.0-canary.21
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +29 -18
- package/dist/core.mjs +670 -95
- package/dist/index.js +9439 -8680
- package/dist/index.mjs +889 -219
- package/dist/modules.mjs +139 -122
- package/dist/react.mjs +912 -225
- package/package.json +1 -1
package/dist/index.mjs
CHANGED
|
@@ -79,14 +79,6 @@ function getDevice() {
|
|
|
79
79
|
|
|
80
80
|
const isSafari = () => /^((?!chrome|android|X11|Linux).)*(safari|iPad|iPhone|Version)/i.test(navigator.userAgent);
|
|
81
81
|
|
|
82
|
-
function needNativeHls() {
|
|
83
|
-
// Don't let Android phones play HLS, even if some of them report supported
|
|
84
|
-
// This covers Samsung & OPPO special cases
|
|
85
|
-
const isAndroid = /android|X11|Linux/i.test(navigator.userAgent);
|
|
86
|
-
return isAndroid || /firefox/i.test(navigator.userAgent) ? '' : // canPlayType isn't reliable across all iOS verion / device combinations, so also check user agent
|
|
87
|
-
isSafari() ? 'maybe' : document.createElement('video').canPlayType('application/vnd.apple.mpegURL');
|
|
88
|
-
}
|
|
89
|
-
|
|
90
82
|
const isDesktop = () => !getDevice().type; // TODO solve lint error:
|
|
91
83
|
// navigator.maxTouchPoints() is not supported in Safari 11, iOS Safari 11.0-11.2 compat/compat
|
|
92
84
|
|
|
@@ -104,38 +96,6 @@ const havePointer = () => {
|
|
|
104
96
|
return havePointer.memo;
|
|
105
97
|
};
|
|
106
98
|
|
|
107
|
-
const timeoutError = () => new Error('request timeout');
|
|
108
|
-
/**
|
|
109
|
-
* @param {URL|RequestInfo} url
|
|
110
|
-
* @param {RequestInit} options
|
|
111
|
-
* @param {{responseType: 'json'|'text'}}
|
|
112
|
-
*/
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
const retryRequest = (url, options = {}, {
|
|
116
|
-
responseType = 'json',
|
|
117
|
-
timeout = 6,
|
|
118
|
-
retryTimes = 6
|
|
119
|
-
} = {}) => new Promise((resolve, reject) => {
|
|
120
|
-
setTimeout(() => reject(timeoutError()), timeout * 1000);
|
|
121
|
-
fetch(url, options).then(response => {
|
|
122
|
-
var _response$responseTyp;
|
|
123
|
-
|
|
124
|
-
return resolve(((_response$responseTyp = response[responseType]) === null || _response$responseTyp === void 0 ? void 0 : _response$responseTyp.call(response)) || response);
|
|
125
|
-
}).catch(reject);
|
|
126
|
-
}).catch(error => {
|
|
127
|
-
console.log(error);
|
|
128
|
-
|
|
129
|
-
if (retryTimes > 0) {
|
|
130
|
-
return retryRequest(url, options, {
|
|
131
|
-
timeout,
|
|
132
|
-
retryTimes: retryTimes - 1
|
|
133
|
-
});
|
|
134
|
-
}
|
|
135
|
-
|
|
136
|
-
return error;
|
|
137
|
-
});
|
|
138
|
-
|
|
139
99
|
const protocolExtensions = {
|
|
140
100
|
hls: 'm3u8',
|
|
141
101
|
dash: 'mpd'
|
|
@@ -288,74 +248,6 @@ const getDrmOptions = source => {
|
|
|
288
248
|
}];
|
|
289
249
|
};
|
|
290
250
|
|
|
291
|
-
function convertToSeconds(timeString) {
|
|
292
|
-
const [hours, minutes, seconds] = timeString.split(':').map(parseFloat);
|
|
293
|
-
return hours * 3600 + minutes * 60 + seconds;
|
|
294
|
-
}
|
|
295
|
-
function getVersion() {
|
|
296
|
-
try {
|
|
297
|
-
// eslint-disable-next-line no-undef
|
|
298
|
-
return "2.25.0-canary.2";
|
|
299
|
-
} catch (e) {
|
|
300
|
-
return undefined;
|
|
301
|
-
}
|
|
302
|
-
}
|
|
303
|
-
function getPopoverPosition(rect, target, boundary) {
|
|
304
|
-
const rectX = rect.x || rect.left;
|
|
305
|
-
const boundaryX = boundary.x || boundary.left;
|
|
306
|
-
const maxLeft = boundary.width - rect.width;
|
|
307
|
-
const targetCenter = (target.left + target.right) / 2 - boundaryX;
|
|
308
|
-
const center = rectX + rect.width / 2 - boundaryX;
|
|
309
|
-
const alignLeft = rectX + (targetCenter - center) - boundaryX;
|
|
310
|
-
return {
|
|
311
|
-
left: Math.max(0, Math.min(alignLeft, maxLeft)),
|
|
312
|
-
top: target.top - rect.height
|
|
313
|
-
};
|
|
314
|
-
} // eslint-disable-next-line consistent-return
|
|
315
|
-
|
|
316
|
-
const nearest = (items, diff) => {
|
|
317
|
-
if (!items.length) {
|
|
318
|
-
return;
|
|
319
|
-
}
|
|
320
|
-
|
|
321
|
-
return items.reduce((a, b) => Math.abs(diff(a)) > Math.abs(diff(b)) ? b : a, items[0]);
|
|
322
|
-
};
|
|
323
|
-
|
|
324
|
-
/* eslint-disable no-param-reassign */
|
|
325
|
-
|
|
326
|
-
const matchAll = (input, pattern) => {
|
|
327
|
-
const flags = [pattern.global && 'g', pattern.ignoreCase && 'i', pattern.multiline && 'm'].filter(Boolean).join('');
|
|
328
|
-
const clone = new RegExp(pattern, flags);
|
|
329
|
-
return Array.from(function* () {
|
|
330
|
-
let matched = true;
|
|
331
|
-
|
|
332
|
-
while (1) {
|
|
333
|
-
matched = clone.exec(input);
|
|
334
|
-
|
|
335
|
-
if (!matched) {
|
|
336
|
-
return;
|
|
337
|
-
}
|
|
338
|
-
|
|
339
|
-
yield matched;
|
|
340
|
-
}
|
|
341
|
-
}());
|
|
342
|
-
};
|
|
343
|
-
|
|
344
|
-
const getHlsQualityOptions = manifest => {
|
|
345
|
-
const resolutionList = matchAll(manifest, /RESOLUTION=\d+x(\d+)/g);
|
|
346
|
-
return Array.from(new Set(resolutionList.map(([, height]) => ({
|
|
347
|
-
height: +height
|
|
348
|
-
})))).sort((a, b) => b.height - a.height);
|
|
349
|
-
};
|
|
350
|
-
|
|
351
|
-
const selectVideoResolution = (_media, {
|
|
352
|
-
player
|
|
353
|
-
}, restrictions) => {
|
|
354
|
-
player.preferredSettings.videoResolution = restrictions;
|
|
355
|
-
player.configure('abr.restrictions', restrictions);
|
|
356
|
-
};
|
|
357
|
-
// for unit test
|
|
358
|
-
|
|
359
251
|
/* eslint-disable no-param-reassign */
|
|
360
252
|
// when the gap is small enough, we consider it is on edge.
|
|
361
253
|
// The magic number 10s comes from observation of YouTube
|
|
@@ -369,7 +261,7 @@ const isLiveDuration = duration => duration >= SHAKA_LIVE_DURATION;
|
|
|
369
261
|
const isEnded = media => !isLiveDuration(media.initialDuration) && media.initialDuration - media.currentTime < 1; // When donwload bandwidth is low, Safari may report time update while buffering, ignore it.
|
|
370
262
|
|
|
371
263
|
|
|
372
|
-
const isBuffered = media =>
|
|
264
|
+
const isBuffered = media => isSafari() || Array.from({
|
|
373
265
|
length: media.buffered.length
|
|
374
266
|
}, (_, index) => ({
|
|
375
267
|
start: media.buffered.start(index),
|
|
@@ -525,22 +417,6 @@ const subscribePlaybackState = (media, updateState, {
|
|
|
525
417
|
return () => registered.forEach(off => off());
|
|
526
418
|
};
|
|
527
419
|
|
|
528
|
-
const tryPatchHlsVideoQualities = async (player, hlsUrl) => {
|
|
529
|
-
if (/(^data)|(mp4$)/.test(hlsUrl)) {
|
|
530
|
-
return;
|
|
531
|
-
} // filtered manifest comes with data URI and should be ignored
|
|
532
|
-
|
|
533
|
-
|
|
534
|
-
const manifest = await retryRequest(hlsUrl, {}, {
|
|
535
|
-
responseType: 'text'
|
|
536
|
-
}).catch(e => console.warn('Failed to get HLS video qualities', e));
|
|
537
|
-
const videoQualities = getHlsQualityOptions(manifest);
|
|
538
|
-
|
|
539
|
-
if (videoQualities) {
|
|
540
|
-
player.getVariantTracks = () => videoQualities;
|
|
541
|
-
}
|
|
542
|
-
};
|
|
543
|
-
|
|
544
420
|
const seek = async (media, {
|
|
545
421
|
player,
|
|
546
422
|
plugins = []
|
|
@@ -690,10 +566,6 @@ const load = async (media, {
|
|
|
690
566
|
loadStartTime = merged.startTime;
|
|
691
567
|
}
|
|
692
568
|
|
|
693
|
-
if (needNativeHls() && merged.type === 'application/x-mpegurl') {
|
|
694
|
-
await tryPatchHlsVideoQualities(player, merged.src);
|
|
695
|
-
}
|
|
696
|
-
|
|
697
569
|
return player.unload().then(() => player.load(merged.src, loadStartTime, merged.type)).then(loadResult => {
|
|
698
570
|
getSourceText(source).forEach(({
|
|
699
571
|
src,
|
|
@@ -843,11 +715,16 @@ const getTextTracks = (_, {
|
|
|
843
715
|
}).filter(track => track.label !== 'unknown' && track.language && !/^un/i.test(track.language));
|
|
844
716
|
};
|
|
845
717
|
|
|
718
|
+
const textTrackOptionOff = {
|
|
719
|
+
label: 'none',
|
|
720
|
+
language: 'none'
|
|
721
|
+
};
|
|
722
|
+
|
|
846
723
|
const getCurrentSubtitle = (_, {
|
|
847
724
|
player
|
|
848
725
|
}) => getTextTracks({}, {
|
|
849
726
|
player
|
|
850
|
-
}).find(track => track.selected) ||
|
|
727
|
+
}).find(track => track.selected) || textTrackOptionOff.label;
|
|
851
728
|
|
|
852
729
|
const textTrackLabel = 'playcraft-text-track';
|
|
853
730
|
/** @param media HTMLMediaElement */
|
|
@@ -865,9 +742,7 @@ const syncTextTrack = (media, {
|
|
|
865
742
|
label,
|
|
866
743
|
language,
|
|
867
744
|
textTracks = []
|
|
868
|
-
} = selected ===
|
|
869
|
-
language: 'off'
|
|
870
|
-
} : selected;
|
|
745
|
+
} = selected === textTrackOptionOff.label ? textTrackOptionOff : selected;
|
|
871
746
|
const cues = (_textTracks$find = textTracks.find(track => {
|
|
872
747
|
var _track$cues;
|
|
873
748
|
|
|
@@ -887,7 +762,7 @@ const syncTextTrack = (media, {
|
|
|
887
762
|
|
|
888
763
|
Array.from(cues).forEach(cue => customTextTrack.addCue(new VTTCue(cue.startTime, cue.endTime, cue.text)));
|
|
889
764
|
customTextTrack.mode = 'showing';
|
|
890
|
-
} else if (language ===
|
|
765
|
+
} else if (language === textTrackOptionOff.language) {
|
|
891
766
|
player.setTextTrackVisibility(false);
|
|
892
767
|
} else {
|
|
893
768
|
if (customTextTrack) {
|
|
@@ -907,7 +782,7 @@ const syncTextTrack = (media, {
|
|
|
907
782
|
}
|
|
908
783
|
|
|
909
784
|
return ((_media$textTracks = media.textTracks) === null || _media$textTracks === void 0 ? void 0 : _media$textTracks.addEventListener) && on(media.textTracks, 'change', () => {
|
|
910
|
-
if (language ===
|
|
785
|
+
if (language === textTrackOptionOff.language) {
|
|
911
786
|
player.setTextTrackVisibility(false);
|
|
912
787
|
}
|
|
913
788
|
});
|
|
@@ -917,7 +792,7 @@ const syncTextTrack = (media, {
|
|
|
917
792
|
const setAudioTrack = (_, {
|
|
918
793
|
player
|
|
919
794
|
}, next) => {
|
|
920
|
-
if (!next) return;
|
|
795
|
+
if (!next || !player) return;
|
|
921
796
|
|
|
922
797
|
try {
|
|
923
798
|
player.selectAudioLanguage(next.language);
|
|
@@ -930,6 +805,39 @@ const setAudioTrack = (_, {
|
|
|
930
805
|
}
|
|
931
806
|
};
|
|
932
807
|
|
|
808
|
+
function convertToSeconds(timeString) {
|
|
809
|
+
const [hours, minutes, seconds] = timeString.split(':').map(parseFloat);
|
|
810
|
+
return hours * 3600 + minutes * 60 + seconds;
|
|
811
|
+
}
|
|
812
|
+
function getVersion() {
|
|
813
|
+
try {
|
|
814
|
+
// eslint-disable-next-line no-undef
|
|
815
|
+
return "2.25.0-canary.21";
|
|
816
|
+
} catch (e) {
|
|
817
|
+
return undefined;
|
|
818
|
+
}
|
|
819
|
+
}
|
|
820
|
+
function getPopoverPosition(rect, target, boundary) {
|
|
821
|
+
const rectX = rect.x || rect.left;
|
|
822
|
+
const boundaryX = boundary.x || boundary.left;
|
|
823
|
+
const maxLeft = boundary.width - rect.width;
|
|
824
|
+
const targetCenter = (target.left + target.right) / 2 - boundaryX;
|
|
825
|
+
const center = rectX + rect.width / 2 - boundaryX;
|
|
826
|
+
const alignLeft = rectX + (targetCenter - center) - boundaryX;
|
|
827
|
+
return {
|
|
828
|
+
left: Math.max(0, Math.min(alignLeft, maxLeft)),
|
|
829
|
+
top: target.top - rect.height
|
|
830
|
+
};
|
|
831
|
+
} // eslint-disable-next-line consistent-return
|
|
832
|
+
|
|
833
|
+
const nearest = (items, diff) => {
|
|
834
|
+
if (!items.length) {
|
|
835
|
+
return;
|
|
836
|
+
}
|
|
837
|
+
|
|
838
|
+
return items.reduce((a, b) => Math.abs(diff(a)) > Math.abs(diff(b)) ? b : a, items[0]);
|
|
839
|
+
};
|
|
840
|
+
|
|
933
841
|
/* eslint-disable no-param-reassign */
|
|
934
842
|
|
|
935
843
|
const getSpeedItems = items => items.map(value => ({
|
|
@@ -997,7 +905,7 @@ const getAudioTracks = (_, {
|
|
|
997
905
|
const all = player.getVariantTracks().reduce((result, track) => {
|
|
998
906
|
var _track$label;
|
|
999
907
|
|
|
1000
|
-
const label = ((_track$label = track.label) === null || _track$label === void 0 ? void 0 : _track$label.trim()) || track.language;
|
|
908
|
+
const label = ((_track$label = track.label) === null || _track$label === void 0 ? void 0 : _track$label.trim()) || track.language || 'unknown';
|
|
1001
909
|
const id = `${track.language}:${label}`;
|
|
1002
910
|
result[id] = result[id] || { ...track,
|
|
1003
911
|
type: 'audio',
|
|
@@ -1015,9 +923,17 @@ const getAudioTracks = (_, {
|
|
|
1015
923
|
|
|
1016
924
|
return result;
|
|
1017
925
|
}, {});
|
|
1018
|
-
const
|
|
1019
|
-
|
|
1020
|
-
|
|
926
|
+
const tracks = Object.values(all);
|
|
927
|
+
|
|
928
|
+
if (tracks.length <= 1) {
|
|
929
|
+
return [{
|
|
930
|
+
type: 'audio',
|
|
931
|
+
...tracks[0],
|
|
932
|
+
selected: true
|
|
933
|
+
}];
|
|
934
|
+
}
|
|
935
|
+
|
|
936
|
+
return tracks;
|
|
1021
937
|
};
|
|
1022
938
|
|
|
1023
939
|
const getCurrentAudioTrack = (media, {
|
|
@@ -1027,6 +943,8 @@ const getCurrentAudioTrack = (media, {
|
|
|
1027
943
|
}).find(track => track.selected);
|
|
1028
944
|
|
|
1029
945
|
const getSelectedAudioName = player => {
|
|
946
|
+
var _currentAudio$value;
|
|
947
|
+
|
|
1030
948
|
const track = getCurrentAudioTrack({}, {
|
|
1031
949
|
player
|
|
1032
950
|
});
|
|
@@ -1042,7 +960,7 @@ const getSelectedAudioName = player => {
|
|
|
1042
960
|
});
|
|
1043
961
|
const currentAudio = track !== undefined ? track : audioList === null || audioList === void 0 ? void 0 : audioList[0];
|
|
1044
962
|
return currentAudio && { ...currentAudio,
|
|
1045
|
-
id: [currentAudio.language, currentAudio.label].join(':')
|
|
963
|
+
id: ((_currentAudio$value = currentAudio.value) === null || _currentAudio$value === void 0 ? void 0 : _currentAudio$value.id) || [currentAudio.language, currentAudio.label].join(':')
|
|
1046
964
|
};
|
|
1047
965
|
};
|
|
1048
966
|
|
|
@@ -1109,7 +1027,8 @@ const getSettingsData = ({
|
|
|
1109
1027
|
quality = {},
|
|
1110
1028
|
speedItems = [0.25, 0.5, 0.75, 1, 1.25, 1.5, 1.75, 2],
|
|
1111
1029
|
loop,
|
|
1112
|
-
preferred = {}
|
|
1030
|
+
preferred = {},
|
|
1031
|
+
otherSections = []
|
|
1113
1032
|
}) => {
|
|
1114
1033
|
// TODO extract base player specific things
|
|
1115
1034
|
const audioItems = getAudioTracks({}, {
|
|
@@ -1119,7 +1038,7 @@ const getSettingsData = ({
|
|
|
1119
1038
|
preferManifestType: 'platform'
|
|
1120
1039
|
}) || {};
|
|
1121
1040
|
const currentSpeedItems = getSpeedItems(speedItems);
|
|
1122
|
-
const sections = [].concat([quality && getQualitySettings({ ...quality,
|
|
1041
|
+
const sections = [].concat(otherSections, [quality && getQualitySettings({ ...quality,
|
|
1123
1042
|
items: selectedSource.qualityOptions
|
|
1124
1043
|
}, player), {
|
|
1125
1044
|
name: 'subtitles',
|
|
@@ -1184,27 +1103,52 @@ const getSettingsData = ({
|
|
|
1184
1103
|
};
|
|
1185
1104
|
};
|
|
1186
1105
|
|
|
1187
|
-
const matchTracks = (tracks,
|
|
1106
|
+
const matchTracks = (tracks, sorted) => {
|
|
1107
|
+
var _tracks$;
|
|
1188
1108
|
|
|
1189
|
-
const
|
|
1190
|
-
|
|
1191
|
-
|
|
1109
|
+
const lookup = {};
|
|
1110
|
+
tracks.forEach(track => {
|
|
1111
|
+
lookup[`${track.language}|${track.label}`] = track;
|
|
1112
|
+
lookup[`${track.language}|`] = track;
|
|
1113
|
+
lookup[`|${track.label}`] = track;
|
|
1192
1114
|
});
|
|
1115
|
+
const matched = sorted.map(info => {
|
|
1116
|
+
const track = lookup[`${info.language}|${info.label}`] || lookup[`${info.language}|`];
|
|
1117
|
+
return track && { ...track,
|
|
1118
|
+
...info
|
|
1119
|
+
};
|
|
1120
|
+
}).filter(Boolean);
|
|
1121
|
+
|
|
1122
|
+
if (((_tracks$ = tracks[0]) === null || _tracks$ === void 0 ? void 0 : _tracks$.type) === 'audio' && matched.length === 0) {
|
|
1123
|
+
return [{ ...tracks[0],
|
|
1124
|
+
...sorted[0]
|
|
1125
|
+
}];
|
|
1126
|
+
}
|
|
1127
|
+
|
|
1128
|
+
return matched;
|
|
1129
|
+
};
|
|
1130
|
+
|
|
1131
|
+
const getLanguageOptions = (player, sorted) => {
|
|
1132
|
+
const audioTracks = player ? getAudioTracks({}, {
|
|
1133
|
+
player
|
|
1134
|
+
}) : [];
|
|
1193
1135
|
const textTracks = getTextTracks({}, {
|
|
1194
1136
|
player
|
|
1195
1137
|
});
|
|
1196
1138
|
const textOptions = [].concat(textTracks, {
|
|
1197
1139
|
type: 'subtitles',
|
|
1198
|
-
label: '
|
|
1140
|
+
label: 'none',
|
|
1141
|
+
language: 'none',
|
|
1199
1142
|
selected: !textTracks.some(track => track.selected),
|
|
1200
1143
|
value: {
|
|
1201
|
-
|
|
1144
|
+
label: 'none',
|
|
1145
|
+
language: 'none',
|
|
1202
1146
|
id: 'off'
|
|
1203
1147
|
}
|
|
1204
1148
|
});
|
|
1205
1149
|
return {
|
|
1206
|
-
audioTracks: matchTracks(audioTracks,
|
|
1207
|
-
textTracks: matchTracks(textOptions,
|
|
1150
|
+
audioTracks: matchTracks(audioTracks, sorted.audioTracks),
|
|
1151
|
+
textTracks: matchTracks(textOptions, sorted.textTracks)
|
|
1208
1152
|
};
|
|
1209
1153
|
};
|
|
1210
1154
|
|
|
@@ -2402,6 +2346,9 @@ var ja = {
|
|
|
2402
2346
|
"KKS.QUALITY": "解像度",
|
|
2403
2347
|
"KKS.AUDIO": "音声",
|
|
2404
2348
|
"KKS.SETTING.OFF": "オフ",
|
|
2349
|
+
"KKS.SETTING.none": "なし",
|
|
2350
|
+
"KKS.SETTING.ja": "日本語",
|
|
2351
|
+
"KKS.SETTING.en": "英語",
|
|
2405
2352
|
"KKS.SUBTITLES": "字幕",
|
|
2406
2353
|
"KKS.SETTING": "設定",
|
|
2407
2354
|
"KKS.SETTING.CONFLICT": "ご視聴する動画は、デフォルト画質設定に一致しません。",
|
|
@@ -3200,13 +3147,10 @@ const nativeTextStyle = {
|
|
|
3200
3147
|
};
|
|
3201
3148
|
const customTextStyle = {
|
|
3202
3149
|
'.shaka-text-container': {
|
|
3150
|
+
boxSizing: 'border-box',
|
|
3203
3151
|
padding: '2cqmin',
|
|
3204
3152
|
font: '3cqmin/1 sans-serif',
|
|
3205
3153
|
color: '#ffffff',
|
|
3206
|
-
fontFamily: 'system-ui-monospaced',
|
|
3207
|
-
'unicode-bidi': 'plaintext',
|
|
3208
|
-
'overflow-wrap': 'break-word',
|
|
3209
|
-
textWrap: 'balance',
|
|
3210
3154
|
textShadow: '1px 1px 0.2cqmin #000'
|
|
3211
3155
|
},
|
|
3212
3156
|
'.shaka-text-container > .shaka-cue-horizontal-tb': {
|
|
@@ -3518,7 +3462,7 @@ var _ref$1 = process.env.NODE_ENV === "production" ? {
|
|
|
3518
3462
|
} : {
|
|
3519
3463
|
name: "9zffum-DefaultLayout",
|
|
3520
3464
|
styles: "&:not(:empty){flex:1;margin-left:calc(var(--spacing, 0.75em) - 0.5em + 0.5rem);};label:DefaultLayout;",
|
|
3521
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["DefaultLayout.js"],"names":[],"mappings":"AAqcc","file":"DefaultLayout.js","sourcesContent":["/* eslint-disable no-param-reassign */\n/* @jsxImportSource @emotion/react */\n/* eslint-disable react/prop-types */\nimport {useRef} from 'react'\nimport {SlotProvider} from './uiExtensions'\nimport {nativeTextStyle, customTextStyle, bottomUiTop} from './subtitles'\n\nconst hidden = {display: 'none'}\nconst invisible = hiddenArea => ({\n  visibility: `var(${hiddenArea}, visible)`,\n})\nconst containerStyle = {\n  '--spacing': '0.5em',\n  '--center-pointer-events': 'auto',\n  width: '100%',\n  height: '100%',\n  fontSize: '16px',\n  boxSizing: 'border-box',\n  display: 'flex',\n  flexDirection: 'column',\n  overflow: 'hidden',\n  color: 'white',\n  background: '#000',\n  // prevent animation glich(afterimage) of descendant elements\n  transform: 'translateX(0)',\n  userSelect: 'none',\n  'a, a:link, a:visited': {\n    color: '#fff',\n    opacity: 0.8,\n    textDecoration: 'none',\n  },\n  button: {\n    fontSize: 'inherit',\n    '> *': {\n      pointerEvents: 'none',\n    },\n  },\n  '--thumbnail-width': '160', // height 90\n  '--theme-color': 'var(--primary-highlight, red)',\n}\n\nconst videoContainerStyle = {\n  '> div:first-of-type': {\n    position: 'absolute',\n    zIndex: '-1',\n    display: 'flex',\n    alignItems: 'center',\n    width: '100%',\n    height: '100%',\n  },\n}\n\nconst drop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0.5) 0,\n    rgba(0,0,0,0) var(--player-ui-drop-shadow-size, 30%) calc(100% - var(--player-ui-drop-shadow-size, 30%)),\n    rgba(0,0,0,0.5) 100%\n  )`,\n  backgroundColor: `rgba(0, 0, 0, 0.3)`,\n}\n\nconst dropTop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0) 0,\n    rgba(0,0,0,0) 8rem calc(100% - 8rem),\n    rgba(0,0,0,0.5) 100%\n  )`,\n}\n\nconst responsiveStyles = {\n  desktop: {\n    fontSize: '24px',\n  }, // add if necessary: big-desktop\n}\n\nconst rowStyle = {\n  boxSizing: 'border-box',\n  width: '100%',\n  padding: '1em',\n  display: 'flex',\n  alignItems: 'center',\n  '> button, > div > button': {\n    padding: '0.5rem',\n  },\n}\n\n/* \n  Hint:\n    displayStyles specificity is higher then controlsStyle\n    Need to pay attention when we revise these styles.\n*/\n\nconst displayHidden = {\n  '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n    opacity: 0,\n    transform: 'translate(-200vw, 0)',\n    transition: 'opacity 0.8s var(--autohide-delay, 0s) ease-out, transform 0s calc(0.8s + var(--autohide-delay, 0s))',\n    'button': {\n      opacity: 0, // extra transition to hide tooltip for Safari\n      transition: 'opacity 0s calc(0.8s + var(--autohide-delay, 0s)) ease-out',\n    }\n  },\n}\n\nconst displayStyles = {\n  hidden: displayHidden,\n  'seekbar-only': displayHidden,\n  shown: {\n    '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n      zIndex: 1,\n      pointerEvents: 'auto',\n      transition: 'opacity 0.3s ease-out, transform 0s 0s',\n    },\n  },\n}\n\nconst controlsDisplayStyles = {\n  hidden: {\n    ...displayStyles.hidden,\n    '~ div:not(.pinned)': displayStyles.hidden,\n  },\n  'seekbar-only': {\n    '> div:first-of-type ~ *': displayStyles.hidden,\n  },\n  shown: {\n    ...displayStyles.shown,\n    '~ div': displayStyles.shown,\n  },\n}\n\n// TODO: transform into sample object instead of function\nconst controlsStyle = type => ({\n  position: 'absolute',\n  zIndex: '2',\n  width: '100%',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'space-evenly',\n  fontSize: '1.5em',\n  '> button': {\n    padding: '0.75em 0.5em',\n    '&:disabled': {\n      opacity: 0,\n      pointerEvents: 'none',\n    },\n    '&.play-button': {\n      padding: '0.75rem 0.5rem',\n      fontSize: 'var(--play-button-size, 1em)',\n    },\n    '&.firstplay-button':\n      type === 'mobile'\n        ? {fontSize: '200%', padding: '4em 10em', marginBottom: '1em'}\n        : {fontSize: '400%'},\n  },\n})\n\n// Remove the height for imporving tapping effect at current, and this height was used for fixed region before.\nconst infoBarSlotStyle = {\n  display: 'flex',\n  order: -1,\n  alignItems: 'center',\n  flex: '1 60%',\n}\n\nconst desktopInfoBarStyle = {\n  marginLeft: 'var(--spacing, 0.5em)',\n  order: 'initial',\n  flex: '1 0%',\n}\n\nconst textEllipsis = {\n  overflow: 'hidden',\n  whiteSpace: 'nowrap',\n  textOverflow: 'ellipsis',\n}\n\nconst infoStyle = {\n  zIndex: 1,\n  overflow: 'visible',\n  pointerEvents: 'none',\n  padding: '0 calc(2em - 20px)',\n  marginBottom: 'calc(0.5em + var(--bottom-spacing, 0))',\n  flex: '3em 0',\n  '> div:first-of-type, > div:nth-of-type(2), > button': {\n    minWidth: 0,\n  },\n  '> div:nth-of-type(2)': {\n    // title and channel title\n    margin: '0 calc(var(--spacing, 0.5em) / 2)',\n    flex: 1,\n    fontSize: '0.75em',\n    visibility: 'var(--player-title-visibility, visible)',\n    span: {\n      marginRight: 'var(--spacing, 0.5em)',\n    },\n    '> div': textEllipsis,\n  },\n  h1: {\n    margin: 0,\n    fontSize: '1.66em',\n    fontWeight: '500',\n    ...textEllipsis,\n    flex: 1,\n    '&:not(:only-child)': {\n      marginTop: '0.5em',\n    },\n  },\n  img: {\n    display: 'block',\n    width: '2em',\n    height: '2em',\n    borderRadius: '0.25em',\n    objectFit: 'cover',\n  },\n}\n\nconst backStyle = {\n  position: 'absolute',\n  zIndex: 0,\n  width: '100%',\n  height: '100%',\n  // move DAI UI slightly up to avoid stacking\n  paddingBottom: 'calc(var(--spacing) * 4)',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  '> iframe': {pointerEvents: 'auto'},\n}\n\nconst skipStyle = {\n  position: 'absolute',\n  right: 0,\n  bottom: '9rem',\n  textAlign: 'right',\n  button: {\n    width: 'auto',\n    height: 'auto',\n  },\n}\n\nconst desktopStyle = {\n  '--spacing': '0.75em',\n  '--center-pointer-events': 'none',\n  '> div button': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em)',\n  },\n  '--thumbnail-width': '288', // height 162\n}\n\nconst desktopControls = {\n  '> div:first-of-type': {\n    flex: '100%',\n    marginBottom: '0.4em',\n  },\n  '> button': {\n    '&[disabled]': {\n    display: 'none',\n    },\n    '&.play-button': {\n      fontSize: '1em',\n    },\n    '&.firstplay-button': {\n      fontSize: '400%'\n    },\n  },\n}\n\nconst seekbarStyles = {\n  flex: '1 var(--seekbar-flex-basis, 100%)',\n  '&:not(:empty)': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n  },\n}\n\nconst adContainerStyle = {\n  flexGrow: 1,\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  pointerEvents: 'var(--center-pointer-events)',\n  button: {pointerEvents: 'auto'},\n  zIndex: 0,\n}\n\nconst ControlsBlock = ({\n  order = 'mobile',\n  liveButton,\n  playButton,\n  rewindButton = '',\n  forwardButton = '',\n  previousEpisodeButton = '',\n  nextEpisodeButton = '',\n}) =>\n  order === 'desktop' ? (\n    <>\n      {liveButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {rewindButton}\n      {forwardButton}\n    </>\n  ) : (\n    <>\n      {rewindButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {forwardButton}\n    </>\n  )\n\nconst DefaultLayout = ({\n  type = 'mobile',\n  style,\n  display,\n  liveButton,\n  controlsDisplay = display,\n  size,\n  title = '',\n  channelTitle = '',\n  video,\n  haveBottomItem,\n  seekbar = '',\n  displayTime,\n  controlButtons,\n  volumeControl,\n  fullscreenButton,\n  backButton = '',\n  adStatus = '',\n  adLink = '',\n  adSkipButton,\n  backItems,\n  children,\n  containerRef,\n  backRef,\n  adContainerRef,\n  ...rest\n}) => {\n  const slotRef = useRef({})\n\n  return (\n    <div\n      style={{\n        '--player-bottom-ui-top':\n          controlsDisplay === 'shown'\n            ? `-${bottomUiTop(containerRef.current)}px`\n            : '-8px',\n        '--player-bottom-ui-height':\n          type === 'desktop' && controlsDisplay === 'shown' ? '3.5rem' : '0',\n      }}\n      css={[\n        containerStyle,\n        videoContainerStyle,\n        responsiveStyles[size],\n        type === 'desktop' && desktopStyle,\n        adStatus && {'--bottom-spacing': 0},\n        nativeTextStyle,\n        customTextStyle,\n        style,\n      ]}\n      ref={containerRef}\n      {...rest}\n    >\n      {video}\n      <div\n        ref={backRef}\n        css={[\n          backStyle,\n          display !== 'hidden' && (haveBottomItem ? dropTop : drop),\n          invisible('--controller-middle-area'),\n        ]}\n      >\n        {type !== 'mobile' && backItems}\n        {adSkipButton && <div css={skipStyle}>{adSkipButton}</div>}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          infoStyle,\n          displayStyles[display],\n          invisible('--controller-top-area'),\n        ]}\n      >\n        {backButton}\n        <div\n          ref={element => {\n            slotRef.current.titleBarLeft = element\n          }}\n        />\n        <div>\n          <h1>{title}</h1>\n          {channelTitle && <div>{channelTitle}</div>}\n        </div>\n        <div\n          ref={element => {\n            slotRef.current.titleBarRight = element\n          }}\n        />\n        {adLink && <div className=\"pinned\">{adLink}</div>}\n      </div>\n      <div\n        ref={adContainerRef}\n        css={[\n          adContainerStyle,\n          // this container covers Google DAI iframe UI, left some space for user to click \"skip ad\"\n          // see https://kkvideo.atlassian.net/browse/CPT-4535 for detail\n          adStatus && {margin: 'calc(var(--spacing) * 6) 0'},\n        ]}\n      >\n        {type === 'mobile' && (\n          <div\n            css={[\n              controlsStyle(type),\n              displayStyles[controlsDisplay],\n              invisible('--controller-middle-area'),\n            ]}\n          >\n            <ControlsBlock order=\"mobile\" {...controlButtons} />\n          </div>\n        )}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          {\n            marginTop: '1em',\n            paddingTop: 0,\n            paddingBottom: 'calc(1em - var(--bottom-spacing, 0) / 5)',\n            flexWrap: 'wrap',\n            '> div:first-of-type': seekbarStyles,\n          },\n          type === 'desktop' && desktopControls,\n          controlsDisplayStyles[controlsDisplay],\n          invisible('--controller-down-area'),\n        ]}\n      >\n        {seekbar || <div />}\n        {type === 'desktop' && (\n          <ControlsBlock\n            order=\"desktop\"\n            liveButton={liveButton}\n            {...controlButtons}\n          />\n        )}\n        {type === 'desktop' ? (\n          <div css={[displayStyles[controlsDisplay]]}>{displayTime}</div>\n        ) : (\n          <>\n            {liveButton}\n            <div\n              css={{\n                '&:not(:empty)': {\n                  flex: 1,\n                  marginLeft: 'calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n                },\n              }}\n            >\n              {displayTime}\n            </div>\n          </>\n        )}\n        {adStatus && (\n          <div\n            className=\"pinned\"\n            css={[infoBarSlotStyle, type === 'desktop' && desktopInfoBarStyle]}\n          >\n            {adStatus}\n          </div>\n        )}\n        <div\n          css={\n            !adStatus && [\n              infoBarSlotStyle,\n              type === 'desktop' && desktopInfoBarStyle,\n            ]\n          }\n          ref={element => {\n            slotRef.current.infoBar = element\n          }}\n        />\n        {volumeControl}\n        <div\n          css={adStatus && hidden}\n          ref={element => {\n            slotRef.current.functionBar = element\n          }}\n        />\n        {fullscreenButton}\n      </div>\n      <SlotProvider slotRef={slotRef}>{children}</SlotProvider>\n    </div>\n  )\n}\n\nexport default DefaultLayout\n"]} */",
|
|
3465
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["DefaultLayout.js"],"names":[],"mappings":"AAqcc","file":"DefaultLayout.js","sourcesContent":["/* eslint-disable no-param-reassign */\n/* @jsxImportSource @emotion/react */\n/* eslint-disable react/prop-types */\nimport {useRef} from 'react'\nimport {SlotProvider} from './uiExtensions'\nimport {nativeTextStyle, customTextStyle, bottomUiTop} from './subtitles'\n\nconst hidden = {display: 'none'}\nconst invisible = hiddenArea => ({\n  visibility: `var(${hiddenArea}, visible)`,\n})\nconst containerStyle = {\n  '--spacing': '0.5em',\n  '--center-pointer-events': 'auto',\n  width: '100%',\n  height: '100%',\n  fontSize: '16px',\n  boxSizing: 'border-box',\n  display: 'flex',\n  flexDirection: 'column',\n  overflow: 'hidden',\n  color: 'white',\n  background: '#000',\n  // prevent animation glich(afterimage) of descendant elements\n  transform: 'translateX(0)',\n  userSelect: 'none',\n  'a, a:link, a:visited': {\n    color: '#fff',\n    opacity: 0.8,\n    textDecoration: 'none',\n  },\n  button: {\n    fontSize: 'inherit',\n    '> *': {\n      pointerEvents: 'none',\n    },\n  },\n  '--thumbnail-width': '160', // height 90\n  '--theme-color': 'var(--primary-highlight, red)',\n}\n\nconst videoContainerStyle = {\n  '> div:first-of-type': {\n    position: 'absolute',\n    zIndex: '-1',\n    display: 'flex',\n    alignItems: 'center',\n    width: '100%',\n    height: '100%',\n  },\n}\n\nconst drop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0.5) 0,\n    rgba(0,0,0,0) var(--player-ui-drop-shadow-size, 30%) calc(100% - var(--player-ui-drop-shadow-size, 30%)),\n    rgba(0,0,0,0.5) 100%\n  )`,\n  backgroundColor: `rgba(0, 0, 0, 0.3)`,\n}\n\nconst dropTop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0) 0,\n    rgba(0,0,0,0) 8rem calc(100% - 8rem),\n    rgba(0,0,0,0.5) 100%\n  )`,\n}\n\nconst responsiveStyles = {\n  desktop: {\n    fontSize: '24px',\n  }, // add if necessary: big-desktop\n}\n\nconst rowStyle = {\n  boxSizing: 'border-box',\n  width: '100%',\n  padding: '1em',\n  display: 'flex',\n  alignItems: 'center',\n  '> button, > div > button': {\n    padding: '0.5rem',\n  },\n}\n\n/* \n  Hint:\n    displayStyles specificity is higher then controlsStyle\n    Need to pay attention when we revise these styles.\n*/\n\nconst displayHidden = {\n  '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n    opacity: 0,\n    transform: 'translate(-200vw, 0)',\n    transition: 'opacity 0.8s var(--autohide-delay, 0s) ease-out, transform 0s calc(0.8s + var(--autohide-delay, 0s))',\n    'button': {\n      opacity: 0, // extra transition to hide tooltip for Safari\n      transition: 'opacity 0s calc(0.8s + var(--autohide-delay, 0s)) ease-out',\n    }\n  },\n}\n\nconst displayStyles = {\n  hidden: displayHidden,\n  'seekbar-only': displayHidden,\n  shown: {\n    '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n      zIndex: 1,\n      pointerEvents: 'auto',\n      transition: 'opacity 0.3s ease-out, transform 0s 0s',\n    },\n  },\n}\n\nconst controlsDisplayStyles = {\n  hidden: {\n    ...displayStyles.hidden,\n    '~ div:not(.pinned)': displayStyles.hidden,\n  },\n  'seekbar-only': {\n    '> div:first-of-type ~ *': displayStyles.hidden,\n  },\n  shown: {\n    ...displayStyles.shown,\n    '~ div': displayStyles.shown,\n  },\n}\n\n// TODO: transform into sample object instead of function\nconst controlsStyle = type => ({\n  position: 'absolute',\n  zIndex: '2',\n  width: '100%',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'space-evenly',\n  fontSize: '1.5em',\n  '> button': {\n    padding: '0.75em 0.5em',\n    '&:disabled': {\n      opacity: 0,\n      pointerEvents: 'none',\n    },\n    '&.play-button': {\n      padding: '0.75rem 0.5rem',\n      fontSize: 'var(--play-button-size, 1em)',\n    },\n    '&.firstplay-button':\n      type === 'mobile'\n        ? {fontSize: '200%', padding: '4em 10em', marginBottom: '1em'}\n        : {fontSize: '400%'},\n  },\n})\n\n// Remove the height for imporving tapping effect at current, and this height was used for fixed region before.\nconst infoBarSlotStyle = {\n  display: 'flex',\n  order: -1,\n  alignItems: 'center',\n  flex: '1 60%',\n}\n\nconst desktopInfoBarStyle = {\n  marginLeft: 'var(--spacing, 0.5em)',\n  order: 'initial',\n  flex: '1 0%',\n}\n\nconst textEllipsis = {\n  overflow: 'hidden',\n  whiteSpace: 'nowrap',\n  textOverflow: 'ellipsis',\n}\n\nconst infoStyle = {\n  zIndex: 1,\n  overflow: 'visible',\n  pointerEvents: 'none',\n  padding: '0 calc(2em - 20px)',\n  marginBottom: 'calc(0.5em + var(--bottom-spacing, 0))',\n  flex: '3em 0',\n  '> div:first-of-type, > div:nth-of-type(2), > button': {\n    minWidth: 0,\n  },\n  '> div:nth-of-type(2)': {\n    // title and channel title\n    margin: '0 calc(var(--spacing, 0.5em) / 2)',\n    flex: 1,\n    fontSize: '0.75em',\n    visibility: 'var(--player-title-visibility, visible)',\n    span: {\n      marginRight: 'var(--spacing, 0.5em)',\n    },\n    '> div': textEllipsis,\n  },\n  h1: {\n    margin: 0,\n    fontSize: '1.66em',\n    fontWeight: '500',\n    ...textEllipsis,\n    flex: 1,\n    '&:not(:only-child)': {\n      marginTop: '0.5em',\n    },\n  },\n  img: {\n    display: 'block',\n    width: '2em',\n    height: '2em',\n    borderRadius: '0.25em',\n    objectFit: 'cover',\n  },\n}\n\nconst backStyle = {\n  position: 'absolute',\n  zIndex: 0,\n  width: '100%',\n  height: '100%',\n  // move DAI UI slightly up to avoid stacking\n  paddingBottom: 'calc(var(--spacing) * 4)',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  '> iframe': {pointerEvents: 'auto'},\n}\n\nconst skipStyle = {\n  position: 'absolute',\n  right: 0,\n  bottom: '9rem',\n  textAlign: 'right',\n  button: {\n    width: 'auto',\n    height: 'auto',\n  },\n}\n\nconst desktopStyle = {\n  '--spacing': '0.75em',\n  '--center-pointer-events': 'none',\n  '> div button': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em)',\n  },\n  '--thumbnail-width': '288', // height 162\n}\n\nconst desktopControls = {\n  '> div:first-of-type': {\n    flex: '100%',\n    marginBottom: '0.4em',\n  },\n  '> button': {\n    '&[disabled]': {\n    display: 'none',\n    },\n    '&.play-button': {\n      fontSize: '1em',\n    },\n    '&.firstplay-button': {\n      fontSize: '400%'\n    },\n  },\n}\n\nconst seekbarStyles = {\n  flex: '1 var(--seekbar-flex-basis, 100%)',\n  '&:not(:empty)': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n  },\n}\n\nconst adContainerStyle = {\n  flexGrow: 1,\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  pointerEvents: 'var(--center-pointer-events)',\n  button: {pointerEvents: 'auto'},\n  zIndex: 0,\n}\n\nconst ControlsBlock = ({\n  order = 'mobile',\n  liveButton,\n  playButton,\n  rewindButton = '',\n  forwardButton = '',\n  previousEpisodeButton = '',\n  nextEpisodeButton = '',\n}) =>\n  order === 'desktop' ? (\n    <>\n      {liveButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {rewindButton}\n      {forwardButton}\n    </>\n  ) : (\n    <>\n      {rewindButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {forwardButton}\n    </>\n  )\n\nconst DefaultLayout = ({\n  type = 'mobile',\n  style,\n  display,\n  liveButton,\n  controlsDisplay = display,\n  size,\n  title = '',\n  channelTitle = '',\n  video,\n  haveBottomItem,\n  seekbar = '',\n  displayTime,\n  controlButtons,\n  volumeControl,\n  fullscreenButton,\n  backButton = '',\n  adStatus = '',\n  adLink = '',\n  adSkipButton,\n  backItems,\n  children,\n  containerRef,\n  backRef,\n  adContainerRef,\n  ...rest\n}) => {\n  const slotRef = useRef({})\n\n  return (\n    <div\n      style={{\n        '--player-bottom-ui-top':\n          controlsDisplay === 'shown'\n            ? `-${bottomUiTop(containerRef.current)}px`\n            : '-8px',\n        '--player-bottom-ui-height':\n          type === 'desktop' && controlsDisplay === 'shown' ? '3.5rem' : '0',\n      }}\n      css={[\n        containerStyle,\n        videoContainerStyle,\n        responsiveStyles[size],\n        type === 'desktop' && desktopStyle,\n        adStatus && {'--bottom-spacing': 0},\n        nativeTextStyle,\n        customTextStyle,\n        style,\n      ]}\n      ref={containerRef}\n      {...rest}\n    >\n      {video}\n      <div\n        ref={backRef}\n        css={[\n          backStyle,\n          display !== 'hidden' && (haveBottomItem ? dropTop : drop),\n          invisible('--controller-middle-area'),\n        ]}\n      >\n        {type !== 'mobile' && backItems}\n        {adSkipButton && <div css={skipStyle}>{adSkipButton}</div>}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          infoStyle,\n          displayStyles[display],\n          invisible('--controller-top-area'),\n        ]}\n      >\n        {backButton}\n        <div\n          ref={element => {\n            slotRef.current.titleBarLeft = element\n          }}\n        />\n        <div>\n          <h1>{title}</h1>\n          {channelTitle && <div>{channelTitle}</div>}\n        </div>\n        <div\n          ref={element => {\n            slotRef.current.titleBarRight = element\n          }}\n        />\n        {adLink && <div className=\"pinned\">{adLink}</div>}\n      </div>\n      <div\n        ref={adContainerRef}\n        css={[\n          adContainerStyle,\n          // this container covers Google DAI iframe UI, left some space for user to click \"skip ad\"\n          // see https://kkvideo.atlassian.net/browse/CPT-4535 for detail\n          adStatus && {margin: 'calc(var(--spacing) * 6) 0'},\n        ]}\n      >\n        {type === 'mobile' && (\n          <div\n            css={[\n              controlsStyle(type),\n              displayStyles[controlsDisplay],\n              invisible('--controller-middle-area'),\n            ]}\n          >\n            <ControlsBlock order=\"mobile\" {...controlButtons} />\n          </div>\n        )}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          {\n            marginTop: '1em',\n            paddingTop: 0,\n            paddingBottom: 'calc(1em - var(--bottom-spacing, 0) / 5)',\n            flexWrap: 'wrap',\n            '> div:first-of-type': seekbarStyles,\n          },\n          type === 'desktop' && desktopControls,\n          controlsDisplayStyles[controlsDisplay],\n          invisible('--controller-down-area'),\n        ]}\n      >\n        {seekbar || <div />}\n        {type === 'desktop' && (\n          <ControlsBlock\n            order=\"desktop\"\n            liveButton={liveButton}\n            {...controlButtons}\n          />\n        )}\n        {type === 'desktop' ? (\n          <div css={[displayStyles[controlsDisplay]]}>{displayTime}</div>\n        ) : (\n          <>\n            {liveButton}\n            <div\n              css={{\n                '&:not(:empty)': {\n                  flex: 1,\n                  marginLeft: 'calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n                },\n              }}\n            >\n              {displayTime}\n            </div>\n          </>\n        )}\n        {adStatus && (\n          <div\n            className=\"pinned\"\n            css={[infoBarSlotStyle, type === 'desktop' && desktopInfoBarStyle]}\n          >\n            {adStatus}\n          </div>\n        )}\n        <div\n          css={\n            !adStatus && [\n              infoBarSlotStyle,\n              type === 'desktop' && desktopInfoBarStyle,\n            ]\n          }\n          ref={element => {\n            slotRef.current.infoBar = element\n          }}\n        />\n        {volumeControl}\n        <div\n          css={[adStatus ? hidden : {display: 'flex'}]}\n          ref={element => {\n            slotRef.current.functionBar = element\n          }}\n        />\n        {fullscreenButton}\n      </div>\n      <SlotProvider slotRef={slotRef}>{children}</SlotProvider>\n    </div>\n  )\n}\n\nexport default DefaultLayout\n"]} */",
|
|
3522
3466
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__$2
|
|
3523
3467
|
};
|
|
3524
3468
|
|
|
@@ -3557,18 +3501,18 @@ const DefaultLayout = ({
|
|
|
3557
3501
|
},
|
|
3558
3502
|
css: [containerStyle$5, videoContainerStyle, responsiveStyles[size], type === 'desktop' && desktopStyle$2, adStatus && {
|
|
3559
3503
|
'--bottom-spacing': 0
|
|
3560
|
-
}, nativeTextStyle, customTextStyle, style, process.env.NODE_ENV === "production" ? "" : ";label:DefaultLayout;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["DefaultLayout.js"],"names":[],"mappings":"AAiWM","file":"DefaultLayout.js","sourcesContent":["/* eslint-disable no-param-reassign */\n/* @jsxImportSource @emotion/react */\n/* eslint-disable react/prop-types */\nimport {useRef} from 'react'\nimport {SlotProvider} from './uiExtensions'\nimport {nativeTextStyle, customTextStyle, bottomUiTop} from './subtitles'\n\nconst hidden = {display: 'none'}\nconst invisible = hiddenArea => ({\n  visibility: `var(${hiddenArea}, visible)`,\n})\nconst containerStyle = {\n  '--spacing': '0.5em',\n  '--center-pointer-events': 'auto',\n  width: '100%',\n  height: '100%',\n  fontSize: '16px',\n  boxSizing: 'border-box',\n  display: 'flex',\n  flexDirection: 'column',\n  overflow: 'hidden',\n  color: 'white',\n  background: '#000',\n  // prevent animation glich(afterimage) of descendant elements\n  transform: 'translateX(0)',\n  userSelect: 'none',\n  'a, a:link, a:visited': {\n    color: '#fff',\n    opacity: 0.8,\n    textDecoration: 'none',\n  },\n  button: {\n    fontSize: 'inherit',\n    '> *': {\n      pointerEvents: 'none',\n    },\n  },\n  '--thumbnail-width': '160', // height 90\n  '--theme-color': 'var(--primary-highlight, red)',\n}\n\nconst videoContainerStyle = {\n  '> div:first-of-type': {\n    position: 'absolute',\n    zIndex: '-1',\n    display: 'flex',\n    alignItems: 'center',\n    width: '100%',\n    height: '100%',\n  },\n}\n\nconst drop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0.5) 0,\n    rgba(0,0,0,0) var(--player-ui-drop-shadow-size, 30%) calc(100% - var(--player-ui-drop-shadow-size, 30%)),\n    rgba(0,0,0,0.5) 100%\n  )`,\n  backgroundColor: `rgba(0, 0, 0, 0.3)`,\n}\n\nconst dropTop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0) 0,\n    rgba(0,0,0,0) 8rem calc(100% - 8rem),\n    rgba(0,0,0,0.5) 100%\n  )`,\n}\n\nconst responsiveStyles = {\n  desktop: {\n    fontSize: '24px',\n  }, // add if necessary: big-desktop\n}\n\nconst rowStyle = {\n  boxSizing: 'border-box',\n  width: '100%',\n  padding: '1em',\n  display: 'flex',\n  alignItems: 'center',\n  '> button, > div > button': {\n    padding: '0.5rem',\n  },\n}\n\n/* \n  Hint:\n    displayStyles specificity is higher then controlsStyle\n    Need to pay attention when we revise these styles.\n*/\n\nconst displayHidden = {\n  '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n    opacity: 0,\n    transform: 'translate(-200vw, 0)',\n    transition: 'opacity 0.8s var(--autohide-delay, 0s) ease-out, transform 0s calc(0.8s + var(--autohide-delay, 0s))',\n    'button': {\n      opacity: 0, // extra transition to hide tooltip for Safari\n      transition: 'opacity 0s calc(0.8s + var(--autohide-delay, 0s)) ease-out',\n    }\n  },\n}\n\nconst displayStyles = {\n  hidden: displayHidden,\n  'seekbar-only': displayHidden,\n  shown: {\n    '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n      zIndex: 1,\n      pointerEvents: 'auto',\n      transition: 'opacity 0.3s ease-out, transform 0s 0s',\n    },\n  },\n}\n\nconst controlsDisplayStyles = {\n  hidden: {\n    ...displayStyles.hidden,\n    '~ div:not(.pinned)': displayStyles.hidden,\n  },\n  'seekbar-only': {\n    '> div:first-of-type ~ *': displayStyles.hidden,\n  },\n  shown: {\n    ...displayStyles.shown,\n    '~ div': displayStyles.shown,\n  },\n}\n\n// TODO: transform into sample object instead of function\nconst controlsStyle = type => ({\n  position: 'absolute',\n  zIndex: '2',\n  width: '100%',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'space-evenly',\n  fontSize: '1.5em',\n  '> button': {\n    padding: '0.75em 0.5em',\n    '&:disabled': {\n      opacity: 0,\n      pointerEvents: 'none',\n    },\n    '&.play-button': {\n      padding: '0.75rem 0.5rem',\n      fontSize: 'var(--play-button-size, 1em)',\n    },\n    '&.firstplay-button':\n      type === 'mobile'\n        ? {fontSize: '200%', padding: '4em 10em', marginBottom: '1em'}\n        : {fontSize: '400%'},\n  },\n})\n\n// Remove the height for imporving tapping effect at current, and this height was used for fixed region before.\nconst infoBarSlotStyle = {\n  display: 'flex',\n  order: -1,\n  alignItems: 'center',\n  flex: '1 60%',\n}\n\nconst desktopInfoBarStyle = {\n  marginLeft: 'var(--spacing, 0.5em)',\n  order: 'initial',\n  flex: '1 0%',\n}\n\nconst textEllipsis = {\n  overflow: 'hidden',\n  whiteSpace: 'nowrap',\n  textOverflow: 'ellipsis',\n}\n\nconst infoStyle = {\n  zIndex: 1,\n  overflow: 'visible',\n  pointerEvents: 'none',\n  padding: '0 calc(2em - 20px)',\n  marginBottom: 'calc(0.5em + var(--bottom-spacing, 0))',\n  flex: '3em 0',\n  '> div:first-of-type, > div:nth-of-type(2), > button': {\n    minWidth: 0,\n  },\n  '> div:nth-of-type(2)': {\n    // title and channel title\n    margin: '0 calc(var(--spacing, 0.5em) / 2)',\n    flex: 1,\n    fontSize: '0.75em',\n    visibility: 'var(--player-title-visibility, visible)',\n    span: {\n      marginRight: 'var(--spacing, 0.5em)',\n    },\n    '> div': textEllipsis,\n  },\n  h1: {\n    margin: 0,\n    fontSize: '1.66em',\n    fontWeight: '500',\n    ...textEllipsis,\n    flex: 1,\n    '&:not(:only-child)': {\n      marginTop: '0.5em',\n    },\n  },\n  img: {\n    display: 'block',\n    width: '2em',\n    height: '2em',\n    borderRadius: '0.25em',\n    objectFit: 'cover',\n  },\n}\n\nconst backStyle = {\n  position: 'absolute',\n  zIndex: 0,\n  width: '100%',\n  height: '100%',\n  // move DAI UI slightly up to avoid stacking\n  paddingBottom: 'calc(var(--spacing) * 4)',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  '> iframe': {pointerEvents: 'auto'},\n}\n\nconst skipStyle = {\n  position: 'absolute',\n  right: 0,\n  bottom: '9rem',\n  textAlign: 'right',\n  button: {\n    width: 'auto',\n    height: 'auto',\n  },\n}\n\nconst desktopStyle = {\n  '--spacing': '0.75em',\n  '--center-pointer-events': 'none',\n  '> div button': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em)',\n  },\n  '--thumbnail-width': '288', // height 162\n}\n\nconst desktopControls = {\n  '> div:first-of-type': {\n    flex: '100%',\n    marginBottom: '0.4em',\n  },\n  '> button': {\n    '&[disabled]': {\n    display: 'none',\n    },\n    '&.play-button': {\n      fontSize: '1em',\n    },\n    '&.firstplay-button': {\n      fontSize: '400%'\n    },\n  },\n}\n\nconst seekbarStyles = {\n  flex: '1 var(--seekbar-flex-basis, 100%)',\n  '&:not(:empty)': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n  },\n}\n\nconst adContainerStyle = {\n  flexGrow: 1,\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  pointerEvents: 'var(--center-pointer-events)',\n  button: {pointerEvents: 'auto'},\n  zIndex: 0,\n}\n\nconst ControlsBlock = ({\n  order = 'mobile',\n  liveButton,\n  playButton,\n  rewindButton = '',\n  forwardButton = '',\n  previousEpisodeButton = '',\n  nextEpisodeButton = '',\n}) =>\n  order === 'desktop' ? (\n    <>\n      {liveButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {rewindButton}\n      {forwardButton}\n    </>\n  ) : (\n    <>\n      {rewindButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {forwardButton}\n    </>\n  )\n\nconst DefaultLayout = ({\n  type = 'mobile',\n  style,\n  display,\n  liveButton,\n  controlsDisplay = display,\n  size,\n  title = '',\n  channelTitle = '',\n  video,\n  haveBottomItem,\n  seekbar = '',\n  displayTime,\n  controlButtons,\n  volumeControl,\n  fullscreenButton,\n  backButton = '',\n  adStatus = '',\n  adLink = '',\n  adSkipButton,\n  backItems,\n  children,\n  containerRef,\n  backRef,\n  adContainerRef,\n  ...rest\n}) => {\n  const slotRef = useRef({})\n\n  return (\n    <div\n      style={{\n        '--player-bottom-ui-top':\n          controlsDisplay === 'shown'\n            ? `-${bottomUiTop(containerRef.current)}px`\n            : '-8px',\n        '--player-bottom-ui-height':\n          type === 'desktop' && controlsDisplay === 'shown' ? '3.5rem' : '0',\n      }}\n      css={[\n        containerStyle,\n        videoContainerStyle,\n        responsiveStyles[size],\n        type === 'desktop' && desktopStyle,\n        adStatus && {'--bottom-spacing': 0},\n        nativeTextStyle,\n        customTextStyle,\n        style,\n      ]}\n      ref={containerRef}\n      {...rest}\n    >\n      {video}\n      <div\n        ref={backRef}\n        css={[\n          backStyle,\n          display !== 'hidden' && (haveBottomItem ? dropTop : drop),\n          invisible('--controller-middle-area'),\n        ]}\n      >\n        {type !== 'mobile' && backItems}\n        {adSkipButton && <div css={skipStyle}>{adSkipButton}</div>}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          infoStyle,\n          displayStyles[display],\n          invisible('--controller-top-area'),\n        ]}\n      >\n        {backButton}\n        <div\n          ref={element => {\n            slotRef.current.titleBarLeft = element\n          }}\n        />\n        <div>\n          <h1>{title}</h1>\n          {channelTitle && <div>{channelTitle}</div>}\n        </div>\n        <div\n          ref={element => {\n            slotRef.current.titleBarRight = element\n          }}\n        />\n        {adLink && <div className=\"pinned\">{adLink}</div>}\n      </div>\n      <div\n        ref={adContainerRef}\n        css={[\n          adContainerStyle,\n          // this container covers Google DAI iframe UI, left some space for user to click \"skip ad\"\n          // see https://kkvideo.atlassian.net/browse/CPT-4535 for detail\n          adStatus && {margin: 'calc(var(--spacing) * 6) 0'},\n        ]}\n      >\n        {type === 'mobile' && (\n          <div\n            css={[\n              controlsStyle(type),\n              displayStyles[controlsDisplay],\n              invisible('--controller-middle-area'),\n            ]}\n          >\n            <ControlsBlock order=\"mobile\" {...controlButtons} />\n          </div>\n        )}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          {\n            marginTop: '1em',\n            paddingTop: 0,\n            paddingBottom: 'calc(1em - var(--bottom-spacing, 0) / 5)',\n            flexWrap: 'wrap',\n            '> div:first-of-type': seekbarStyles,\n          },\n          type === 'desktop' && desktopControls,\n          controlsDisplayStyles[controlsDisplay],\n          invisible('--controller-down-area'),\n        ]}\n      >\n        {seekbar || <div />}\n        {type === 'desktop' && (\n          <ControlsBlock\n            order=\"desktop\"\n            liveButton={liveButton}\n            {...controlButtons}\n          />\n        )}\n        {type === 'desktop' ? (\n          <div css={[displayStyles[controlsDisplay]]}>{displayTime}</div>\n        ) : (\n          <>\n            {liveButton}\n            <div\n              css={{\n                '&:not(:empty)': {\n                  flex: 1,\n                  marginLeft: 'calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n                },\n              }}\n            >\n              {displayTime}\n            </div>\n          </>\n        )}\n        {adStatus && (\n          <div\n            className=\"pinned\"\n            css={[infoBarSlotStyle, type === 'desktop' && desktopInfoBarStyle]}\n          >\n            {adStatus}\n          </div>\n        )}\n        <div\n          css={\n            !adStatus && [\n              infoBarSlotStyle,\n              type === 'desktop' && desktopInfoBarStyle,\n            ]\n          }\n          ref={element => {\n            slotRef.current.infoBar = element\n          }}\n        />\n        {volumeControl}\n        <div\n          css={adStatus && hidden}\n          ref={element => {\n            slotRef.current.functionBar = element\n          }}\n        />\n        {fullscreenButton}\n      </div>\n      <SlotProvider slotRef={slotRef}>{children}</SlotProvider>\n    </div>\n  )\n}\n\nexport default DefaultLayout\n"]} */"],
|
|
3504
|
+
}, nativeTextStyle, customTextStyle, style, process.env.NODE_ENV === "production" ? "" : ";label:DefaultLayout;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["DefaultLayout.js"],"names":[],"mappings":"AAiWM","file":"DefaultLayout.js","sourcesContent":["/* eslint-disable no-param-reassign */\n/* @jsxImportSource @emotion/react */\n/* eslint-disable react/prop-types */\nimport {useRef} from 'react'\nimport {SlotProvider} from './uiExtensions'\nimport {nativeTextStyle, customTextStyle, bottomUiTop} from './subtitles'\n\nconst hidden = {display: 'none'}\nconst invisible = hiddenArea => ({\n  visibility: `var(${hiddenArea}, visible)`,\n})\nconst containerStyle = {\n  '--spacing': '0.5em',\n  '--center-pointer-events': 'auto',\n  width: '100%',\n  height: '100%',\n  fontSize: '16px',\n  boxSizing: 'border-box',\n  display: 'flex',\n  flexDirection: 'column',\n  overflow: 'hidden',\n  color: 'white',\n  background: '#000',\n  // prevent animation glich(afterimage) of descendant elements\n  transform: 'translateX(0)',\n  userSelect: 'none',\n  'a, a:link, a:visited': {\n    color: '#fff',\n    opacity: 0.8,\n    textDecoration: 'none',\n  },\n  button: {\n    fontSize: 'inherit',\n    '> *': {\n      pointerEvents: 'none',\n    },\n  },\n  '--thumbnail-width': '160', // height 90\n  '--theme-color': 'var(--primary-highlight, red)',\n}\n\nconst videoContainerStyle = {\n  '> div:first-of-type': {\n    position: 'absolute',\n    zIndex: '-1',\n    display: 'flex',\n    alignItems: 'center',\n    width: '100%',\n    height: '100%',\n  },\n}\n\nconst drop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0.5) 0,\n    rgba(0,0,0,0) var(--player-ui-drop-shadow-size, 30%) calc(100% - var(--player-ui-drop-shadow-size, 30%)),\n    rgba(0,0,0,0.5) 100%\n  )`,\n  backgroundColor: `rgba(0, 0, 0, 0.3)`,\n}\n\nconst dropTop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0) 0,\n    rgba(0,0,0,0) 8rem calc(100% - 8rem),\n    rgba(0,0,0,0.5) 100%\n  )`,\n}\n\nconst responsiveStyles = {\n  desktop: {\n    fontSize: '24px',\n  }, // add if necessary: big-desktop\n}\n\nconst rowStyle = {\n  boxSizing: 'border-box',\n  width: '100%',\n  padding: '1em',\n  display: 'flex',\n  alignItems: 'center',\n  '> button, > div > button': {\n    padding: '0.5rem',\n  },\n}\n\n/* \n  Hint:\n    displayStyles specificity is higher then controlsStyle\n    Need to pay attention when we revise these styles.\n*/\n\nconst displayHidden = {\n  '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n    opacity: 0,\n    transform: 'translate(-200vw, 0)',\n    transition: 'opacity 0.8s var(--autohide-delay, 0s) ease-out, transform 0s calc(0.8s + var(--autohide-delay, 0s))',\n    'button': {\n      opacity: 0, // extra transition to hide tooltip for Safari\n      transition: 'opacity 0s calc(0.8s + var(--autohide-delay, 0s)) ease-out',\n    }\n  },\n}\n\nconst displayStyles = {\n  hidden: displayHidden,\n  'seekbar-only': displayHidden,\n  shown: {\n    '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n      zIndex: 1,\n      pointerEvents: 'auto',\n      transition: 'opacity 0.3s ease-out, transform 0s 0s',\n    },\n  },\n}\n\nconst controlsDisplayStyles = {\n  hidden: {\n    ...displayStyles.hidden,\n    '~ div:not(.pinned)': displayStyles.hidden,\n  },\n  'seekbar-only': {\n    '> div:first-of-type ~ *': displayStyles.hidden,\n  },\n  shown: {\n    ...displayStyles.shown,\n    '~ div': displayStyles.shown,\n  },\n}\n\n// TODO: transform into sample object instead of function\nconst controlsStyle = type => ({\n  position: 'absolute',\n  zIndex: '2',\n  width: '100%',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'space-evenly',\n  fontSize: '1.5em',\n  '> button': {\n    padding: '0.75em 0.5em',\n    '&:disabled': {\n      opacity: 0,\n      pointerEvents: 'none',\n    },\n    '&.play-button': {\n      padding: '0.75rem 0.5rem',\n      fontSize: 'var(--play-button-size, 1em)',\n    },\n    '&.firstplay-button':\n      type === 'mobile'\n        ? {fontSize: '200%', padding: '4em 10em', marginBottom: '1em'}\n        : {fontSize: '400%'},\n  },\n})\n\n// Remove the height for imporving tapping effect at current, and this height was used for fixed region before.\nconst infoBarSlotStyle = {\n  display: 'flex',\n  order: -1,\n  alignItems: 'center',\n  flex: '1 60%',\n}\n\nconst desktopInfoBarStyle = {\n  marginLeft: 'var(--spacing, 0.5em)',\n  order: 'initial',\n  flex: '1 0%',\n}\n\nconst textEllipsis = {\n  overflow: 'hidden',\n  whiteSpace: 'nowrap',\n  textOverflow: 'ellipsis',\n}\n\nconst infoStyle = {\n  zIndex: 1,\n  overflow: 'visible',\n  pointerEvents: 'none',\n  padding: '0 calc(2em - 20px)',\n  marginBottom: 'calc(0.5em + var(--bottom-spacing, 0))',\n  flex: '3em 0',\n  '> div:first-of-type, > div:nth-of-type(2), > button': {\n    minWidth: 0,\n  },\n  '> div:nth-of-type(2)': {\n    // title and channel title\n    margin: '0 calc(var(--spacing, 0.5em) / 2)',\n    flex: 1,\n    fontSize: '0.75em',\n    visibility: 'var(--player-title-visibility, visible)',\n    span: {\n      marginRight: 'var(--spacing, 0.5em)',\n    },\n    '> div': textEllipsis,\n  },\n  h1: {\n    margin: 0,\n    fontSize: '1.66em',\n    fontWeight: '500',\n    ...textEllipsis,\n    flex: 1,\n    '&:not(:only-child)': {\n      marginTop: '0.5em',\n    },\n  },\n  img: {\n    display: 'block',\n    width: '2em',\n    height: '2em',\n    borderRadius: '0.25em',\n    objectFit: 'cover',\n  },\n}\n\nconst backStyle = {\n  position: 'absolute',\n  zIndex: 0,\n  width: '100%',\n  height: '100%',\n  // move DAI UI slightly up to avoid stacking\n  paddingBottom: 'calc(var(--spacing) * 4)',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  '> iframe': {pointerEvents: 'auto'},\n}\n\nconst skipStyle = {\n  position: 'absolute',\n  right: 0,\n  bottom: '9rem',\n  textAlign: 'right',\n  button: {\n    width: 'auto',\n    height: 'auto',\n  },\n}\n\nconst desktopStyle = {\n  '--spacing': '0.75em',\n  '--center-pointer-events': 'none',\n  '> div button': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em)',\n  },\n  '--thumbnail-width': '288', // height 162\n}\n\nconst desktopControls = {\n  '> div:first-of-type': {\n    flex: '100%',\n    marginBottom: '0.4em',\n  },\n  '> button': {\n    '&[disabled]': {\n    display: 'none',\n    },\n    '&.play-button': {\n      fontSize: '1em',\n    },\n    '&.firstplay-button': {\n      fontSize: '400%'\n    },\n  },\n}\n\nconst seekbarStyles = {\n  flex: '1 var(--seekbar-flex-basis, 100%)',\n  '&:not(:empty)': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n  },\n}\n\nconst adContainerStyle = {\n  flexGrow: 1,\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  pointerEvents: 'var(--center-pointer-events)',\n  button: {pointerEvents: 'auto'},\n  zIndex: 0,\n}\n\nconst ControlsBlock = ({\n  order = 'mobile',\n  liveButton,\n  playButton,\n  rewindButton = '',\n  forwardButton = '',\n  previousEpisodeButton = '',\n  nextEpisodeButton = '',\n}) =>\n  order === 'desktop' ? (\n    <>\n      {liveButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {rewindButton}\n      {forwardButton}\n    </>\n  ) : (\n    <>\n      {rewindButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {forwardButton}\n    </>\n  )\n\nconst DefaultLayout = ({\n  type = 'mobile',\n  style,\n  display,\n  liveButton,\n  controlsDisplay = display,\n  size,\n  title = '',\n  channelTitle = '',\n  video,\n  haveBottomItem,\n  seekbar = '',\n  displayTime,\n  controlButtons,\n  volumeControl,\n  fullscreenButton,\n  backButton = '',\n  adStatus = '',\n  adLink = '',\n  adSkipButton,\n  backItems,\n  children,\n  containerRef,\n  backRef,\n  adContainerRef,\n  ...rest\n}) => {\n  const slotRef = useRef({})\n\n  return (\n    <div\n      style={{\n        '--player-bottom-ui-top':\n          controlsDisplay === 'shown'\n            ? `-${bottomUiTop(containerRef.current)}px`\n            : '-8px',\n        '--player-bottom-ui-height':\n          type === 'desktop' && controlsDisplay === 'shown' ? '3.5rem' : '0',\n      }}\n      css={[\n        containerStyle,\n        videoContainerStyle,\n        responsiveStyles[size],\n        type === 'desktop' && desktopStyle,\n        adStatus && {'--bottom-spacing': 0},\n        nativeTextStyle,\n        customTextStyle,\n        style,\n      ]}\n      ref={containerRef}\n      {...rest}\n    >\n      {video}\n      <div\n        ref={backRef}\n        css={[\n          backStyle,\n          display !== 'hidden' && (haveBottomItem ? dropTop : drop),\n          invisible('--controller-middle-area'),\n        ]}\n      >\n        {type !== 'mobile' && backItems}\n        {adSkipButton && <div css={skipStyle}>{adSkipButton}</div>}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          infoStyle,\n          displayStyles[display],\n          invisible('--controller-top-area'),\n        ]}\n      >\n        {backButton}\n        <div\n          ref={element => {\n            slotRef.current.titleBarLeft = element\n          }}\n        />\n        <div>\n          <h1>{title}</h1>\n          {channelTitle && <div>{channelTitle}</div>}\n        </div>\n        <div\n          ref={element => {\n            slotRef.current.titleBarRight = element\n          }}\n        />\n        {adLink && <div className=\"pinned\">{adLink}</div>}\n      </div>\n      <div\n        ref={adContainerRef}\n        css={[\n          adContainerStyle,\n          // this container covers Google DAI iframe UI, left some space for user to click \"skip ad\"\n          // see https://kkvideo.atlassian.net/browse/CPT-4535 for detail\n          adStatus && {margin: 'calc(var(--spacing) * 6) 0'},\n        ]}\n      >\n        {type === 'mobile' && (\n          <div\n            css={[\n              controlsStyle(type),\n              displayStyles[controlsDisplay],\n              invisible('--controller-middle-area'),\n            ]}\n          >\n            <ControlsBlock order=\"mobile\" {...controlButtons} />\n          </div>\n        )}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          {\n            marginTop: '1em',\n            paddingTop: 0,\n            paddingBottom: 'calc(1em - var(--bottom-spacing, 0) / 5)',\n            flexWrap: 'wrap',\n            '> div:first-of-type': seekbarStyles,\n          },\n          type === 'desktop' && desktopControls,\n          controlsDisplayStyles[controlsDisplay],\n          invisible('--controller-down-area'),\n        ]}\n      >\n        {seekbar || <div />}\n        {type === 'desktop' && (\n          <ControlsBlock\n            order=\"desktop\"\n            liveButton={liveButton}\n            {...controlButtons}\n          />\n        )}\n        {type === 'desktop' ? (\n          <div css={[displayStyles[controlsDisplay]]}>{displayTime}</div>\n        ) : (\n          <>\n            {liveButton}\n            <div\n              css={{\n                '&:not(:empty)': {\n                  flex: 1,\n                  marginLeft: 'calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n                },\n              }}\n            >\n              {displayTime}\n            </div>\n          </>\n        )}\n        {adStatus && (\n          <div\n            className=\"pinned\"\n            css={[infoBarSlotStyle, type === 'desktop' && desktopInfoBarStyle]}\n          >\n            {adStatus}\n          </div>\n        )}\n        <div\n          css={\n            !adStatus && [\n              infoBarSlotStyle,\n              type === 'desktop' && desktopInfoBarStyle,\n            ]\n          }\n          ref={element => {\n            slotRef.current.infoBar = element\n          }}\n        />\n        {volumeControl}\n        <div\n          css={[adStatus ? hidden : {display: 'flex'}]}\n          ref={element => {\n            slotRef.current.functionBar = element\n          }}\n        />\n        {fullscreenButton}\n      </div>\n      <SlotProvider slotRef={slotRef}>{children}</SlotProvider>\n    </div>\n  )\n}\n\nexport default DefaultLayout\n"]} */"],
|
|
3561
3505
|
ref: containerRef,
|
|
3562
3506
|
...rest,
|
|
3563
3507
|
children: [video, jsxs("div", {
|
|
3564
3508
|
ref: backRef,
|
|
3565
|
-
css: [backStyle, display !== 'hidden' && (haveBottomItem ? dropTop : drop), invisible('--controller-middle-area'), process.env.NODE_ENV === "production" ? "" : ";label:DefaultLayout;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["DefaultLayout.js"],"names":[],"mappings":"AAiXQ","file":"DefaultLayout.js","sourcesContent":["/* eslint-disable no-param-reassign */\n/* @jsxImportSource @emotion/react */\n/* eslint-disable react/prop-types */\nimport {useRef} from 'react'\nimport {SlotProvider} from './uiExtensions'\nimport {nativeTextStyle, customTextStyle, bottomUiTop} from './subtitles'\n\nconst hidden = {display: 'none'}\nconst invisible = hiddenArea => ({\n  visibility: `var(${hiddenArea}, visible)`,\n})\nconst containerStyle = {\n  '--spacing': '0.5em',\n  '--center-pointer-events': 'auto',\n  width: '100%',\n  height: '100%',\n  fontSize: '16px',\n  boxSizing: 'border-box',\n  display: 'flex',\n  flexDirection: 'column',\n  overflow: 'hidden',\n  color: 'white',\n  background: '#000',\n  // prevent animation glich(afterimage) of descendant elements\n  transform: 'translateX(0)',\n  userSelect: 'none',\n  'a, a:link, a:visited': {\n    color: '#fff',\n    opacity: 0.8,\n    textDecoration: 'none',\n  },\n  button: {\n    fontSize: 'inherit',\n    '> *': {\n      pointerEvents: 'none',\n    },\n  },\n  '--thumbnail-width': '160', // height 90\n  '--theme-color': 'var(--primary-highlight, red)',\n}\n\nconst videoContainerStyle = {\n  '> div:first-of-type': {\n    position: 'absolute',\n    zIndex: '-1',\n    display: 'flex',\n    alignItems: 'center',\n    width: '100%',\n    height: '100%',\n  },\n}\n\nconst drop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0.5) 0,\n    rgba(0,0,0,0) var(--player-ui-drop-shadow-size, 30%) calc(100% - var(--player-ui-drop-shadow-size, 30%)),\n    rgba(0,0,0,0.5) 100%\n  )`,\n  backgroundColor: `rgba(0, 0, 0, 0.3)`,\n}\n\nconst dropTop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0) 0,\n    rgba(0,0,0,0) 8rem calc(100% - 8rem),\n    rgba(0,0,0,0.5) 100%\n  )`,\n}\n\nconst responsiveStyles = {\n  desktop: {\n    fontSize: '24px',\n  }, // add if necessary: big-desktop\n}\n\nconst rowStyle = {\n  boxSizing: 'border-box',\n  width: '100%',\n  padding: '1em',\n  display: 'flex',\n  alignItems: 'center',\n  '> button, > div > button': {\n    padding: '0.5rem',\n  },\n}\n\n/* \n  Hint:\n    displayStyles specificity is higher then controlsStyle\n    Need to pay attention when we revise these styles.\n*/\n\nconst displayHidden = {\n  '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n    opacity: 0,\n    transform: 'translate(-200vw, 0)',\n    transition: 'opacity 0.8s var(--autohide-delay, 0s) ease-out, transform 0s calc(0.8s + var(--autohide-delay, 0s))',\n    'button': {\n      opacity: 0, // extra transition to hide tooltip for Safari\n      transition: 'opacity 0s calc(0.8s + var(--autohide-delay, 0s)) ease-out',\n    }\n  },\n}\n\nconst displayStyles = {\n  hidden: displayHidden,\n  'seekbar-only': displayHidden,\n  shown: {\n    '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n      zIndex: 1,\n      pointerEvents: 'auto',\n      transition: 'opacity 0.3s ease-out, transform 0s 0s',\n    },\n  },\n}\n\nconst controlsDisplayStyles = {\n  hidden: {\n    ...displayStyles.hidden,\n    '~ div:not(.pinned)': displayStyles.hidden,\n  },\n  'seekbar-only': {\n    '> div:first-of-type ~ *': displayStyles.hidden,\n  },\n  shown: {\n    ...displayStyles.shown,\n    '~ div': displayStyles.shown,\n  },\n}\n\n// TODO: transform into sample object instead of function\nconst controlsStyle = type => ({\n  position: 'absolute',\n  zIndex: '2',\n  width: '100%',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'space-evenly',\n  fontSize: '1.5em',\n  '> button': {\n    padding: '0.75em 0.5em',\n    '&:disabled': {\n      opacity: 0,\n      pointerEvents: 'none',\n    },\n    '&.play-button': {\n      padding: '0.75rem 0.5rem',\n      fontSize: 'var(--play-button-size, 1em)',\n    },\n    '&.firstplay-button':\n      type === 'mobile'\n        ? {fontSize: '200%', padding: '4em 10em', marginBottom: '1em'}\n        : {fontSize: '400%'},\n  },\n})\n\n// Remove the height for imporving tapping effect at current, and this height was used for fixed region before.\nconst infoBarSlotStyle = {\n  display: 'flex',\n  order: -1,\n  alignItems: 'center',\n  flex: '1 60%',\n}\n\nconst desktopInfoBarStyle = {\n  marginLeft: 'var(--spacing, 0.5em)',\n  order: 'initial',\n  flex: '1 0%',\n}\n\nconst textEllipsis = {\n  overflow: 'hidden',\n  whiteSpace: 'nowrap',\n  textOverflow: 'ellipsis',\n}\n\nconst infoStyle = {\n  zIndex: 1,\n  overflow: 'visible',\n  pointerEvents: 'none',\n  padding: '0 calc(2em - 20px)',\n  marginBottom: 'calc(0.5em + var(--bottom-spacing, 0))',\n  flex: '3em 0',\n  '> div:first-of-type, > div:nth-of-type(2), > button': {\n    minWidth: 0,\n  },\n  '> div:nth-of-type(2)': {\n    // title and channel title\n    margin: '0 calc(var(--spacing, 0.5em) / 2)',\n    flex: 1,\n    fontSize: '0.75em',\n    visibility: 'var(--player-title-visibility, visible)',\n    span: {\n      marginRight: 'var(--spacing, 0.5em)',\n    },\n    '> div': textEllipsis,\n  },\n  h1: {\n    margin: 0,\n    fontSize: '1.66em',\n    fontWeight: '500',\n    ...textEllipsis,\n    flex: 1,\n    '&:not(:only-child)': {\n      marginTop: '0.5em',\n    },\n  },\n  img: {\n    display: 'block',\n    width: '2em',\n    height: '2em',\n    borderRadius: '0.25em',\n    objectFit: 'cover',\n  },\n}\n\nconst backStyle = {\n  position: 'absolute',\n  zIndex: 0,\n  width: '100%',\n  height: '100%',\n  // move DAI UI slightly up to avoid stacking\n  paddingBottom: 'calc(var(--spacing) * 4)',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  '> iframe': {pointerEvents: 'auto'},\n}\n\nconst skipStyle = {\n  position: 'absolute',\n  right: 0,\n  bottom: '9rem',\n  textAlign: 'right',\n  button: {\n    width: 'auto',\n    height: 'auto',\n  },\n}\n\nconst desktopStyle = {\n  '--spacing': '0.75em',\n  '--center-pointer-events': 'none',\n  '> div button': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em)',\n  },\n  '--thumbnail-width': '288', // height 162\n}\n\nconst desktopControls = {\n  '> div:first-of-type': {\n    flex: '100%',\n    marginBottom: '0.4em',\n  },\n  '> button': {\n    '&[disabled]': {\n    display: 'none',\n    },\n    '&.play-button': {\n      fontSize: '1em',\n    },\n    '&.firstplay-button': {\n      fontSize: '400%'\n    },\n  },\n}\n\nconst seekbarStyles = {\n  flex: '1 var(--seekbar-flex-basis, 100%)',\n  '&:not(:empty)': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n  },\n}\n\nconst adContainerStyle = {\n  flexGrow: 1,\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  pointerEvents: 'var(--center-pointer-events)',\n  button: {pointerEvents: 'auto'},\n  zIndex: 0,\n}\n\nconst ControlsBlock = ({\n  order = 'mobile',\n  liveButton,\n  playButton,\n  rewindButton = '',\n  forwardButton = '',\n  previousEpisodeButton = '',\n  nextEpisodeButton = '',\n}) =>\n  order === 'desktop' ? (\n    <>\n      {liveButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {rewindButton}\n      {forwardButton}\n    </>\n  ) : (\n    <>\n      {rewindButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {forwardButton}\n    </>\n  )\n\nconst DefaultLayout = ({\n  type = 'mobile',\n  style,\n  display,\n  liveButton,\n  controlsDisplay = display,\n  size,\n  title = '',\n  channelTitle = '',\n  video,\n  haveBottomItem,\n  seekbar = '',\n  displayTime,\n  controlButtons,\n  volumeControl,\n  fullscreenButton,\n  backButton = '',\n  adStatus = '',\n  adLink = '',\n  adSkipButton,\n  backItems,\n  children,\n  containerRef,\n  backRef,\n  adContainerRef,\n  ...rest\n}) => {\n  const slotRef = useRef({})\n\n  return (\n    <div\n      style={{\n        '--player-bottom-ui-top':\n          controlsDisplay === 'shown'\n            ? `-${bottomUiTop(containerRef.current)}px`\n            : '-8px',\n        '--player-bottom-ui-height':\n          type === 'desktop' && controlsDisplay === 'shown' ? '3.5rem' : '0',\n      }}\n      css={[\n        containerStyle,\n        videoContainerStyle,\n        responsiveStyles[size],\n        type === 'desktop' && desktopStyle,\n        adStatus && {'--bottom-spacing': 0},\n        nativeTextStyle,\n        customTextStyle,\n        style,\n      ]}\n      ref={containerRef}\n      {...rest}\n    >\n      {video}\n      <div\n        ref={backRef}\n        css={[\n          backStyle,\n          display !== 'hidden' && (haveBottomItem ? dropTop : drop),\n          invisible('--controller-middle-area'),\n        ]}\n      >\n        {type !== 'mobile' && backItems}\n        {adSkipButton && <div css={skipStyle}>{adSkipButton}</div>}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          infoStyle,\n          displayStyles[display],\n          invisible('--controller-top-area'),\n        ]}\n      >\n        {backButton}\n        <div\n          ref={element => {\n            slotRef.current.titleBarLeft = element\n          }}\n        />\n        <div>\n          <h1>{title}</h1>\n          {channelTitle && <div>{channelTitle}</div>}\n        </div>\n        <div\n          ref={element => {\n            slotRef.current.titleBarRight = element\n          }}\n        />\n        {adLink && <div className=\"pinned\">{adLink}</div>}\n      </div>\n      <div\n        ref={adContainerRef}\n        css={[\n          adContainerStyle,\n          // this container covers Google DAI iframe UI, left some space for user to click \"skip ad\"\n          // see https://kkvideo.atlassian.net/browse/CPT-4535 for detail\n          adStatus && {margin: 'calc(var(--spacing) * 6) 0'},\n        ]}\n      >\n        {type === 'mobile' && (\n          <div\n            css={[\n              controlsStyle(type),\n              displayStyles[controlsDisplay],\n              invisible('--controller-middle-area'),\n            ]}\n          >\n            <ControlsBlock order=\"mobile\" {...controlButtons} />\n          </div>\n        )}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          {\n            marginTop: '1em',\n            paddingTop: 0,\n            paddingBottom: 'calc(1em - var(--bottom-spacing, 0) / 5)',\n            flexWrap: 'wrap',\n            '> div:first-of-type': seekbarStyles,\n          },\n          type === 'desktop' && desktopControls,\n          controlsDisplayStyles[controlsDisplay],\n          invisible('--controller-down-area'),\n        ]}\n      >\n        {seekbar || <div />}\n        {type === 'desktop' && (\n          <ControlsBlock\n            order=\"desktop\"\n            liveButton={liveButton}\n            {...controlButtons}\n          />\n        )}\n        {type === 'desktop' ? (\n          <div css={[displayStyles[controlsDisplay]]}>{displayTime}</div>\n        ) : (\n          <>\n            {liveButton}\n            <div\n              css={{\n                '&:not(:empty)': {\n                  flex: 1,\n                  marginLeft: 'calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n                },\n              }}\n            >\n              {displayTime}\n            </div>\n          </>\n        )}\n        {adStatus && (\n          <div\n            className=\"pinned\"\n            css={[infoBarSlotStyle, type === 'desktop' && desktopInfoBarStyle]}\n          >\n            {adStatus}\n          </div>\n        )}\n        <div\n          css={\n            !adStatus && [\n              infoBarSlotStyle,\n              type === 'desktop' && desktopInfoBarStyle,\n            ]\n          }\n          ref={element => {\n            slotRef.current.infoBar = element\n          }}\n        />\n        {volumeControl}\n        <div\n          css={adStatus && hidden}\n          ref={element => {\n            slotRef.current.functionBar = element\n          }}\n        />\n        {fullscreenButton}\n      </div>\n      <SlotProvider slotRef={slotRef}>{children}</SlotProvider>\n    </div>\n  )\n}\n\nexport default DefaultLayout\n"]} */"],
|
|
3509
|
+
css: [backStyle, display !== 'hidden' && (haveBottomItem ? dropTop : drop), invisible('--controller-middle-area'), process.env.NODE_ENV === "production" ? "" : ";label:DefaultLayout;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["DefaultLayout.js"],"names":[],"mappings":"AAiXQ","file":"DefaultLayout.js","sourcesContent":["/* eslint-disable no-param-reassign */\n/* @jsxImportSource @emotion/react */\n/* eslint-disable react/prop-types */\nimport {useRef} from 'react'\nimport {SlotProvider} from './uiExtensions'\nimport {nativeTextStyle, customTextStyle, bottomUiTop} from './subtitles'\n\nconst hidden = {display: 'none'}\nconst invisible = hiddenArea => ({\n  visibility: `var(${hiddenArea}, visible)`,\n})\nconst containerStyle = {\n  '--spacing': '0.5em',\n  '--center-pointer-events': 'auto',\n  width: '100%',\n  height: '100%',\n  fontSize: '16px',\n  boxSizing: 'border-box',\n  display: 'flex',\n  flexDirection: 'column',\n  overflow: 'hidden',\n  color: 'white',\n  background: '#000',\n  // prevent animation glich(afterimage) of descendant elements\n  transform: 'translateX(0)',\n  userSelect: 'none',\n  'a, a:link, a:visited': {\n    color: '#fff',\n    opacity: 0.8,\n    textDecoration: 'none',\n  },\n  button: {\n    fontSize: 'inherit',\n    '> *': {\n      pointerEvents: 'none',\n    },\n  },\n  '--thumbnail-width': '160', // height 90\n  '--theme-color': 'var(--primary-highlight, red)',\n}\n\nconst videoContainerStyle = {\n  '> div:first-of-type': {\n    position: 'absolute',\n    zIndex: '-1',\n    display: 'flex',\n    alignItems: 'center',\n    width: '100%',\n    height: '100%',\n  },\n}\n\nconst drop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0.5) 0,\n    rgba(0,0,0,0) var(--player-ui-drop-shadow-size, 30%) calc(100% - var(--player-ui-drop-shadow-size, 30%)),\n    rgba(0,0,0,0.5) 100%\n  )`,\n  backgroundColor: `rgba(0, 0, 0, 0.3)`,\n}\n\nconst dropTop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0) 0,\n    rgba(0,0,0,0) 8rem calc(100% - 8rem),\n    rgba(0,0,0,0.5) 100%\n  )`,\n}\n\nconst responsiveStyles = {\n  desktop: {\n    fontSize: '24px',\n  }, // add if necessary: big-desktop\n}\n\nconst rowStyle = {\n  boxSizing: 'border-box',\n  width: '100%',\n  padding: '1em',\n  display: 'flex',\n  alignItems: 'center',\n  '> button, > div > button': {\n    padding: '0.5rem',\n  },\n}\n\n/* \n  Hint:\n    displayStyles specificity is higher then controlsStyle\n    Need to pay attention when we revise these styles.\n*/\n\nconst displayHidden = {\n  '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n    opacity: 0,\n    transform: 'translate(-200vw, 0)',\n    transition: 'opacity 0.8s var(--autohide-delay, 0s) ease-out, transform 0s calc(0.8s + var(--autohide-delay, 0s))',\n    'button': {\n      opacity: 0, // extra transition to hide tooltip for Safari\n      transition: 'opacity 0s calc(0.8s + var(--autohide-delay, 0s)) ease-out',\n    }\n  },\n}\n\nconst displayStyles = {\n  hidden: displayHidden,\n  'seekbar-only': displayHidden,\n  shown: {\n    '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n      zIndex: 1,\n      pointerEvents: 'auto',\n      transition: 'opacity 0.3s ease-out, transform 0s 0s',\n    },\n  },\n}\n\nconst controlsDisplayStyles = {\n  hidden: {\n    ...displayStyles.hidden,\n    '~ div:not(.pinned)': displayStyles.hidden,\n  },\n  'seekbar-only': {\n    '> div:first-of-type ~ *': displayStyles.hidden,\n  },\n  shown: {\n    ...displayStyles.shown,\n    '~ div': displayStyles.shown,\n  },\n}\n\n// TODO: transform into sample object instead of function\nconst controlsStyle = type => ({\n  position: 'absolute',\n  zIndex: '2',\n  width: '100%',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'space-evenly',\n  fontSize: '1.5em',\n  '> button': {\n    padding: '0.75em 0.5em',\n    '&:disabled': {\n      opacity: 0,\n      pointerEvents: 'none',\n    },\n    '&.play-button': {\n      padding: '0.75rem 0.5rem',\n      fontSize: 'var(--play-button-size, 1em)',\n    },\n    '&.firstplay-button':\n      type === 'mobile'\n        ? {fontSize: '200%', padding: '4em 10em', marginBottom: '1em'}\n        : {fontSize: '400%'},\n  },\n})\n\n// Remove the height for imporving tapping effect at current, and this height was used for fixed region before.\nconst infoBarSlotStyle = {\n  display: 'flex',\n  order: -1,\n  alignItems: 'center',\n  flex: '1 60%',\n}\n\nconst desktopInfoBarStyle = {\n  marginLeft: 'var(--spacing, 0.5em)',\n  order: 'initial',\n  flex: '1 0%',\n}\n\nconst textEllipsis = {\n  overflow: 'hidden',\n  whiteSpace: 'nowrap',\n  textOverflow: 'ellipsis',\n}\n\nconst infoStyle = {\n  zIndex: 1,\n  overflow: 'visible',\n  pointerEvents: 'none',\n  padding: '0 calc(2em - 20px)',\n  marginBottom: 'calc(0.5em + var(--bottom-spacing, 0))',\n  flex: '3em 0',\n  '> div:first-of-type, > div:nth-of-type(2), > button': {\n    minWidth: 0,\n  },\n  '> div:nth-of-type(2)': {\n    // title and channel title\n    margin: '0 calc(var(--spacing, 0.5em) / 2)',\n    flex: 1,\n    fontSize: '0.75em',\n    visibility: 'var(--player-title-visibility, visible)',\n    span: {\n      marginRight: 'var(--spacing, 0.5em)',\n    },\n    '> div': textEllipsis,\n  },\n  h1: {\n    margin: 0,\n    fontSize: '1.66em',\n    fontWeight: '500',\n    ...textEllipsis,\n    flex: 1,\n    '&:not(:only-child)': {\n      marginTop: '0.5em',\n    },\n  },\n  img: {\n    display: 'block',\n    width: '2em',\n    height: '2em',\n    borderRadius: '0.25em',\n    objectFit: 'cover',\n  },\n}\n\nconst backStyle = {\n  position: 'absolute',\n  zIndex: 0,\n  width: '100%',\n  height: '100%',\n  // move DAI UI slightly up to avoid stacking\n  paddingBottom: 'calc(var(--spacing) * 4)',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  '> iframe': {pointerEvents: 'auto'},\n}\n\nconst skipStyle = {\n  position: 'absolute',\n  right: 0,\n  bottom: '9rem',\n  textAlign: 'right',\n  button: {\n    width: 'auto',\n    height: 'auto',\n  },\n}\n\nconst desktopStyle = {\n  '--spacing': '0.75em',\n  '--center-pointer-events': 'none',\n  '> div button': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em)',\n  },\n  '--thumbnail-width': '288', // height 162\n}\n\nconst desktopControls = {\n  '> div:first-of-type': {\n    flex: '100%',\n    marginBottom: '0.4em',\n  },\n  '> button': {\n    '&[disabled]': {\n    display: 'none',\n    },\n    '&.play-button': {\n      fontSize: '1em',\n    },\n    '&.firstplay-button': {\n      fontSize: '400%'\n    },\n  },\n}\n\nconst seekbarStyles = {\n  flex: '1 var(--seekbar-flex-basis, 100%)',\n  '&:not(:empty)': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n  },\n}\n\nconst adContainerStyle = {\n  flexGrow: 1,\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  pointerEvents: 'var(--center-pointer-events)',\n  button: {pointerEvents: 'auto'},\n  zIndex: 0,\n}\n\nconst ControlsBlock = ({\n  order = 'mobile',\n  liveButton,\n  playButton,\n  rewindButton = '',\n  forwardButton = '',\n  previousEpisodeButton = '',\n  nextEpisodeButton = '',\n}) =>\n  order === 'desktop' ? (\n    <>\n      {liveButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {rewindButton}\n      {forwardButton}\n    </>\n  ) : (\n    <>\n      {rewindButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {forwardButton}\n    </>\n  )\n\nconst DefaultLayout = ({\n  type = 'mobile',\n  style,\n  display,\n  liveButton,\n  controlsDisplay = display,\n  size,\n  title = '',\n  channelTitle = '',\n  video,\n  haveBottomItem,\n  seekbar = '',\n  displayTime,\n  controlButtons,\n  volumeControl,\n  fullscreenButton,\n  backButton = '',\n  adStatus = '',\n  adLink = '',\n  adSkipButton,\n  backItems,\n  children,\n  containerRef,\n  backRef,\n  adContainerRef,\n  ...rest\n}) => {\n  const slotRef = useRef({})\n\n  return (\n    <div\n      style={{\n        '--player-bottom-ui-top':\n          controlsDisplay === 'shown'\n            ? `-${bottomUiTop(containerRef.current)}px`\n            : '-8px',\n        '--player-bottom-ui-height':\n          type === 'desktop' && controlsDisplay === 'shown' ? '3.5rem' : '0',\n      }}\n      css={[\n        containerStyle,\n        videoContainerStyle,\n        responsiveStyles[size],\n        type === 'desktop' && desktopStyle,\n        adStatus && {'--bottom-spacing': 0},\n        nativeTextStyle,\n        customTextStyle,\n        style,\n      ]}\n      ref={containerRef}\n      {...rest}\n    >\n      {video}\n      <div\n        ref={backRef}\n        css={[\n          backStyle,\n          display !== 'hidden' && (haveBottomItem ? dropTop : drop),\n          invisible('--controller-middle-area'),\n        ]}\n      >\n        {type !== 'mobile' && backItems}\n        {adSkipButton && <div css={skipStyle}>{adSkipButton}</div>}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          infoStyle,\n          displayStyles[display],\n          invisible('--controller-top-area'),\n        ]}\n      >\n        {backButton}\n        <div\n          ref={element => {\n            slotRef.current.titleBarLeft = element\n          }}\n        />\n        <div>\n          <h1>{title}</h1>\n          {channelTitle && <div>{channelTitle}</div>}\n        </div>\n        <div\n          ref={element => {\n            slotRef.current.titleBarRight = element\n          }}\n        />\n        {adLink && <div className=\"pinned\">{adLink}</div>}\n      </div>\n      <div\n        ref={adContainerRef}\n        css={[\n          adContainerStyle,\n          // this container covers Google DAI iframe UI, left some space for user to click \"skip ad\"\n          // see https://kkvideo.atlassian.net/browse/CPT-4535 for detail\n          adStatus && {margin: 'calc(var(--spacing) * 6) 0'},\n        ]}\n      >\n        {type === 'mobile' && (\n          <div\n            css={[\n              controlsStyle(type),\n              displayStyles[controlsDisplay],\n              invisible('--controller-middle-area'),\n            ]}\n          >\n            <ControlsBlock order=\"mobile\" {...controlButtons} />\n          </div>\n        )}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          {\n            marginTop: '1em',\n            paddingTop: 0,\n            paddingBottom: 'calc(1em - var(--bottom-spacing, 0) / 5)',\n            flexWrap: 'wrap',\n            '> div:first-of-type': seekbarStyles,\n          },\n          type === 'desktop' && desktopControls,\n          controlsDisplayStyles[controlsDisplay],\n          invisible('--controller-down-area'),\n        ]}\n      >\n        {seekbar || <div />}\n        {type === 'desktop' && (\n          <ControlsBlock\n            order=\"desktop\"\n            liveButton={liveButton}\n            {...controlButtons}\n          />\n        )}\n        {type === 'desktop' ? (\n          <div css={[displayStyles[controlsDisplay]]}>{displayTime}</div>\n        ) : (\n          <>\n            {liveButton}\n            <div\n              css={{\n                '&:not(:empty)': {\n                  flex: 1,\n                  marginLeft: 'calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n                },\n              }}\n            >\n              {displayTime}\n            </div>\n          </>\n        )}\n        {adStatus && (\n          <div\n            className=\"pinned\"\n            css={[infoBarSlotStyle, type === 'desktop' && desktopInfoBarStyle]}\n          >\n            {adStatus}\n          </div>\n        )}\n        <div\n          css={\n            !adStatus && [\n              infoBarSlotStyle,\n              type === 'desktop' && desktopInfoBarStyle,\n            ]\n          }\n          ref={element => {\n            slotRef.current.infoBar = element\n          }}\n        />\n        {volumeControl}\n        <div\n          css={[adStatus ? hidden : {display: 'flex'}]}\n          ref={element => {\n            slotRef.current.functionBar = element\n          }}\n        />\n        {fullscreenButton}\n      </div>\n      <SlotProvider slotRef={slotRef}>{children}</SlotProvider>\n    </div>\n  )\n}\n\nexport default DefaultLayout\n"]} */"],
|
|
3566
3510
|
children: [type !== 'mobile' && backItems, adSkipButton && jsx$1("div", {
|
|
3567
3511
|
css: skipStyle,
|
|
3568
3512
|
children: adSkipButton
|
|
3569
3513
|
})]
|
|
3570
3514
|
}), jsxs("div", {
|
|
3571
|
-
css: [rowStyle, infoStyle, displayStyles[display], invisible('--controller-top-area'), process.env.NODE_ENV === "production" ? "" : ";label:DefaultLayout;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["DefaultLayout.js"],"names":[],"mappings":"AA2XQ","file":"DefaultLayout.js","sourcesContent":["/* eslint-disable no-param-reassign */\n/* @jsxImportSource @emotion/react */\n/* eslint-disable react/prop-types */\nimport {useRef} from 'react'\nimport {SlotProvider} from './uiExtensions'\nimport {nativeTextStyle, customTextStyle, bottomUiTop} from './subtitles'\n\nconst hidden = {display: 'none'}\nconst invisible = hiddenArea => ({\n  visibility: `var(${hiddenArea}, visible)`,\n})\nconst containerStyle = {\n  '--spacing': '0.5em',\n  '--center-pointer-events': 'auto',\n  width: '100%',\n  height: '100%',\n  fontSize: '16px',\n  boxSizing: 'border-box',\n  display: 'flex',\n  flexDirection: 'column',\n  overflow: 'hidden',\n  color: 'white',\n  background: '#000',\n  // prevent animation glich(afterimage) of descendant elements\n  transform: 'translateX(0)',\n  userSelect: 'none',\n  'a, a:link, a:visited': {\n    color: '#fff',\n    opacity: 0.8,\n    textDecoration: 'none',\n  },\n  button: {\n    fontSize: 'inherit',\n    '> *': {\n      pointerEvents: 'none',\n    },\n  },\n  '--thumbnail-width': '160', // height 90\n  '--theme-color': 'var(--primary-highlight, red)',\n}\n\nconst videoContainerStyle = {\n  '> div:first-of-type': {\n    position: 'absolute',\n    zIndex: '-1',\n    display: 'flex',\n    alignItems: 'center',\n    width: '100%',\n    height: '100%',\n  },\n}\n\nconst drop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0.5) 0,\n    rgba(0,0,0,0) var(--player-ui-drop-shadow-size, 30%) calc(100% - var(--player-ui-drop-shadow-size, 30%)),\n    rgba(0,0,0,0.5) 100%\n  )`,\n  backgroundColor: `rgba(0, 0, 0, 0.3)`,\n}\n\nconst dropTop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0) 0,\n    rgba(0,0,0,0) 8rem calc(100% - 8rem),\n    rgba(0,0,0,0.5) 100%\n  )`,\n}\n\nconst responsiveStyles = {\n  desktop: {\n    fontSize: '24px',\n  }, // add if necessary: big-desktop\n}\n\nconst rowStyle = {\n  boxSizing: 'border-box',\n  width: '100%',\n  padding: '1em',\n  display: 'flex',\n  alignItems: 'center',\n  '> button, > div > button': {\n    padding: '0.5rem',\n  },\n}\n\n/* \n  Hint:\n    displayStyles specificity is higher then controlsStyle\n    Need to pay attention when we revise these styles.\n*/\n\nconst displayHidden = {\n  '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n    opacity: 0,\n    transform: 'translate(-200vw, 0)',\n    transition: 'opacity 0.8s var(--autohide-delay, 0s) ease-out, transform 0s calc(0.8s + var(--autohide-delay, 0s))',\n    'button': {\n      opacity: 0, // extra transition to hide tooltip for Safari\n      transition: 'opacity 0s calc(0.8s + var(--autohide-delay, 0s)) ease-out',\n    }\n  },\n}\n\nconst displayStyles = {\n  hidden: displayHidden,\n  'seekbar-only': displayHidden,\n  shown: {\n    '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n      zIndex: 1,\n      pointerEvents: 'auto',\n      transition: 'opacity 0.3s ease-out, transform 0s 0s',\n    },\n  },\n}\n\nconst controlsDisplayStyles = {\n  hidden: {\n    ...displayStyles.hidden,\n    '~ div:not(.pinned)': displayStyles.hidden,\n  },\n  'seekbar-only': {\n    '> div:first-of-type ~ *': displayStyles.hidden,\n  },\n  shown: {\n    ...displayStyles.shown,\n    '~ div': displayStyles.shown,\n  },\n}\n\n// TODO: transform into sample object instead of function\nconst controlsStyle = type => ({\n  position: 'absolute',\n  zIndex: '2',\n  width: '100%',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'space-evenly',\n  fontSize: '1.5em',\n  '> button': {\n    padding: '0.75em 0.5em',\n    '&:disabled': {\n      opacity: 0,\n      pointerEvents: 'none',\n    },\n    '&.play-button': {\n      padding: '0.75rem 0.5rem',\n      fontSize: 'var(--play-button-size, 1em)',\n    },\n    '&.firstplay-button':\n      type === 'mobile'\n        ? {fontSize: '200%', padding: '4em 10em', marginBottom: '1em'}\n        : {fontSize: '400%'},\n  },\n})\n\n// Remove the height for imporving tapping effect at current, and this height was used for fixed region before.\nconst infoBarSlotStyle = {\n  display: 'flex',\n  order: -1,\n  alignItems: 'center',\n  flex: '1 60%',\n}\n\nconst desktopInfoBarStyle = {\n  marginLeft: 'var(--spacing, 0.5em)',\n  order: 'initial',\n  flex: '1 0%',\n}\n\nconst textEllipsis = {\n  overflow: 'hidden',\n  whiteSpace: 'nowrap',\n  textOverflow: 'ellipsis',\n}\n\nconst infoStyle = {\n  zIndex: 1,\n  overflow: 'visible',\n  pointerEvents: 'none',\n  padding: '0 calc(2em - 20px)',\n  marginBottom: 'calc(0.5em + var(--bottom-spacing, 0))',\n  flex: '3em 0',\n  '> div:first-of-type, > div:nth-of-type(2), > button': {\n    minWidth: 0,\n  },\n  '> div:nth-of-type(2)': {\n    // title and channel title\n    margin: '0 calc(var(--spacing, 0.5em) / 2)',\n    flex: 1,\n    fontSize: '0.75em',\n    visibility: 'var(--player-title-visibility, visible)',\n    span: {\n      marginRight: 'var(--spacing, 0.5em)',\n    },\n    '> div': textEllipsis,\n  },\n  h1: {\n    margin: 0,\n    fontSize: '1.66em',\n    fontWeight: '500',\n    ...textEllipsis,\n    flex: 1,\n    '&:not(:only-child)': {\n      marginTop: '0.5em',\n    },\n  },\n  img: {\n    display: 'block',\n    width: '2em',\n    height: '2em',\n    borderRadius: '0.25em',\n    objectFit: 'cover',\n  },\n}\n\nconst backStyle = {\n  position: 'absolute',\n  zIndex: 0,\n  width: '100%',\n  height: '100%',\n  // move DAI UI slightly up to avoid stacking\n  paddingBottom: 'calc(var(--spacing) * 4)',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  '> iframe': {pointerEvents: 'auto'},\n}\n\nconst skipStyle = {\n  position: 'absolute',\n  right: 0,\n  bottom: '9rem',\n  textAlign: 'right',\n  button: {\n    width: 'auto',\n    height: 'auto',\n  },\n}\n\nconst desktopStyle = {\n  '--spacing': '0.75em',\n  '--center-pointer-events': 'none',\n  '> div button': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em)',\n  },\n  '--thumbnail-width': '288', // height 162\n}\n\nconst desktopControls = {\n  '> div:first-of-type': {\n    flex: '100%',\n    marginBottom: '0.4em',\n  },\n  '> button': {\n    '&[disabled]': {\n    display: 'none',\n    },\n    '&.play-button': {\n      fontSize: '1em',\n    },\n    '&.firstplay-button': {\n      fontSize: '400%'\n    },\n  },\n}\n\nconst seekbarStyles = {\n  flex: '1 var(--seekbar-flex-basis, 100%)',\n  '&:not(:empty)': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n  },\n}\n\nconst adContainerStyle = {\n  flexGrow: 1,\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  pointerEvents: 'var(--center-pointer-events)',\n  button: {pointerEvents: 'auto'},\n  zIndex: 0,\n}\n\nconst ControlsBlock = ({\n  order = 'mobile',\n  liveButton,\n  playButton,\n  rewindButton = '',\n  forwardButton = '',\n  previousEpisodeButton = '',\n  nextEpisodeButton = '',\n}) =>\n  order === 'desktop' ? (\n    <>\n      {liveButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {rewindButton}\n      {forwardButton}\n    </>\n  ) : (\n    <>\n      {rewindButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {forwardButton}\n    </>\n  )\n\nconst DefaultLayout = ({\n  type = 'mobile',\n  style,\n  display,\n  liveButton,\n  controlsDisplay = display,\n  size,\n  title = '',\n  channelTitle = '',\n  video,\n  haveBottomItem,\n  seekbar = '',\n  displayTime,\n  controlButtons,\n  volumeControl,\n  fullscreenButton,\n  backButton = '',\n  adStatus = '',\n  adLink = '',\n  adSkipButton,\n  backItems,\n  children,\n  containerRef,\n  backRef,\n  adContainerRef,\n  ...rest\n}) => {\n  const slotRef = useRef({})\n\n  return (\n    <div\n      style={{\n        '--player-bottom-ui-top':\n          controlsDisplay === 'shown'\n            ? `-${bottomUiTop(containerRef.current)}px`\n            : '-8px',\n        '--player-bottom-ui-height':\n          type === 'desktop' && controlsDisplay === 'shown' ? '3.5rem' : '0',\n      }}\n      css={[\n        containerStyle,\n        videoContainerStyle,\n        responsiveStyles[size],\n        type === 'desktop' && desktopStyle,\n        adStatus && {'--bottom-spacing': 0},\n        nativeTextStyle,\n        customTextStyle,\n        style,\n      ]}\n      ref={containerRef}\n      {...rest}\n    >\n      {video}\n      <div\n        ref={backRef}\n        css={[\n          backStyle,\n          display !== 'hidden' && (haveBottomItem ? dropTop : drop),\n          invisible('--controller-middle-area'),\n        ]}\n      >\n        {type !== 'mobile' && backItems}\n        {adSkipButton && <div css={skipStyle}>{adSkipButton}</div>}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          infoStyle,\n          displayStyles[display],\n          invisible('--controller-top-area'),\n        ]}\n      >\n        {backButton}\n        <div\n          ref={element => {\n            slotRef.current.titleBarLeft = element\n          }}\n        />\n        <div>\n          <h1>{title}</h1>\n          {channelTitle && <div>{channelTitle}</div>}\n        </div>\n        <div\n          ref={element => {\n            slotRef.current.titleBarRight = element\n          }}\n        />\n        {adLink && <div className=\"pinned\">{adLink}</div>}\n      </div>\n      <div\n        ref={adContainerRef}\n        css={[\n          adContainerStyle,\n          // this container covers Google DAI iframe UI, left some space for user to click \"skip ad\"\n          // see https://kkvideo.atlassian.net/browse/CPT-4535 for detail\n          adStatus && {margin: 'calc(var(--spacing) * 6) 0'},\n        ]}\n      >\n        {type === 'mobile' && (\n          <div\n            css={[\n              controlsStyle(type),\n              displayStyles[controlsDisplay],\n              invisible('--controller-middle-area'),\n            ]}\n          >\n            <ControlsBlock order=\"mobile\" {...controlButtons} />\n          </div>\n        )}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          {\n            marginTop: '1em',\n            paddingTop: 0,\n            paddingBottom: 'calc(1em - var(--bottom-spacing, 0) / 5)',\n            flexWrap: 'wrap',\n            '> div:first-of-type': seekbarStyles,\n          },\n          type === 'desktop' && desktopControls,\n          controlsDisplayStyles[controlsDisplay],\n          invisible('--controller-down-area'),\n        ]}\n      >\n        {seekbar || <div />}\n        {type === 'desktop' && (\n          <ControlsBlock\n            order=\"desktop\"\n            liveButton={liveButton}\n            {...controlButtons}\n          />\n        )}\n        {type === 'desktop' ? (\n          <div css={[displayStyles[controlsDisplay]]}>{displayTime}</div>\n        ) : (\n          <>\n            {liveButton}\n            <div\n              css={{\n                '&:not(:empty)': {\n                  flex: 1,\n                  marginLeft: 'calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n                },\n              }}\n            >\n              {displayTime}\n            </div>\n          </>\n        )}\n        {adStatus && (\n          <div\n            className=\"pinned\"\n            css={[infoBarSlotStyle, type === 'desktop' && desktopInfoBarStyle]}\n          >\n            {adStatus}\n          </div>\n        )}\n        <div\n          css={\n            !adStatus && [\n              infoBarSlotStyle,\n              type === 'desktop' && desktopInfoBarStyle,\n            ]\n          }\n          ref={element => {\n            slotRef.current.infoBar = element\n          }}\n        />\n        {volumeControl}\n        <div\n          css={adStatus && hidden}\n          ref={element => {\n            slotRef.current.functionBar = element\n          }}\n        />\n        {fullscreenButton}\n      </div>\n      <SlotProvider slotRef={slotRef}>{children}</SlotProvider>\n    </div>\n  )\n}\n\nexport default DefaultLayout\n"]} */"],
|
|
3515
|
+
css: [rowStyle, infoStyle, displayStyles[display], invisible('--controller-top-area'), process.env.NODE_ENV === "production" ? "" : ";label:DefaultLayout;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["DefaultLayout.js"],"names":[],"mappings":"AA2XQ","file":"DefaultLayout.js","sourcesContent":["/* eslint-disable no-param-reassign */\n/* @jsxImportSource @emotion/react */\n/* eslint-disable react/prop-types */\nimport {useRef} from 'react'\nimport {SlotProvider} from './uiExtensions'\nimport {nativeTextStyle, customTextStyle, bottomUiTop} from './subtitles'\n\nconst hidden = {display: 'none'}\nconst invisible = hiddenArea => ({\n  visibility: `var(${hiddenArea}, visible)`,\n})\nconst containerStyle = {\n  '--spacing': '0.5em',\n  '--center-pointer-events': 'auto',\n  width: '100%',\n  height: '100%',\n  fontSize: '16px',\n  boxSizing: 'border-box',\n  display: 'flex',\n  flexDirection: 'column',\n  overflow: 'hidden',\n  color: 'white',\n  background: '#000',\n  // prevent animation glich(afterimage) of descendant elements\n  transform: 'translateX(0)',\n  userSelect: 'none',\n  'a, a:link, a:visited': {\n    color: '#fff',\n    opacity: 0.8,\n    textDecoration: 'none',\n  },\n  button: {\n    fontSize: 'inherit',\n    '> *': {\n      pointerEvents: 'none',\n    },\n  },\n  '--thumbnail-width': '160', // height 90\n  '--theme-color': 'var(--primary-highlight, red)',\n}\n\nconst videoContainerStyle = {\n  '> div:first-of-type': {\n    position: 'absolute',\n    zIndex: '-1',\n    display: 'flex',\n    alignItems: 'center',\n    width: '100%',\n    height: '100%',\n  },\n}\n\nconst drop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0.5) 0,\n    rgba(0,0,0,0) var(--player-ui-drop-shadow-size, 30%) calc(100% - var(--player-ui-drop-shadow-size, 30%)),\n    rgba(0,0,0,0.5) 100%\n  )`,\n  backgroundColor: `rgba(0, 0, 0, 0.3)`,\n}\n\nconst dropTop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0) 0,\n    rgba(0,0,0,0) 8rem calc(100% - 8rem),\n    rgba(0,0,0,0.5) 100%\n  )`,\n}\n\nconst responsiveStyles = {\n  desktop: {\n    fontSize: '24px',\n  }, // add if necessary: big-desktop\n}\n\nconst rowStyle = {\n  boxSizing: 'border-box',\n  width: '100%',\n  padding: '1em',\n  display: 'flex',\n  alignItems: 'center',\n  '> button, > div > button': {\n    padding: '0.5rem',\n  },\n}\n\n/* \n  Hint:\n    displayStyles specificity is higher then controlsStyle\n    Need to pay attention when we revise these styles.\n*/\n\nconst displayHidden = {\n  '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n    opacity: 0,\n    transform: 'translate(-200vw, 0)',\n    transition: 'opacity 0.8s var(--autohide-delay, 0s) ease-out, transform 0s calc(0.8s + var(--autohide-delay, 0s))',\n    'button': {\n      opacity: 0, // extra transition to hide tooltip for Safari\n      transition: 'opacity 0s calc(0.8s + var(--autohide-delay, 0s)) ease-out',\n    }\n  },\n}\n\nconst displayStyles = {\n  hidden: displayHidden,\n  'seekbar-only': displayHidden,\n  shown: {\n    '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n      zIndex: 1,\n      pointerEvents: 'auto',\n      transition: 'opacity 0.3s ease-out, transform 0s 0s',\n    },\n  },\n}\n\nconst controlsDisplayStyles = {\n  hidden: {\n    ...displayStyles.hidden,\n    '~ div:not(.pinned)': displayStyles.hidden,\n  },\n  'seekbar-only': {\n    '> div:first-of-type ~ *': displayStyles.hidden,\n  },\n  shown: {\n    ...displayStyles.shown,\n    '~ div': displayStyles.shown,\n  },\n}\n\n// TODO: transform into sample object instead of function\nconst controlsStyle = type => ({\n  position: 'absolute',\n  zIndex: '2',\n  width: '100%',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'space-evenly',\n  fontSize: '1.5em',\n  '> button': {\n    padding: '0.75em 0.5em',\n    '&:disabled': {\n      opacity: 0,\n      pointerEvents: 'none',\n    },\n    '&.play-button': {\n      padding: '0.75rem 0.5rem',\n      fontSize: 'var(--play-button-size, 1em)',\n    },\n    '&.firstplay-button':\n      type === 'mobile'\n        ? {fontSize: '200%', padding: '4em 10em', marginBottom: '1em'}\n        : {fontSize: '400%'},\n  },\n})\n\n// Remove the height for imporving tapping effect at current, and this height was used for fixed region before.\nconst infoBarSlotStyle = {\n  display: 'flex',\n  order: -1,\n  alignItems: 'center',\n  flex: '1 60%',\n}\n\nconst desktopInfoBarStyle = {\n  marginLeft: 'var(--spacing, 0.5em)',\n  order: 'initial',\n  flex: '1 0%',\n}\n\nconst textEllipsis = {\n  overflow: 'hidden',\n  whiteSpace: 'nowrap',\n  textOverflow: 'ellipsis',\n}\n\nconst infoStyle = {\n  zIndex: 1,\n  overflow: 'visible',\n  pointerEvents: 'none',\n  padding: '0 calc(2em - 20px)',\n  marginBottom: 'calc(0.5em + var(--bottom-spacing, 0))',\n  flex: '3em 0',\n  '> div:first-of-type, > div:nth-of-type(2), > button': {\n    minWidth: 0,\n  },\n  '> div:nth-of-type(2)': {\n    // title and channel title\n    margin: '0 calc(var(--spacing, 0.5em) / 2)',\n    flex: 1,\n    fontSize: '0.75em',\n    visibility: 'var(--player-title-visibility, visible)',\n    span: {\n      marginRight: 'var(--spacing, 0.5em)',\n    },\n    '> div': textEllipsis,\n  },\n  h1: {\n    margin: 0,\n    fontSize: '1.66em',\n    fontWeight: '500',\n    ...textEllipsis,\n    flex: 1,\n    '&:not(:only-child)': {\n      marginTop: '0.5em',\n    },\n  },\n  img: {\n    display: 'block',\n    width: '2em',\n    height: '2em',\n    borderRadius: '0.25em',\n    objectFit: 'cover',\n  },\n}\n\nconst backStyle = {\n  position: 'absolute',\n  zIndex: 0,\n  width: '100%',\n  height: '100%',\n  // move DAI UI slightly up to avoid stacking\n  paddingBottom: 'calc(var(--spacing) * 4)',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  '> iframe': {pointerEvents: 'auto'},\n}\n\nconst skipStyle = {\n  position: 'absolute',\n  right: 0,\n  bottom: '9rem',\n  textAlign: 'right',\n  button: {\n    width: 'auto',\n    height: 'auto',\n  },\n}\n\nconst desktopStyle = {\n  '--spacing': '0.75em',\n  '--center-pointer-events': 'none',\n  '> div button': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em)',\n  },\n  '--thumbnail-width': '288', // height 162\n}\n\nconst desktopControls = {\n  '> div:first-of-type': {\n    flex: '100%',\n    marginBottom: '0.4em',\n  },\n  '> button': {\n    '&[disabled]': {\n    display: 'none',\n    },\n    '&.play-button': {\n      fontSize: '1em',\n    },\n    '&.firstplay-button': {\n      fontSize: '400%'\n    },\n  },\n}\n\nconst seekbarStyles = {\n  flex: '1 var(--seekbar-flex-basis, 100%)',\n  '&:not(:empty)': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n  },\n}\n\nconst adContainerStyle = {\n  flexGrow: 1,\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  pointerEvents: 'var(--center-pointer-events)',\n  button: {pointerEvents: 'auto'},\n  zIndex: 0,\n}\n\nconst ControlsBlock = ({\n  order = 'mobile',\n  liveButton,\n  playButton,\n  rewindButton = '',\n  forwardButton = '',\n  previousEpisodeButton = '',\n  nextEpisodeButton = '',\n}) =>\n  order === 'desktop' ? (\n    <>\n      {liveButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {rewindButton}\n      {forwardButton}\n    </>\n  ) : (\n    <>\n      {rewindButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {forwardButton}\n    </>\n  )\n\nconst DefaultLayout = ({\n  type = 'mobile',\n  style,\n  display,\n  liveButton,\n  controlsDisplay = display,\n  size,\n  title = '',\n  channelTitle = '',\n  video,\n  haveBottomItem,\n  seekbar = '',\n  displayTime,\n  controlButtons,\n  volumeControl,\n  fullscreenButton,\n  backButton = '',\n  adStatus = '',\n  adLink = '',\n  adSkipButton,\n  backItems,\n  children,\n  containerRef,\n  backRef,\n  adContainerRef,\n  ...rest\n}) => {\n  const slotRef = useRef({})\n\n  return (\n    <div\n      style={{\n        '--player-bottom-ui-top':\n          controlsDisplay === 'shown'\n            ? `-${bottomUiTop(containerRef.current)}px`\n            : '-8px',\n        '--player-bottom-ui-height':\n          type === 'desktop' && controlsDisplay === 'shown' ? '3.5rem' : '0',\n      }}\n      css={[\n        containerStyle,\n        videoContainerStyle,\n        responsiveStyles[size],\n        type === 'desktop' && desktopStyle,\n        adStatus && {'--bottom-spacing': 0},\n        nativeTextStyle,\n        customTextStyle,\n        style,\n      ]}\n      ref={containerRef}\n      {...rest}\n    >\n      {video}\n      <div\n        ref={backRef}\n        css={[\n          backStyle,\n          display !== 'hidden' && (haveBottomItem ? dropTop : drop),\n          invisible('--controller-middle-area'),\n        ]}\n      >\n        {type !== 'mobile' && backItems}\n        {adSkipButton && <div css={skipStyle}>{adSkipButton}</div>}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          infoStyle,\n          displayStyles[display],\n          invisible('--controller-top-area'),\n        ]}\n      >\n        {backButton}\n        <div\n          ref={element => {\n            slotRef.current.titleBarLeft = element\n          }}\n        />\n        <div>\n          <h1>{title}</h1>\n          {channelTitle && <div>{channelTitle}</div>}\n        </div>\n        <div\n          ref={element => {\n            slotRef.current.titleBarRight = element\n          }}\n        />\n        {adLink && <div className=\"pinned\">{adLink}</div>}\n      </div>\n      <div\n        ref={adContainerRef}\n        css={[\n          adContainerStyle,\n          // this container covers Google DAI iframe UI, left some space for user to click \"skip ad\"\n          // see https://kkvideo.atlassian.net/browse/CPT-4535 for detail\n          adStatus && {margin: 'calc(var(--spacing) * 6) 0'},\n        ]}\n      >\n        {type === 'mobile' && (\n          <div\n            css={[\n              controlsStyle(type),\n              displayStyles[controlsDisplay],\n              invisible('--controller-middle-area'),\n            ]}\n          >\n            <ControlsBlock order=\"mobile\" {...controlButtons} />\n          </div>\n        )}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          {\n            marginTop: '1em',\n            paddingTop: 0,\n            paddingBottom: 'calc(1em - var(--bottom-spacing, 0) / 5)',\n            flexWrap: 'wrap',\n            '> div:first-of-type': seekbarStyles,\n          },\n          type === 'desktop' && desktopControls,\n          controlsDisplayStyles[controlsDisplay],\n          invisible('--controller-down-area'),\n        ]}\n      >\n        {seekbar || <div />}\n        {type === 'desktop' && (\n          <ControlsBlock\n            order=\"desktop\"\n            liveButton={liveButton}\n            {...controlButtons}\n          />\n        )}\n        {type === 'desktop' ? (\n          <div css={[displayStyles[controlsDisplay]]}>{displayTime}</div>\n        ) : (\n          <>\n            {liveButton}\n            <div\n              css={{\n                '&:not(:empty)': {\n                  flex: 1,\n                  marginLeft: 'calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n                },\n              }}\n            >\n              {displayTime}\n            </div>\n          </>\n        )}\n        {adStatus && (\n          <div\n            className=\"pinned\"\n            css={[infoBarSlotStyle, type === 'desktop' && desktopInfoBarStyle]}\n          >\n            {adStatus}\n          </div>\n        )}\n        <div\n          css={\n            !adStatus && [\n              infoBarSlotStyle,\n              type === 'desktop' && desktopInfoBarStyle,\n            ]\n          }\n          ref={element => {\n            slotRef.current.infoBar = element\n          }}\n        />\n        {volumeControl}\n        <div\n          css={[adStatus ? hidden : {display: 'flex'}]}\n          ref={element => {\n            slotRef.current.functionBar = element\n          }}\n        />\n        {fullscreenButton}\n      </div>\n      <SlotProvider slotRef={slotRef}>{children}</SlotProvider>\n    </div>\n  )\n}\n\nexport default DefaultLayout\n"]} */"],
|
|
3572
3516
|
children: [backButton, jsx$1("div", {
|
|
3573
3517
|
ref: element => {
|
|
3574
3518
|
slotRef.current.titleBarLeft = element;
|
|
@@ -3593,9 +3537,9 @@ const DefaultLayout = ({
|
|
|
3593
3537
|
// see https://kkvideo.atlassian.net/browse/CPT-4535 for detail
|
|
3594
3538
|
adStatus && {
|
|
3595
3539
|
margin: 'calc(var(--spacing) * 6) 0'
|
|
3596
|
-
}, process.env.NODE_ENV === "production" ? "" : ";label:DefaultLayout;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["DefaultLayout.js"],"names":[],"mappings":"AAqZQ","file":"DefaultLayout.js","sourcesContent":["/* eslint-disable no-param-reassign */\n/* @jsxImportSource @emotion/react */\n/* eslint-disable react/prop-types */\nimport {useRef} from 'react'\nimport {SlotProvider} from './uiExtensions'\nimport {nativeTextStyle, customTextStyle, bottomUiTop} from './subtitles'\n\nconst hidden = {display: 'none'}\nconst invisible = hiddenArea => ({\n  visibility: `var(${hiddenArea}, visible)`,\n})\nconst containerStyle = {\n  '--spacing': '0.5em',\n  '--center-pointer-events': 'auto',\n  width: '100%',\n  height: '100%',\n  fontSize: '16px',\n  boxSizing: 'border-box',\n  display: 'flex',\n  flexDirection: 'column',\n  overflow: 'hidden',\n  color: 'white',\n  background: '#000',\n  // prevent animation glich(afterimage) of descendant elements\n  transform: 'translateX(0)',\n  userSelect: 'none',\n  'a, a:link, a:visited': {\n    color: '#fff',\n    opacity: 0.8,\n    textDecoration: 'none',\n  },\n  button: {\n    fontSize: 'inherit',\n    '> *': {\n      pointerEvents: 'none',\n    },\n  },\n  '--thumbnail-width': '160', // height 90\n  '--theme-color': 'var(--primary-highlight, red)',\n}\n\nconst videoContainerStyle = {\n  '> div:first-of-type': {\n    position: 'absolute',\n    zIndex: '-1',\n    display: 'flex',\n    alignItems: 'center',\n    width: '100%',\n    height: '100%',\n  },\n}\n\nconst drop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0.5) 0,\n    rgba(0,0,0,0) var(--player-ui-drop-shadow-size, 30%) calc(100% - var(--player-ui-drop-shadow-size, 30%)),\n    rgba(0,0,0,0.5) 100%\n  )`,\n  backgroundColor: `rgba(0, 0, 0, 0.3)`,\n}\n\nconst dropTop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0) 0,\n    rgba(0,0,0,0) 8rem calc(100% - 8rem),\n    rgba(0,0,0,0.5) 100%\n  )`,\n}\n\nconst responsiveStyles = {\n  desktop: {\n    fontSize: '24px',\n  }, // add if necessary: big-desktop\n}\n\nconst rowStyle = {\n  boxSizing: 'border-box',\n  width: '100%',\n  padding: '1em',\n  display: 'flex',\n  alignItems: 'center',\n  '> button, > div > button': {\n    padding: '0.5rem',\n  },\n}\n\n/* \n  Hint:\n    displayStyles specificity is higher then controlsStyle\n    Need to pay attention when we revise these styles.\n*/\n\nconst displayHidden = {\n  '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n    opacity: 0,\n    transform: 'translate(-200vw, 0)',\n    transition: 'opacity 0.8s var(--autohide-delay, 0s) ease-out, transform 0s calc(0.8s + var(--autohide-delay, 0s))',\n    'button': {\n      opacity: 0, // extra transition to hide tooltip for Safari\n      transition: 'opacity 0s calc(0.8s + var(--autohide-delay, 0s)) ease-out',\n    }\n  },\n}\n\nconst displayStyles = {\n  hidden: displayHidden,\n  'seekbar-only': displayHidden,\n  shown: {\n    '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n      zIndex: 1,\n      pointerEvents: 'auto',\n      transition: 'opacity 0.3s ease-out, transform 0s 0s',\n    },\n  },\n}\n\nconst controlsDisplayStyles = {\n  hidden: {\n    ...displayStyles.hidden,\n    '~ div:not(.pinned)': displayStyles.hidden,\n  },\n  'seekbar-only': {\n    '> div:first-of-type ~ *': displayStyles.hidden,\n  },\n  shown: {\n    ...displayStyles.shown,\n    '~ div': displayStyles.shown,\n  },\n}\n\n// TODO: transform into sample object instead of function\nconst controlsStyle = type => ({\n  position: 'absolute',\n  zIndex: '2',\n  width: '100%',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'space-evenly',\n  fontSize: '1.5em',\n  '> button': {\n    padding: '0.75em 0.5em',\n    '&:disabled': {\n      opacity: 0,\n      pointerEvents: 'none',\n    },\n    '&.play-button': {\n      padding: '0.75rem 0.5rem',\n      fontSize: 'var(--play-button-size, 1em)',\n    },\n    '&.firstplay-button':\n      type === 'mobile'\n        ? {fontSize: '200%', padding: '4em 10em', marginBottom: '1em'}\n        : {fontSize: '400%'},\n  },\n})\n\n// Remove the height for imporving tapping effect at current, and this height was used for fixed region before.\nconst infoBarSlotStyle = {\n  display: 'flex',\n  order: -1,\n  alignItems: 'center',\n  flex: '1 60%',\n}\n\nconst desktopInfoBarStyle = {\n  marginLeft: 'var(--spacing, 0.5em)',\n  order: 'initial',\n  flex: '1 0%',\n}\n\nconst textEllipsis = {\n  overflow: 'hidden',\n  whiteSpace: 'nowrap',\n  textOverflow: 'ellipsis',\n}\n\nconst infoStyle = {\n  zIndex: 1,\n  overflow: 'visible',\n  pointerEvents: 'none',\n  padding: '0 calc(2em - 20px)',\n  marginBottom: 'calc(0.5em + var(--bottom-spacing, 0))',\n  flex: '3em 0',\n  '> div:first-of-type, > div:nth-of-type(2), > button': {\n    minWidth: 0,\n  },\n  '> div:nth-of-type(2)': {\n    // title and channel title\n    margin: '0 calc(var(--spacing, 0.5em) / 2)',\n    flex: 1,\n    fontSize: '0.75em',\n    visibility: 'var(--player-title-visibility, visible)',\n    span: {\n      marginRight: 'var(--spacing, 0.5em)',\n    },\n    '> div': textEllipsis,\n  },\n  h1: {\n    margin: 0,\n    fontSize: '1.66em',\n    fontWeight: '500',\n    ...textEllipsis,\n    flex: 1,\n    '&:not(:only-child)': {\n      marginTop: '0.5em',\n    },\n  },\n  img: {\n    display: 'block',\n    width: '2em',\n    height: '2em',\n    borderRadius: '0.25em',\n    objectFit: 'cover',\n  },\n}\n\nconst backStyle = {\n  position: 'absolute',\n  zIndex: 0,\n  width: '100%',\n  height: '100%',\n  // move DAI UI slightly up to avoid stacking\n  paddingBottom: 'calc(var(--spacing) * 4)',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  '> iframe': {pointerEvents: 'auto'},\n}\n\nconst skipStyle = {\n  position: 'absolute',\n  right: 0,\n  bottom: '9rem',\n  textAlign: 'right',\n  button: {\n    width: 'auto',\n    height: 'auto',\n  },\n}\n\nconst desktopStyle = {\n  '--spacing': '0.75em',\n  '--center-pointer-events': 'none',\n  '> div button': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em)',\n  },\n  '--thumbnail-width': '288', // height 162\n}\n\nconst desktopControls = {\n  '> div:first-of-type': {\n    flex: '100%',\n    marginBottom: '0.4em',\n  },\n  '> button': {\n    '&[disabled]': {\n    display: 'none',\n    },\n    '&.play-button': {\n      fontSize: '1em',\n    },\n    '&.firstplay-button': {\n      fontSize: '400%'\n    },\n  },\n}\n\nconst seekbarStyles = {\n  flex: '1 var(--seekbar-flex-basis, 100%)',\n  '&:not(:empty)': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n  },\n}\n\nconst adContainerStyle = {\n  flexGrow: 1,\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  pointerEvents: 'var(--center-pointer-events)',\n  button: {pointerEvents: 'auto'},\n  zIndex: 0,\n}\n\nconst ControlsBlock = ({\n  order = 'mobile',\n  liveButton,\n  playButton,\n  rewindButton = '',\n  forwardButton = '',\n  previousEpisodeButton = '',\n  nextEpisodeButton = '',\n}) =>\n  order === 'desktop' ? (\n    <>\n      {liveButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {rewindButton}\n      {forwardButton}\n    </>\n  ) : (\n    <>\n      {rewindButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {forwardButton}\n    </>\n  )\n\nconst DefaultLayout = ({\n  type = 'mobile',\n  style,\n  display,\n  liveButton,\n  controlsDisplay = display,\n  size,\n  title = '',\n  channelTitle = '',\n  video,\n  haveBottomItem,\n  seekbar = '',\n  displayTime,\n  controlButtons,\n  volumeControl,\n  fullscreenButton,\n  backButton = '',\n  adStatus = '',\n  adLink = '',\n  adSkipButton,\n  backItems,\n  children,\n  containerRef,\n  backRef,\n  adContainerRef,\n  ...rest\n}) => {\n  const slotRef = useRef({})\n\n  return (\n    <div\n      style={{\n        '--player-bottom-ui-top':\n          controlsDisplay === 'shown'\n            ? `-${bottomUiTop(containerRef.current)}px`\n            : '-8px',\n        '--player-bottom-ui-height':\n          type === 'desktop' && controlsDisplay === 'shown' ? '3.5rem' : '0',\n      }}\n      css={[\n        containerStyle,\n        videoContainerStyle,\n        responsiveStyles[size],\n        type === 'desktop' && desktopStyle,\n        adStatus && {'--bottom-spacing': 0},\n        nativeTextStyle,\n        customTextStyle,\n        style,\n      ]}\n      ref={containerRef}\n      {...rest}\n    >\n      {video}\n      <div\n        ref={backRef}\n        css={[\n          backStyle,\n          display !== 'hidden' && (haveBottomItem ? dropTop : drop),\n          invisible('--controller-middle-area'),\n        ]}\n      >\n        {type !== 'mobile' && backItems}\n        {adSkipButton && <div css={skipStyle}>{adSkipButton}</div>}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          infoStyle,\n          displayStyles[display],\n          invisible('--controller-top-area'),\n        ]}\n      >\n        {backButton}\n        <div\n          ref={element => {\n            slotRef.current.titleBarLeft = element\n          }}\n        />\n        <div>\n          <h1>{title}</h1>\n          {channelTitle && <div>{channelTitle}</div>}\n        </div>\n        <div\n          ref={element => {\n            slotRef.current.titleBarRight = element\n          }}\n        />\n        {adLink && <div className=\"pinned\">{adLink}</div>}\n      </div>\n      <div\n        ref={adContainerRef}\n        css={[\n          adContainerStyle,\n          // this container covers Google DAI iframe UI, left some space for user to click \"skip ad\"\n          // see https://kkvideo.atlassian.net/browse/CPT-4535 for detail\n          adStatus && {margin: 'calc(var(--spacing) * 6) 0'},\n        ]}\n      >\n        {type === 'mobile' && (\n          <div\n            css={[\n              controlsStyle(type),\n              displayStyles[controlsDisplay],\n              invisible('--controller-middle-area'),\n            ]}\n          >\n            <ControlsBlock order=\"mobile\" {...controlButtons} />\n          </div>\n        )}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          {\n            marginTop: '1em',\n            paddingTop: 0,\n            paddingBottom: 'calc(1em - var(--bottom-spacing, 0) / 5)',\n            flexWrap: 'wrap',\n            '> div:first-of-type': seekbarStyles,\n          },\n          type === 'desktop' && desktopControls,\n          controlsDisplayStyles[controlsDisplay],\n          invisible('--controller-down-area'),\n        ]}\n      >\n        {seekbar || <div />}\n        {type === 'desktop' && (\n          <ControlsBlock\n            order=\"desktop\"\n            liveButton={liveButton}\n            {...controlButtons}\n          />\n        )}\n        {type === 'desktop' ? (\n          <div css={[displayStyles[controlsDisplay]]}>{displayTime}</div>\n        ) : (\n          <>\n            {liveButton}\n            <div\n              css={{\n                '&:not(:empty)': {\n                  flex: 1,\n                  marginLeft: 'calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n                },\n              }}\n            >\n              {displayTime}\n            </div>\n          </>\n        )}\n        {adStatus && (\n          <div\n            className=\"pinned\"\n            css={[infoBarSlotStyle, type === 'desktop' && desktopInfoBarStyle]}\n          >\n            {adStatus}\n          </div>\n        )}\n        <div\n          css={\n            !adStatus && [\n              infoBarSlotStyle,\n              type === 'desktop' && desktopInfoBarStyle,\n            ]\n          }\n          ref={element => {\n            slotRef.current.infoBar = element\n          }}\n        />\n        {volumeControl}\n        <div\n          css={adStatus && hidden}\n          ref={element => {\n            slotRef.current.functionBar = element\n          }}\n        />\n        {fullscreenButton}\n      </div>\n      <SlotProvider slotRef={slotRef}>{children}</SlotProvider>\n    </div>\n  )\n}\n\nexport default DefaultLayout\n"]} */"],
|
|
3540
|
+
}, process.env.NODE_ENV === "production" ? "" : ";label:DefaultLayout;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["DefaultLayout.js"],"names":[],"mappings":"AAqZQ","file":"DefaultLayout.js","sourcesContent":["/* eslint-disable no-param-reassign */\n/* @jsxImportSource @emotion/react */\n/* eslint-disable react/prop-types */\nimport {useRef} from 'react'\nimport {SlotProvider} from './uiExtensions'\nimport {nativeTextStyle, customTextStyle, bottomUiTop} from './subtitles'\n\nconst hidden = {display: 'none'}\nconst invisible = hiddenArea => ({\n  visibility: `var(${hiddenArea}, visible)`,\n})\nconst containerStyle = {\n  '--spacing': '0.5em',\n  '--center-pointer-events': 'auto',\n  width: '100%',\n  height: '100%',\n  fontSize: '16px',\n  boxSizing: 'border-box',\n  display: 'flex',\n  flexDirection: 'column',\n  overflow: 'hidden',\n  color: 'white',\n  background: '#000',\n  // prevent animation glich(afterimage) of descendant elements\n  transform: 'translateX(0)',\n  userSelect: 'none',\n  'a, a:link, a:visited': {\n    color: '#fff',\n    opacity: 0.8,\n    textDecoration: 'none',\n  },\n  button: {\n    fontSize: 'inherit',\n    '> *': {\n      pointerEvents: 'none',\n    },\n  },\n  '--thumbnail-width': '160', // height 90\n  '--theme-color': 'var(--primary-highlight, red)',\n}\n\nconst videoContainerStyle = {\n  '> div:first-of-type': {\n    position: 'absolute',\n    zIndex: '-1',\n    display: 'flex',\n    alignItems: 'center',\n    width: '100%',\n    height: '100%',\n  },\n}\n\nconst drop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0.5) 0,\n    rgba(0,0,0,0) var(--player-ui-drop-shadow-size, 30%) calc(100% - var(--player-ui-drop-shadow-size, 30%)),\n    rgba(0,0,0,0.5) 100%\n  )`,\n  backgroundColor: `rgba(0, 0, 0, 0.3)`,\n}\n\nconst dropTop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0) 0,\n    rgba(0,0,0,0) 8rem calc(100% - 8rem),\n    rgba(0,0,0,0.5) 100%\n  )`,\n}\n\nconst responsiveStyles = {\n  desktop: {\n    fontSize: '24px',\n  }, // add if necessary: big-desktop\n}\n\nconst rowStyle = {\n  boxSizing: 'border-box',\n  width: '100%',\n  padding: '1em',\n  display: 'flex',\n  alignItems: 'center',\n  '> button, > div > button': {\n    padding: '0.5rem',\n  },\n}\n\n/* \n  Hint:\n    displayStyles specificity is higher then controlsStyle\n    Need to pay attention when we revise these styles.\n*/\n\nconst displayHidden = {\n  '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n    opacity: 0,\n    transform: 'translate(-200vw, 0)',\n    transition: 'opacity 0.8s var(--autohide-delay, 0s) ease-out, transform 0s calc(0.8s + var(--autohide-delay, 0s))',\n    'button': {\n      opacity: 0, // extra transition to hide tooltip for Safari\n      transition: 'opacity 0s calc(0.8s + var(--autohide-delay, 0s)) ease-out',\n    }\n  },\n}\n\nconst displayStyles = {\n  hidden: displayHidden,\n  'seekbar-only': displayHidden,\n  shown: {\n    '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n      zIndex: 1,\n      pointerEvents: 'auto',\n      transition: 'opacity 0.3s ease-out, transform 0s 0s',\n    },\n  },\n}\n\nconst controlsDisplayStyles = {\n  hidden: {\n    ...displayStyles.hidden,\n    '~ div:not(.pinned)': displayStyles.hidden,\n  },\n  'seekbar-only': {\n    '> div:first-of-type ~ *': displayStyles.hidden,\n  },\n  shown: {\n    ...displayStyles.shown,\n    '~ div': displayStyles.shown,\n  },\n}\n\n// TODO: transform into sample object instead of function\nconst controlsStyle = type => ({\n  position: 'absolute',\n  zIndex: '2',\n  width: '100%',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'space-evenly',\n  fontSize: '1.5em',\n  '> button': {\n    padding: '0.75em 0.5em',\n    '&:disabled': {\n      opacity: 0,\n      pointerEvents: 'none',\n    },\n    '&.play-button': {\n      padding: '0.75rem 0.5rem',\n      fontSize: 'var(--play-button-size, 1em)',\n    },\n    '&.firstplay-button':\n      type === 'mobile'\n        ? {fontSize: '200%', padding: '4em 10em', marginBottom: '1em'}\n        : {fontSize: '400%'},\n  },\n})\n\n// Remove the height for imporving tapping effect at current, and this height was used for fixed region before.\nconst infoBarSlotStyle = {\n  display: 'flex',\n  order: -1,\n  alignItems: 'center',\n  flex: '1 60%',\n}\n\nconst desktopInfoBarStyle = {\n  marginLeft: 'var(--spacing, 0.5em)',\n  order: 'initial',\n  flex: '1 0%',\n}\n\nconst textEllipsis = {\n  overflow: 'hidden',\n  whiteSpace: 'nowrap',\n  textOverflow: 'ellipsis',\n}\n\nconst infoStyle = {\n  zIndex: 1,\n  overflow: 'visible',\n  pointerEvents: 'none',\n  padding: '0 calc(2em - 20px)',\n  marginBottom: 'calc(0.5em + var(--bottom-spacing, 0))',\n  flex: '3em 0',\n  '> div:first-of-type, > div:nth-of-type(2), > button': {\n    minWidth: 0,\n  },\n  '> div:nth-of-type(2)': {\n    // title and channel title\n    margin: '0 calc(var(--spacing, 0.5em) / 2)',\n    flex: 1,\n    fontSize: '0.75em',\n    visibility: 'var(--player-title-visibility, visible)',\n    span: {\n      marginRight: 'var(--spacing, 0.5em)',\n    },\n    '> div': textEllipsis,\n  },\n  h1: {\n    margin: 0,\n    fontSize: '1.66em',\n    fontWeight: '500',\n    ...textEllipsis,\n    flex: 1,\n    '&:not(:only-child)': {\n      marginTop: '0.5em',\n    },\n  },\n  img: {\n    display: 'block',\n    width: '2em',\n    height: '2em',\n    borderRadius: '0.25em',\n    objectFit: 'cover',\n  },\n}\n\nconst backStyle = {\n  position: 'absolute',\n  zIndex: 0,\n  width: '100%',\n  height: '100%',\n  // move DAI UI slightly up to avoid stacking\n  paddingBottom: 'calc(var(--spacing) * 4)',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  '> iframe': {pointerEvents: 'auto'},\n}\n\nconst skipStyle = {\n  position: 'absolute',\n  right: 0,\n  bottom: '9rem',\n  textAlign: 'right',\n  button: {\n    width: 'auto',\n    height: 'auto',\n  },\n}\n\nconst desktopStyle = {\n  '--spacing': '0.75em',\n  '--center-pointer-events': 'none',\n  '> div button': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em)',\n  },\n  '--thumbnail-width': '288', // height 162\n}\n\nconst desktopControls = {\n  '> div:first-of-type': {\n    flex: '100%',\n    marginBottom: '0.4em',\n  },\n  '> button': {\n    '&[disabled]': {\n    display: 'none',\n    },\n    '&.play-button': {\n      fontSize: '1em',\n    },\n    '&.firstplay-button': {\n      fontSize: '400%'\n    },\n  },\n}\n\nconst seekbarStyles = {\n  flex: '1 var(--seekbar-flex-basis, 100%)',\n  '&:not(:empty)': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n  },\n}\n\nconst adContainerStyle = {\n  flexGrow: 1,\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  pointerEvents: 'var(--center-pointer-events)',\n  button: {pointerEvents: 'auto'},\n  zIndex: 0,\n}\n\nconst ControlsBlock = ({\n  order = 'mobile',\n  liveButton,\n  playButton,\n  rewindButton = '',\n  forwardButton = '',\n  previousEpisodeButton = '',\n  nextEpisodeButton = '',\n}) =>\n  order === 'desktop' ? (\n    <>\n      {liveButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {rewindButton}\n      {forwardButton}\n    </>\n  ) : (\n    <>\n      {rewindButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {forwardButton}\n    </>\n  )\n\nconst DefaultLayout = ({\n  type = 'mobile',\n  style,\n  display,\n  liveButton,\n  controlsDisplay = display,\n  size,\n  title = '',\n  channelTitle = '',\n  video,\n  haveBottomItem,\n  seekbar = '',\n  displayTime,\n  controlButtons,\n  volumeControl,\n  fullscreenButton,\n  backButton = '',\n  adStatus = '',\n  adLink = '',\n  adSkipButton,\n  backItems,\n  children,\n  containerRef,\n  backRef,\n  adContainerRef,\n  ...rest\n}) => {\n  const slotRef = useRef({})\n\n  return (\n    <div\n      style={{\n        '--player-bottom-ui-top':\n          controlsDisplay === 'shown'\n            ? `-${bottomUiTop(containerRef.current)}px`\n            : '-8px',\n        '--player-bottom-ui-height':\n          type === 'desktop' && controlsDisplay === 'shown' ? '3.5rem' : '0',\n      }}\n      css={[\n        containerStyle,\n        videoContainerStyle,\n        responsiveStyles[size],\n        type === 'desktop' && desktopStyle,\n        adStatus && {'--bottom-spacing': 0},\n        nativeTextStyle,\n        customTextStyle,\n        style,\n      ]}\n      ref={containerRef}\n      {...rest}\n    >\n      {video}\n      <div\n        ref={backRef}\n        css={[\n          backStyle,\n          display !== 'hidden' && (haveBottomItem ? dropTop : drop),\n          invisible('--controller-middle-area'),\n        ]}\n      >\n        {type !== 'mobile' && backItems}\n        {adSkipButton && <div css={skipStyle}>{adSkipButton}</div>}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          infoStyle,\n          displayStyles[display],\n          invisible('--controller-top-area'),\n        ]}\n      >\n        {backButton}\n        <div\n          ref={element => {\n            slotRef.current.titleBarLeft = element\n          }}\n        />\n        <div>\n          <h1>{title}</h1>\n          {channelTitle && <div>{channelTitle}</div>}\n        </div>\n        <div\n          ref={element => {\n            slotRef.current.titleBarRight = element\n          }}\n        />\n        {adLink && <div className=\"pinned\">{adLink}</div>}\n      </div>\n      <div\n        ref={adContainerRef}\n        css={[\n          adContainerStyle,\n          // this container covers Google DAI iframe UI, left some space for user to click \"skip ad\"\n          // see https://kkvideo.atlassian.net/browse/CPT-4535 for detail\n          adStatus && {margin: 'calc(var(--spacing) * 6) 0'},\n        ]}\n      >\n        {type === 'mobile' && (\n          <div\n            css={[\n              controlsStyle(type),\n              displayStyles[controlsDisplay],\n              invisible('--controller-middle-area'),\n            ]}\n          >\n            <ControlsBlock order=\"mobile\" {...controlButtons} />\n          </div>\n        )}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          {\n            marginTop: '1em',\n            paddingTop: 0,\n            paddingBottom: 'calc(1em - var(--bottom-spacing, 0) / 5)',\n            flexWrap: 'wrap',\n            '> div:first-of-type': seekbarStyles,\n          },\n          type === 'desktop' && desktopControls,\n          controlsDisplayStyles[controlsDisplay],\n          invisible('--controller-down-area'),\n        ]}\n      >\n        {seekbar || <div />}\n        {type === 'desktop' && (\n          <ControlsBlock\n            order=\"desktop\"\n            liveButton={liveButton}\n            {...controlButtons}\n          />\n        )}\n        {type === 'desktop' ? (\n          <div css={[displayStyles[controlsDisplay]]}>{displayTime}</div>\n        ) : (\n          <>\n            {liveButton}\n            <div\n              css={{\n                '&:not(:empty)': {\n                  flex: 1,\n                  marginLeft: 'calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n                },\n              }}\n            >\n              {displayTime}\n            </div>\n          </>\n        )}\n        {adStatus && (\n          <div\n            className=\"pinned\"\n            css={[infoBarSlotStyle, type === 'desktop' && desktopInfoBarStyle]}\n          >\n            {adStatus}\n          </div>\n        )}\n        <div\n          css={\n            !adStatus && [\n              infoBarSlotStyle,\n              type === 'desktop' && desktopInfoBarStyle,\n            ]\n          }\n          ref={element => {\n            slotRef.current.infoBar = element\n          }}\n        />\n        {volumeControl}\n        <div\n          css={[adStatus ? hidden : {display: 'flex'}]}\n          ref={element => {\n            slotRef.current.functionBar = element\n          }}\n        />\n        {fullscreenButton}\n      </div>\n      <SlotProvider slotRef={slotRef}>{children}</SlotProvider>\n    </div>\n  )\n}\n\nexport default DefaultLayout\n"]} */"],
|
|
3597
3541
|
children: type === 'mobile' && jsx$1("div", {
|
|
3598
|
-
css: [controlsStyle(type), displayStyles[controlsDisplay], invisible('--controller-middle-area'), process.env.NODE_ENV === "production" ? "" : ";label:DefaultLayout;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["DefaultLayout.js"],"names":[],"mappings":"AA8ZY","file":"DefaultLayout.js","sourcesContent":["/* eslint-disable no-param-reassign */\n/* @jsxImportSource @emotion/react */\n/* eslint-disable react/prop-types */\nimport {useRef} from 'react'\nimport {SlotProvider} from './uiExtensions'\nimport {nativeTextStyle, customTextStyle, bottomUiTop} from './subtitles'\n\nconst hidden = {display: 'none'}\nconst invisible = hiddenArea => ({\n  visibility: `var(${hiddenArea}, visible)`,\n})\nconst containerStyle = {\n  '--spacing': '0.5em',\n  '--center-pointer-events': 'auto',\n  width: '100%',\n  height: '100%',\n  fontSize: '16px',\n  boxSizing: 'border-box',\n  display: 'flex',\n  flexDirection: 'column',\n  overflow: 'hidden',\n  color: 'white',\n  background: '#000',\n  // prevent animation glich(afterimage) of descendant elements\n  transform: 'translateX(0)',\n  userSelect: 'none',\n  'a, a:link, a:visited': {\n    color: '#fff',\n    opacity: 0.8,\n    textDecoration: 'none',\n  },\n  button: {\n    fontSize: 'inherit',\n    '> *': {\n      pointerEvents: 'none',\n    },\n  },\n  '--thumbnail-width': '160', // height 90\n  '--theme-color': 'var(--primary-highlight, red)',\n}\n\nconst videoContainerStyle = {\n  '> div:first-of-type': {\n    position: 'absolute',\n    zIndex: '-1',\n    display: 'flex',\n    alignItems: 'center',\n    width: '100%',\n    height: '100%',\n  },\n}\n\nconst drop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0.5) 0,\n    rgba(0,0,0,0) var(--player-ui-drop-shadow-size, 30%) calc(100% - var(--player-ui-drop-shadow-size, 30%)),\n    rgba(0,0,0,0.5) 100%\n  )`,\n  backgroundColor: `rgba(0, 0, 0, 0.3)`,\n}\n\nconst dropTop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0) 0,\n    rgba(0,0,0,0) 8rem calc(100% - 8rem),\n    rgba(0,0,0,0.5) 100%\n  )`,\n}\n\nconst responsiveStyles = {\n  desktop: {\n    fontSize: '24px',\n  }, // add if necessary: big-desktop\n}\n\nconst rowStyle = {\n  boxSizing: 'border-box',\n  width: '100%',\n  padding: '1em',\n  display: 'flex',\n  alignItems: 'center',\n  '> button, > div > button': {\n    padding: '0.5rem',\n  },\n}\n\n/* \n  Hint:\n    displayStyles specificity is higher then controlsStyle\n    Need to pay attention when we revise these styles.\n*/\n\nconst displayHidden = {\n  '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n    opacity: 0,\n    transform: 'translate(-200vw, 0)',\n    transition: 'opacity 0.8s var(--autohide-delay, 0s) ease-out, transform 0s calc(0.8s + var(--autohide-delay, 0s))',\n    'button': {\n      opacity: 0, // extra transition to hide tooltip for Safari\n      transition: 'opacity 0s calc(0.8s + var(--autohide-delay, 0s)) ease-out',\n    }\n  },\n}\n\nconst displayStyles = {\n  hidden: displayHidden,\n  'seekbar-only': displayHidden,\n  shown: {\n    '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n      zIndex: 1,\n      pointerEvents: 'auto',\n      transition: 'opacity 0.3s ease-out, transform 0s 0s',\n    },\n  },\n}\n\nconst controlsDisplayStyles = {\n  hidden: {\n    ...displayStyles.hidden,\n    '~ div:not(.pinned)': displayStyles.hidden,\n  },\n  'seekbar-only': {\n    '> div:first-of-type ~ *': displayStyles.hidden,\n  },\n  shown: {\n    ...displayStyles.shown,\n    '~ div': displayStyles.shown,\n  },\n}\n\n// TODO: transform into sample object instead of function\nconst controlsStyle = type => ({\n  position: 'absolute',\n  zIndex: '2',\n  width: '100%',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'space-evenly',\n  fontSize: '1.5em',\n  '> button': {\n    padding: '0.75em 0.5em',\n    '&:disabled': {\n      opacity: 0,\n      pointerEvents: 'none',\n    },\n    '&.play-button': {\n      padding: '0.75rem 0.5rem',\n      fontSize: 'var(--play-button-size, 1em)',\n    },\n    '&.firstplay-button':\n      type === 'mobile'\n        ? {fontSize: '200%', padding: '4em 10em', marginBottom: '1em'}\n        : {fontSize: '400%'},\n  },\n})\n\n// Remove the height for imporving tapping effect at current, and this height was used for fixed region before.\nconst infoBarSlotStyle = {\n  display: 'flex',\n  order: -1,\n  alignItems: 'center',\n  flex: '1 60%',\n}\n\nconst desktopInfoBarStyle = {\n  marginLeft: 'var(--spacing, 0.5em)',\n  order: 'initial',\n  flex: '1 0%',\n}\n\nconst textEllipsis = {\n  overflow: 'hidden',\n  whiteSpace: 'nowrap',\n  textOverflow: 'ellipsis',\n}\n\nconst infoStyle = {\n  zIndex: 1,\n  overflow: 'visible',\n  pointerEvents: 'none',\n  padding: '0 calc(2em - 20px)',\n  marginBottom: 'calc(0.5em + var(--bottom-spacing, 0))',\n  flex: '3em 0',\n  '> div:first-of-type, > div:nth-of-type(2), > button': {\n    minWidth: 0,\n  },\n  '> div:nth-of-type(2)': {\n    // title and channel title\n    margin: '0 calc(var(--spacing, 0.5em) / 2)',\n    flex: 1,\n    fontSize: '0.75em',\n    visibility: 'var(--player-title-visibility, visible)',\n    span: {\n      marginRight: 'var(--spacing, 0.5em)',\n    },\n    '> div': textEllipsis,\n  },\n  h1: {\n    margin: 0,\n    fontSize: '1.66em',\n    fontWeight: '500',\n    ...textEllipsis,\n    flex: 1,\n    '&:not(:only-child)': {\n      marginTop: '0.5em',\n    },\n  },\n  img: {\n    display: 'block',\n    width: '2em',\n    height: '2em',\n    borderRadius: '0.25em',\n    objectFit: 'cover',\n  },\n}\n\nconst backStyle = {\n  position: 'absolute',\n  zIndex: 0,\n  width: '100%',\n  height: '100%',\n  // move DAI UI slightly up to avoid stacking\n  paddingBottom: 'calc(var(--spacing) * 4)',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  '> iframe': {pointerEvents: 'auto'},\n}\n\nconst skipStyle = {\n  position: 'absolute',\n  right: 0,\n  bottom: '9rem',\n  textAlign: 'right',\n  button: {\n    width: 'auto',\n    height: 'auto',\n  },\n}\n\nconst desktopStyle = {\n  '--spacing': '0.75em',\n  '--center-pointer-events': 'none',\n  '> div button': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em)',\n  },\n  '--thumbnail-width': '288', // height 162\n}\n\nconst desktopControls = {\n  '> div:first-of-type': {\n    flex: '100%',\n    marginBottom: '0.4em',\n  },\n  '> button': {\n    '&[disabled]': {\n    display: 'none',\n    },\n    '&.play-button': {\n      fontSize: '1em',\n    },\n    '&.firstplay-button': {\n      fontSize: '400%'\n    },\n  },\n}\n\nconst seekbarStyles = {\n  flex: '1 var(--seekbar-flex-basis, 100%)',\n  '&:not(:empty)': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n  },\n}\n\nconst adContainerStyle = {\n  flexGrow: 1,\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  pointerEvents: 'var(--center-pointer-events)',\n  button: {pointerEvents: 'auto'},\n  zIndex: 0,\n}\n\nconst ControlsBlock = ({\n  order = 'mobile',\n  liveButton,\n  playButton,\n  rewindButton = '',\n  forwardButton = '',\n  previousEpisodeButton = '',\n  nextEpisodeButton = '',\n}) =>\n  order === 'desktop' ? (\n    <>\n      {liveButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {rewindButton}\n      {forwardButton}\n    </>\n  ) : (\n    <>\n      {rewindButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {forwardButton}\n    </>\n  )\n\nconst DefaultLayout = ({\n  type = 'mobile',\n  style,\n  display,\n  liveButton,\n  controlsDisplay = display,\n  size,\n  title = '',\n  channelTitle = '',\n  video,\n  haveBottomItem,\n  seekbar = '',\n  displayTime,\n  controlButtons,\n  volumeControl,\n  fullscreenButton,\n  backButton = '',\n  adStatus = '',\n  adLink = '',\n  adSkipButton,\n  backItems,\n  children,\n  containerRef,\n  backRef,\n  adContainerRef,\n  ...rest\n}) => {\n  const slotRef = useRef({})\n\n  return (\n    <div\n      style={{\n        '--player-bottom-ui-top':\n          controlsDisplay === 'shown'\n            ? `-${bottomUiTop(containerRef.current)}px`\n            : '-8px',\n        '--player-bottom-ui-height':\n          type === 'desktop' && controlsDisplay === 'shown' ? '3.5rem' : '0',\n      }}\n      css={[\n        containerStyle,\n        videoContainerStyle,\n        responsiveStyles[size],\n        type === 'desktop' && desktopStyle,\n        adStatus && {'--bottom-spacing': 0},\n        nativeTextStyle,\n        customTextStyle,\n        style,\n      ]}\n      ref={containerRef}\n      {...rest}\n    >\n      {video}\n      <div\n        ref={backRef}\n        css={[\n          backStyle,\n          display !== 'hidden' && (haveBottomItem ? dropTop : drop),\n          invisible('--controller-middle-area'),\n        ]}\n      >\n        {type !== 'mobile' && backItems}\n        {adSkipButton && <div css={skipStyle}>{adSkipButton}</div>}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          infoStyle,\n          displayStyles[display],\n          invisible('--controller-top-area'),\n        ]}\n      >\n        {backButton}\n        <div\n          ref={element => {\n            slotRef.current.titleBarLeft = element\n          }}\n        />\n        <div>\n          <h1>{title}</h1>\n          {channelTitle && <div>{channelTitle}</div>}\n        </div>\n        <div\n          ref={element => {\n            slotRef.current.titleBarRight = element\n          }}\n        />\n        {adLink && <div className=\"pinned\">{adLink}</div>}\n      </div>\n      <div\n        ref={adContainerRef}\n        css={[\n          adContainerStyle,\n          // this container covers Google DAI iframe UI, left some space for user to click \"skip ad\"\n          // see https://kkvideo.atlassian.net/browse/CPT-4535 for detail\n          adStatus && {margin: 'calc(var(--spacing) * 6) 0'},\n        ]}\n      >\n        {type === 'mobile' && (\n          <div\n            css={[\n              controlsStyle(type),\n              displayStyles[controlsDisplay],\n              invisible('--controller-middle-area'),\n            ]}\n          >\n            <ControlsBlock order=\"mobile\" {...controlButtons} />\n          </div>\n        )}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          {\n            marginTop: '1em',\n            paddingTop: 0,\n            paddingBottom: 'calc(1em - var(--bottom-spacing, 0) / 5)',\n            flexWrap: 'wrap',\n            '> div:first-of-type': seekbarStyles,\n          },\n          type === 'desktop' && desktopControls,\n          controlsDisplayStyles[controlsDisplay],\n          invisible('--controller-down-area'),\n        ]}\n      >\n        {seekbar || <div />}\n        {type === 'desktop' && (\n          <ControlsBlock\n            order=\"desktop\"\n            liveButton={liveButton}\n            {...controlButtons}\n          />\n        )}\n        {type === 'desktop' ? (\n          <div css={[displayStyles[controlsDisplay]]}>{displayTime}</div>\n        ) : (\n          <>\n            {liveButton}\n            <div\n              css={{\n                '&:not(:empty)': {\n                  flex: 1,\n                  marginLeft: 'calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n                },\n              }}\n            >\n              {displayTime}\n            </div>\n          </>\n        )}\n        {adStatus && (\n          <div\n            className=\"pinned\"\n            css={[infoBarSlotStyle, type === 'desktop' && desktopInfoBarStyle]}\n          >\n            {adStatus}\n          </div>\n        )}\n        <div\n          css={\n            !adStatus && [\n              infoBarSlotStyle,\n              type === 'desktop' && desktopInfoBarStyle,\n            ]\n          }\n          ref={element => {\n            slotRef.current.infoBar = element\n          }}\n        />\n        {volumeControl}\n        <div\n          css={adStatus && hidden}\n          ref={element => {\n            slotRef.current.functionBar = element\n          }}\n        />\n        {fullscreenButton}\n      </div>\n      <SlotProvider slotRef={slotRef}>{children}</SlotProvider>\n    </div>\n  )\n}\n\nexport default DefaultLayout\n"]} */"],
|
|
3542
|
+
css: [controlsStyle(type), displayStyles[controlsDisplay], invisible('--controller-middle-area'), process.env.NODE_ENV === "production" ? "" : ";label:DefaultLayout;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["DefaultLayout.js"],"names":[],"mappings":"AA8ZY","file":"DefaultLayout.js","sourcesContent":["/* eslint-disable no-param-reassign */\n/* @jsxImportSource @emotion/react */\n/* eslint-disable react/prop-types */\nimport {useRef} from 'react'\nimport {SlotProvider} from './uiExtensions'\nimport {nativeTextStyle, customTextStyle, bottomUiTop} from './subtitles'\n\nconst hidden = {display: 'none'}\nconst invisible = hiddenArea => ({\n  visibility: `var(${hiddenArea}, visible)`,\n})\nconst containerStyle = {\n  '--spacing': '0.5em',\n  '--center-pointer-events': 'auto',\n  width: '100%',\n  height: '100%',\n  fontSize: '16px',\n  boxSizing: 'border-box',\n  display: 'flex',\n  flexDirection: 'column',\n  overflow: 'hidden',\n  color: 'white',\n  background: '#000',\n  // prevent animation glich(afterimage) of descendant elements\n  transform: 'translateX(0)',\n  userSelect: 'none',\n  'a, a:link, a:visited': {\n    color: '#fff',\n    opacity: 0.8,\n    textDecoration: 'none',\n  },\n  button: {\n    fontSize: 'inherit',\n    '> *': {\n      pointerEvents: 'none',\n    },\n  },\n  '--thumbnail-width': '160', // height 90\n  '--theme-color': 'var(--primary-highlight, red)',\n}\n\nconst videoContainerStyle = {\n  '> div:first-of-type': {\n    position: 'absolute',\n    zIndex: '-1',\n    display: 'flex',\n    alignItems: 'center',\n    width: '100%',\n    height: '100%',\n  },\n}\n\nconst drop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0.5) 0,\n    rgba(0,0,0,0) var(--player-ui-drop-shadow-size, 30%) calc(100% - var(--player-ui-drop-shadow-size, 30%)),\n    rgba(0,0,0,0.5) 100%\n  )`,\n  backgroundColor: `rgba(0, 0, 0, 0.3)`,\n}\n\nconst dropTop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0) 0,\n    rgba(0,0,0,0) 8rem calc(100% - 8rem),\n    rgba(0,0,0,0.5) 100%\n  )`,\n}\n\nconst responsiveStyles = {\n  desktop: {\n    fontSize: '24px',\n  }, // add if necessary: big-desktop\n}\n\nconst rowStyle = {\n  boxSizing: 'border-box',\n  width: '100%',\n  padding: '1em',\n  display: 'flex',\n  alignItems: 'center',\n  '> button, > div > button': {\n    padding: '0.5rem',\n  },\n}\n\n/* \n  Hint:\n    displayStyles specificity is higher then controlsStyle\n    Need to pay attention when we revise these styles.\n*/\n\nconst displayHidden = {\n  '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n    opacity: 0,\n    transform: 'translate(-200vw, 0)',\n    transition: 'opacity 0.8s var(--autohide-delay, 0s) ease-out, transform 0s calc(0.8s + var(--autohide-delay, 0s))',\n    'button': {\n      opacity: 0, // extra transition to hide tooltip for Safari\n      transition: 'opacity 0s calc(0.8s + var(--autohide-delay, 0s)) ease-out',\n    }\n  },\n}\n\nconst displayStyles = {\n  hidden: displayHidden,\n  'seekbar-only': displayHidden,\n  shown: {\n    '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n      zIndex: 1,\n      pointerEvents: 'auto',\n      transition: 'opacity 0.3s ease-out, transform 0s 0s',\n    },\n  },\n}\n\nconst controlsDisplayStyles = {\n  hidden: {\n    ...displayStyles.hidden,\n    '~ div:not(.pinned)': displayStyles.hidden,\n  },\n  'seekbar-only': {\n    '> div:first-of-type ~ *': displayStyles.hidden,\n  },\n  shown: {\n    ...displayStyles.shown,\n    '~ div': displayStyles.shown,\n  },\n}\n\n// TODO: transform into sample object instead of function\nconst controlsStyle = type => ({\n  position: 'absolute',\n  zIndex: '2',\n  width: '100%',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'space-evenly',\n  fontSize: '1.5em',\n  '> button': {\n    padding: '0.75em 0.5em',\n    '&:disabled': {\n      opacity: 0,\n      pointerEvents: 'none',\n    },\n    '&.play-button': {\n      padding: '0.75rem 0.5rem',\n      fontSize: 'var(--play-button-size, 1em)',\n    },\n    '&.firstplay-button':\n      type === 'mobile'\n        ? {fontSize: '200%', padding: '4em 10em', marginBottom: '1em'}\n        : {fontSize: '400%'},\n  },\n})\n\n// Remove the height for imporving tapping effect at current, and this height was used for fixed region before.\nconst infoBarSlotStyle = {\n  display: 'flex',\n  order: -1,\n  alignItems: 'center',\n  flex: '1 60%',\n}\n\nconst desktopInfoBarStyle = {\n  marginLeft: 'var(--spacing, 0.5em)',\n  order: 'initial',\n  flex: '1 0%',\n}\n\nconst textEllipsis = {\n  overflow: 'hidden',\n  whiteSpace: 'nowrap',\n  textOverflow: 'ellipsis',\n}\n\nconst infoStyle = {\n  zIndex: 1,\n  overflow: 'visible',\n  pointerEvents: 'none',\n  padding: '0 calc(2em - 20px)',\n  marginBottom: 'calc(0.5em + var(--bottom-spacing, 0))',\n  flex: '3em 0',\n  '> div:first-of-type, > div:nth-of-type(2), > button': {\n    minWidth: 0,\n  },\n  '> div:nth-of-type(2)': {\n    // title and channel title\n    margin: '0 calc(var(--spacing, 0.5em) / 2)',\n    flex: 1,\n    fontSize: '0.75em',\n    visibility: 'var(--player-title-visibility, visible)',\n    span: {\n      marginRight: 'var(--spacing, 0.5em)',\n    },\n    '> div': textEllipsis,\n  },\n  h1: {\n    margin: 0,\n    fontSize: '1.66em',\n    fontWeight: '500',\n    ...textEllipsis,\n    flex: 1,\n    '&:not(:only-child)': {\n      marginTop: '0.5em',\n    },\n  },\n  img: {\n    display: 'block',\n    width: '2em',\n    height: '2em',\n    borderRadius: '0.25em',\n    objectFit: 'cover',\n  },\n}\n\nconst backStyle = {\n  position: 'absolute',\n  zIndex: 0,\n  width: '100%',\n  height: '100%',\n  // move DAI UI slightly up to avoid stacking\n  paddingBottom: 'calc(var(--spacing) * 4)',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  '> iframe': {pointerEvents: 'auto'},\n}\n\nconst skipStyle = {\n  position: 'absolute',\n  right: 0,\n  bottom: '9rem',\n  textAlign: 'right',\n  button: {\n    width: 'auto',\n    height: 'auto',\n  },\n}\n\nconst desktopStyle = {\n  '--spacing': '0.75em',\n  '--center-pointer-events': 'none',\n  '> div button': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em)',\n  },\n  '--thumbnail-width': '288', // height 162\n}\n\nconst desktopControls = {\n  '> div:first-of-type': {\n    flex: '100%',\n    marginBottom: '0.4em',\n  },\n  '> button': {\n    '&[disabled]': {\n    display: 'none',\n    },\n    '&.play-button': {\n      fontSize: '1em',\n    },\n    '&.firstplay-button': {\n      fontSize: '400%'\n    },\n  },\n}\n\nconst seekbarStyles = {\n  flex: '1 var(--seekbar-flex-basis, 100%)',\n  '&:not(:empty)': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n  },\n}\n\nconst adContainerStyle = {\n  flexGrow: 1,\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  pointerEvents: 'var(--center-pointer-events)',\n  button: {pointerEvents: 'auto'},\n  zIndex: 0,\n}\n\nconst ControlsBlock = ({\n  order = 'mobile',\n  liveButton,\n  playButton,\n  rewindButton = '',\n  forwardButton = '',\n  previousEpisodeButton = '',\n  nextEpisodeButton = '',\n}) =>\n  order === 'desktop' ? (\n    <>\n      {liveButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {rewindButton}\n      {forwardButton}\n    </>\n  ) : (\n    <>\n      {rewindButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {forwardButton}\n    </>\n  )\n\nconst DefaultLayout = ({\n  type = 'mobile',\n  style,\n  display,\n  liveButton,\n  controlsDisplay = display,\n  size,\n  title = '',\n  channelTitle = '',\n  video,\n  haveBottomItem,\n  seekbar = '',\n  displayTime,\n  controlButtons,\n  volumeControl,\n  fullscreenButton,\n  backButton = '',\n  adStatus = '',\n  adLink = '',\n  adSkipButton,\n  backItems,\n  children,\n  containerRef,\n  backRef,\n  adContainerRef,\n  ...rest\n}) => {\n  const slotRef = useRef({})\n\n  return (\n    <div\n      style={{\n        '--player-bottom-ui-top':\n          controlsDisplay === 'shown'\n            ? `-${bottomUiTop(containerRef.current)}px`\n            : '-8px',\n        '--player-bottom-ui-height':\n          type === 'desktop' && controlsDisplay === 'shown' ? '3.5rem' : '0',\n      }}\n      css={[\n        containerStyle,\n        videoContainerStyle,\n        responsiveStyles[size],\n        type === 'desktop' && desktopStyle,\n        adStatus && {'--bottom-spacing': 0},\n        nativeTextStyle,\n        customTextStyle,\n        style,\n      ]}\n      ref={containerRef}\n      {...rest}\n    >\n      {video}\n      <div\n        ref={backRef}\n        css={[\n          backStyle,\n          display !== 'hidden' && (haveBottomItem ? dropTop : drop),\n          invisible('--controller-middle-area'),\n        ]}\n      >\n        {type !== 'mobile' && backItems}\n        {adSkipButton && <div css={skipStyle}>{adSkipButton}</div>}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          infoStyle,\n          displayStyles[display],\n          invisible('--controller-top-area'),\n        ]}\n      >\n        {backButton}\n        <div\n          ref={element => {\n            slotRef.current.titleBarLeft = element\n          }}\n        />\n        <div>\n          <h1>{title}</h1>\n          {channelTitle && <div>{channelTitle}</div>}\n        </div>\n        <div\n          ref={element => {\n            slotRef.current.titleBarRight = element\n          }}\n        />\n        {adLink && <div className=\"pinned\">{adLink}</div>}\n      </div>\n      <div\n        ref={adContainerRef}\n        css={[\n          adContainerStyle,\n          // this container covers Google DAI iframe UI, left some space for user to click \"skip ad\"\n          // see https://kkvideo.atlassian.net/browse/CPT-4535 for detail\n          adStatus && {margin: 'calc(var(--spacing) * 6) 0'},\n        ]}\n      >\n        {type === 'mobile' && (\n          <div\n            css={[\n              controlsStyle(type),\n              displayStyles[controlsDisplay],\n              invisible('--controller-middle-area'),\n            ]}\n          >\n            <ControlsBlock order=\"mobile\" {...controlButtons} />\n          </div>\n        )}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          {\n            marginTop: '1em',\n            paddingTop: 0,\n            paddingBottom: 'calc(1em - var(--bottom-spacing, 0) / 5)',\n            flexWrap: 'wrap',\n            '> div:first-of-type': seekbarStyles,\n          },\n          type === 'desktop' && desktopControls,\n          controlsDisplayStyles[controlsDisplay],\n          invisible('--controller-down-area'),\n        ]}\n      >\n        {seekbar || <div />}\n        {type === 'desktop' && (\n          <ControlsBlock\n            order=\"desktop\"\n            liveButton={liveButton}\n            {...controlButtons}\n          />\n        )}\n        {type === 'desktop' ? (\n          <div css={[displayStyles[controlsDisplay]]}>{displayTime}</div>\n        ) : (\n          <>\n            {liveButton}\n            <div\n              css={{\n                '&:not(:empty)': {\n                  flex: 1,\n                  marginLeft: 'calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n                },\n              }}\n            >\n              {displayTime}\n            </div>\n          </>\n        )}\n        {adStatus && (\n          <div\n            className=\"pinned\"\n            css={[infoBarSlotStyle, type === 'desktop' && desktopInfoBarStyle]}\n          >\n            {adStatus}\n          </div>\n        )}\n        <div\n          css={\n            !adStatus && [\n              infoBarSlotStyle,\n              type === 'desktop' && desktopInfoBarStyle,\n            ]\n          }\n          ref={element => {\n            slotRef.current.infoBar = element\n          }}\n        />\n        {volumeControl}\n        <div\n          css={[adStatus ? hidden : {display: 'flex'}]}\n          ref={element => {\n            slotRef.current.functionBar = element\n          }}\n        />\n        {fullscreenButton}\n      </div>\n      <SlotProvider slotRef={slotRef}>{children}</SlotProvider>\n    </div>\n  )\n}\n\nexport default DefaultLayout\n"]} */"],
|
|
3599
3543
|
children: jsx$1(ControlsBlock, {
|
|
3600
3544
|
order: "mobile",
|
|
3601
3545
|
...controlButtons
|
|
@@ -3608,13 +3552,13 @@ const DefaultLayout = ({
|
|
|
3608
3552
|
paddingBottom: 'calc(1em - var(--bottom-spacing, 0) / 5)',
|
|
3609
3553
|
flexWrap: 'wrap',
|
|
3610
3554
|
'> div:first-of-type': seekbarStyles
|
|
3611
|
-
}, type === 'desktop' && desktopControls, controlsDisplayStyles[controlsDisplay], invisible('--controller-down-area'), process.env.NODE_ENV === "production" ? "" : ";label:DefaultLayout;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["DefaultLayout.js"],"names":[],"mappings":"AAyaQ","file":"DefaultLayout.js","sourcesContent":["/* eslint-disable no-param-reassign */\n/* @jsxImportSource @emotion/react */\n/* eslint-disable react/prop-types */\nimport {useRef} from 'react'\nimport {SlotProvider} from './uiExtensions'\nimport {nativeTextStyle, customTextStyle, bottomUiTop} from './subtitles'\n\nconst hidden = {display: 'none'}\nconst invisible = hiddenArea => ({\n  visibility: `var(${hiddenArea}, visible)`,\n})\nconst containerStyle = {\n  '--spacing': '0.5em',\n  '--center-pointer-events': 'auto',\n  width: '100%',\n  height: '100%',\n  fontSize: '16px',\n  boxSizing: 'border-box',\n  display: 'flex',\n  flexDirection: 'column',\n  overflow: 'hidden',\n  color: 'white',\n  background: '#000',\n  // prevent animation glich(afterimage) of descendant elements\n  transform: 'translateX(0)',\n  userSelect: 'none',\n  'a, a:link, a:visited': {\n    color: '#fff',\n    opacity: 0.8,\n    textDecoration: 'none',\n  },\n  button: {\n    fontSize: 'inherit',\n    '> *': {\n      pointerEvents: 'none',\n    },\n  },\n  '--thumbnail-width': '160', // height 90\n  '--theme-color': 'var(--primary-highlight, red)',\n}\n\nconst videoContainerStyle = {\n  '> div:first-of-type': {\n    position: 'absolute',\n    zIndex: '-1',\n    display: 'flex',\n    alignItems: 'center',\n    width: '100%',\n    height: '100%',\n  },\n}\n\nconst drop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0.5) 0,\n    rgba(0,0,0,0) var(--player-ui-drop-shadow-size, 30%) calc(100% - var(--player-ui-drop-shadow-size, 30%)),\n    rgba(0,0,0,0.5) 100%\n  )`,\n  backgroundColor: `rgba(0, 0, 0, 0.3)`,\n}\n\nconst dropTop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0) 0,\n    rgba(0,0,0,0) 8rem calc(100% - 8rem),\n    rgba(0,0,0,0.5) 100%\n  )`,\n}\n\nconst responsiveStyles = {\n  desktop: {\n    fontSize: '24px',\n  }, // add if necessary: big-desktop\n}\n\nconst rowStyle = {\n  boxSizing: 'border-box',\n  width: '100%',\n  padding: '1em',\n  display: 'flex',\n  alignItems: 'center',\n  '> button, > div > button': {\n    padding: '0.5rem',\n  },\n}\n\n/* \n  Hint:\n    displayStyles specificity is higher then controlsStyle\n    Need to pay attention when we revise these styles.\n*/\n\nconst displayHidden = {\n  '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n    opacity: 0,\n    transform: 'translate(-200vw, 0)',\n    transition: 'opacity 0.8s var(--autohide-delay, 0s) ease-out, transform 0s calc(0.8s + var(--autohide-delay, 0s))',\n    'button': {\n      opacity: 0, // extra transition to hide tooltip for Safari\n      transition: 'opacity 0s calc(0.8s + var(--autohide-delay, 0s)) ease-out',\n    }\n  },\n}\n\nconst displayStyles = {\n  hidden: displayHidden,\n  'seekbar-only': displayHidden,\n  shown: {\n    '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n      zIndex: 1,\n      pointerEvents: 'auto',\n      transition: 'opacity 0.3s ease-out, transform 0s 0s',\n    },\n  },\n}\n\nconst controlsDisplayStyles = {\n  hidden: {\n    ...displayStyles.hidden,\n    '~ div:not(.pinned)': displayStyles.hidden,\n  },\n  'seekbar-only': {\n    '> div:first-of-type ~ *': displayStyles.hidden,\n  },\n  shown: {\n    ...displayStyles.shown,\n    '~ div': displayStyles.shown,\n  },\n}\n\n// TODO: transform into sample object instead of function\nconst controlsStyle = type => ({\n  position: 'absolute',\n  zIndex: '2',\n  width: '100%',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'space-evenly',\n  fontSize: '1.5em',\n  '> button': {\n    padding: '0.75em 0.5em',\n    '&:disabled': {\n      opacity: 0,\n      pointerEvents: 'none',\n    },\n    '&.play-button': {\n      padding: '0.75rem 0.5rem',\n      fontSize: 'var(--play-button-size, 1em)',\n    },\n    '&.firstplay-button':\n      type === 'mobile'\n        ? {fontSize: '200%', padding: '4em 10em', marginBottom: '1em'}\n        : {fontSize: '400%'},\n  },\n})\n\n// Remove the height for imporving tapping effect at current, and this height was used for fixed region before.\nconst infoBarSlotStyle = {\n  display: 'flex',\n  order: -1,\n  alignItems: 'center',\n  flex: '1 60%',\n}\n\nconst desktopInfoBarStyle = {\n  marginLeft: 'var(--spacing, 0.5em)',\n  order: 'initial',\n  flex: '1 0%',\n}\n\nconst textEllipsis = {\n  overflow: 'hidden',\n  whiteSpace: 'nowrap',\n  textOverflow: 'ellipsis',\n}\n\nconst infoStyle = {\n  zIndex: 1,\n  overflow: 'visible',\n  pointerEvents: 'none',\n  padding: '0 calc(2em - 20px)',\n  marginBottom: 'calc(0.5em + var(--bottom-spacing, 0))',\n  flex: '3em 0',\n  '> div:first-of-type, > div:nth-of-type(2), > button': {\n    minWidth: 0,\n  },\n  '> div:nth-of-type(2)': {\n    // title and channel title\n    margin: '0 calc(var(--spacing, 0.5em) / 2)',\n    flex: 1,\n    fontSize: '0.75em',\n    visibility: 'var(--player-title-visibility, visible)',\n    span: {\n      marginRight: 'var(--spacing, 0.5em)',\n    },\n    '> div': textEllipsis,\n  },\n  h1: {\n    margin: 0,\n    fontSize: '1.66em',\n    fontWeight: '500',\n    ...textEllipsis,\n    flex: 1,\n    '&:not(:only-child)': {\n      marginTop: '0.5em',\n    },\n  },\n  img: {\n    display: 'block',\n    width: '2em',\n    height: '2em',\n    borderRadius: '0.25em',\n    objectFit: 'cover',\n  },\n}\n\nconst backStyle = {\n  position: 'absolute',\n  zIndex: 0,\n  width: '100%',\n  height: '100%',\n  // move DAI UI slightly up to avoid stacking\n  paddingBottom: 'calc(var(--spacing) * 4)',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  '> iframe': {pointerEvents: 'auto'},\n}\n\nconst skipStyle = {\n  position: 'absolute',\n  right: 0,\n  bottom: '9rem',\n  textAlign: 'right',\n  button: {\n    width: 'auto',\n    height: 'auto',\n  },\n}\n\nconst desktopStyle = {\n  '--spacing': '0.75em',\n  '--center-pointer-events': 'none',\n  '> div button': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em)',\n  },\n  '--thumbnail-width': '288', // height 162\n}\n\nconst desktopControls = {\n  '> div:first-of-type': {\n    flex: '100%',\n    marginBottom: '0.4em',\n  },\n  '> button': {\n    '&[disabled]': {\n    display: 'none',\n    },\n    '&.play-button': {\n      fontSize: '1em',\n    },\n    '&.firstplay-button': {\n      fontSize: '400%'\n    },\n  },\n}\n\nconst seekbarStyles = {\n  flex: '1 var(--seekbar-flex-basis, 100%)',\n  '&:not(:empty)': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n  },\n}\n\nconst adContainerStyle = {\n  flexGrow: 1,\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  pointerEvents: 'var(--center-pointer-events)',\n  button: {pointerEvents: 'auto'},\n  zIndex: 0,\n}\n\nconst ControlsBlock = ({\n  order = 'mobile',\n  liveButton,\n  playButton,\n  rewindButton = '',\n  forwardButton = '',\n  previousEpisodeButton = '',\n  nextEpisodeButton = '',\n}) =>\n  order === 'desktop' ? (\n    <>\n      {liveButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {rewindButton}\n      {forwardButton}\n    </>\n  ) : (\n    <>\n      {rewindButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {forwardButton}\n    </>\n  )\n\nconst DefaultLayout = ({\n  type = 'mobile',\n  style,\n  display,\n  liveButton,\n  controlsDisplay = display,\n  size,\n  title = '',\n  channelTitle = '',\n  video,\n  haveBottomItem,\n  seekbar = '',\n  displayTime,\n  controlButtons,\n  volumeControl,\n  fullscreenButton,\n  backButton = '',\n  adStatus = '',\n  adLink = '',\n  adSkipButton,\n  backItems,\n  children,\n  containerRef,\n  backRef,\n  adContainerRef,\n  ...rest\n}) => {\n  const slotRef = useRef({})\n\n  return (\n    <div\n      style={{\n        '--player-bottom-ui-top':\n          controlsDisplay === 'shown'\n            ? `-${bottomUiTop(containerRef.current)}px`\n            : '-8px',\n        '--player-bottom-ui-height':\n          type === 'desktop' && controlsDisplay === 'shown' ? '3.5rem' : '0',\n      }}\n      css={[\n        containerStyle,\n        videoContainerStyle,\n        responsiveStyles[size],\n        type === 'desktop' && desktopStyle,\n        adStatus && {'--bottom-spacing': 0},\n        nativeTextStyle,\n        customTextStyle,\n        style,\n      ]}\n      ref={containerRef}\n      {...rest}\n    >\n      {video}\n      <div\n        ref={backRef}\n        css={[\n          backStyle,\n          display !== 'hidden' && (haveBottomItem ? dropTop : drop),\n          invisible('--controller-middle-area'),\n        ]}\n      >\n        {type !== 'mobile' && backItems}\n        {adSkipButton && <div css={skipStyle}>{adSkipButton}</div>}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          infoStyle,\n          displayStyles[display],\n          invisible('--controller-top-area'),\n        ]}\n      >\n        {backButton}\n        <div\n          ref={element => {\n            slotRef.current.titleBarLeft = element\n          }}\n        />\n        <div>\n          <h1>{title}</h1>\n          {channelTitle && <div>{channelTitle}</div>}\n        </div>\n        <div\n          ref={element => {\n            slotRef.current.titleBarRight = element\n          }}\n        />\n        {adLink && <div className=\"pinned\">{adLink}</div>}\n      </div>\n      <div\n        ref={adContainerRef}\n        css={[\n          adContainerStyle,\n          // this container covers Google DAI iframe UI, left some space for user to click \"skip ad\"\n          // see https://kkvideo.atlassian.net/browse/CPT-4535 for detail\n          adStatus && {margin: 'calc(var(--spacing) * 6) 0'},\n        ]}\n      >\n        {type === 'mobile' && (\n          <div\n            css={[\n              controlsStyle(type),\n              displayStyles[controlsDisplay],\n              invisible('--controller-middle-area'),\n            ]}\n          >\n            <ControlsBlock order=\"mobile\" {...controlButtons} />\n          </div>\n        )}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          {\n            marginTop: '1em',\n            paddingTop: 0,\n            paddingBottom: 'calc(1em - var(--bottom-spacing, 0) / 5)',\n            flexWrap: 'wrap',\n            '> div:first-of-type': seekbarStyles,\n          },\n          type === 'desktop' && desktopControls,\n          controlsDisplayStyles[controlsDisplay],\n          invisible('--controller-down-area'),\n        ]}\n      >\n        {seekbar || <div />}\n        {type === 'desktop' && (\n          <ControlsBlock\n            order=\"desktop\"\n            liveButton={liveButton}\n            {...controlButtons}\n          />\n        )}\n        {type === 'desktop' ? (\n          <div css={[displayStyles[controlsDisplay]]}>{displayTime}</div>\n        ) : (\n          <>\n            {liveButton}\n            <div\n              css={{\n                '&:not(:empty)': {\n                  flex: 1,\n                  marginLeft: 'calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n                },\n              }}\n            >\n              {displayTime}\n            </div>\n          </>\n        )}\n        {adStatus && (\n          <div\n            className=\"pinned\"\n            css={[infoBarSlotStyle, type === 'desktop' && desktopInfoBarStyle]}\n          >\n            {adStatus}\n          </div>\n        )}\n        <div\n          css={\n            !adStatus && [\n              infoBarSlotStyle,\n              type === 'desktop' && desktopInfoBarStyle,\n            ]\n          }\n          ref={element => {\n            slotRef.current.infoBar = element\n          }}\n        />\n        {volumeControl}\n        <div\n          css={adStatus && hidden}\n          ref={element => {\n            slotRef.current.functionBar = element\n          }}\n        />\n        {fullscreenButton}\n      </div>\n      <SlotProvider slotRef={slotRef}>{children}</SlotProvider>\n    </div>\n  )\n}\n\nexport default DefaultLayout\n"]} */"],
|
|
3555
|
+
}, type === 'desktop' && desktopControls, controlsDisplayStyles[controlsDisplay], invisible('--controller-down-area'), process.env.NODE_ENV === "production" ? "" : ";label:DefaultLayout;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["DefaultLayout.js"],"names":[],"mappings":"AAyaQ","file":"DefaultLayout.js","sourcesContent":["/* eslint-disable no-param-reassign */\n/* @jsxImportSource @emotion/react */\n/* eslint-disable react/prop-types */\nimport {useRef} from 'react'\nimport {SlotProvider} from './uiExtensions'\nimport {nativeTextStyle, customTextStyle, bottomUiTop} from './subtitles'\n\nconst hidden = {display: 'none'}\nconst invisible = hiddenArea => ({\n  visibility: `var(${hiddenArea}, visible)`,\n})\nconst containerStyle = {\n  '--spacing': '0.5em',\n  '--center-pointer-events': 'auto',\n  width: '100%',\n  height: '100%',\n  fontSize: '16px',\n  boxSizing: 'border-box',\n  display: 'flex',\n  flexDirection: 'column',\n  overflow: 'hidden',\n  color: 'white',\n  background: '#000',\n  // prevent animation glich(afterimage) of descendant elements\n  transform: 'translateX(0)',\n  userSelect: 'none',\n  'a, a:link, a:visited': {\n    color: '#fff',\n    opacity: 0.8,\n    textDecoration: 'none',\n  },\n  button: {\n    fontSize: 'inherit',\n    '> *': {\n      pointerEvents: 'none',\n    },\n  },\n  '--thumbnail-width': '160', // height 90\n  '--theme-color': 'var(--primary-highlight, red)',\n}\n\nconst videoContainerStyle = {\n  '> div:first-of-type': {\n    position: 'absolute',\n    zIndex: '-1',\n    display: 'flex',\n    alignItems: 'center',\n    width: '100%',\n    height: '100%',\n  },\n}\n\nconst drop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0.5) 0,\n    rgba(0,0,0,0) var(--player-ui-drop-shadow-size, 30%) calc(100% - var(--player-ui-drop-shadow-size, 30%)),\n    rgba(0,0,0,0.5) 100%\n  )`,\n  backgroundColor: `rgba(0, 0, 0, 0.3)`,\n}\n\nconst dropTop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0) 0,\n    rgba(0,0,0,0) 8rem calc(100% - 8rem),\n    rgba(0,0,0,0.5) 100%\n  )`,\n}\n\nconst responsiveStyles = {\n  desktop: {\n    fontSize: '24px',\n  }, // add if necessary: big-desktop\n}\n\nconst rowStyle = {\n  boxSizing: 'border-box',\n  width: '100%',\n  padding: '1em',\n  display: 'flex',\n  alignItems: 'center',\n  '> button, > div > button': {\n    padding: '0.5rem',\n  },\n}\n\n/* \n  Hint:\n    displayStyles specificity is higher then controlsStyle\n    Need to pay attention when we revise these styles.\n*/\n\nconst displayHidden = {\n  '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n    opacity: 0,\n    transform: 'translate(-200vw, 0)',\n    transition: 'opacity 0.8s var(--autohide-delay, 0s) ease-out, transform 0s calc(0.8s + var(--autohide-delay, 0s))',\n    'button': {\n      opacity: 0, // extra transition to hide tooltip for Safari\n      transition: 'opacity 0s calc(0.8s + var(--autohide-delay, 0s)) ease-out',\n    }\n  },\n}\n\nconst displayStyles = {\n  hidden: displayHidden,\n  'seekbar-only': displayHidden,\n  shown: {\n    '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n      zIndex: 1,\n      pointerEvents: 'auto',\n      transition: 'opacity 0.3s ease-out, transform 0s 0s',\n    },\n  },\n}\n\nconst controlsDisplayStyles = {\n  hidden: {\n    ...displayStyles.hidden,\n    '~ div:not(.pinned)': displayStyles.hidden,\n  },\n  'seekbar-only': {\n    '> div:first-of-type ~ *': displayStyles.hidden,\n  },\n  shown: {\n    ...displayStyles.shown,\n    '~ div': displayStyles.shown,\n  },\n}\n\n// TODO: transform into sample object instead of function\nconst controlsStyle = type => ({\n  position: 'absolute',\n  zIndex: '2',\n  width: '100%',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'space-evenly',\n  fontSize: '1.5em',\n  '> button': {\n    padding: '0.75em 0.5em',\n    '&:disabled': {\n      opacity: 0,\n      pointerEvents: 'none',\n    },\n    '&.play-button': {\n      padding: '0.75rem 0.5rem',\n      fontSize: 'var(--play-button-size, 1em)',\n    },\n    '&.firstplay-button':\n      type === 'mobile'\n        ? {fontSize: '200%', padding: '4em 10em', marginBottom: '1em'}\n        : {fontSize: '400%'},\n  },\n})\n\n// Remove the height for imporving tapping effect at current, and this height was used for fixed region before.\nconst infoBarSlotStyle = {\n  display: 'flex',\n  order: -1,\n  alignItems: 'center',\n  flex: '1 60%',\n}\n\nconst desktopInfoBarStyle = {\n  marginLeft: 'var(--spacing, 0.5em)',\n  order: 'initial',\n  flex: '1 0%',\n}\n\nconst textEllipsis = {\n  overflow: 'hidden',\n  whiteSpace: 'nowrap',\n  textOverflow: 'ellipsis',\n}\n\nconst infoStyle = {\n  zIndex: 1,\n  overflow: 'visible',\n  pointerEvents: 'none',\n  padding: '0 calc(2em - 20px)',\n  marginBottom: 'calc(0.5em + var(--bottom-spacing, 0))',\n  flex: '3em 0',\n  '> div:first-of-type, > div:nth-of-type(2), > button': {\n    minWidth: 0,\n  },\n  '> div:nth-of-type(2)': {\n    // title and channel title\n    margin: '0 calc(var(--spacing, 0.5em) / 2)',\n    flex: 1,\n    fontSize: '0.75em',\n    visibility: 'var(--player-title-visibility, visible)',\n    span: {\n      marginRight: 'var(--spacing, 0.5em)',\n    },\n    '> div': textEllipsis,\n  },\n  h1: {\n    margin: 0,\n    fontSize: '1.66em',\n    fontWeight: '500',\n    ...textEllipsis,\n    flex: 1,\n    '&:not(:only-child)': {\n      marginTop: '0.5em',\n    },\n  },\n  img: {\n    display: 'block',\n    width: '2em',\n    height: '2em',\n    borderRadius: '0.25em',\n    objectFit: 'cover',\n  },\n}\n\nconst backStyle = {\n  position: 'absolute',\n  zIndex: 0,\n  width: '100%',\n  height: '100%',\n  // move DAI UI slightly up to avoid stacking\n  paddingBottom: 'calc(var(--spacing) * 4)',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  '> iframe': {pointerEvents: 'auto'},\n}\n\nconst skipStyle = {\n  position: 'absolute',\n  right: 0,\n  bottom: '9rem',\n  textAlign: 'right',\n  button: {\n    width: 'auto',\n    height: 'auto',\n  },\n}\n\nconst desktopStyle = {\n  '--spacing': '0.75em',\n  '--center-pointer-events': 'none',\n  '> div button': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em)',\n  },\n  '--thumbnail-width': '288', // height 162\n}\n\nconst desktopControls = {\n  '> div:first-of-type': {\n    flex: '100%',\n    marginBottom: '0.4em',\n  },\n  '> button': {\n    '&[disabled]': {\n    display: 'none',\n    },\n    '&.play-button': {\n      fontSize: '1em',\n    },\n    '&.firstplay-button': {\n      fontSize: '400%'\n    },\n  },\n}\n\nconst seekbarStyles = {\n  flex: '1 var(--seekbar-flex-basis, 100%)',\n  '&:not(:empty)': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n  },\n}\n\nconst adContainerStyle = {\n  flexGrow: 1,\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  pointerEvents: 'var(--center-pointer-events)',\n  button: {pointerEvents: 'auto'},\n  zIndex: 0,\n}\n\nconst ControlsBlock = ({\n  order = 'mobile',\n  liveButton,\n  playButton,\n  rewindButton = '',\n  forwardButton = '',\n  previousEpisodeButton = '',\n  nextEpisodeButton = '',\n}) =>\n  order === 'desktop' ? (\n    <>\n      {liveButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {rewindButton}\n      {forwardButton}\n    </>\n  ) : (\n    <>\n      {rewindButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {forwardButton}\n    </>\n  )\n\nconst DefaultLayout = ({\n  type = 'mobile',\n  style,\n  display,\n  liveButton,\n  controlsDisplay = display,\n  size,\n  title = '',\n  channelTitle = '',\n  video,\n  haveBottomItem,\n  seekbar = '',\n  displayTime,\n  controlButtons,\n  volumeControl,\n  fullscreenButton,\n  backButton = '',\n  adStatus = '',\n  adLink = '',\n  adSkipButton,\n  backItems,\n  children,\n  containerRef,\n  backRef,\n  adContainerRef,\n  ...rest\n}) => {\n  const slotRef = useRef({})\n\n  return (\n    <div\n      style={{\n        '--player-bottom-ui-top':\n          controlsDisplay === 'shown'\n            ? `-${bottomUiTop(containerRef.current)}px`\n            : '-8px',\n        '--player-bottom-ui-height':\n          type === 'desktop' && controlsDisplay === 'shown' ? '3.5rem' : '0',\n      }}\n      css={[\n        containerStyle,\n        videoContainerStyle,\n        responsiveStyles[size],\n        type === 'desktop' && desktopStyle,\n        adStatus && {'--bottom-spacing': 0},\n        nativeTextStyle,\n        customTextStyle,\n        style,\n      ]}\n      ref={containerRef}\n      {...rest}\n    >\n      {video}\n      <div\n        ref={backRef}\n        css={[\n          backStyle,\n          display !== 'hidden' && (haveBottomItem ? dropTop : drop),\n          invisible('--controller-middle-area'),\n        ]}\n      >\n        {type !== 'mobile' && backItems}\n        {adSkipButton && <div css={skipStyle}>{adSkipButton}</div>}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          infoStyle,\n          displayStyles[display],\n          invisible('--controller-top-area'),\n        ]}\n      >\n        {backButton}\n        <div\n          ref={element => {\n            slotRef.current.titleBarLeft = element\n          }}\n        />\n        <div>\n          <h1>{title}</h1>\n          {channelTitle && <div>{channelTitle}</div>}\n        </div>\n        <div\n          ref={element => {\n            slotRef.current.titleBarRight = element\n          }}\n        />\n        {adLink && <div className=\"pinned\">{adLink}</div>}\n      </div>\n      <div\n        ref={adContainerRef}\n        css={[\n          adContainerStyle,\n          // this container covers Google DAI iframe UI, left some space for user to click \"skip ad\"\n          // see https://kkvideo.atlassian.net/browse/CPT-4535 for detail\n          adStatus && {margin: 'calc(var(--spacing) * 6) 0'},\n        ]}\n      >\n        {type === 'mobile' && (\n          <div\n            css={[\n              controlsStyle(type),\n              displayStyles[controlsDisplay],\n              invisible('--controller-middle-area'),\n            ]}\n          >\n            <ControlsBlock order=\"mobile\" {...controlButtons} />\n          </div>\n        )}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          {\n            marginTop: '1em',\n            paddingTop: 0,\n            paddingBottom: 'calc(1em - var(--bottom-spacing, 0) / 5)',\n            flexWrap: 'wrap',\n            '> div:first-of-type': seekbarStyles,\n          },\n          type === 'desktop' && desktopControls,\n          controlsDisplayStyles[controlsDisplay],\n          invisible('--controller-down-area'),\n        ]}\n      >\n        {seekbar || <div />}\n        {type === 'desktop' && (\n          <ControlsBlock\n            order=\"desktop\"\n            liveButton={liveButton}\n            {...controlButtons}\n          />\n        )}\n        {type === 'desktop' ? (\n          <div css={[displayStyles[controlsDisplay]]}>{displayTime}</div>\n        ) : (\n          <>\n            {liveButton}\n            <div\n              css={{\n                '&:not(:empty)': {\n                  flex: 1,\n                  marginLeft: 'calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n                },\n              }}\n            >\n              {displayTime}\n            </div>\n          </>\n        )}\n        {adStatus && (\n          <div\n            className=\"pinned\"\n            css={[infoBarSlotStyle, type === 'desktop' && desktopInfoBarStyle]}\n          >\n            {adStatus}\n          </div>\n        )}\n        <div\n          css={\n            !adStatus && [\n              infoBarSlotStyle,\n              type === 'desktop' && desktopInfoBarStyle,\n            ]\n          }\n          ref={element => {\n            slotRef.current.infoBar = element\n          }}\n        />\n        {volumeControl}\n        <div\n          css={[adStatus ? hidden : {display: 'flex'}]}\n          ref={element => {\n            slotRef.current.functionBar = element\n          }}\n        />\n        {fullscreenButton}\n      </div>\n      <SlotProvider slotRef={slotRef}>{children}</SlotProvider>\n    </div>\n  )\n}\n\nexport default DefaultLayout\n"]} */"],
|
|
3612
3556
|
children: [seekbar || jsx$1("div", {}), type === 'desktop' && jsx$1(ControlsBlock, {
|
|
3613
3557
|
order: "desktop",
|
|
3614
3558
|
liveButton: liveButton,
|
|
3615
3559
|
...controlButtons
|
|
3616
3560
|
}), type === 'desktop' ? jsx$1("div", {
|
|
3617
|
-
css: [displayStyles[controlsDisplay], process.env.NODE_ENV === "production" ? "" : ";label:DefaultLayout;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["DefaultLayout.js"],"names":[],"mappings":"AAgce","file":"DefaultLayout.js","sourcesContent":["/* eslint-disable no-param-reassign */\n/* @jsxImportSource @emotion/react */\n/* eslint-disable react/prop-types */\nimport {useRef} from 'react'\nimport {SlotProvider} from './uiExtensions'\nimport {nativeTextStyle, customTextStyle, bottomUiTop} from './subtitles'\n\nconst hidden = {display: 'none'}\nconst invisible = hiddenArea => ({\n  visibility: `var(${hiddenArea}, visible)`,\n})\nconst containerStyle = {\n  '--spacing': '0.5em',\n  '--center-pointer-events': 'auto',\n  width: '100%',\n  height: '100%',\n  fontSize: '16px',\n  boxSizing: 'border-box',\n  display: 'flex',\n  flexDirection: 'column',\n  overflow: 'hidden',\n  color: 'white',\n  background: '#000',\n  // prevent animation glich(afterimage) of descendant elements\n  transform: 'translateX(0)',\n  userSelect: 'none',\n  'a, a:link, a:visited': {\n    color: '#fff',\n    opacity: 0.8,\n    textDecoration: 'none',\n  },\n  button: {\n    fontSize: 'inherit',\n    '> *': {\n      pointerEvents: 'none',\n    },\n  },\n  '--thumbnail-width': '160', // height 90\n  '--theme-color': 'var(--primary-highlight, red)',\n}\n\nconst videoContainerStyle = {\n  '> div:first-of-type': {\n    position: 'absolute',\n    zIndex: '-1',\n    display: 'flex',\n    alignItems: 'center',\n    width: '100%',\n    height: '100%',\n  },\n}\n\nconst drop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0.5) 0,\n    rgba(0,0,0,0) var(--player-ui-drop-shadow-size, 30%) calc(100% - var(--player-ui-drop-shadow-size, 30%)),\n    rgba(0,0,0,0.5) 100%\n  )`,\n  backgroundColor: `rgba(0, 0, 0, 0.3)`,\n}\n\nconst dropTop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0) 0,\n    rgba(0,0,0,0) 8rem calc(100% - 8rem),\n    rgba(0,0,0,0.5) 100%\n  )`,\n}\n\nconst responsiveStyles = {\n  desktop: {\n    fontSize: '24px',\n  }, // add if necessary: big-desktop\n}\n\nconst rowStyle = {\n  boxSizing: 'border-box',\n  width: '100%',\n  padding: '1em',\n  display: 'flex',\n  alignItems: 'center',\n  '> button, > div > button': {\n    padding: '0.5rem',\n  },\n}\n\n/* \n  Hint:\n    displayStyles specificity is higher then controlsStyle\n    Need to pay attention when we revise these styles.\n*/\n\nconst displayHidden = {\n  '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n    opacity: 0,\n    transform: 'translate(-200vw, 0)',\n    transition: 'opacity 0.8s var(--autohide-delay, 0s) ease-out, transform 0s calc(0.8s + var(--autohide-delay, 0s))',\n    'button': {\n      opacity: 0, // extra transition to hide tooltip for Safari\n      transition: 'opacity 0s calc(0.8s + var(--autohide-delay, 0s)) ease-out',\n    }\n  },\n}\n\nconst displayStyles = {\n  hidden: displayHidden,\n  'seekbar-only': displayHidden,\n  shown: {\n    '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n      zIndex: 1,\n      pointerEvents: 'auto',\n      transition: 'opacity 0.3s ease-out, transform 0s 0s',\n    },\n  },\n}\n\nconst controlsDisplayStyles = {\n  hidden: {\n    ...displayStyles.hidden,\n    '~ div:not(.pinned)': displayStyles.hidden,\n  },\n  'seekbar-only': {\n    '> div:first-of-type ~ *': displayStyles.hidden,\n  },\n  shown: {\n    ...displayStyles.shown,\n    '~ div': displayStyles.shown,\n  },\n}\n\n// TODO: transform into sample object instead of function\nconst controlsStyle = type => ({\n  position: 'absolute',\n  zIndex: '2',\n  width: '100%',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'space-evenly',\n  fontSize: '1.5em',\n  '> button': {\n    padding: '0.75em 0.5em',\n    '&:disabled': {\n      opacity: 0,\n      pointerEvents: 'none',\n    },\n    '&.play-button': {\n      padding: '0.75rem 0.5rem',\n      fontSize: 'var(--play-button-size, 1em)',\n    },\n    '&.firstplay-button':\n      type === 'mobile'\n        ? {fontSize: '200%', padding: '4em 10em', marginBottom: '1em'}\n        : {fontSize: '400%'},\n  },\n})\n\n// Remove the height for imporving tapping effect at current, and this height was used for fixed region before.\nconst infoBarSlotStyle = {\n  display: 'flex',\n  order: -1,\n  alignItems: 'center',\n  flex: '1 60%',\n}\n\nconst desktopInfoBarStyle = {\n  marginLeft: 'var(--spacing, 0.5em)',\n  order: 'initial',\n  flex: '1 0%',\n}\n\nconst textEllipsis = {\n  overflow: 'hidden',\n  whiteSpace: 'nowrap',\n  textOverflow: 'ellipsis',\n}\n\nconst infoStyle = {\n  zIndex: 1,\n  overflow: 'visible',\n  pointerEvents: 'none',\n  padding: '0 calc(2em - 20px)',\n  marginBottom: 'calc(0.5em + var(--bottom-spacing, 0))',\n  flex: '3em 0',\n  '> div:first-of-type, > div:nth-of-type(2), > button': {\n    minWidth: 0,\n  },\n  '> div:nth-of-type(2)': {\n    // title and channel title\n    margin: '0 calc(var(--spacing, 0.5em) / 2)',\n    flex: 1,\n    fontSize: '0.75em',\n    visibility: 'var(--player-title-visibility, visible)',\n    span: {\n      marginRight: 'var(--spacing, 0.5em)',\n    },\n    '> div': textEllipsis,\n  },\n  h1: {\n    margin: 0,\n    fontSize: '1.66em',\n    fontWeight: '500',\n    ...textEllipsis,\n    flex: 1,\n    '&:not(:only-child)': {\n      marginTop: '0.5em',\n    },\n  },\n  img: {\n    display: 'block',\n    width: '2em',\n    height: '2em',\n    borderRadius: '0.25em',\n    objectFit: 'cover',\n  },\n}\n\nconst backStyle = {\n  position: 'absolute',\n  zIndex: 0,\n  width: '100%',\n  height: '100%',\n  // move DAI UI slightly up to avoid stacking\n  paddingBottom: 'calc(var(--spacing) * 4)',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  '> iframe': {pointerEvents: 'auto'},\n}\n\nconst skipStyle = {\n  position: 'absolute',\n  right: 0,\n  bottom: '9rem',\n  textAlign: 'right',\n  button: {\n    width: 'auto',\n    height: 'auto',\n  },\n}\n\nconst desktopStyle = {\n  '--spacing': '0.75em',\n  '--center-pointer-events': 'none',\n  '> div button': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em)',\n  },\n  '--thumbnail-width': '288', // height 162\n}\n\nconst desktopControls = {\n  '> div:first-of-type': {\n    flex: '100%',\n    marginBottom: '0.4em',\n  },\n  '> button': {\n    '&[disabled]': {\n    display: 'none',\n    },\n    '&.play-button': {\n      fontSize: '1em',\n    },\n    '&.firstplay-button': {\n      fontSize: '400%'\n    },\n  },\n}\n\nconst seekbarStyles = {\n  flex: '1 var(--seekbar-flex-basis, 100%)',\n  '&:not(:empty)': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n  },\n}\n\nconst adContainerStyle = {\n  flexGrow: 1,\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  pointerEvents: 'var(--center-pointer-events)',\n  button: {pointerEvents: 'auto'},\n  zIndex: 0,\n}\n\nconst ControlsBlock = ({\n  order = 'mobile',\n  liveButton,\n  playButton,\n  rewindButton = '',\n  forwardButton = '',\n  previousEpisodeButton = '',\n  nextEpisodeButton = '',\n}) =>\n  order === 'desktop' ? (\n    <>\n      {liveButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {rewindButton}\n      {forwardButton}\n    </>\n  ) : (\n    <>\n      {rewindButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {forwardButton}\n    </>\n  )\n\nconst DefaultLayout = ({\n  type = 'mobile',\n  style,\n  display,\n  liveButton,\n  controlsDisplay = display,\n  size,\n  title = '',\n  channelTitle = '',\n  video,\n  haveBottomItem,\n  seekbar = '',\n  displayTime,\n  controlButtons,\n  volumeControl,\n  fullscreenButton,\n  backButton = '',\n  adStatus = '',\n  adLink = '',\n  adSkipButton,\n  backItems,\n  children,\n  containerRef,\n  backRef,\n  adContainerRef,\n  ...rest\n}) => {\n  const slotRef = useRef({})\n\n  return (\n    <div\n      style={{\n        '--player-bottom-ui-top':\n          controlsDisplay === 'shown'\n            ? `-${bottomUiTop(containerRef.current)}px`\n            : '-8px',\n        '--player-bottom-ui-height':\n          type === 'desktop' && controlsDisplay === 'shown' ? '3.5rem' : '0',\n      }}\n      css={[\n        containerStyle,\n        videoContainerStyle,\n        responsiveStyles[size],\n        type === 'desktop' && desktopStyle,\n        adStatus && {'--bottom-spacing': 0},\n        nativeTextStyle,\n        customTextStyle,\n        style,\n      ]}\n      ref={containerRef}\n      {...rest}\n    >\n      {video}\n      <div\n        ref={backRef}\n        css={[\n          backStyle,\n          display !== 'hidden' && (haveBottomItem ? dropTop : drop),\n          invisible('--controller-middle-area'),\n        ]}\n      >\n        {type !== 'mobile' && backItems}\n        {adSkipButton && <div css={skipStyle}>{adSkipButton}</div>}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          infoStyle,\n          displayStyles[display],\n          invisible('--controller-top-area'),\n        ]}\n      >\n        {backButton}\n        <div\n          ref={element => {\n            slotRef.current.titleBarLeft = element\n          }}\n        />\n        <div>\n          <h1>{title}</h1>\n          {channelTitle && <div>{channelTitle}</div>}\n        </div>\n        <div\n          ref={element => {\n            slotRef.current.titleBarRight = element\n          }}\n        />\n        {adLink && <div className=\"pinned\">{adLink}</div>}\n      </div>\n      <div\n        ref={adContainerRef}\n        css={[\n          adContainerStyle,\n          // this container covers Google DAI iframe UI, left some space for user to click \"skip ad\"\n          // see https://kkvideo.atlassian.net/browse/CPT-4535 for detail\n          adStatus && {margin: 'calc(var(--spacing) * 6) 0'},\n        ]}\n      >\n        {type === 'mobile' && (\n          <div\n            css={[\n              controlsStyle(type),\n              displayStyles[controlsDisplay],\n              invisible('--controller-middle-area'),\n            ]}\n          >\n            <ControlsBlock order=\"mobile\" {...controlButtons} />\n          </div>\n        )}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          {\n            marginTop: '1em',\n            paddingTop: 0,\n            paddingBottom: 'calc(1em - var(--bottom-spacing, 0) / 5)',\n            flexWrap: 'wrap',\n            '> div:first-of-type': seekbarStyles,\n          },\n          type === 'desktop' && desktopControls,\n          controlsDisplayStyles[controlsDisplay],\n          invisible('--controller-down-area'),\n        ]}\n      >\n        {seekbar || <div />}\n        {type === 'desktop' && (\n          <ControlsBlock\n            order=\"desktop\"\n            liveButton={liveButton}\n            {...controlButtons}\n          />\n        )}\n        {type === 'desktop' ? (\n          <div css={[displayStyles[controlsDisplay]]}>{displayTime}</div>\n        ) : (\n          <>\n            {liveButton}\n            <div\n              css={{\n                '&:not(:empty)': {\n                  flex: 1,\n                  marginLeft: 'calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n                },\n              }}\n            >\n              {displayTime}\n            </div>\n          </>\n        )}\n        {adStatus && (\n          <div\n            className=\"pinned\"\n            css={[infoBarSlotStyle, type === 'desktop' && desktopInfoBarStyle]}\n          >\n            {adStatus}\n          </div>\n        )}\n        <div\n          css={\n            !adStatus && [\n              infoBarSlotStyle,\n              type === 'desktop' && desktopInfoBarStyle,\n            ]\n          }\n          ref={element => {\n            slotRef.current.infoBar = element\n          }}\n        />\n        {volumeControl}\n        <div\n          css={adStatus && hidden}\n          ref={element => {\n            slotRef.current.functionBar = element\n          }}\n        />\n        {fullscreenButton}\n      </div>\n      <SlotProvider slotRef={slotRef}>{children}</SlotProvider>\n    </div>\n  )\n}\n\nexport default DefaultLayout\n"]} */"],
|
|
3561
|
+
css: [displayStyles[controlsDisplay], process.env.NODE_ENV === "production" ? "" : ";label:DefaultLayout;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["DefaultLayout.js"],"names":[],"mappings":"AAgce","file":"DefaultLayout.js","sourcesContent":["/* eslint-disable no-param-reassign */\n/* @jsxImportSource @emotion/react */\n/* eslint-disable react/prop-types */\nimport {useRef} from 'react'\nimport {SlotProvider} from './uiExtensions'\nimport {nativeTextStyle, customTextStyle, bottomUiTop} from './subtitles'\n\nconst hidden = {display: 'none'}\nconst invisible = hiddenArea => ({\n  visibility: `var(${hiddenArea}, visible)`,\n})\nconst containerStyle = {\n  '--spacing': '0.5em',\n  '--center-pointer-events': 'auto',\n  width: '100%',\n  height: '100%',\n  fontSize: '16px',\n  boxSizing: 'border-box',\n  display: 'flex',\n  flexDirection: 'column',\n  overflow: 'hidden',\n  color: 'white',\n  background: '#000',\n  // prevent animation glich(afterimage) of descendant elements\n  transform: 'translateX(0)',\n  userSelect: 'none',\n  'a, a:link, a:visited': {\n    color: '#fff',\n    opacity: 0.8,\n    textDecoration: 'none',\n  },\n  button: {\n    fontSize: 'inherit',\n    '> *': {\n      pointerEvents: 'none',\n    },\n  },\n  '--thumbnail-width': '160', // height 90\n  '--theme-color': 'var(--primary-highlight, red)',\n}\n\nconst videoContainerStyle = {\n  '> div:first-of-type': {\n    position: 'absolute',\n    zIndex: '-1',\n    display: 'flex',\n    alignItems: 'center',\n    width: '100%',\n    height: '100%',\n  },\n}\n\nconst drop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0.5) 0,\n    rgba(0,0,0,0) var(--player-ui-drop-shadow-size, 30%) calc(100% - var(--player-ui-drop-shadow-size, 30%)),\n    rgba(0,0,0,0.5) 100%\n  )`,\n  backgroundColor: `rgba(0, 0, 0, 0.3)`,\n}\n\nconst dropTop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0) 0,\n    rgba(0,0,0,0) 8rem calc(100% - 8rem),\n    rgba(0,0,0,0.5) 100%\n  )`,\n}\n\nconst responsiveStyles = {\n  desktop: {\n    fontSize: '24px',\n  }, // add if necessary: big-desktop\n}\n\nconst rowStyle = {\n  boxSizing: 'border-box',\n  width: '100%',\n  padding: '1em',\n  display: 'flex',\n  alignItems: 'center',\n  '> button, > div > button': {\n    padding: '0.5rem',\n  },\n}\n\n/* \n  Hint:\n    displayStyles specificity is higher then controlsStyle\n    Need to pay attention when we revise these styles.\n*/\n\nconst displayHidden = {\n  '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n    opacity: 0,\n    transform: 'translate(-200vw, 0)',\n    transition: 'opacity 0.8s var(--autohide-delay, 0s) ease-out, transform 0s calc(0.8s + var(--autohide-delay, 0s))',\n    'button': {\n      opacity: 0, // extra transition to hide tooltip for Safari\n      transition: 'opacity 0s calc(0.8s + var(--autohide-delay, 0s)) ease-out',\n    }\n  },\n}\n\nconst displayStyles = {\n  hidden: displayHidden,\n  'seekbar-only': displayHidden,\n  shown: {\n    '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n      zIndex: 1,\n      pointerEvents: 'auto',\n      transition: 'opacity 0.3s ease-out, transform 0s 0s',\n    },\n  },\n}\n\nconst controlsDisplayStyles = {\n  hidden: {\n    ...displayStyles.hidden,\n    '~ div:not(.pinned)': displayStyles.hidden,\n  },\n  'seekbar-only': {\n    '> div:first-of-type ~ *': displayStyles.hidden,\n  },\n  shown: {\n    ...displayStyles.shown,\n    '~ div': displayStyles.shown,\n  },\n}\n\n// TODO: transform into sample object instead of function\nconst controlsStyle = type => ({\n  position: 'absolute',\n  zIndex: '2',\n  width: '100%',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'space-evenly',\n  fontSize: '1.5em',\n  '> button': {\n    padding: '0.75em 0.5em',\n    '&:disabled': {\n      opacity: 0,\n      pointerEvents: 'none',\n    },\n    '&.play-button': {\n      padding: '0.75rem 0.5rem',\n      fontSize: 'var(--play-button-size, 1em)',\n    },\n    '&.firstplay-button':\n      type === 'mobile'\n        ? {fontSize: '200%', padding: '4em 10em', marginBottom: '1em'}\n        : {fontSize: '400%'},\n  },\n})\n\n// Remove the height for imporving tapping effect at current, and this height was used for fixed region before.\nconst infoBarSlotStyle = {\n  display: 'flex',\n  order: -1,\n  alignItems: 'center',\n  flex: '1 60%',\n}\n\nconst desktopInfoBarStyle = {\n  marginLeft: 'var(--spacing, 0.5em)',\n  order: 'initial',\n  flex: '1 0%',\n}\n\nconst textEllipsis = {\n  overflow: 'hidden',\n  whiteSpace: 'nowrap',\n  textOverflow: 'ellipsis',\n}\n\nconst infoStyle = {\n  zIndex: 1,\n  overflow: 'visible',\n  pointerEvents: 'none',\n  padding: '0 calc(2em - 20px)',\n  marginBottom: 'calc(0.5em + var(--bottom-spacing, 0))',\n  flex: '3em 0',\n  '> div:first-of-type, > div:nth-of-type(2), > button': {\n    minWidth: 0,\n  },\n  '> div:nth-of-type(2)': {\n    // title and channel title\n    margin: '0 calc(var(--spacing, 0.5em) / 2)',\n    flex: 1,\n    fontSize: '0.75em',\n    visibility: 'var(--player-title-visibility, visible)',\n    span: {\n      marginRight: 'var(--spacing, 0.5em)',\n    },\n    '> div': textEllipsis,\n  },\n  h1: {\n    margin: 0,\n    fontSize: '1.66em',\n    fontWeight: '500',\n    ...textEllipsis,\n    flex: 1,\n    '&:not(:only-child)': {\n      marginTop: '0.5em',\n    },\n  },\n  img: {\n    display: 'block',\n    width: '2em',\n    height: '2em',\n    borderRadius: '0.25em',\n    objectFit: 'cover',\n  },\n}\n\nconst backStyle = {\n  position: 'absolute',\n  zIndex: 0,\n  width: '100%',\n  height: '100%',\n  // move DAI UI slightly up to avoid stacking\n  paddingBottom: 'calc(var(--spacing) * 4)',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  '> iframe': {pointerEvents: 'auto'},\n}\n\nconst skipStyle = {\n  position: 'absolute',\n  right: 0,\n  bottom: '9rem',\n  textAlign: 'right',\n  button: {\n    width: 'auto',\n    height: 'auto',\n  },\n}\n\nconst desktopStyle = {\n  '--spacing': '0.75em',\n  '--center-pointer-events': 'none',\n  '> div button': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em)',\n  },\n  '--thumbnail-width': '288', // height 162\n}\n\nconst desktopControls = {\n  '> div:first-of-type': {\n    flex: '100%',\n    marginBottom: '0.4em',\n  },\n  '> button': {\n    '&[disabled]': {\n    display: 'none',\n    },\n    '&.play-button': {\n      fontSize: '1em',\n    },\n    '&.firstplay-button': {\n      fontSize: '400%'\n    },\n  },\n}\n\nconst seekbarStyles = {\n  flex: '1 var(--seekbar-flex-basis, 100%)',\n  '&:not(:empty)': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n  },\n}\n\nconst adContainerStyle = {\n  flexGrow: 1,\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  pointerEvents: 'var(--center-pointer-events)',\n  button: {pointerEvents: 'auto'},\n  zIndex: 0,\n}\n\nconst ControlsBlock = ({\n  order = 'mobile',\n  liveButton,\n  playButton,\n  rewindButton = '',\n  forwardButton = '',\n  previousEpisodeButton = '',\n  nextEpisodeButton = '',\n}) =>\n  order === 'desktop' ? (\n    <>\n      {liveButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {rewindButton}\n      {forwardButton}\n    </>\n  ) : (\n    <>\n      {rewindButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {forwardButton}\n    </>\n  )\n\nconst DefaultLayout = ({\n  type = 'mobile',\n  style,\n  display,\n  liveButton,\n  controlsDisplay = display,\n  size,\n  title = '',\n  channelTitle = '',\n  video,\n  haveBottomItem,\n  seekbar = '',\n  displayTime,\n  controlButtons,\n  volumeControl,\n  fullscreenButton,\n  backButton = '',\n  adStatus = '',\n  adLink = '',\n  adSkipButton,\n  backItems,\n  children,\n  containerRef,\n  backRef,\n  adContainerRef,\n  ...rest\n}) => {\n  const slotRef = useRef({})\n\n  return (\n    <div\n      style={{\n        '--player-bottom-ui-top':\n          controlsDisplay === 'shown'\n            ? `-${bottomUiTop(containerRef.current)}px`\n            : '-8px',\n        '--player-bottom-ui-height':\n          type === 'desktop' && controlsDisplay === 'shown' ? '3.5rem' : '0',\n      }}\n      css={[\n        containerStyle,\n        videoContainerStyle,\n        responsiveStyles[size],\n        type === 'desktop' && desktopStyle,\n        adStatus && {'--bottom-spacing': 0},\n        nativeTextStyle,\n        customTextStyle,\n        style,\n      ]}\n      ref={containerRef}\n      {...rest}\n    >\n      {video}\n      <div\n        ref={backRef}\n        css={[\n          backStyle,\n          display !== 'hidden' && (haveBottomItem ? dropTop : drop),\n          invisible('--controller-middle-area'),\n        ]}\n      >\n        {type !== 'mobile' && backItems}\n        {adSkipButton && <div css={skipStyle}>{adSkipButton}</div>}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          infoStyle,\n          displayStyles[display],\n          invisible('--controller-top-area'),\n        ]}\n      >\n        {backButton}\n        <div\n          ref={element => {\n            slotRef.current.titleBarLeft = element\n          }}\n        />\n        <div>\n          <h1>{title}</h1>\n          {channelTitle && <div>{channelTitle}</div>}\n        </div>\n        <div\n          ref={element => {\n            slotRef.current.titleBarRight = element\n          }}\n        />\n        {adLink && <div className=\"pinned\">{adLink}</div>}\n      </div>\n      <div\n        ref={adContainerRef}\n        css={[\n          adContainerStyle,\n          // this container covers Google DAI iframe UI, left some space for user to click \"skip ad\"\n          // see https://kkvideo.atlassian.net/browse/CPT-4535 for detail\n          adStatus && {margin: 'calc(var(--spacing) * 6) 0'},\n        ]}\n      >\n        {type === 'mobile' && (\n          <div\n            css={[\n              controlsStyle(type),\n              displayStyles[controlsDisplay],\n              invisible('--controller-middle-area'),\n            ]}\n          >\n            <ControlsBlock order=\"mobile\" {...controlButtons} />\n          </div>\n        )}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          {\n            marginTop: '1em',\n            paddingTop: 0,\n            paddingBottom: 'calc(1em - var(--bottom-spacing, 0) / 5)',\n            flexWrap: 'wrap',\n            '> div:first-of-type': seekbarStyles,\n          },\n          type === 'desktop' && desktopControls,\n          controlsDisplayStyles[controlsDisplay],\n          invisible('--controller-down-area'),\n        ]}\n      >\n        {seekbar || <div />}\n        {type === 'desktop' && (\n          <ControlsBlock\n            order=\"desktop\"\n            liveButton={liveButton}\n            {...controlButtons}\n          />\n        )}\n        {type === 'desktop' ? (\n          <div css={[displayStyles[controlsDisplay]]}>{displayTime}</div>\n        ) : (\n          <>\n            {liveButton}\n            <div\n              css={{\n                '&:not(:empty)': {\n                  flex: 1,\n                  marginLeft: 'calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n                },\n              }}\n            >\n              {displayTime}\n            </div>\n          </>\n        )}\n        {adStatus && (\n          <div\n            className=\"pinned\"\n            css={[infoBarSlotStyle, type === 'desktop' && desktopInfoBarStyle]}\n          >\n            {adStatus}\n          </div>\n        )}\n        <div\n          css={\n            !adStatus && [\n              infoBarSlotStyle,\n              type === 'desktop' && desktopInfoBarStyle,\n            ]\n          }\n          ref={element => {\n            slotRef.current.infoBar = element\n          }}\n        />\n        {volumeControl}\n        <div\n          css={[adStatus ? hidden : {display: 'flex'}]}\n          ref={element => {\n            slotRef.current.functionBar = element\n          }}\n        />\n        {fullscreenButton}\n      </div>\n      <SlotProvider slotRef={slotRef}>{children}</SlotProvider>\n    </div>\n  )\n}\n\nexport default DefaultLayout\n"]} */"],
|
|
3618
3562
|
children: displayTime
|
|
3619
3563
|
}) : jsxs(Fragment, {
|
|
3620
3564
|
children: [liveButton, jsx$1("div", {
|
|
@@ -3623,7 +3567,7 @@ const DefaultLayout = ({
|
|
|
3623
3567
|
})]
|
|
3624
3568
|
}), adStatus && jsx$1("div", {
|
|
3625
3569
|
className: "pinned",
|
|
3626
|
-
css: [infoBarSlotStyle, type === 'desktop' && desktopInfoBarStyle, process.env.NODE_ENV === "production" ? "" : ";label:DefaultLayout;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["DefaultLayout.js"],"names":[],"mappings":"AAmdY","file":"DefaultLayout.js","sourcesContent":["/* eslint-disable no-param-reassign */\n/* @jsxImportSource @emotion/react */\n/* eslint-disable react/prop-types */\nimport {useRef} from 'react'\nimport {SlotProvider} from './uiExtensions'\nimport {nativeTextStyle, customTextStyle, bottomUiTop} from './subtitles'\n\nconst hidden = {display: 'none'}\nconst invisible = hiddenArea => ({\n  visibility: `var(${hiddenArea}, visible)`,\n})\nconst containerStyle = {\n  '--spacing': '0.5em',\n  '--center-pointer-events': 'auto',\n  width: '100%',\n  height: '100%',\n  fontSize: '16px',\n  boxSizing: 'border-box',\n  display: 'flex',\n  flexDirection: 'column',\n  overflow: 'hidden',\n  color: 'white',\n  background: '#000',\n  // prevent animation glich(afterimage) of descendant elements\n  transform: 'translateX(0)',\n  userSelect: 'none',\n  'a, a:link, a:visited': {\n    color: '#fff',\n    opacity: 0.8,\n    textDecoration: 'none',\n  },\n  button: {\n    fontSize: 'inherit',\n    '> *': {\n      pointerEvents: 'none',\n    },\n  },\n  '--thumbnail-width': '160', // height 90\n  '--theme-color': 'var(--primary-highlight, red)',\n}\n\nconst videoContainerStyle = {\n  '> div:first-of-type': {\n    position: 'absolute',\n    zIndex: '-1',\n    display: 'flex',\n    alignItems: 'center',\n    width: '100%',\n    height: '100%',\n  },\n}\n\nconst drop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0.5) 0,\n    rgba(0,0,0,0) var(--player-ui-drop-shadow-size, 30%) calc(100% - var(--player-ui-drop-shadow-size, 30%)),\n    rgba(0,0,0,0.5) 100%\n  )`,\n  backgroundColor: `rgba(0, 0, 0, 0.3)`,\n}\n\nconst dropTop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0) 0,\n    rgba(0,0,0,0) 8rem calc(100% - 8rem),\n    rgba(0,0,0,0.5) 100%\n  )`,\n}\n\nconst responsiveStyles = {\n  desktop: {\n    fontSize: '24px',\n  }, // add if necessary: big-desktop\n}\n\nconst rowStyle = {\n  boxSizing: 'border-box',\n  width: '100%',\n  padding: '1em',\n  display: 'flex',\n  alignItems: 'center',\n  '> button, > div > button': {\n    padding: '0.5rem',\n  },\n}\n\n/* \n  Hint:\n    displayStyles specificity is higher then controlsStyle\n    Need to pay attention when we revise these styles.\n*/\n\nconst displayHidden = {\n  '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n    opacity: 0,\n    transform: 'translate(-200vw, 0)',\n    transition: 'opacity 0.8s var(--autohide-delay, 0s) ease-out, transform 0s calc(0.8s + var(--autohide-delay, 0s))',\n    'button': {\n      opacity: 0, // extra transition to hide tooltip for Safari\n      transition: 'opacity 0s calc(0.8s + var(--autohide-delay, 0s)) ease-out',\n    }\n  },\n}\n\nconst displayStyles = {\n  hidden: displayHidden,\n  'seekbar-only': displayHidden,\n  shown: {\n    '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n      zIndex: 1,\n      pointerEvents: 'auto',\n      transition: 'opacity 0.3s ease-out, transform 0s 0s',\n    },\n  },\n}\n\nconst controlsDisplayStyles = {\n  hidden: {\n    ...displayStyles.hidden,\n    '~ div:not(.pinned)': displayStyles.hidden,\n  },\n  'seekbar-only': {\n    '> div:first-of-type ~ *': displayStyles.hidden,\n  },\n  shown: {\n    ...displayStyles.shown,\n    '~ div': displayStyles.shown,\n  },\n}\n\n// TODO: transform into sample object instead of function\nconst controlsStyle = type => ({\n  position: 'absolute',\n  zIndex: '2',\n  width: '100%',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'space-evenly',\n  fontSize: '1.5em',\n  '> button': {\n    padding: '0.75em 0.5em',\n    '&:disabled': {\n      opacity: 0,\n      pointerEvents: 'none',\n    },\n    '&.play-button': {\n      padding: '0.75rem 0.5rem',\n      fontSize: 'var(--play-button-size, 1em)',\n    },\n    '&.firstplay-button':\n      type === 'mobile'\n        ? {fontSize: '200%', padding: '4em 10em', marginBottom: '1em'}\n        : {fontSize: '400%'},\n  },\n})\n\n// Remove the height for imporving tapping effect at current, and this height was used for fixed region before.\nconst infoBarSlotStyle = {\n  display: 'flex',\n  order: -1,\n  alignItems: 'center',\n  flex: '1 60%',\n}\n\nconst desktopInfoBarStyle = {\n  marginLeft: 'var(--spacing, 0.5em)',\n  order: 'initial',\n  flex: '1 0%',\n}\n\nconst textEllipsis = {\n  overflow: 'hidden',\n  whiteSpace: 'nowrap',\n  textOverflow: 'ellipsis',\n}\n\nconst infoStyle = {\n  zIndex: 1,\n  overflow: 'visible',\n  pointerEvents: 'none',\n  padding: '0 calc(2em - 20px)',\n  marginBottom: 'calc(0.5em + var(--bottom-spacing, 0))',\n  flex: '3em 0',\n  '> div:first-of-type, > div:nth-of-type(2), > button': {\n    minWidth: 0,\n  },\n  '> div:nth-of-type(2)': {\n    // title and channel title\n    margin: '0 calc(var(--spacing, 0.5em) / 2)',\n    flex: 1,\n    fontSize: '0.75em',\n    visibility: 'var(--player-title-visibility, visible)',\n    span: {\n      marginRight: 'var(--spacing, 0.5em)',\n    },\n    '> div': textEllipsis,\n  },\n  h1: {\n    margin: 0,\n    fontSize: '1.66em',\n    fontWeight: '500',\n    ...textEllipsis,\n    flex: 1,\n    '&:not(:only-child)': {\n      marginTop: '0.5em',\n    },\n  },\n  img: {\n    display: 'block',\n    width: '2em',\n    height: '2em',\n    borderRadius: '0.25em',\n    objectFit: 'cover',\n  },\n}\n\nconst backStyle = {\n  position: 'absolute',\n  zIndex: 0,\n  width: '100%',\n  height: '100%',\n  // move DAI UI slightly up to avoid stacking\n  paddingBottom: 'calc(var(--spacing) * 4)',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  '> iframe': {pointerEvents: 'auto'},\n}\n\nconst skipStyle = {\n  position: 'absolute',\n  right: 0,\n  bottom: '9rem',\n  textAlign: 'right',\n  button: {\n    width: 'auto',\n    height: 'auto',\n  },\n}\n\nconst desktopStyle = {\n  '--spacing': '0.75em',\n  '--center-pointer-events': 'none',\n  '> div button': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em)',\n  },\n  '--thumbnail-width': '288', // height 162\n}\n\nconst desktopControls = {\n  '> div:first-of-type': {\n    flex: '100%',\n    marginBottom: '0.4em',\n  },\n  '> button': {\n    '&[disabled]': {\n    display: 'none',\n    },\n    '&.play-button': {\n      fontSize: '1em',\n    },\n    '&.firstplay-button': {\n      fontSize: '400%'\n    },\n  },\n}\n\nconst seekbarStyles = {\n  flex: '1 var(--seekbar-flex-basis, 100%)',\n  '&:not(:empty)': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n  },\n}\n\nconst adContainerStyle = {\n  flexGrow: 1,\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  pointerEvents: 'var(--center-pointer-events)',\n  button: {pointerEvents: 'auto'},\n  zIndex: 0,\n}\n\nconst ControlsBlock = ({\n  order = 'mobile',\n  liveButton,\n  playButton,\n  rewindButton = '',\n  forwardButton = '',\n  previousEpisodeButton = '',\n  nextEpisodeButton = '',\n}) =>\n  order === 'desktop' ? (\n    <>\n      {liveButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {rewindButton}\n      {forwardButton}\n    </>\n  ) : (\n    <>\n      {rewindButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {forwardButton}\n    </>\n  )\n\nconst DefaultLayout = ({\n  type = 'mobile',\n  style,\n  display,\n  liveButton,\n  controlsDisplay = display,\n  size,\n  title = '',\n  channelTitle = '',\n  video,\n  haveBottomItem,\n  seekbar = '',\n  displayTime,\n  controlButtons,\n  volumeControl,\n  fullscreenButton,\n  backButton = '',\n  adStatus = '',\n  adLink = '',\n  adSkipButton,\n  backItems,\n  children,\n  containerRef,\n  backRef,\n  adContainerRef,\n  ...rest\n}) => {\n  const slotRef = useRef({})\n\n  return (\n    <div\n      style={{\n        '--player-bottom-ui-top':\n          controlsDisplay === 'shown'\n            ? `-${bottomUiTop(containerRef.current)}px`\n            : '-8px',\n        '--player-bottom-ui-height':\n          type === 'desktop' && controlsDisplay === 'shown' ? '3.5rem' : '0',\n      }}\n      css={[\n        containerStyle,\n        videoContainerStyle,\n        responsiveStyles[size],\n        type === 'desktop' && desktopStyle,\n        adStatus && {'--bottom-spacing': 0},\n        nativeTextStyle,\n        customTextStyle,\n        style,\n      ]}\n      ref={containerRef}\n      {...rest}\n    >\n      {video}\n      <div\n        ref={backRef}\n        css={[\n          backStyle,\n          display !== 'hidden' && (haveBottomItem ? dropTop : drop),\n          invisible('--controller-middle-area'),\n        ]}\n      >\n        {type !== 'mobile' && backItems}\n        {adSkipButton && <div css={skipStyle}>{adSkipButton}</div>}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          infoStyle,\n          displayStyles[display],\n          invisible('--controller-top-area'),\n        ]}\n      >\n        {backButton}\n        <div\n          ref={element => {\n            slotRef.current.titleBarLeft = element\n          }}\n        />\n        <div>\n          <h1>{title}</h1>\n          {channelTitle && <div>{channelTitle}</div>}\n        </div>\n        <div\n          ref={element => {\n            slotRef.current.titleBarRight = element\n          }}\n        />\n        {adLink && <div className=\"pinned\">{adLink}</div>}\n      </div>\n      <div\n        ref={adContainerRef}\n        css={[\n          adContainerStyle,\n          // this container covers Google DAI iframe UI, left some space for user to click \"skip ad\"\n          // see https://kkvideo.atlassian.net/browse/CPT-4535 for detail\n          adStatus && {margin: 'calc(var(--spacing) * 6) 0'},\n        ]}\n      >\n        {type === 'mobile' && (\n          <div\n            css={[\n              controlsStyle(type),\n              displayStyles[controlsDisplay],\n              invisible('--controller-middle-area'),\n            ]}\n          >\n            <ControlsBlock order=\"mobile\" {...controlButtons} />\n          </div>\n        )}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          {\n            marginTop: '1em',\n            paddingTop: 0,\n            paddingBottom: 'calc(1em - var(--bottom-spacing, 0) / 5)',\n            flexWrap: 'wrap',\n            '> div:first-of-type': seekbarStyles,\n          },\n          type === 'desktop' && desktopControls,\n          controlsDisplayStyles[controlsDisplay],\n          invisible('--controller-down-area'),\n        ]}\n      >\n        {seekbar || <div />}\n        {type === 'desktop' && (\n          <ControlsBlock\n            order=\"desktop\"\n            liveButton={liveButton}\n            {...controlButtons}\n          />\n        )}\n        {type === 'desktop' ? (\n          <div css={[displayStyles[controlsDisplay]]}>{displayTime}</div>\n        ) : (\n          <>\n            {liveButton}\n            <div\n              css={{\n                '&:not(:empty)': {\n                  flex: 1,\n                  marginLeft: 'calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n                },\n              }}\n            >\n              {displayTime}\n            </div>\n          </>\n        )}\n        {adStatus && (\n          <div\n            className=\"pinned\"\n            css={[infoBarSlotStyle, type === 'desktop' && desktopInfoBarStyle]}\n          >\n            {adStatus}\n          </div>\n        )}\n        <div\n          css={\n            !adStatus && [\n              infoBarSlotStyle,\n              type === 'desktop' && desktopInfoBarStyle,\n            ]\n          }\n          ref={element => {\n            slotRef.current.infoBar = element\n          }}\n        />\n        {volumeControl}\n        <div\n          css={adStatus && hidden}\n          ref={element => {\n            slotRef.current.functionBar = element\n          }}\n        />\n        {fullscreenButton}\n      </div>\n      <SlotProvider slotRef={slotRef}>{children}</SlotProvider>\n    </div>\n  )\n}\n\nexport default DefaultLayout\n"]} */"],
|
|
3570
|
+
css: [infoBarSlotStyle, type === 'desktop' && desktopInfoBarStyle, process.env.NODE_ENV === "production" ? "" : ";label:DefaultLayout;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["DefaultLayout.js"],"names":[],"mappings":"AAmdY","file":"DefaultLayout.js","sourcesContent":["/* eslint-disable no-param-reassign */\n/* @jsxImportSource @emotion/react */\n/* eslint-disable react/prop-types */\nimport {useRef} from 'react'\nimport {SlotProvider} from './uiExtensions'\nimport {nativeTextStyle, customTextStyle, bottomUiTop} from './subtitles'\n\nconst hidden = {display: 'none'}\nconst invisible = hiddenArea => ({\n  visibility: `var(${hiddenArea}, visible)`,\n})\nconst containerStyle = {\n  '--spacing': '0.5em',\n  '--center-pointer-events': 'auto',\n  width: '100%',\n  height: '100%',\n  fontSize: '16px',\n  boxSizing: 'border-box',\n  display: 'flex',\n  flexDirection: 'column',\n  overflow: 'hidden',\n  color: 'white',\n  background: '#000',\n  // prevent animation glich(afterimage) of descendant elements\n  transform: 'translateX(0)',\n  userSelect: 'none',\n  'a, a:link, a:visited': {\n    color: '#fff',\n    opacity: 0.8,\n    textDecoration: 'none',\n  },\n  button: {\n    fontSize: 'inherit',\n    '> *': {\n      pointerEvents: 'none',\n    },\n  },\n  '--thumbnail-width': '160', // height 90\n  '--theme-color': 'var(--primary-highlight, red)',\n}\n\nconst videoContainerStyle = {\n  '> div:first-of-type': {\n    position: 'absolute',\n    zIndex: '-1',\n    display: 'flex',\n    alignItems: 'center',\n    width: '100%',\n    height: '100%',\n  },\n}\n\nconst drop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0.5) 0,\n    rgba(0,0,0,0) var(--player-ui-drop-shadow-size, 30%) calc(100% - var(--player-ui-drop-shadow-size, 30%)),\n    rgba(0,0,0,0.5) 100%\n  )`,\n  backgroundColor: `rgba(0, 0, 0, 0.3)`,\n}\n\nconst dropTop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0) 0,\n    rgba(0,0,0,0) 8rem calc(100% - 8rem),\n    rgba(0,0,0,0.5) 100%\n  )`,\n}\n\nconst responsiveStyles = {\n  desktop: {\n    fontSize: '24px',\n  }, // add if necessary: big-desktop\n}\n\nconst rowStyle = {\n  boxSizing: 'border-box',\n  width: '100%',\n  padding: '1em',\n  display: 'flex',\n  alignItems: 'center',\n  '> button, > div > button': {\n    padding: '0.5rem',\n  },\n}\n\n/* \n  Hint:\n    displayStyles specificity is higher then controlsStyle\n    Need to pay attention when we revise these styles.\n*/\n\nconst displayHidden = {\n  '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n    opacity: 0,\n    transform: 'translate(-200vw, 0)',\n    transition: 'opacity 0.8s var(--autohide-delay, 0s) ease-out, transform 0s calc(0.8s + var(--autohide-delay, 0s))',\n    'button': {\n      opacity: 0, // extra transition to hide tooltip for Safari\n      transition: 'opacity 0s calc(0.8s + var(--autohide-delay, 0s)) ease-out',\n    }\n  },\n}\n\nconst displayStyles = {\n  hidden: displayHidden,\n  'seekbar-only': displayHidden,\n  shown: {\n    '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n      zIndex: 1,\n      pointerEvents: 'auto',\n      transition: 'opacity 0.3s ease-out, transform 0s 0s',\n    },\n  },\n}\n\nconst controlsDisplayStyles = {\n  hidden: {\n    ...displayStyles.hidden,\n    '~ div:not(.pinned)': displayStyles.hidden,\n  },\n  'seekbar-only': {\n    '> div:first-of-type ~ *': displayStyles.hidden,\n  },\n  shown: {\n    ...displayStyles.shown,\n    '~ div': displayStyles.shown,\n  },\n}\n\n// TODO: transform into sample object instead of function\nconst controlsStyle = type => ({\n  position: 'absolute',\n  zIndex: '2',\n  width: '100%',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'space-evenly',\n  fontSize: '1.5em',\n  '> button': {\n    padding: '0.75em 0.5em',\n    '&:disabled': {\n      opacity: 0,\n      pointerEvents: 'none',\n    },\n    '&.play-button': {\n      padding: '0.75rem 0.5rem',\n      fontSize: 'var(--play-button-size, 1em)',\n    },\n    '&.firstplay-button':\n      type === 'mobile'\n        ? {fontSize: '200%', padding: '4em 10em', marginBottom: '1em'}\n        : {fontSize: '400%'},\n  },\n})\n\n// Remove the height for imporving tapping effect at current, and this height was used for fixed region before.\nconst infoBarSlotStyle = {\n  display: 'flex',\n  order: -1,\n  alignItems: 'center',\n  flex: '1 60%',\n}\n\nconst desktopInfoBarStyle = {\n  marginLeft: 'var(--spacing, 0.5em)',\n  order: 'initial',\n  flex: '1 0%',\n}\n\nconst textEllipsis = {\n  overflow: 'hidden',\n  whiteSpace: 'nowrap',\n  textOverflow: 'ellipsis',\n}\n\nconst infoStyle = {\n  zIndex: 1,\n  overflow: 'visible',\n  pointerEvents: 'none',\n  padding: '0 calc(2em - 20px)',\n  marginBottom: 'calc(0.5em + var(--bottom-spacing, 0))',\n  flex: '3em 0',\n  '> div:first-of-type, > div:nth-of-type(2), > button': {\n    minWidth: 0,\n  },\n  '> div:nth-of-type(2)': {\n    // title and channel title\n    margin: '0 calc(var(--spacing, 0.5em) / 2)',\n    flex: 1,\n    fontSize: '0.75em',\n    visibility: 'var(--player-title-visibility, visible)',\n    span: {\n      marginRight: 'var(--spacing, 0.5em)',\n    },\n    '> div': textEllipsis,\n  },\n  h1: {\n    margin: 0,\n    fontSize: '1.66em',\n    fontWeight: '500',\n    ...textEllipsis,\n    flex: 1,\n    '&:not(:only-child)': {\n      marginTop: '0.5em',\n    },\n  },\n  img: {\n    display: 'block',\n    width: '2em',\n    height: '2em',\n    borderRadius: '0.25em',\n    objectFit: 'cover',\n  },\n}\n\nconst backStyle = {\n  position: 'absolute',\n  zIndex: 0,\n  width: '100%',\n  height: '100%',\n  // move DAI UI slightly up to avoid stacking\n  paddingBottom: 'calc(var(--spacing) * 4)',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  '> iframe': {pointerEvents: 'auto'},\n}\n\nconst skipStyle = {\n  position: 'absolute',\n  right: 0,\n  bottom: '9rem',\n  textAlign: 'right',\n  button: {\n    width: 'auto',\n    height: 'auto',\n  },\n}\n\nconst desktopStyle = {\n  '--spacing': '0.75em',\n  '--center-pointer-events': 'none',\n  '> div button': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em)',\n  },\n  '--thumbnail-width': '288', // height 162\n}\n\nconst desktopControls = {\n  '> div:first-of-type': {\n    flex: '100%',\n    marginBottom: '0.4em',\n  },\n  '> button': {\n    '&[disabled]': {\n    display: 'none',\n    },\n    '&.play-button': {\n      fontSize: '1em',\n    },\n    '&.firstplay-button': {\n      fontSize: '400%'\n    },\n  },\n}\n\nconst seekbarStyles = {\n  flex: '1 var(--seekbar-flex-basis, 100%)',\n  '&:not(:empty)': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n  },\n}\n\nconst adContainerStyle = {\n  flexGrow: 1,\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  pointerEvents: 'var(--center-pointer-events)',\n  button: {pointerEvents: 'auto'},\n  zIndex: 0,\n}\n\nconst ControlsBlock = ({\n  order = 'mobile',\n  liveButton,\n  playButton,\n  rewindButton = '',\n  forwardButton = '',\n  previousEpisodeButton = '',\n  nextEpisodeButton = '',\n}) =>\n  order === 'desktop' ? (\n    <>\n      {liveButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {rewindButton}\n      {forwardButton}\n    </>\n  ) : (\n    <>\n      {rewindButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {forwardButton}\n    </>\n  )\n\nconst DefaultLayout = ({\n  type = 'mobile',\n  style,\n  display,\n  liveButton,\n  controlsDisplay = display,\n  size,\n  title = '',\n  channelTitle = '',\n  video,\n  haveBottomItem,\n  seekbar = '',\n  displayTime,\n  controlButtons,\n  volumeControl,\n  fullscreenButton,\n  backButton = '',\n  adStatus = '',\n  adLink = '',\n  adSkipButton,\n  backItems,\n  children,\n  containerRef,\n  backRef,\n  adContainerRef,\n  ...rest\n}) => {\n  const slotRef = useRef({})\n\n  return (\n    <div\n      style={{\n        '--player-bottom-ui-top':\n          controlsDisplay === 'shown'\n            ? `-${bottomUiTop(containerRef.current)}px`\n            : '-8px',\n        '--player-bottom-ui-height':\n          type === 'desktop' && controlsDisplay === 'shown' ? '3.5rem' : '0',\n      }}\n      css={[\n        containerStyle,\n        videoContainerStyle,\n        responsiveStyles[size],\n        type === 'desktop' && desktopStyle,\n        adStatus && {'--bottom-spacing': 0},\n        nativeTextStyle,\n        customTextStyle,\n        style,\n      ]}\n      ref={containerRef}\n      {...rest}\n    >\n      {video}\n      <div\n        ref={backRef}\n        css={[\n          backStyle,\n          display !== 'hidden' && (haveBottomItem ? dropTop : drop),\n          invisible('--controller-middle-area'),\n        ]}\n      >\n        {type !== 'mobile' && backItems}\n        {adSkipButton && <div css={skipStyle}>{adSkipButton}</div>}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          infoStyle,\n          displayStyles[display],\n          invisible('--controller-top-area'),\n        ]}\n      >\n        {backButton}\n        <div\n          ref={element => {\n            slotRef.current.titleBarLeft = element\n          }}\n        />\n        <div>\n          <h1>{title}</h1>\n          {channelTitle && <div>{channelTitle}</div>}\n        </div>\n        <div\n          ref={element => {\n            slotRef.current.titleBarRight = element\n          }}\n        />\n        {adLink && <div className=\"pinned\">{adLink}</div>}\n      </div>\n      <div\n        ref={adContainerRef}\n        css={[\n          adContainerStyle,\n          // this container covers Google DAI iframe UI, left some space for user to click \"skip ad\"\n          // see https://kkvideo.atlassian.net/browse/CPT-4535 for detail\n          adStatus && {margin: 'calc(var(--spacing) * 6) 0'},\n        ]}\n      >\n        {type === 'mobile' && (\n          <div\n            css={[\n              controlsStyle(type),\n              displayStyles[controlsDisplay],\n              invisible('--controller-middle-area'),\n            ]}\n          >\n            <ControlsBlock order=\"mobile\" {...controlButtons} />\n          </div>\n        )}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          {\n            marginTop: '1em',\n            paddingTop: 0,\n            paddingBottom: 'calc(1em - var(--bottom-spacing, 0) / 5)',\n            flexWrap: 'wrap',\n            '> div:first-of-type': seekbarStyles,\n          },\n          type === 'desktop' && desktopControls,\n          controlsDisplayStyles[controlsDisplay],\n          invisible('--controller-down-area'),\n        ]}\n      >\n        {seekbar || <div />}\n        {type === 'desktop' && (\n          <ControlsBlock\n            order=\"desktop\"\n            liveButton={liveButton}\n            {...controlButtons}\n          />\n        )}\n        {type === 'desktop' ? (\n          <div css={[displayStyles[controlsDisplay]]}>{displayTime}</div>\n        ) : (\n          <>\n            {liveButton}\n            <div\n              css={{\n                '&:not(:empty)': {\n                  flex: 1,\n                  marginLeft: 'calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n                },\n              }}\n            >\n              {displayTime}\n            </div>\n          </>\n        )}\n        {adStatus && (\n          <div\n            className=\"pinned\"\n            css={[infoBarSlotStyle, type === 'desktop' && desktopInfoBarStyle]}\n          >\n            {adStatus}\n          </div>\n        )}\n        <div\n          css={\n            !adStatus && [\n              infoBarSlotStyle,\n              type === 'desktop' && desktopInfoBarStyle,\n            ]\n          }\n          ref={element => {\n            slotRef.current.infoBar = element\n          }}\n        />\n        {volumeControl}\n        <div\n          css={[adStatus ? hidden : {display: 'flex'}]}\n          ref={element => {\n            slotRef.current.functionBar = element\n          }}\n        />\n        {fullscreenButton}\n      </div>\n      <SlotProvider slotRef={slotRef}>{children}</SlotProvider>\n    </div>\n  )\n}\n\nexport default DefaultLayout\n"]} */"],
|
|
3627
3571
|
children: adStatus
|
|
3628
3572
|
}), jsx$1("div", {
|
|
3629
3573
|
css: !adStatus && [infoBarSlotStyle, type === 'desktop' && desktopInfoBarStyle],
|
|
@@ -3631,7 +3575,9 @@ const DefaultLayout = ({
|
|
|
3631
3575
|
slotRef.current.infoBar = element;
|
|
3632
3576
|
}
|
|
3633
3577
|
}), volumeControl, jsx$1("div", {
|
|
3634
|
-
css: adStatus
|
|
3578
|
+
css: [adStatus ? hidden : {
|
|
3579
|
+
display: 'flex'
|
|
3580
|
+
}, process.env.NODE_ENV === "production" ? "" : ";label:DefaultLayout;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["DefaultLayout.js"],"names":[],"mappings":"AAqeU","file":"DefaultLayout.js","sourcesContent":["/* eslint-disable no-param-reassign */\n/* @jsxImportSource @emotion/react */\n/* eslint-disable react/prop-types */\nimport {useRef} from 'react'\nimport {SlotProvider} from './uiExtensions'\nimport {nativeTextStyle, customTextStyle, bottomUiTop} from './subtitles'\n\nconst hidden = {display: 'none'}\nconst invisible = hiddenArea => ({\n  visibility: `var(${hiddenArea}, visible)`,\n})\nconst containerStyle = {\n  '--spacing': '0.5em',\n  '--center-pointer-events': 'auto',\n  width: '100%',\n  height: '100%',\n  fontSize: '16px',\n  boxSizing: 'border-box',\n  display: 'flex',\n  flexDirection: 'column',\n  overflow: 'hidden',\n  color: 'white',\n  background: '#000',\n  // prevent animation glich(afterimage) of descendant elements\n  transform: 'translateX(0)',\n  userSelect: 'none',\n  'a, a:link, a:visited': {\n    color: '#fff',\n    opacity: 0.8,\n    textDecoration: 'none',\n  },\n  button: {\n    fontSize: 'inherit',\n    '> *': {\n      pointerEvents: 'none',\n    },\n  },\n  '--thumbnail-width': '160', // height 90\n  '--theme-color': 'var(--primary-highlight, red)',\n}\n\nconst videoContainerStyle = {\n  '> div:first-of-type': {\n    position: 'absolute',\n    zIndex: '-1',\n    display: 'flex',\n    alignItems: 'center',\n    width: '100%',\n    height: '100%',\n  },\n}\n\nconst drop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0.5) 0,\n    rgba(0,0,0,0) var(--player-ui-drop-shadow-size, 30%) calc(100% - var(--player-ui-drop-shadow-size, 30%)),\n    rgba(0,0,0,0.5) 100%\n  )`,\n  backgroundColor: `rgba(0, 0, 0, 0.3)`,\n}\n\nconst dropTop = {\n  backgroundImage: `linear-gradient(\n    0deg,\n    rgba(0,0,0,0) 0,\n    rgba(0,0,0,0) 8rem calc(100% - 8rem),\n    rgba(0,0,0,0.5) 100%\n  )`,\n}\n\nconst responsiveStyles = {\n  desktop: {\n    fontSize: '24px',\n  }, // add if necessary: big-desktop\n}\n\nconst rowStyle = {\n  boxSizing: 'border-box',\n  width: '100%',\n  padding: '1em',\n  display: 'flex',\n  alignItems: 'center',\n  '> button, > div > button': {\n    padding: '0.5rem',\n  },\n}\n\n/* \n  Hint:\n    displayStyles specificity is higher then controlsStyle\n    Need to pay attention when we revise these styles.\n*/\n\nconst displayHidden = {\n  '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n    opacity: 0,\n    transform: 'translate(-200vw, 0)',\n    transition: 'opacity 0.8s var(--autohide-delay, 0s) ease-out, transform 0s calc(0.8s + var(--autohide-delay, 0s))',\n    'button': {\n      opacity: 0, // extra transition to hide tooltip for Safari\n      transition: 'opacity 0s calc(0.8s + var(--autohide-delay, 0s)) ease-out',\n    }\n  },\n}\n\nconst displayStyles = {\n  hidden: displayHidden,\n  'seekbar-only': displayHidden,\n  shown: {\n    '> div:not(.pinned), > button:not(.pinned), > img:not(.pinned)': {\n      zIndex: 1,\n      pointerEvents: 'auto',\n      transition: 'opacity 0.3s ease-out, transform 0s 0s',\n    },\n  },\n}\n\nconst controlsDisplayStyles = {\n  hidden: {\n    ...displayStyles.hidden,\n    '~ div:not(.pinned)': displayStyles.hidden,\n  },\n  'seekbar-only': {\n    '> div:first-of-type ~ *': displayStyles.hidden,\n  },\n  shown: {\n    ...displayStyles.shown,\n    '~ div': displayStyles.shown,\n  },\n}\n\n// TODO: transform into sample object instead of function\nconst controlsStyle = type => ({\n  position: 'absolute',\n  zIndex: '2',\n  width: '100%',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'space-evenly',\n  fontSize: '1.5em',\n  '> button': {\n    padding: '0.75em 0.5em',\n    '&:disabled': {\n      opacity: 0,\n      pointerEvents: 'none',\n    },\n    '&.play-button': {\n      padding: '0.75rem 0.5rem',\n      fontSize: 'var(--play-button-size, 1em)',\n    },\n    '&.firstplay-button':\n      type === 'mobile'\n        ? {fontSize: '200%', padding: '4em 10em', marginBottom: '1em'}\n        : {fontSize: '400%'},\n  },\n})\n\n// Remove the height for imporving tapping effect at current, and this height was used for fixed region before.\nconst infoBarSlotStyle = {\n  display: 'flex',\n  order: -1,\n  alignItems: 'center',\n  flex: '1 60%',\n}\n\nconst desktopInfoBarStyle = {\n  marginLeft: 'var(--spacing, 0.5em)',\n  order: 'initial',\n  flex: '1 0%',\n}\n\nconst textEllipsis = {\n  overflow: 'hidden',\n  whiteSpace: 'nowrap',\n  textOverflow: 'ellipsis',\n}\n\nconst infoStyle = {\n  zIndex: 1,\n  overflow: 'visible',\n  pointerEvents: 'none',\n  padding: '0 calc(2em - 20px)',\n  marginBottom: 'calc(0.5em + var(--bottom-spacing, 0))',\n  flex: '3em 0',\n  '> div:first-of-type, > div:nth-of-type(2), > button': {\n    minWidth: 0,\n  },\n  '> div:nth-of-type(2)': {\n    // title and channel title\n    margin: '0 calc(var(--spacing, 0.5em) / 2)',\n    flex: 1,\n    fontSize: '0.75em',\n    visibility: 'var(--player-title-visibility, visible)',\n    span: {\n      marginRight: 'var(--spacing, 0.5em)',\n    },\n    '> div': textEllipsis,\n  },\n  h1: {\n    margin: 0,\n    fontSize: '1.66em',\n    fontWeight: '500',\n    ...textEllipsis,\n    flex: 1,\n    '&:not(:only-child)': {\n      marginTop: '0.5em',\n    },\n  },\n  img: {\n    display: 'block',\n    width: '2em',\n    height: '2em',\n    borderRadius: '0.25em',\n    objectFit: 'cover',\n  },\n}\n\nconst backStyle = {\n  position: 'absolute',\n  zIndex: 0,\n  width: '100%',\n  height: '100%',\n  // move DAI UI slightly up to avoid stacking\n  paddingBottom: 'calc(var(--spacing) * 4)',\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  '> iframe': {pointerEvents: 'auto'},\n}\n\nconst skipStyle = {\n  position: 'absolute',\n  right: 0,\n  bottom: '9rem',\n  textAlign: 'right',\n  button: {\n    width: 'auto',\n    height: 'auto',\n  },\n}\n\nconst desktopStyle = {\n  '--spacing': '0.75em',\n  '--center-pointer-events': 'none',\n  '> div button': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em)',\n  },\n  '--thumbnail-width': '288', // height 162\n}\n\nconst desktopControls = {\n  '> div:first-of-type': {\n    flex: '100%',\n    marginBottom: '0.4em',\n  },\n  '> button': {\n    '&[disabled]': {\n    display: 'none',\n    },\n    '&.play-button': {\n      fontSize: '1em',\n    },\n    '&.firstplay-button': {\n      fontSize: '400%'\n    },\n  },\n}\n\nconst seekbarStyles = {\n  flex: '1 var(--seekbar-flex-basis, 100%)',\n  '&:not(:empty)': {\n    margin: '0 calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n  },\n}\n\nconst adContainerStyle = {\n  flexGrow: 1,\n  display: 'flex',\n  alignItems: 'center',\n  justifyContent: 'center',\n  pointerEvents: 'var(--center-pointer-events)',\n  button: {pointerEvents: 'auto'},\n  zIndex: 0,\n}\n\nconst ControlsBlock = ({\n  order = 'mobile',\n  liveButton,\n  playButton,\n  rewindButton = '',\n  forwardButton = '',\n  previousEpisodeButton = '',\n  nextEpisodeButton = '',\n}) =>\n  order === 'desktop' ? (\n    <>\n      {liveButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {rewindButton}\n      {forwardButton}\n    </>\n  ) : (\n    <>\n      {rewindButton}\n      {previousEpisodeButton}\n      {playButton}\n      {nextEpisodeButton}\n      {forwardButton}\n    </>\n  )\n\nconst DefaultLayout = ({\n  type = 'mobile',\n  style,\n  display,\n  liveButton,\n  controlsDisplay = display,\n  size,\n  title = '',\n  channelTitle = '',\n  video,\n  haveBottomItem,\n  seekbar = '',\n  displayTime,\n  controlButtons,\n  volumeControl,\n  fullscreenButton,\n  backButton = '',\n  adStatus = '',\n  adLink = '',\n  adSkipButton,\n  backItems,\n  children,\n  containerRef,\n  backRef,\n  adContainerRef,\n  ...rest\n}) => {\n  const slotRef = useRef({})\n\n  return (\n    <div\n      style={{\n        '--player-bottom-ui-top':\n          controlsDisplay === 'shown'\n            ? `-${bottomUiTop(containerRef.current)}px`\n            : '-8px',\n        '--player-bottom-ui-height':\n          type === 'desktop' && controlsDisplay === 'shown' ? '3.5rem' : '0',\n      }}\n      css={[\n        containerStyle,\n        videoContainerStyle,\n        responsiveStyles[size],\n        type === 'desktop' && desktopStyle,\n        adStatus && {'--bottom-spacing': 0},\n        nativeTextStyle,\n        customTextStyle,\n        style,\n      ]}\n      ref={containerRef}\n      {...rest}\n    >\n      {video}\n      <div\n        ref={backRef}\n        css={[\n          backStyle,\n          display !== 'hidden' && (haveBottomItem ? dropTop : drop),\n          invisible('--controller-middle-area'),\n        ]}\n      >\n        {type !== 'mobile' && backItems}\n        {adSkipButton && <div css={skipStyle}>{adSkipButton}</div>}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          infoStyle,\n          displayStyles[display],\n          invisible('--controller-top-area'),\n        ]}\n      >\n        {backButton}\n        <div\n          ref={element => {\n            slotRef.current.titleBarLeft = element\n          }}\n        />\n        <div>\n          <h1>{title}</h1>\n          {channelTitle && <div>{channelTitle}</div>}\n        </div>\n        <div\n          ref={element => {\n            slotRef.current.titleBarRight = element\n          }}\n        />\n        {adLink && <div className=\"pinned\">{adLink}</div>}\n      </div>\n      <div\n        ref={adContainerRef}\n        css={[\n          adContainerStyle,\n          // this container covers Google DAI iframe UI, left some space for user to click \"skip ad\"\n          // see https://kkvideo.atlassian.net/browse/CPT-4535 for detail\n          adStatus && {margin: 'calc(var(--spacing) * 6) 0'},\n        ]}\n      >\n        {type === 'mobile' && (\n          <div\n            css={[\n              controlsStyle(type),\n              displayStyles[controlsDisplay],\n              invisible('--controller-middle-area'),\n            ]}\n          >\n            <ControlsBlock order=\"mobile\" {...controlButtons} />\n          </div>\n        )}\n      </div>\n      <div\n        css={[\n          rowStyle,\n          {\n            marginTop: '1em',\n            paddingTop: 0,\n            paddingBottom: 'calc(1em - var(--bottom-spacing, 0) / 5)',\n            flexWrap: 'wrap',\n            '> div:first-of-type': seekbarStyles,\n          },\n          type === 'desktop' && desktopControls,\n          controlsDisplayStyles[controlsDisplay],\n          invisible('--controller-down-area'),\n        ]}\n      >\n        {seekbar || <div />}\n        {type === 'desktop' && (\n          <ControlsBlock\n            order=\"desktop\"\n            liveButton={liveButton}\n            {...controlButtons}\n          />\n        )}\n        {type === 'desktop' ? (\n          <div css={[displayStyles[controlsDisplay]]}>{displayTime}</div>\n        ) : (\n          <>\n            {liveButton}\n            <div\n              css={{\n                '&:not(:empty)': {\n                  flex: 1,\n                  marginLeft: 'calc(var(--spacing, 0.75em) - 0.5em + 0.5rem)',\n                },\n              }}\n            >\n              {displayTime}\n            </div>\n          </>\n        )}\n        {adStatus && (\n          <div\n            className=\"pinned\"\n            css={[infoBarSlotStyle, type === 'desktop' && desktopInfoBarStyle]}\n          >\n            {adStatus}\n          </div>\n        )}\n        <div\n          css={\n            !adStatus && [\n              infoBarSlotStyle,\n              type === 'desktop' && desktopInfoBarStyle,\n            ]\n          }\n          ref={element => {\n            slotRef.current.infoBar = element\n          }}\n        />\n        {volumeControl}\n        <div\n          css={[adStatus ? hidden : {display: 'flex'}]}\n          ref={element => {\n            slotRef.current.functionBar = element\n          }}\n        />\n        {fullscreenButton}\n      </div>\n      <SlotProvider slotRef={slotRef}>{children}</SlotProvider>\n    </div>\n  )\n}\n\nexport default DefaultLayout\n"]} */"],
|
|
3635
3581
|
ref: element => {
|
|
3636
3582
|
slotRef.current.functionBar = element;
|
|
3637
3583
|
}
|
|
@@ -4176,8 +4122,7 @@ const sliderStyle = {
|
|
|
4176
4122
|
},
|
|
4177
4123
|
'> div:last-of-type': {
|
|
4178
4124
|
width: '1.2em',
|
|
4179
|
-
height: '1.2em'
|
|
4180
|
-
background: 'var(--theme-color, red)'
|
|
4125
|
+
height: '1.2em'
|
|
4181
4126
|
},
|
|
4182
4127
|
'&:hover > div:last-of-type': {
|
|
4183
4128
|
opacity: 1
|
|
@@ -4256,7 +4201,7 @@ const SeekbarTrack = ({
|
|
|
4256
4201
|
current,
|
|
4257
4202
|
buffered
|
|
4258
4203
|
}) => jsxs("div", {
|
|
4259
|
-
css: [style, segmentedTrackStyle, process.env.NODE_ENV === "production" ? "" : ";label:SeekbarTrack;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
4204
|
+
css: [style, segmentedTrackStyle, process.env.NODE_ENV === "production" ? "" : ";label:SeekbarTrack;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIlNlZWtiYXIuanMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBb0ZNIiwiZmlsZSI6IlNlZWtiYXIuanMiLCJzb3VyY2VzQ29udGVudCI6WyIvKiBAanN4SW1wb3J0U291cmNlIEBlbW90aW9uL3JlYWN0ICovXG5pbXBvcnQge3VzZVJlZHVjZXIsIHVzZVJlZiwgY2xvbmVFbGVtZW50fSBmcm9tICdyZWFjdCdcbmltcG9ydCB1c2VEaW1lbnNpb25zIGZyb20gJ3JlYWN0LWNvb2wtZGltZW5zaW9ucydcbmltcG9ydCBTaW1wbGVTbGlkZXIgZnJvbSAncGxheWVyVWkvU2ltcGxlU2xpZGVyJ1xuaW1wb3J0IHtGb3JtYXR0ZWRUaW1lfSBmcm9tICdjb250ZXh0L0kxOG4nXG5pbXBvcnQge2dldFRpbWVsaW5lU2VnbWVudHN9IGZyb20gJ3ByZW1pdW0vdGltZWxpbmUnXG5cbmNvbnN0IHNlZWtiYXJTdHlsZSA9IHtcbiAgcG9zaXRpb246ICdyZWxhdGl2ZScsXG4gIGRpc3BsYXk6ICdmbGV4JyxcbiAgYWxpZ25JdGVtczogJ2NlbnRlcicsXG4gIG1pbldpZHRoOiAnMCcsXG4gIGhlaWdodDogJzI0cHgnLFxuICBmb250U2l6ZTogJzc1JScsXG4gIGxldHRlclNwYWNpbmc6ICcxcHgnLFxuICBjb2xvcjogJyNmZmYnLFxufVxuXG5jb25zdCBzbGlkZXJTdHlsZSA9IHtcbiAgbWluV2lkdGg6ICcwJyxcbiAgZmxleDogMSxcbiAgJ0BtZWRpYSAoaG92ZXI6IGhvdmVyKSwgc2NyZWVuIGFuZCAoLW1zLWhpZ2gtY29udHJhc3Q6IGFjdGl2ZSksICgtbXMtaGlnaC1jb250cmFzdDogbm9uZSknOlxuICAgIHtcbiAgICAgICc+IGRpdjpsYXN0LW9mLXR5cGUnOiB7XG4gICAgICAgIG9wYWNpdHk6IDAsXG4gICAgICAgIHRyYW5zaXRpb246ICdvcGFjaXR5IDAuMnMgZWFzZS1vdXQnLFxuICAgICAgfSxcbiAgICB9LFxuICAnPiBkaXY6bGFzdC1vZi10eXBlJzoge1xuICAgIHdpZHRoOiAnMS4yZW0nLFxuICAgIGhlaWdodDogJzEuMmVtJyxcbiAgfSxcbiAgJyY6aG92ZXIgPiBkaXY6bGFzdC1vZi10eXBlJzoge1xuICAgIG9wYWNpdHk6IDEsXG4gIH0sXG59XG5cbmNvbnN0IHJlZHVjZVBvaW50ZXIgPSAoc3RhdGUsIHt0eXBlLCB2YWx1ZSwgeH0pID0+IHtcbiAgc3dpdGNoICh0eXBlKSB7XG4gICAgY2FzZSAnbW92ZSc6XG4gICAgICByZXR1cm4gey4uLnN0YXRlLCBob3ZlcjogdHJ1ZSwgdmFsdWUsIHh9XG4gICAgY2FzZSAnY2hhbmdlJzpcbiAgICAgIHJldHVybiB7Li4uc3RhdGUsIGZvY3VzZWQ6IHRydWUsIHZhbHVlfVxuICAgIGNhc2UgJ3JlbGVhc2UnOlxuICAgICAgcmV0dXJuIHsuLi5zdGF0ZSwgZm9jdXNlZDogZmFsc2UsIHZhbHVlfVxuICAgIGNhc2UgJ2xlYXZlJzpcbiAgICAgIHJldHVybiB7Li4uc3RhdGUsIGhvdmVyOiBmYWxzZX1cbiAgICBkZWZhdWx0OlxuICAgICAgcmV0dXJuIHN0YXRlXG4gIH1cbn1cblxuY29uc3QgU2Vla2JhclJhaWwgPSAoKSA9PiAnJ1xuXG5jb25zdCBzZWdtZW50ZWRUcmFja1N0eWxlID0ge1xuICBib3hTaXppbmc6ICdib3JkZXItYm94JyxcbiAgcG9zaXRpb246ICdyZWxhdGl2ZScsXG4gIG1hcmdpbjogJzAgMnB4JyxcbiAgaGVpZ2h0OiAnMjRweCcsXG4gIGJvcmRlclRvcDogJ2NhbGMoMTJweCAtIDAuM2VtKSBzb2xpZCB0cmFuc3BhcmVudCcsXG4gIGJvcmRlckJvdHRvbTogJ2NhbGMoMTJweCAtIDAuM2VtKSBzb2xpZCB0cmFuc3BhcmVudCcsXG4gIGJhY2tncm91bmQ6ICd0cmFuc3BhcmVudCcsXG4gICcmOmhvdmVyJzoge1xuICAgIHRyYW5zZm9ybTogJ3NjYWxlKDEsIDEuNSknLFxuICB9LFxuICAnPiBkaXYnOiB7XG4gICAgaGVpZ2h0OiAnMC42ZW0nLFxuICAgIGJhY2tncm91bmRDb2xvcjogJ3JnYmEoMjU1LCAyNTUsIDI1NSwgMC4yKScsXG4gICAgJyY6bm90KDpmaXJzdC1vZi10eXBlKSc6IHtcbiAgICAgIHBvc2l0aW9uOiAnYWJzb2x1dGUnLFxuICAgICAgdG9wOiAnMCcsXG4gICAgfSxcbiAgfSxcbiAgJz4gZGl2Om50aC1vZi10eXBlKDIpJzoge1xuICAgIGJhY2tncm91bmRDb2xvcjogJ3JnYmEoMjU1LCAyNTUsIDI1NSwgMC4zKScsXG4gIH0sXG4gICc+IGRpdjpudGgtb2YtdHlwZSgzKSc6IHtcbiAgICBiYWNrZ3JvdW5kQ29sb3I6ICd2YXIoLS10aGVtZS1jb2xvciwgcmVkKScsXG4gIH0sXG59XG5cbmNvbnN0IFNlZWtiYXJUcmFjayA9ICh7c2VnbWVudHMsIHN0eWxlfSkgPT5cbiAgc2VnbWVudHMubWFwKCh7bGVuZ3RoLCBjdXJyZW50LCBidWZmZXJlZH0pID0+IChcbiAgICA8ZGl2XG4gICAgICBjc3M9e1tzdHlsZSwgc2VnbWVudGVkVHJhY2tTdHlsZV19XG4gICAgICBzdHlsZT17e1xuICAgICAgICBmbGV4OiBgMCAke2xlbmd0aCAqIDEwMH0lYCxcbiAgICAgICAgd2lkdGg6IGBjYWxjKCR7bGVuZ3RoICogMTAwfSUgLSA0cHgpYCxcbiAgICAgIH19XG4gICAgPlxuICAgICAgPGRpdiBzdHlsZT17e3dpZHRoOiAnMTAwJSd9fSAvPlxuICAgICAgPGRpdiBzdHlsZT17e3dpZHRoOiBgJHtidWZmZXJlZCAqIDEwMH0lYH19IC8+XG4gICAgICA8ZGl2IHN0eWxlPXt7d2lkdGg6IGAke2N1cnJlbnQgKiAxMDB9JWB9fSAvPlxuICAgIDwvZGl2PlxuICApKVxuXG4vLyBUT0RPIHVzZSBjbGFzc05hbWUgaW5zdGVhZCBvZiBjbGFzc2VzID9cbmNvbnN0IFNlZWtiYXIgPSAoe1xuICBzdHlsZSxcbiAgY2xhc3NlcyxcbiAgc3RhcnRUaW1lID0gMCxcbiAgY3VycmVudFRpbWUsXG4gIGJ1ZmZlclRpbWUsXG4gIGR1cmF0aW9uLFxuICBjaGFwdGVycyA9IFtdLFxuICBtYXJrcyxcbiAgdGltZURpc3BsYXkgPSBmYWxzZSwgLy8gVE9ETyBtb3JlIHNjYWxhYmxlIHdheT9cbiAgb25DaGFuZ2UsXG4gIG9uQ2hhbmdlQ29tbWl0dGVkLFxuICBjaGlsZHJlbixcbiAgb25Qb2ludGVyTW92ZSxcbiAgb25Qb2ludGVyTGVhdmUsXG4gIC4uLnJlc3Rcbn0pID0+IHtcbiAgY29uc3QgW3BvaW50ZXJTdGF0ZSwgZGlzcGF0Y2hQb2ludGVyXSA9IHVzZVJlZHVjZXIocmVkdWNlUG9pbnRlciwge30pXG4gIGNvbnN0IHBvaW50ZXJBY3RpdmUgPSBwb2ludGVyU3RhdGUuaG92ZXIgfHwgcG9pbnRlclN0YXRlLmZvY3VzZWRcbiAgLy8gdG8gcmVmbGVjdCBib3VuZGFyeSB3aGVuIGNvbnRhaW5lciByZXNpemVkXG4gIGNvbnN0IHtvYnNlcnZlfSA9IHVzZURpbWVuc2lvbnMoKVxuICBjb25zdCByZWYgPSB1c2VSZWYoKVxuICBjb25zdCByZWN0ID0gcmVmLmN1cnJlbnQ/LmdldEJvdW5kaW5nQ2xpZW50UmVjdCgpXG4gIGNvbnN0IGhhbmRsZXJzID0gb25DaGFuZ2VDb21taXR0ZWQgJiYge1xuICAgIG9uUG9pbnRlck1vdmU6IChldmVudCwge3ZhbHVlLCB4fSkgPT4ge1xuICAgICAgb25Qb2ludGVyTW92ZT8uKGV2ZW50LCB7dmFsdWUsIHh9KVxuICAgICAgZGlzcGF0Y2hQb2ludGVyKHt0eXBlOiAnbW92ZScsIHZhbHVlLCB4fSlcbiAgICB9LFxuICAgIG9uUG9pbnRlckxlYXZlOiBldmVudCA9PiB7XG4gICAgICBvblBvaW50ZXJMZWF2ZT8uKGV2ZW50KVxuICAgICAgZGlzcGF0Y2hQb2ludGVyKHt0eXBlOiAnbGVhdmUnfSlcbiAgICB9LFxuICAgIG9uQ2hhbmdlOiAoXywge3ZhbHVlfSkgPT4ge1xuICAgICAgb25DaGFuZ2U/Lih2YWx1ZSlcbiAgICAgIGRpc3BhdGNoUG9pbnRlcih7dHlwZTogJ2NoYW5nZScsIHZhbHVlfSlcbiAgICB9LFxuICAgIG9uQ2hhbmdlQ29tbWl0dGVkOiAoXywge3ZhbHVlfSkgPT4ge1xuICAgICAgZGlzcGF0Y2hQb2ludGVyKHt0eXBlOiAncmVsZWFzZScsIHZhbHVlfSlcbiAgICAgIG9uQ2hhbmdlQ29tbWl0dGVkPy4odmFsdWUpXG4gICAgfSxcbiAgfVxuICBjb25zdCBlbmRUaW1lID0gc3RhcnRUaW1lICsgZHVyYXRpb25cbiAgY29uc3Qgc2VnbWVudHMgPSBnZXRUaW1lbGluZVNlZ21lbnRzKGNoYXB0ZXJzLCB7XG4gICAgc3RhcnRUaW1lLFxuICAgIGN1cnJlbnQ6IHBvaW50ZXJTdGF0ZS5mb2N1c2VkID8gcG9pbnRlclN0YXRlLnZhbHVlIDogY3VycmVudFRpbWUsXG4gICAgYnVmZmVyZWQ6IGJ1ZmZlclRpbWUsXG4gICAgZHVyYXRpb24sXG4gIH0pXG5cbiAgcmV0dXJuICEoZHVyYXRpb24gPiAwKSA/IChcbiAgICA8ZGl2IC8+XG4gICkgOiAoXG4gICAgPGRpdlxuICAgICAgcmVmPXtlbGVtZW50ID0+IHtcbiAgICAgICAgb2JzZXJ2ZShlbGVtZW50KVxuICAgICAgICByZWYuY3VycmVudCA9IGVsZW1lbnRcbiAgICAgIH19XG4gICAgICBjc3M9e1tzZWVrYmFyU3R5bGUsIHN0eWxlXX1cbiAgICAgIHN0eWxlPXtcbiAgICAgICAgcmVjdCAmJiB7XG4gICAgICAgICAgJy0tc2Vla2Jhci1sZWZ0JzogYCR7cmVjdC5sZWZ0fXB4YCxcbiAgICAgICAgICAnLS1zZWVrYmFyLXJpZ2h0JzogYCR7cmVjdC5yaWdodH1weGAsXG4gICAgICAgICAgJy0tcG9pbnRlci14JzogYCR7cG9pbnRlclN0YXRlLnh9cHhgLFxuICAgICAgICB9XG4gICAgICB9XG4gICAgPlxuICAgICAgeyF0aW1lRGlzcGxheSA/IChcbiAgICAgICAgJydcbiAgICAgICkgOiBwb2ludGVyQWN0aXZlID8gKFxuICAgICAgICA8Rm9ybWF0dGVkVGltZSB0aW1lPXtwb2ludGVyU3RhdGUudmFsdWV9IC8+XG4gICAgICApIDogKFxuICAgICAgICA8Rm9ybWF0dGVkVGltZSB0aW1lPXtNYXRoLm1pbihNYXRoLm1heCgwLCBjdXJyZW50VGltZSksIGR1cmF0aW9uKX0gLz5cbiAgICAgICl9XG4gICAgICA8U2ltcGxlU2xpZGVyXG4gICAgICAgIGNzcz17W3NsaWRlclN0eWxlLCB0aW1lRGlzcGxheSAmJiB7bWFyZ2luOiAnMCAxZW0nfV19XG4gICAgICAgIGNsYXNzZXM9e2NsYXNzZXN9XG4gICAgICAgIGRpc2FibGVkPXshb25DaGFuZ2VDb21taXR0ZWR9XG4gICAgICAgIC8vIGRpc3BsYXkgZmlsbGVkIHdoZW4gc2VlayBoYW5kbGVyIGlzIG5vdCBwcm92aWRlZCwgZnJvbSBQbGF5Qm95IGJlaGF2aW9yXG4gICAgICAgIHZhbHVlPXtvbkNoYW5nZUNvbW1pdHRlZCA/IGN1cnJlbnRUaW1lIDogZW5kVGltZX1cbiAgICAgICAgbWluPXtzdGFydFRpbWV9XG4gICAgICAgIG1heD17ZW5kVGltZX1cbiAgICAgICAgbWFya3M9e21hcmtzfVxuICAgICAgICB7Li4uaGFuZGxlcnN9XG4gICAgICAgIHsuLi5yZXN0fVxuICAgICAgICBzbG90cz17e1JhaWw6IFNlZWtiYXJSYWlsLCBUcmFjazogU2Vla2JhclRyYWNrfX1cbiAgICAgICAgc2xvdFByb3BzPXt7dHJhY2s6IHtzZWdtZW50c319fVxuICAgICAgLz5cbiAgICAgIHt0aW1lRGlzcGxheSAmJiA8Rm9ybWF0dGVkVGltZSB0aW1lPXtkdXJhdGlvbn0gLz59XG4gICAgICB7Y2hpbGRyZW4gJiZcbiAgICAgICAgW11cbiAgICAgICAgICAuY29uY2F0KGNoaWxkcmVuKVxuICAgICAgICAgIC5tYXAoY2hpbGQgPT5cbiAgICAgICAgICAgIGNsb25lRWxlbWVudChjaGlsZCwge3RpbWU6IHBvaW50ZXJBY3RpdmUgJiYgcG9pbnRlclN0YXRlLnZhbHVlfSlcbiAgICAgICAgICApfVxuICAgIDwvZGl2PlxuICApXG59XG5cbmV4cG9ydCBkZWZhdWx0IFNlZWtiYXJcbiJdfQ== */"],
|
|
4260
4205
|
style: {
|
|
4261
4206
|
flex: `0 ${length * 100}%`,
|
|
4262
4207
|
width: `calc(${length * 100}% - 4px)`
|
|
@@ -4357,7 +4302,7 @@ const Seekbar = ({
|
|
|
4357
4302
|
observe(element);
|
|
4358
4303
|
ref.current = element;
|
|
4359
4304
|
},
|
|
4360
|
-
css: [seekbarStyle, style, process.env.NODE_ENV === "production" ? "" : ";label:Seekbar;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
4305
|
+
css: [seekbarStyle, style, process.env.NODE_ENV === "production" ? "" : ";label:Seekbar;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIlNlZWtiYXIuanMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBMEpNIiwiZmlsZSI6IlNlZWtiYXIuanMiLCJzb3VyY2VzQ29udGVudCI6WyIvKiBAanN4SW1wb3J0U291cmNlIEBlbW90aW9uL3JlYWN0ICovXG5pbXBvcnQge3VzZVJlZHVjZXIsIHVzZVJlZiwgY2xvbmVFbGVtZW50fSBmcm9tICdyZWFjdCdcbmltcG9ydCB1c2VEaW1lbnNpb25zIGZyb20gJ3JlYWN0LWNvb2wtZGltZW5zaW9ucydcbmltcG9ydCBTaW1wbGVTbGlkZXIgZnJvbSAncGxheWVyVWkvU2ltcGxlU2xpZGVyJ1xuaW1wb3J0IHtGb3JtYXR0ZWRUaW1lfSBmcm9tICdjb250ZXh0L0kxOG4nXG5pbXBvcnQge2dldFRpbWVsaW5lU2VnbWVudHN9IGZyb20gJ3ByZW1pdW0vdGltZWxpbmUnXG5cbmNvbnN0IHNlZWtiYXJTdHlsZSA9IHtcbiAgcG9zaXRpb246ICdyZWxhdGl2ZScsXG4gIGRpc3BsYXk6ICdmbGV4JyxcbiAgYWxpZ25JdGVtczogJ2NlbnRlcicsXG4gIG1pbldpZHRoOiAnMCcsXG4gIGhlaWdodDogJzI0cHgnLFxuICBmb250U2l6ZTogJzc1JScsXG4gIGxldHRlclNwYWNpbmc6ICcxcHgnLFxuICBjb2xvcjogJyNmZmYnLFxufVxuXG5jb25zdCBzbGlkZXJTdHlsZSA9IHtcbiAgbWluV2lkdGg6ICcwJyxcbiAgZmxleDogMSxcbiAgJ0BtZWRpYSAoaG92ZXI6IGhvdmVyKSwgc2NyZWVuIGFuZCAoLW1zLWhpZ2gtY29udHJhc3Q6IGFjdGl2ZSksICgtbXMtaGlnaC1jb250cmFzdDogbm9uZSknOlxuICAgIHtcbiAgICAgICc+IGRpdjpsYXN0LW9mLXR5cGUnOiB7XG4gICAgICAgIG9wYWNpdHk6IDAsXG4gICAgICAgIHRyYW5zaXRpb246ICdvcGFjaXR5IDAuMnMgZWFzZS1vdXQnLFxuICAgICAgfSxcbiAgICB9LFxuICAnPiBkaXY6bGFzdC1vZi10eXBlJzoge1xuICAgIHdpZHRoOiAnMS4yZW0nLFxuICAgIGhlaWdodDogJzEuMmVtJyxcbiAgfSxcbiAgJyY6aG92ZXIgPiBkaXY6bGFzdC1vZi10eXBlJzoge1xuICAgIG9wYWNpdHk6IDEsXG4gIH0sXG59XG5cbmNvbnN0IHJlZHVjZVBvaW50ZXIgPSAoc3RhdGUsIHt0eXBlLCB2YWx1ZSwgeH0pID0+IHtcbiAgc3dpdGNoICh0eXBlKSB7XG4gICAgY2FzZSAnbW92ZSc6XG4gICAgICByZXR1cm4gey4uLnN0YXRlLCBob3ZlcjogdHJ1ZSwgdmFsdWUsIHh9XG4gICAgY2FzZSAnY2hhbmdlJzpcbiAgICAgIHJldHVybiB7Li4uc3RhdGUsIGZvY3VzZWQ6IHRydWUsIHZhbHVlfVxuICAgIGNhc2UgJ3JlbGVhc2UnOlxuICAgICAgcmV0dXJuIHsuLi5zdGF0ZSwgZm9jdXNlZDogZmFsc2UsIHZhbHVlfVxuICAgIGNhc2UgJ2xlYXZlJzpcbiAgICAgIHJldHVybiB7Li4uc3RhdGUsIGhvdmVyOiBmYWxzZX1cbiAgICBkZWZhdWx0OlxuICAgICAgcmV0dXJuIHN0YXRlXG4gIH1cbn1cblxuY29uc3QgU2Vla2JhclJhaWwgPSAoKSA9PiAnJ1xuXG5jb25zdCBzZWdtZW50ZWRUcmFja1N0eWxlID0ge1xuICBib3hTaXppbmc6ICdib3JkZXItYm94JyxcbiAgcG9zaXRpb246ICdyZWxhdGl2ZScsXG4gIG1hcmdpbjogJzAgMnB4JyxcbiAgaGVpZ2h0OiAnMjRweCcsXG4gIGJvcmRlclRvcDogJ2NhbGMoMTJweCAtIDAuM2VtKSBzb2xpZCB0cmFuc3BhcmVudCcsXG4gIGJvcmRlckJvdHRvbTogJ2NhbGMoMTJweCAtIDAuM2VtKSBzb2xpZCB0cmFuc3BhcmVudCcsXG4gIGJhY2tncm91bmQ6ICd0cmFuc3BhcmVudCcsXG4gICcmOmhvdmVyJzoge1xuICAgIHRyYW5zZm9ybTogJ3NjYWxlKDEsIDEuNSknLFxuICB9LFxuICAnPiBkaXYnOiB7XG4gICAgaGVpZ2h0OiAnMC42ZW0nLFxuICAgIGJhY2tncm91bmRDb2xvcjogJ3JnYmEoMjU1LCAyNTUsIDI1NSwgMC4yKScsXG4gICAgJyY6bm90KDpmaXJzdC1vZi10eXBlKSc6IHtcbiAgICAgIHBvc2l0aW9uOiAnYWJzb2x1dGUnLFxuICAgICAgdG9wOiAnMCcsXG4gICAgfSxcbiAgfSxcbiAgJz4gZGl2Om50aC1vZi10eXBlKDIpJzoge1xuICAgIGJhY2tncm91bmRDb2xvcjogJ3JnYmEoMjU1LCAyNTUsIDI1NSwgMC4zKScsXG4gIH0sXG4gICc+IGRpdjpudGgtb2YtdHlwZSgzKSc6IHtcbiAgICBiYWNrZ3JvdW5kQ29sb3I6ICd2YXIoLS10aGVtZS1jb2xvciwgcmVkKScsXG4gIH0sXG59XG5cbmNvbnN0IFNlZWtiYXJUcmFjayA9ICh7c2VnbWVudHMsIHN0eWxlfSkgPT5cbiAgc2VnbWVudHMubWFwKCh7bGVuZ3RoLCBjdXJyZW50LCBidWZmZXJlZH0pID0+IChcbiAgICA8ZGl2XG4gICAgICBjc3M9e1tzdHlsZSwgc2VnbWVudGVkVHJhY2tTdHlsZV19XG4gICAgICBzdHlsZT17e1xuICAgICAgICBmbGV4OiBgMCAke2xlbmd0aCAqIDEwMH0lYCxcbiAgICAgICAgd2lkdGg6IGBjYWxjKCR7bGVuZ3RoICogMTAwfSUgLSA0cHgpYCxcbiAgICAgIH19XG4gICAgPlxuICAgICAgPGRpdiBzdHlsZT17e3dpZHRoOiAnMTAwJSd9fSAvPlxuICAgICAgPGRpdiBzdHlsZT17e3dpZHRoOiBgJHtidWZmZXJlZCAqIDEwMH0lYH19IC8+XG4gICAgICA8ZGl2IHN0eWxlPXt7d2lkdGg6IGAke2N1cnJlbnQgKiAxMDB9JWB9fSAvPlxuICAgIDwvZGl2PlxuICApKVxuXG4vLyBUT0RPIHVzZSBjbGFzc05hbWUgaW5zdGVhZCBvZiBjbGFzc2VzID9cbmNvbnN0IFNlZWtiYXIgPSAoe1xuICBzdHlsZSxcbiAgY2xhc3NlcyxcbiAgc3RhcnRUaW1lID0gMCxcbiAgY3VycmVudFRpbWUsXG4gIGJ1ZmZlclRpbWUsXG4gIGR1cmF0aW9uLFxuICBjaGFwdGVycyA9IFtdLFxuICBtYXJrcyxcbiAgdGltZURpc3BsYXkgPSBmYWxzZSwgLy8gVE9ETyBtb3JlIHNjYWxhYmxlIHdheT9cbiAgb25DaGFuZ2UsXG4gIG9uQ2hhbmdlQ29tbWl0dGVkLFxuICBjaGlsZHJlbixcbiAgb25Qb2ludGVyTW92ZSxcbiAgb25Qb2ludGVyTGVhdmUsXG4gIC4uLnJlc3Rcbn0pID0+IHtcbiAgY29uc3QgW3BvaW50ZXJTdGF0ZSwgZGlzcGF0Y2hQb2ludGVyXSA9IHVzZVJlZHVjZXIocmVkdWNlUG9pbnRlciwge30pXG4gIGNvbnN0IHBvaW50ZXJBY3RpdmUgPSBwb2ludGVyU3RhdGUuaG92ZXIgfHwgcG9pbnRlclN0YXRlLmZvY3VzZWRcbiAgLy8gdG8gcmVmbGVjdCBib3VuZGFyeSB3aGVuIGNvbnRhaW5lciByZXNpemVkXG4gIGNvbnN0IHtvYnNlcnZlfSA9IHVzZURpbWVuc2lvbnMoKVxuICBjb25zdCByZWYgPSB1c2VSZWYoKVxuICBjb25zdCByZWN0ID0gcmVmLmN1cnJlbnQ/LmdldEJvdW5kaW5nQ2xpZW50UmVjdCgpXG4gIGNvbnN0IGhhbmRsZXJzID0gb25DaGFuZ2VDb21taXR0ZWQgJiYge1xuICAgIG9uUG9pbnRlck1vdmU6IChldmVudCwge3ZhbHVlLCB4fSkgPT4ge1xuICAgICAgb25Qb2ludGVyTW92ZT8uKGV2ZW50LCB7dmFsdWUsIHh9KVxuICAgICAgZGlzcGF0Y2hQb2ludGVyKHt0eXBlOiAnbW92ZScsIHZhbHVlLCB4fSlcbiAgICB9LFxuICAgIG9uUG9pbnRlckxlYXZlOiBldmVudCA9PiB7XG4gICAgICBvblBvaW50ZXJMZWF2ZT8uKGV2ZW50KVxuICAgICAgZGlzcGF0Y2hQb2ludGVyKHt0eXBlOiAnbGVhdmUnfSlcbiAgICB9LFxuICAgIG9uQ2hhbmdlOiAoXywge3ZhbHVlfSkgPT4ge1xuICAgICAgb25DaGFuZ2U/Lih2YWx1ZSlcbiAgICAgIGRpc3BhdGNoUG9pbnRlcih7dHlwZTogJ2NoYW5nZScsIHZhbHVlfSlcbiAgICB9LFxuICAgIG9uQ2hhbmdlQ29tbWl0dGVkOiAoXywge3ZhbHVlfSkgPT4ge1xuICAgICAgZGlzcGF0Y2hQb2ludGVyKHt0eXBlOiAncmVsZWFzZScsIHZhbHVlfSlcbiAgICAgIG9uQ2hhbmdlQ29tbWl0dGVkPy4odmFsdWUpXG4gICAgfSxcbiAgfVxuICBjb25zdCBlbmRUaW1lID0gc3RhcnRUaW1lICsgZHVyYXRpb25cbiAgY29uc3Qgc2VnbWVudHMgPSBnZXRUaW1lbGluZVNlZ21lbnRzKGNoYXB0ZXJzLCB7XG4gICAgc3RhcnRUaW1lLFxuICAgIGN1cnJlbnQ6IHBvaW50ZXJTdGF0ZS5mb2N1c2VkID8gcG9pbnRlclN0YXRlLnZhbHVlIDogY3VycmVudFRpbWUsXG4gICAgYnVmZmVyZWQ6IGJ1ZmZlclRpbWUsXG4gICAgZHVyYXRpb24sXG4gIH0pXG5cbiAgcmV0dXJuICEoZHVyYXRpb24gPiAwKSA/IChcbiAgICA8ZGl2IC8+XG4gICkgOiAoXG4gICAgPGRpdlxuICAgICAgcmVmPXtlbGVtZW50ID0+IHtcbiAgICAgICAgb2JzZXJ2ZShlbGVtZW50KVxuICAgICAgICByZWYuY3VycmVudCA9IGVsZW1lbnRcbiAgICAgIH19XG4gICAgICBjc3M9e1tzZWVrYmFyU3R5bGUsIHN0eWxlXX1cbiAgICAgIHN0eWxlPXtcbiAgICAgICAgcmVjdCAmJiB7XG4gICAgICAgICAgJy0tc2Vla2Jhci1sZWZ0JzogYCR7cmVjdC5sZWZ0fXB4YCxcbiAgICAgICAgICAnLS1zZWVrYmFyLXJpZ2h0JzogYCR7cmVjdC5yaWdodH1weGAsXG4gICAgICAgICAgJy0tcG9pbnRlci14JzogYCR7cG9pbnRlclN0YXRlLnh9cHhgLFxuICAgICAgICB9XG4gICAgICB9XG4gICAgPlxuICAgICAgeyF0aW1lRGlzcGxheSA/IChcbiAgICAgICAgJydcbiAgICAgICkgOiBwb2ludGVyQWN0aXZlID8gKFxuICAgICAgICA8Rm9ybWF0dGVkVGltZSB0aW1lPXtwb2ludGVyU3RhdGUudmFsdWV9IC8+XG4gICAgICApIDogKFxuICAgICAgICA8Rm9ybWF0dGVkVGltZSB0aW1lPXtNYXRoLm1pbihNYXRoLm1heCgwLCBjdXJyZW50VGltZSksIGR1cmF0aW9uKX0gLz5cbiAgICAgICl9XG4gICAgICA8U2ltcGxlU2xpZGVyXG4gICAgICAgIGNzcz17W3NsaWRlclN0eWxlLCB0aW1lRGlzcGxheSAmJiB7bWFyZ2luOiAnMCAxZW0nfV19XG4gICAgICAgIGNsYXNzZXM9e2NsYXNzZXN9XG4gICAgICAgIGRpc2FibGVkPXshb25DaGFuZ2VDb21taXR0ZWR9XG4gICAgICAgIC8vIGRpc3BsYXkgZmlsbGVkIHdoZW4gc2VlayBoYW5kbGVyIGlzIG5vdCBwcm92aWRlZCwgZnJvbSBQbGF5Qm95IGJlaGF2aW9yXG4gICAgICAgIHZhbHVlPXtvbkNoYW5nZUNvbW1pdHRlZCA/IGN1cnJlbnRUaW1lIDogZW5kVGltZX1cbiAgICAgICAgbWluPXtzdGFydFRpbWV9XG4gICAgICAgIG1heD17ZW5kVGltZX1cbiAgICAgICAgbWFya3M9e21hcmtzfVxuICAgICAgICB7Li4uaGFuZGxlcnN9XG4gICAgICAgIHsuLi5yZXN0fVxuICAgICAgICBzbG90cz17e1JhaWw6IFNlZWtiYXJSYWlsLCBUcmFjazogU2Vla2JhclRyYWNrfX1cbiAgICAgICAgc2xvdFByb3BzPXt7dHJhY2s6IHtzZWdtZW50c319fVxuICAgICAgLz5cbiAgICAgIHt0aW1lRGlzcGxheSAmJiA8Rm9ybWF0dGVkVGltZSB0aW1lPXtkdXJhdGlvbn0gLz59XG4gICAgICB7Y2hpbGRyZW4gJiZcbiAgICAgICAgW11cbiAgICAgICAgICAuY29uY2F0KGNoaWxkcmVuKVxuICAgICAgICAgIC5tYXAoY2hpbGQgPT5cbiAgICAgICAgICAgIGNsb25lRWxlbWVudChjaGlsZCwge3RpbWU6IHBvaW50ZXJBY3RpdmUgJiYgcG9pbnRlclN0YXRlLnZhbHVlfSlcbiAgICAgICAgICApfVxuICAgIDwvZGl2PlxuICApXG59XG5cbmV4cG9ydCBkZWZhdWx0IFNlZWtiYXJcbiJdfQ== */"],
|
|
4361
4306
|
style: rect && {
|
|
4362
4307
|
'--seekbar-left': `${rect.left}px`,
|
|
4363
4308
|
'--seekbar-right': `${rect.right}px`,
|
|
@@ -4370,7 +4315,7 @@ const Seekbar = ({
|
|
|
4370
4315
|
}), jsx$1(SimpleSlider, {
|
|
4371
4316
|
css: [sliderStyle, timeDisplay && {
|
|
4372
4317
|
margin: '0 1em'
|
|
4373
|
-
}, process.env.NODE_ENV === "production" ? "" : ";label:Seekbar;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
4318
|
+
}, process.env.NODE_ENV === "production" ? "" : ";label:Seekbar;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIlNlZWtiYXIuanMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBMktRIiwiZmlsZSI6IlNlZWtiYXIuanMiLCJzb3VyY2VzQ29udGVudCI6WyIvKiBAanN4SW1wb3J0U291cmNlIEBlbW90aW9uL3JlYWN0ICovXG5pbXBvcnQge3VzZVJlZHVjZXIsIHVzZVJlZiwgY2xvbmVFbGVtZW50fSBmcm9tICdyZWFjdCdcbmltcG9ydCB1c2VEaW1lbnNpb25zIGZyb20gJ3JlYWN0LWNvb2wtZGltZW5zaW9ucydcbmltcG9ydCBTaW1wbGVTbGlkZXIgZnJvbSAncGxheWVyVWkvU2ltcGxlU2xpZGVyJ1xuaW1wb3J0IHtGb3JtYXR0ZWRUaW1lfSBmcm9tICdjb250ZXh0L0kxOG4nXG5pbXBvcnQge2dldFRpbWVsaW5lU2VnbWVudHN9IGZyb20gJ3ByZW1pdW0vdGltZWxpbmUnXG5cbmNvbnN0IHNlZWtiYXJTdHlsZSA9IHtcbiAgcG9zaXRpb246ICdyZWxhdGl2ZScsXG4gIGRpc3BsYXk6ICdmbGV4JyxcbiAgYWxpZ25JdGVtczogJ2NlbnRlcicsXG4gIG1pbldpZHRoOiAnMCcsXG4gIGhlaWdodDogJzI0cHgnLFxuICBmb250U2l6ZTogJzc1JScsXG4gIGxldHRlclNwYWNpbmc6ICcxcHgnLFxuICBjb2xvcjogJyNmZmYnLFxufVxuXG5jb25zdCBzbGlkZXJTdHlsZSA9IHtcbiAgbWluV2lkdGg6ICcwJyxcbiAgZmxleDogMSxcbiAgJ0BtZWRpYSAoaG92ZXI6IGhvdmVyKSwgc2NyZWVuIGFuZCAoLW1zLWhpZ2gtY29udHJhc3Q6IGFjdGl2ZSksICgtbXMtaGlnaC1jb250cmFzdDogbm9uZSknOlxuICAgIHtcbiAgICAgICc+IGRpdjpsYXN0LW9mLXR5cGUnOiB7XG4gICAgICAgIG9wYWNpdHk6IDAsXG4gICAgICAgIHRyYW5zaXRpb246ICdvcGFjaXR5IDAuMnMgZWFzZS1vdXQnLFxuICAgICAgfSxcbiAgICB9LFxuICAnPiBkaXY6bGFzdC1vZi10eXBlJzoge1xuICAgIHdpZHRoOiAnMS4yZW0nLFxuICAgIGhlaWdodDogJzEuMmVtJyxcbiAgfSxcbiAgJyY6aG92ZXIgPiBkaXY6bGFzdC1vZi10eXBlJzoge1xuICAgIG9wYWNpdHk6IDEsXG4gIH0sXG59XG5cbmNvbnN0IHJlZHVjZVBvaW50ZXIgPSAoc3RhdGUsIHt0eXBlLCB2YWx1ZSwgeH0pID0+IHtcbiAgc3dpdGNoICh0eXBlKSB7XG4gICAgY2FzZSAnbW92ZSc6XG4gICAgICByZXR1cm4gey4uLnN0YXRlLCBob3ZlcjogdHJ1ZSwgdmFsdWUsIHh9XG4gICAgY2FzZSAnY2hhbmdlJzpcbiAgICAgIHJldHVybiB7Li4uc3RhdGUsIGZvY3VzZWQ6IHRydWUsIHZhbHVlfVxuICAgIGNhc2UgJ3JlbGVhc2UnOlxuICAgICAgcmV0dXJuIHsuLi5zdGF0ZSwgZm9jdXNlZDogZmFsc2UsIHZhbHVlfVxuICAgIGNhc2UgJ2xlYXZlJzpcbiAgICAgIHJldHVybiB7Li4uc3RhdGUsIGhvdmVyOiBmYWxzZX1cbiAgICBkZWZhdWx0OlxuICAgICAgcmV0dXJuIHN0YXRlXG4gIH1cbn1cblxuY29uc3QgU2Vla2JhclJhaWwgPSAoKSA9PiAnJ1xuXG5jb25zdCBzZWdtZW50ZWRUcmFja1N0eWxlID0ge1xuICBib3hTaXppbmc6ICdib3JkZXItYm94JyxcbiAgcG9zaXRpb246ICdyZWxhdGl2ZScsXG4gIG1hcmdpbjogJzAgMnB4JyxcbiAgaGVpZ2h0OiAnMjRweCcsXG4gIGJvcmRlclRvcDogJ2NhbGMoMTJweCAtIDAuM2VtKSBzb2xpZCB0cmFuc3BhcmVudCcsXG4gIGJvcmRlckJvdHRvbTogJ2NhbGMoMTJweCAtIDAuM2VtKSBzb2xpZCB0cmFuc3BhcmVudCcsXG4gIGJhY2tncm91bmQ6ICd0cmFuc3BhcmVudCcsXG4gICcmOmhvdmVyJzoge1xuICAgIHRyYW5zZm9ybTogJ3NjYWxlKDEsIDEuNSknLFxuICB9LFxuICAnPiBkaXYnOiB7XG4gICAgaGVpZ2h0OiAnMC42ZW0nLFxuICAgIGJhY2tncm91bmRDb2xvcjogJ3JnYmEoMjU1LCAyNTUsIDI1NSwgMC4yKScsXG4gICAgJyY6bm90KDpmaXJzdC1vZi10eXBlKSc6IHtcbiAgICAgIHBvc2l0aW9uOiAnYWJzb2x1dGUnLFxuICAgICAgdG9wOiAnMCcsXG4gICAgfSxcbiAgfSxcbiAgJz4gZGl2Om50aC1vZi10eXBlKDIpJzoge1xuICAgIGJhY2tncm91bmRDb2xvcjogJ3JnYmEoMjU1LCAyNTUsIDI1NSwgMC4zKScsXG4gIH0sXG4gICc+IGRpdjpudGgtb2YtdHlwZSgzKSc6IHtcbiAgICBiYWNrZ3JvdW5kQ29sb3I6ICd2YXIoLS10aGVtZS1jb2xvciwgcmVkKScsXG4gIH0sXG59XG5cbmNvbnN0IFNlZWtiYXJUcmFjayA9ICh7c2VnbWVudHMsIHN0eWxlfSkgPT5cbiAgc2VnbWVudHMubWFwKCh7bGVuZ3RoLCBjdXJyZW50LCBidWZmZXJlZH0pID0+IChcbiAgICA8ZGl2XG4gICAgICBjc3M9e1tzdHlsZSwgc2VnbWVudGVkVHJhY2tTdHlsZV19XG4gICAgICBzdHlsZT17e1xuICAgICAgICBmbGV4OiBgMCAke2xlbmd0aCAqIDEwMH0lYCxcbiAgICAgICAgd2lkdGg6IGBjYWxjKCR7bGVuZ3RoICogMTAwfSUgLSA0cHgpYCxcbiAgICAgIH19XG4gICAgPlxuICAgICAgPGRpdiBzdHlsZT17e3dpZHRoOiAnMTAwJSd9fSAvPlxuICAgICAgPGRpdiBzdHlsZT17e3dpZHRoOiBgJHtidWZmZXJlZCAqIDEwMH0lYH19IC8+XG4gICAgICA8ZGl2IHN0eWxlPXt7d2lkdGg6IGAke2N1cnJlbnQgKiAxMDB9JWB9fSAvPlxuICAgIDwvZGl2PlxuICApKVxuXG4vLyBUT0RPIHVzZSBjbGFzc05hbWUgaW5zdGVhZCBvZiBjbGFzc2VzID9cbmNvbnN0IFNlZWtiYXIgPSAoe1xuICBzdHlsZSxcbiAgY2xhc3NlcyxcbiAgc3RhcnRUaW1lID0gMCxcbiAgY3VycmVudFRpbWUsXG4gIGJ1ZmZlclRpbWUsXG4gIGR1cmF0aW9uLFxuICBjaGFwdGVycyA9IFtdLFxuICBtYXJrcyxcbiAgdGltZURpc3BsYXkgPSBmYWxzZSwgLy8gVE9ETyBtb3JlIHNjYWxhYmxlIHdheT9cbiAgb25DaGFuZ2UsXG4gIG9uQ2hhbmdlQ29tbWl0dGVkLFxuICBjaGlsZHJlbixcbiAgb25Qb2ludGVyTW92ZSxcbiAgb25Qb2ludGVyTGVhdmUsXG4gIC4uLnJlc3Rcbn0pID0+IHtcbiAgY29uc3QgW3BvaW50ZXJTdGF0ZSwgZGlzcGF0Y2hQb2ludGVyXSA9IHVzZVJlZHVjZXIocmVkdWNlUG9pbnRlciwge30pXG4gIGNvbnN0IHBvaW50ZXJBY3RpdmUgPSBwb2ludGVyU3RhdGUuaG92ZXIgfHwgcG9pbnRlclN0YXRlLmZvY3VzZWRcbiAgLy8gdG8gcmVmbGVjdCBib3VuZGFyeSB3aGVuIGNvbnRhaW5lciByZXNpemVkXG4gIGNvbnN0IHtvYnNlcnZlfSA9IHVzZURpbWVuc2lvbnMoKVxuICBjb25zdCByZWYgPSB1c2VSZWYoKVxuICBjb25zdCByZWN0ID0gcmVmLmN1cnJlbnQ/LmdldEJvdW5kaW5nQ2xpZW50UmVjdCgpXG4gIGNvbnN0IGhhbmRsZXJzID0gb25DaGFuZ2VDb21taXR0ZWQgJiYge1xuICAgIG9uUG9pbnRlck1vdmU6IChldmVudCwge3ZhbHVlLCB4fSkgPT4ge1xuICAgICAgb25Qb2ludGVyTW92ZT8uKGV2ZW50LCB7dmFsdWUsIHh9KVxuICAgICAgZGlzcGF0Y2hQb2ludGVyKHt0eXBlOiAnbW92ZScsIHZhbHVlLCB4fSlcbiAgICB9LFxuICAgIG9uUG9pbnRlckxlYXZlOiBldmVudCA9PiB7XG4gICAgICBvblBvaW50ZXJMZWF2ZT8uKGV2ZW50KVxuICAgICAgZGlzcGF0Y2hQb2ludGVyKHt0eXBlOiAnbGVhdmUnfSlcbiAgICB9LFxuICAgIG9uQ2hhbmdlOiAoXywge3ZhbHVlfSkgPT4ge1xuICAgICAgb25DaGFuZ2U/Lih2YWx1ZSlcbiAgICAgIGRpc3BhdGNoUG9pbnRlcih7dHlwZTogJ2NoYW5nZScsIHZhbHVlfSlcbiAgICB9LFxuICAgIG9uQ2hhbmdlQ29tbWl0dGVkOiAoXywge3ZhbHVlfSkgPT4ge1xuICAgICAgZGlzcGF0Y2hQb2ludGVyKHt0eXBlOiAncmVsZWFzZScsIHZhbHVlfSlcbiAgICAgIG9uQ2hhbmdlQ29tbWl0dGVkPy4odmFsdWUpXG4gICAgfSxcbiAgfVxuICBjb25zdCBlbmRUaW1lID0gc3RhcnRUaW1lICsgZHVyYXRpb25cbiAgY29uc3Qgc2VnbWVudHMgPSBnZXRUaW1lbGluZVNlZ21lbnRzKGNoYXB0ZXJzLCB7XG4gICAgc3RhcnRUaW1lLFxuICAgIGN1cnJlbnQ6IHBvaW50ZXJTdGF0ZS5mb2N1c2VkID8gcG9pbnRlclN0YXRlLnZhbHVlIDogY3VycmVudFRpbWUsXG4gICAgYnVmZmVyZWQ6IGJ1ZmZlclRpbWUsXG4gICAgZHVyYXRpb24sXG4gIH0pXG5cbiAgcmV0dXJuICEoZHVyYXRpb24gPiAwKSA/IChcbiAgICA8ZGl2IC8+XG4gICkgOiAoXG4gICAgPGRpdlxuICAgICAgcmVmPXtlbGVtZW50ID0+IHtcbiAgICAgICAgb2JzZXJ2ZShlbGVtZW50KVxuICAgICAgICByZWYuY3VycmVudCA9IGVsZW1lbnRcbiAgICAgIH19XG4gICAgICBjc3M9e1tzZWVrYmFyU3R5bGUsIHN0eWxlXX1cbiAgICAgIHN0eWxlPXtcbiAgICAgICAgcmVjdCAmJiB7XG4gICAgICAgICAgJy0tc2Vla2Jhci1sZWZ0JzogYCR7cmVjdC5sZWZ0fXB4YCxcbiAgICAgICAgICAnLS1zZWVrYmFyLXJpZ2h0JzogYCR7cmVjdC5yaWdodH1weGAsXG4gICAgICAgICAgJy0tcG9pbnRlci14JzogYCR7cG9pbnRlclN0YXRlLnh9cHhgLFxuICAgICAgICB9XG4gICAgICB9XG4gICAgPlxuICAgICAgeyF0aW1lRGlzcGxheSA/IChcbiAgICAgICAgJydcbiAgICAgICkgOiBwb2ludGVyQWN0aXZlID8gKFxuICAgICAgICA8Rm9ybWF0dGVkVGltZSB0aW1lPXtwb2ludGVyU3RhdGUudmFsdWV9IC8+XG4gICAgICApIDogKFxuICAgICAgICA8Rm9ybWF0dGVkVGltZSB0aW1lPXtNYXRoLm1pbihNYXRoLm1heCgwLCBjdXJyZW50VGltZSksIGR1cmF0aW9uKX0gLz5cbiAgICAgICl9XG4gICAgICA8U2ltcGxlU2xpZGVyXG4gICAgICAgIGNzcz17W3NsaWRlclN0eWxlLCB0aW1lRGlzcGxheSAmJiB7bWFyZ2luOiAnMCAxZW0nfV19XG4gICAgICAgIGNsYXNzZXM9e2NsYXNzZXN9XG4gICAgICAgIGRpc2FibGVkPXshb25DaGFuZ2VDb21taXR0ZWR9XG4gICAgICAgIC8vIGRpc3BsYXkgZmlsbGVkIHdoZW4gc2VlayBoYW5kbGVyIGlzIG5vdCBwcm92aWRlZCwgZnJvbSBQbGF5Qm95IGJlaGF2aW9yXG4gICAgICAgIHZhbHVlPXtvbkNoYW5nZUNvbW1pdHRlZCA/IGN1cnJlbnRUaW1lIDogZW5kVGltZX1cbiAgICAgICAgbWluPXtzdGFydFRpbWV9XG4gICAgICAgIG1heD17ZW5kVGltZX1cbiAgICAgICAgbWFya3M9e21hcmtzfVxuICAgICAgICB7Li4uaGFuZGxlcnN9XG4gICAgICAgIHsuLi5yZXN0fVxuICAgICAgICBzbG90cz17e1JhaWw6IFNlZWtiYXJSYWlsLCBUcmFjazogU2Vla2JhclRyYWNrfX1cbiAgICAgICAgc2xvdFByb3BzPXt7dHJhY2s6IHtzZWdtZW50c319fVxuICAgICAgLz5cbiAgICAgIHt0aW1lRGlzcGxheSAmJiA8Rm9ybWF0dGVkVGltZSB0aW1lPXtkdXJhdGlvbn0gLz59XG4gICAgICB7Y2hpbGRyZW4gJiZcbiAgICAgICAgW11cbiAgICAgICAgICAuY29uY2F0KGNoaWxkcmVuKVxuICAgICAgICAgIC5tYXAoY2hpbGQgPT5cbiAgICAgICAgICAgIGNsb25lRWxlbWVudChjaGlsZCwge3RpbWU6IHBvaW50ZXJBY3RpdmUgJiYgcG9pbnRlclN0YXRlLnZhbHVlfSlcbiAgICAgICAgICApfVxuICAgIDwvZGl2PlxuICApXG59XG5cbmV4cG9ydCBkZWZhdWx0IFNlZWtiYXJcbiJdfQ== */"],
|
|
4374
4319
|
classes: classes,
|
|
4375
4320
|
disabled: !onChangeCommitted // display filled when seek handler is not provided, from PlayBoy behavior
|
|
4376
4321
|
,
|
|
@@ -4743,7 +4688,7 @@ const LoopSwitch = ({
|
|
|
4743
4688
|
onChange
|
|
4744
4689
|
}) => jsxs("li", {
|
|
4745
4690
|
role: "menuitem",
|
|
4746
|
-
css: [style, "::before{display:none;}::after{display:none;}" + (process.env.NODE_ENV === "production" ? "" : ";label:LoopSwitch;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Settings.js"],"names":[],"mappings":"AA8KI","file":"Settings.js","sourcesContent":["/* @jsxImportSource @emotion/react */\n/* eslint-disable jsx-a11y/no-noninteractive-element-interactions */\nimport {useState, useEffect} from 'react'\nimport useOnclickOutside from 'react-cool-onclickoutside'\n\nimport icon from 'style/icon'\nimport {FormattedMessage} from 'context/I18n'\nimport {Button, Switch} from './buttons'\nimport SwipeableDrawer from './SwipeableDrawer'\nimport {FunctionBarExtension, TitleBarExtension} from './uiExtensions'\n\nconst ulReset = {\n  marginBlockStart: 0,\n  marginBlockEnd: 0,\n  paddingInlineStart: 0,\n}\n\nconst mobileStyle = {\n  head: {\n    position: 'sticky',\n    zIndex: '1',\n    top: '0',\n    display: 'flex',\n    alignItems: 'center',\n    padding: '1rem 1.5rem',\n    color: 'white',\n    background: '#161C24',\n    fontSize: '16px',\n    fontWeight: 'bold',\n    button: {\n      marginRight: '1rem',\n      padding: '0',\n      width: '1.5rem',\n      height: '1.5rem',\n      border: 'none',\n    },\n  },\n  overlay: {\n    '> div': {\n      // drawer container\n      background: 'var(--setting-ui-background, #161C24)',\n      maxHeight: 'calc(100% - var(--setting-ui-vertical-padding, 2rem))',\n    },\n  },\n  container: {\n    ...ulReset,\n    // TODO use dvh once we may drop iOS 14\n    maxHeight: 'calc(90vh - var(--setting-ui-vertical-padding, 2rem))',\n    color: '#ccc',\n    whiteSpace: 'nowrap',\n    borderRadius: '4px',\n    userSelect: 'none',\n    overflow: 'auto',\n    overscrollBehavior: 'contain',\n  },\n  title: {\n    padding: '12px 18px',\n  },\n  dismiss: {\n    background: `center / 1rem no-repeat url(${icon.close}), transparent`,\n  },\n  back: {\n    background: '#ccc',\n    maskImage: `var(--icon-setting-back, url(\"${icon.back}\"))`,\n    maskSize: 'contain',\n  },\n  row: {\n    cursor: 'pointer',\n    display: 'flex',\n    position: 'relative',\n    padding: '1rem 1.5rem',\n    fontSize: '16px',\n    background: '#161C24',\n    '::after': {\n      content: '\" \"',\n      marginLeft: '1rem',\n      width: '20px',\n      height: '20px',\n      display: 'inline-block',\n      color: 'white',\n      backgroundPosition: 'center',\n      backgroundSize: 'cover',\n    },\n  },\n  space: {\n    flex: '1',\n  },\n  hasOptions: {\n    '::after': {\n      backgroundImage: `url(${icon.arrowTop})`,\n      transform: 'rotate(90deg)',\n    },\n  },\n  selected: {\n    color: 'white',\n    fontWeight: 'bold',\n    '::after': {\n      backgroundColor: 'var(--setting-check-icon-color, var(--primary-highlight, white))',\n      maskImage: `url(${icon.check})`,\n    },\n  },\n}\n\n// TODO some of styles are for older version UI design, can be simplified\nconst desktopStyle = {\n  overlay: {\n    position: 'absolute',\n    zIndex: '1',\n    bottom: 'calc(5em + var(--bottom-spacing, 0rem))',\n    right: '1rem',\n    display: 'flex',\n    alignItems: 'flex-end',\n    width: '18rem',\n    height: 'calc(100% - 8rem - var(--bottom-spacing, 0rem))',\n    outline: 'none',\n    opacity: '1',\n    transform: 'translateY(-100vh)',\n  },\n  open: {\n    position: 'fixed',\n    opacity: '1',\n    transform: 'translateY(0)',\n    transition: 'opacity 0.2s ease, transform 0s',\n  },\n  container: {\n    ...mobileStyle.container,\n    flex: '0 var(--setting-ui-width, calc(100% - 2rem))',\n    maxHeight: '100%',\n  },\n  head: {\n    ...mobileStyle.head,\n    color: 'var(--setting-head-color, inherit)',\n    background: 'var(--setting-head-background, rgba(22, 28, 36, 0.8))',\n  },\n  row: {\n    ...mobileStyle.row,\n    background: 'rgba(22, 28, 36, 0.8)',\n    '::after': {\n      order: 'var(--setting-check-icon-order, 0)',\n      ...mobileStyle.row['::after'],\n      marginLeft: '0',\n      marginRight: '0.5em',\n    },\n  },\n  hasOptions: {\n    '::after': {\n      order: '0',\n      backgroundImage: `url(${icon.arrowTop})`,\n      transform: 'rotate(90deg)',\n    },\n  },\n  selected: {\n    color: 'white',\n    fontWeight: 'bold',\n    '::after': {\n      backgroundColor:\n        'var(--setting-check-icon-color, var(--primary-highlight, white))',\n      maskImage: `url(${icon.check})`,\n    },\n  },\n}\n\nconst MenuItemText = ({text = ''}) => (\n  <FormattedMessage\n    id={text}\n    defaultMessage={\n      <FormattedMessage id={`KKS.SETTING.${text}`} defaultMessage={text} />\n    }\n  />\n)\n\nconst LoopSwitch = ({style, checked, onChange}) => (\n  <li\n    role=\"menuitem\"\n    css={[style, {'::before': {display: 'none'}, '::after': {display: 'none'}}]}\n    onClick={() => {\n      onChange({name: 'loop', value: !checked, keepOpen: true})\n    }}\n  >\n    <MenuItemText text=\"KKS.PLAYER.LOOP\" />\n    <div css={mobileStyle.space} />\n    <Switch checked={checked} />\n  </li>\n)\n\nconst CloseButton = props => (\n  <button\n    type=\"button\"\n    aria-label=\"Close Settings\"\n    css={mobileStyle.dismiss}\n    {...props}\n  />\n)\n\nconst DesktopContainer = ({open, style, children, onClose, ...rest}) => (\n  <div css={[style, open && desktopStyle.open]} {...rest}>\n    {children}\n  </div>\n)\n\nconst matchValue = (a, b) => a === b || (a?.id ?? a) === (b?.id ?? b)\n\nconst Settings = ({\n  open,\n  values,\n  sections: originalSections,\n  type: uiType,\n  closeBy,\n  buttonPosition,\n  enabledSettingSections = {\n    audio: true,\n    subtitles: true,\n    quality: true,\n    speed: true,\n    loop: true,\n  }, \n  slots = {\n    root: uiType === 'desktop' ? DesktopContainer : SwipeableDrawer,\n  },\n  onChange,\n  onOpen,\n  onClose,\n}) => {\n  const sections = originalSections.filter(section => enabledSettingSections[section.name])\n  const Container = slots.root\n  const ButtonWrap =\n    buttonPosition === 'top-right' ? TitleBarExtension : FunctionBarExtension\n  const commonStyle = uiType === 'desktop' ? desktopStyle : mobileStyle\n  const [path, setPath] = useState('/')\n  useEffect(() => {\n    if (!open) {\n      setPath('/')\n    }\n  }, [open])\n\n  const ref = useOnclickOutside(\n    () => {\n      if (open && uiType === 'desktop') {\n        onClose()\n      }\n    },\n    {eventTypes: ['click']}\n  )\n  const currentSection = sections.find(it => path === `/${it.name}`)\n  const menu =\n    path === '/'\n      ? {\n          title: 'KKS.SETTING',\n          items: sections.map(({type, name, title, items = []}) => ({\n            type,\n            link: `/${name}`,\n            label: title,\n            value:\n              items.find(item => matchValue(item.value, values[name]))?.label ||\n              values[name],\n          })),\n        }\n      : {\n          title: currentSection.title,\n          items: currentSection.items.map(({value, label}) => ({\n            label,\n            checked: matchValue(values[currentSection.name], value),\n            data: value,\n          })),\n          previous: '/',\n        }\n  const navigate = dest => requestAnimationFrame(() => setPath(dest))\n  return (\n    sections.length > 0 && (\n    <>\n      <ButtonWrap position=\"right\">\n        <Button\n          startIcon=\"setting\"\n          title=\"KKS.SETTING\"\n            onClick={event =>\n              setTimeout(() => (open ? onClose(event) : onOpen(event)), 1)\n            }\n        />\n      </ButtonWrap>\n      <Container style={commonStyle.overlay} open={open} onClose={onClose}>\n        <ul role=\"menu\" ref={ref} css={commonStyle.container}>\n          <div css={commonStyle.head}>\n            {menu.previous ? (\n              <Button startIcon=\"back1\" onClick={() => navigate('/')} />\n            ) : (\n              uiType !== 'desktop' &&\n              closeBy === 'button' && <CloseButton onClick={onClose} />\n            )}\n            <FormattedMessage id={menu.title} />\n          </div>\n          {menu.items.map(({type, label, link, value, data, checked}) =>\n            type === 'switch' ? (\n              <LoopSwitch\n                style={commonStyle.row}\n                checked={values.loop}\n                onChange={() =>\n                    onChange({\n                      name: 'loop',\n                      value: !values.loop,\n                      keepOpen: true,\n                    })\n                }\n              />\n            ) : (\n              <li\n                role={link ? 'menuitem' : 'menuitemradio'}\n                aria-checked={checked}\n                css={[\n                  commonStyle.row,\n                  link && commonStyle.hasOptions,\n                  checked && commonStyle.selected,\n                ]}\n                onClick={() =>\n                  link\n                    ? navigate(link)\n                    : onChange({name: currentSection.name, value: data})\n                }\n              >\n                <MenuItemText text={label} />\n                <div css={mobileStyle.space} />\n                {value && <MenuItemText text={value.toString()} />}\n              </li>\n            )\n          )}\n        </ul>\n      </Container>\n    </>\n    )\n  )\n}\n\nexport default Settings\n"]} */"],
|
|
4691
|
+
css: [style, "::before{display:none;}::after{display:none;}" + (process.env.NODE_ENV === "production" ? "" : ";label:LoopSwitch;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Settings.js"],"names":[],"mappings":"AA8KI","file":"Settings.js","sourcesContent":["/* @jsxImportSource @emotion/react */\n/* eslint-disable jsx-a11y/no-noninteractive-element-interactions */\nimport {useState, useEffect} from 'react'\nimport useOnclickOutside from 'react-cool-onclickoutside'\n\nimport icon from 'style/icon'\nimport {FormattedMessage} from 'context/I18n'\nimport {Button, Switch} from './buttons'\nimport SwipeableDrawer from './SwipeableDrawer'\nimport {FunctionBarExtension, TitleBarExtension} from './uiExtensions'\n\nconst ulReset = {\n  marginBlockStart: 0,\n  marginBlockEnd: 0,\n  paddingInlineStart: 0,\n}\n\nconst mobileStyle = {\n  head: {\n    position: 'sticky',\n    zIndex: '1',\n    top: '0',\n    display: 'flex',\n    alignItems: 'center',\n    padding: '1rem 1.5rem',\n    color: 'white',\n    background: '#161C24',\n    fontSize: '16px',\n    fontWeight: 'bold',\n    button: {\n      marginRight: '1rem',\n      padding: '0',\n      width: '1.5rem',\n      height: '1.5rem',\n      border: 'none',\n    },\n  },\n  overlay: {\n    '> div': {\n      // drawer container\n      background: 'var(--setting-ui-background, #161C24)',\n      maxHeight: 'calc(100% - var(--setting-ui-vertical-padding, 2rem))',\n    },\n  },\n  container: {\n    ...ulReset,\n    // TODO use dvh once we may drop iOS 14\n    maxHeight: 'calc(90vh - var(--setting-ui-vertical-padding, 2rem))',\n    color: '#ccc',\n    whiteSpace: 'nowrap',\n    borderRadius: '4px',\n    userSelect: 'none',\n    overflow: 'auto',\n    overscrollBehavior: 'contain',\n  },\n  title: {\n    padding: '12px 18px',\n  },\n  dismiss: {\n    background: `center / 1rem no-repeat url(${icon.close}), transparent`,\n  },\n  back: {\n    background: '#ccc',\n    maskImage: `var(--icon-setting-back, url(\"${icon.back}\"))`,\n    maskSize: 'contain',\n  },\n  row: {\n    cursor: 'pointer',\n    display: 'flex',\n    position: 'relative',\n    padding: '1rem 1.5rem',\n    fontSize: '16px',\n    background: '#161C24',\n    '::after': {\n      content: '\" \"',\n      marginLeft: '1rem',\n      width: '20px',\n      height: '20px',\n      display: 'inline-block',\n      color: 'white',\n      backgroundPosition: 'center',\n      backgroundSize: 'cover',\n    },\n  },\n  space: {\n    flex: '1',\n  },\n  hasOptions: {\n    '::after': {\n      backgroundImage: `url(${icon.arrowTop})`,\n      transform: 'rotate(90deg)',\n    },\n  },\n  selected: {\n    color: 'white',\n    fontWeight: 'bold',\n    '::after': {\n      backgroundColor: 'var(--setting-check-icon-color, var(--primary-highlight, white))',\n      maskImage: `url(${icon.check})`,\n    },\n  },\n}\n\n// TODO some of styles are for older version UI design, can be simplified\nconst desktopStyle = {\n  overlay: {\n    position: 'absolute',\n    zIndex: '1',\n    bottom: 'calc(5em + var(--bottom-spacing, 0rem))',\n    right: '1rem',\n    display: 'flex',\n    alignItems: 'flex-end',\n    width: '18rem',\n    height: 'calc(100% - 8rem - var(--bottom-spacing, 0rem))',\n    outline: 'none',\n    opacity: '1',\n    transform: 'translateY(-100vh)',\n  },\n  open: {\n    position: 'fixed',\n    opacity: '1',\n    transform: 'translateY(0)',\n    transition: 'opacity 0.2s ease, transform 0s',\n  },\n  container: {\n    ...mobileStyle.container,\n    flex: '0 var(--setting-ui-width, calc(100% - 2rem))',\n    maxHeight: '100%',\n  },\n  head: {\n    ...mobileStyle.head,\n    color: 'var(--setting-head-color, inherit)',\n    background: 'var(--setting-head-background, rgba(22, 28, 36, 0.8))',\n  },\n  row: {\n    ...mobileStyle.row,\n    background: 'rgba(22, 28, 36, 0.8)',\n    '::after': {\n      order: 'var(--setting-check-icon-order, 0)',\n      ...mobileStyle.row['::after'],\n      marginLeft: '0',\n      marginRight: '0.5em',\n    },\n  },\n  hasOptions: {\n    '::after': {\n      order: '0',\n      backgroundImage: `url(${icon.arrowTop})`,\n      transform: 'rotate(90deg)',\n    },\n  },\n  selected: {\n    color: 'white',\n    fontWeight: 'bold',\n    '::after': {\n      backgroundColor:\n        'var(--setting-check-icon-color, var(--primary-highlight, white))',\n      maskImage: `url(${icon.check})`,\n    },\n  },\n}\n\nconst MenuItemText = ({text = ''}) => (\n  <FormattedMessage\n    id={text}\n    defaultMessage={\n      <FormattedMessage id={`KKS.SETTING.${text}`} defaultMessage={text} />\n    }\n  />\n)\n\nconst LoopSwitch = ({style, checked, onChange}) => (\n  <li\n    role=\"menuitem\"\n    css={[style, {'::before': {display: 'none'}, '::after': {display: 'none'}}]}\n    onClick={() => {\n      onChange({name: 'loop', value: !checked, keepOpen: true})\n    }}\n  >\n    <MenuItemText text=\"KKS.PLAYER.LOOP\" />\n    <div css={mobileStyle.space} />\n    <Switch checked={checked} />\n  </li>\n)\n\nconst CloseButton = props => (\n  <button\n    type=\"button\"\n    aria-label=\"Close Settings\"\n    css={mobileStyle.dismiss}\n    {...props}\n  />\n)\n\nconst DesktopContainer = ({open, style, children, onClose, ...rest}) => (\n  <div css={[style, open && desktopStyle.open]} {...rest}>\n    {children}\n  </div>\n)\n\nconst matchValue = (a, b) => a === b || (a?.id ?? a) === (b?.id ?? b)\n\nconst Settings = ({\n  open,\n  values,\n  sections: originalSections,\n  type: uiType,\n  closeBy,\n  buttonPosition,\n  enabledSettingSections = {\n    audio: true,\n    subtitles: true,\n    quality: true,\n    speed: true,\n    loop: true,\n  }, \n  slots = {\n    root: uiType === 'desktop' ? DesktopContainer : SwipeableDrawer,\n  },\n  onChange,\n  onOpen,\n  onClose,\n}) => {\n  const sections = originalSections.filter(section => enabledSettingSections[section.name])\n  const Container = slots.root\n  const ButtonWrap =\n    buttonPosition === 'top-right' ? TitleBarExtension : FunctionBarExtension\n  const commonStyle = uiType === 'desktop' ? desktopStyle : mobileStyle\n  const [path, setPath] = useState('/')\n  useEffect(() => {\n    if (!open) {\n      setPath('/')\n    }\n  }, [open])\n\n  const ref = useOnclickOutside(\n    () => {\n      if (open && uiType === 'desktop') {\n        onClose()\n      }\n    },\n    {eventTypes: ['click']}\n  )\n  const currentSection = sections.find(it => path === `/${it.name}`)\n  const menu =\n    path === '/'\n      ? {\n          title: 'KKS.SETTING',\n          items: sections.map(({type, name, title, items = []}) => ({\n            type,\n            link: `/${name}`,\n            label: title,\n            value:\n              items.find(item => matchValue(item.value, values[name]))?.label ||\n              values[name],\n          })),\n        }\n      : {\n          title: currentSection.title,\n          items: currentSection.items.map(({value, label}) => ({\n            label,\n            checked: matchValue(values[currentSection.name], value),\n            data: value,\n          })),\n          previous: '/',\n        }\n  const navigate = dest => requestAnimationFrame(() => setPath(dest))\n  return (\n    sections.length > 0 && (\n    <>\n      <ButtonWrap position=\"right\">\n        <Button\n          startIcon=\"setting\"\n          style={{order: 2}}\n          title=\"KKS.SETTING\"\n            onClick={event =>\n              setTimeout(() => (open ? onClose(event) : onOpen(event)), 1)\n            }\n        />\n      </ButtonWrap>\n      <Container style={commonStyle.overlay} open={open} onClose={onClose}>\n        <ul role=\"menu\" ref={ref} css={commonStyle.container}>\n          <div css={commonStyle.head}>\n            {menu.previous ? (\n              <Button startIcon=\"back1\" onClick={() => navigate('/')} />\n            ) : (\n              uiType !== 'desktop' &&\n              closeBy === 'button' && <CloseButton onClick={onClose} />\n            )}\n            <FormattedMessage id={menu.title} />\n          </div>\n          {menu.items.map(({type, label, link, value, data, checked}) =>\n            type === 'switch' ? (\n              <LoopSwitch\n                style={commonStyle.row}\n                checked={values.loop}\n                onChange={() =>\n                    onChange({\n                      name: 'loop',\n                      value: !values.loop,\n                      keepOpen: true,\n                    })\n                }\n              />\n            ) : (\n              <li\n                role={link ? 'menuitem' : 'menuitemradio'}\n                aria-checked={checked}\n                css={[\n                  commonStyle.row,\n                  link && commonStyle.hasOptions,\n                  checked && commonStyle.selected,\n                ]}\n                onClick={() =>\n                  link\n                    ? navigate(link)\n                    : onChange({name: currentSection.name, value: data})\n                }\n              >\n                <MenuItemText text={label} />\n                <div css={mobileStyle.space} />\n                {value && <MenuItemText text={value.toString()} />}\n              </li>\n            )\n          )}\n        </ul>\n      </Container>\n    </>\n    )\n  )\n}\n\nexport default Settings\n"]} */"],
|
|
4747
4692
|
onClick: () => {
|
|
4748
4693
|
onChange({
|
|
4749
4694
|
name: 'loop',
|
|
@@ -4774,7 +4719,7 @@ const DesktopContainer$1 = ({
|
|
|
4774
4719
|
onClose,
|
|
4775
4720
|
...rest
|
|
4776
4721
|
}) => jsx$1("div", {
|
|
4777
|
-
css: [style, open && desktopStyle$1.open, process.env.NODE_ENV === "production" ? "" : ";label:DesktopContainer;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Settings.js"],"names":[],"mappings":"AAmMO","file":"Settings.js","sourcesContent":["/* @jsxImportSource @emotion/react */\n/* eslint-disable jsx-a11y/no-noninteractive-element-interactions */\nimport {useState, useEffect} from 'react'\nimport useOnclickOutside from 'react-cool-onclickoutside'\n\nimport icon from 'style/icon'\nimport {FormattedMessage} from 'context/I18n'\nimport {Button, Switch} from './buttons'\nimport SwipeableDrawer from './SwipeableDrawer'\nimport {FunctionBarExtension, TitleBarExtension} from './uiExtensions'\n\nconst ulReset = {\n  marginBlockStart: 0,\n  marginBlockEnd: 0,\n  paddingInlineStart: 0,\n}\n\nconst mobileStyle = {\n  head: {\n    position: 'sticky',\n    zIndex: '1',\n    top: '0',\n    display: 'flex',\n    alignItems: 'center',\n    padding: '1rem 1.5rem',\n    color: 'white',\n    background: '#161C24',\n    fontSize: '16px',\n    fontWeight: 'bold',\n    button: {\n      marginRight: '1rem',\n      padding: '0',\n      width: '1.5rem',\n      height: '1.5rem',\n      border: 'none',\n    },\n  },\n  overlay: {\n    '> div': {\n      // drawer container\n      background: 'var(--setting-ui-background, #161C24)',\n      maxHeight: 'calc(100% - var(--setting-ui-vertical-padding, 2rem))',\n    },\n  },\n  container: {\n    ...ulReset,\n    // TODO use dvh once we may drop iOS 14\n    maxHeight: 'calc(90vh - var(--setting-ui-vertical-padding, 2rem))',\n    color: '#ccc',\n    whiteSpace: 'nowrap',\n    borderRadius: '4px',\n    userSelect: 'none',\n    overflow: 'auto',\n    overscrollBehavior: 'contain',\n  },\n  title: {\n    padding: '12px 18px',\n  },\n  dismiss: {\n    background: `center / 1rem no-repeat url(${icon.close}), transparent`,\n  },\n  back: {\n    background: '#ccc',\n    maskImage: `var(--icon-setting-back, url(\"${icon.back}\"))`,\n    maskSize: 'contain',\n  },\n  row: {\n    cursor: 'pointer',\n    display: 'flex',\n    position: 'relative',\n    padding: '1rem 1.5rem',\n    fontSize: '16px',\n    background: '#161C24',\n    '::after': {\n      content: '\" \"',\n      marginLeft: '1rem',\n      width: '20px',\n      height: '20px',\n      display: 'inline-block',\n      color: 'white',\n      backgroundPosition: 'center',\n      backgroundSize: 'cover',\n    },\n  },\n  space: {\n    flex: '1',\n  },\n  hasOptions: {\n    '::after': {\n      backgroundImage: `url(${icon.arrowTop})`,\n      transform: 'rotate(90deg)',\n    },\n  },\n  selected: {\n    color: 'white',\n    fontWeight: 'bold',\n    '::after': {\n      backgroundColor: 'var(--setting-check-icon-color, var(--primary-highlight, white))',\n      maskImage: `url(${icon.check})`,\n    },\n  },\n}\n\n// TODO some of styles are for older version UI design, can be simplified\nconst desktopStyle = {\n  overlay: {\n    position: 'absolute',\n    zIndex: '1',\n    bottom: 'calc(5em + var(--bottom-spacing, 0rem))',\n    right: '1rem',\n    display: 'flex',\n    alignItems: 'flex-end',\n    width: '18rem',\n    height: 'calc(100% - 8rem - var(--bottom-spacing, 0rem))',\n    outline: 'none',\n    opacity: '1',\n    transform: 'translateY(-100vh)',\n  },\n  open: {\n    position: 'fixed',\n    opacity: '1',\n    transform: 'translateY(0)',\n    transition: 'opacity 0.2s ease, transform 0s',\n  },\n  container: {\n    ...mobileStyle.container,\n    flex: '0 var(--setting-ui-width, calc(100% - 2rem))',\n    maxHeight: '100%',\n  },\n  head: {\n    ...mobileStyle.head,\n    color: 'var(--setting-head-color, inherit)',\n    background: 'var(--setting-head-background, rgba(22, 28, 36, 0.8))',\n  },\n  row: {\n    ...mobileStyle.row,\n    background: 'rgba(22, 28, 36, 0.8)',\n    '::after': {\n      order: 'var(--setting-check-icon-order, 0)',\n      ...mobileStyle.row['::after'],\n      marginLeft: '0',\n      marginRight: '0.5em',\n    },\n  },\n  hasOptions: {\n    '::after': {\n      order: '0',\n      backgroundImage: `url(${icon.arrowTop})`,\n      transform: 'rotate(90deg)',\n    },\n  },\n  selected: {\n    color: 'white',\n    fontWeight: 'bold',\n    '::after': {\n      backgroundColor:\n        'var(--setting-check-icon-color, var(--primary-highlight, white))',\n      maskImage: `url(${icon.check})`,\n    },\n  },\n}\n\nconst MenuItemText = ({text = ''}) => (\n  <FormattedMessage\n    id={text}\n    defaultMessage={\n      <FormattedMessage id={`KKS.SETTING.${text}`} defaultMessage={text} />\n    }\n  />\n)\n\nconst LoopSwitch = ({style, checked, onChange}) => (\n  <li\n    role=\"menuitem\"\n    css={[style, {'::before': {display: 'none'}, '::after': {display: 'none'}}]}\n    onClick={() => {\n      onChange({name: 'loop', value: !checked, keepOpen: true})\n    }}\n  >\n    <MenuItemText text=\"KKS.PLAYER.LOOP\" />\n    <div css={mobileStyle.space} />\n    <Switch checked={checked} />\n  </li>\n)\n\nconst CloseButton = props => (\n  <button\n    type=\"button\"\n    aria-label=\"Close Settings\"\n    css={mobileStyle.dismiss}\n    {...props}\n  />\n)\n\nconst DesktopContainer = ({open, style, children, onClose, ...rest}) => (\n  <div css={[style, open && desktopStyle.open]} {...rest}>\n    {children}\n  </div>\n)\n\nconst matchValue = (a, b) => a === b || (a?.id ?? a) === (b?.id ?? b)\n\nconst Settings = ({\n  open,\n  values,\n  sections: originalSections,\n  type: uiType,\n  closeBy,\n  buttonPosition,\n  enabledSettingSections = {\n    audio: true,\n    subtitles: true,\n    quality: true,\n    speed: true,\n    loop: true,\n  }, \n  slots = {\n    root: uiType === 'desktop' ? DesktopContainer : SwipeableDrawer,\n  },\n  onChange,\n  onOpen,\n  onClose,\n}) => {\n  const sections = originalSections.filter(section => enabledSettingSections[section.name])\n  const Container = slots.root\n  const ButtonWrap =\n    buttonPosition === 'top-right' ? TitleBarExtension : FunctionBarExtension\n  const commonStyle = uiType === 'desktop' ? desktopStyle : mobileStyle\n  const [path, setPath] = useState('/')\n  useEffect(() => {\n    if (!open) {\n      setPath('/')\n    }\n  }, [open])\n\n  const ref = useOnclickOutside(\n    () => {\n      if (open && uiType === 'desktop') {\n        onClose()\n      }\n    },\n    {eventTypes: ['click']}\n  )\n  const currentSection = sections.find(it => path === `/${it.name}`)\n  const menu =\n    path === '/'\n      ? {\n          title: 'KKS.SETTING',\n          items: sections.map(({type, name, title, items = []}) => ({\n            type,\n            link: `/${name}`,\n            label: title,\n            value:\n              items.find(item => matchValue(item.value, values[name]))?.label ||\n              values[name],\n          })),\n        }\n      : {\n          title: currentSection.title,\n          items: currentSection.items.map(({value, label}) => ({\n            label,\n            checked: matchValue(values[currentSection.name], value),\n            data: value,\n          })),\n          previous: '/',\n        }\n  const navigate = dest => requestAnimationFrame(() => setPath(dest))\n  return (\n    sections.length > 0 && (\n    <>\n      <ButtonWrap position=\"right\">\n        <Button\n          startIcon=\"setting\"\n          title=\"KKS.SETTING\"\n            onClick={event =>\n              setTimeout(() => (open ? onClose(event) : onOpen(event)), 1)\n            }\n        />\n      </ButtonWrap>\n      <Container style={commonStyle.overlay} open={open} onClose={onClose}>\n        <ul role=\"menu\" ref={ref} css={commonStyle.container}>\n          <div css={commonStyle.head}>\n            {menu.previous ? (\n              <Button startIcon=\"back1\" onClick={() => navigate('/')} />\n            ) : (\n              uiType !== 'desktop' &&\n              closeBy === 'button' && <CloseButton onClick={onClose} />\n            )}\n            <FormattedMessage id={menu.title} />\n          </div>\n          {menu.items.map(({type, label, link, value, data, checked}) =>\n            type === 'switch' ? (\n              <LoopSwitch\n                style={commonStyle.row}\n                checked={values.loop}\n                onChange={() =>\n                    onChange({\n                      name: 'loop',\n                      value: !values.loop,\n                      keepOpen: true,\n                    })\n                }\n              />\n            ) : (\n              <li\n                role={link ? 'menuitem' : 'menuitemradio'}\n                aria-checked={checked}\n                css={[\n                  commonStyle.row,\n                  link && commonStyle.hasOptions,\n                  checked && commonStyle.selected,\n                ]}\n                onClick={() =>\n                  link\n                    ? navigate(link)\n                    : onChange({name: currentSection.name, value: data})\n                }\n              >\n                <MenuItemText text={label} />\n                <div css={mobileStyle.space} />\n                {value && <MenuItemText text={value.toString()} />}\n              </li>\n            )\n          )}\n        </ul>\n      </Container>\n    </>\n    )\n  )\n}\n\nexport default Settings\n"]} */"],
|
|
4722
|
+
css: [style, open && desktopStyle$1.open, process.env.NODE_ENV === "production" ? "" : ";label:DesktopContainer;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Settings.js"],"names":[],"mappings":"AAmMO","file":"Settings.js","sourcesContent":["/* @jsxImportSource @emotion/react */\n/* eslint-disable jsx-a11y/no-noninteractive-element-interactions */\nimport {useState, useEffect} from 'react'\nimport useOnclickOutside from 'react-cool-onclickoutside'\n\nimport icon from 'style/icon'\nimport {FormattedMessage} from 'context/I18n'\nimport {Button, Switch} from './buttons'\nimport SwipeableDrawer from './SwipeableDrawer'\nimport {FunctionBarExtension, TitleBarExtension} from './uiExtensions'\n\nconst ulReset = {\n  marginBlockStart: 0,\n  marginBlockEnd: 0,\n  paddingInlineStart: 0,\n}\n\nconst mobileStyle = {\n  head: {\n    position: 'sticky',\n    zIndex: '1',\n    top: '0',\n    display: 'flex',\n    alignItems: 'center',\n    padding: '1rem 1.5rem',\n    color: 'white',\n    background: '#161C24',\n    fontSize: '16px',\n    fontWeight: 'bold',\n    button: {\n      marginRight: '1rem',\n      padding: '0',\n      width: '1.5rem',\n      height: '1.5rem',\n      border: 'none',\n    },\n  },\n  overlay: {\n    '> div': {\n      // drawer container\n      background: 'var(--setting-ui-background, #161C24)',\n      maxHeight: 'calc(100% - var(--setting-ui-vertical-padding, 2rem))',\n    },\n  },\n  container: {\n    ...ulReset,\n    // TODO use dvh once we may drop iOS 14\n    maxHeight: 'calc(90vh - var(--setting-ui-vertical-padding, 2rem))',\n    color: '#ccc',\n    whiteSpace: 'nowrap',\n    borderRadius: '4px',\n    userSelect: 'none',\n    overflow: 'auto',\n    overscrollBehavior: 'contain',\n  },\n  title: {\n    padding: '12px 18px',\n  },\n  dismiss: {\n    background: `center / 1rem no-repeat url(${icon.close}), transparent`,\n  },\n  back: {\n    background: '#ccc',\n    maskImage: `var(--icon-setting-back, url(\"${icon.back}\"))`,\n    maskSize: 'contain',\n  },\n  row: {\n    cursor: 'pointer',\n    display: 'flex',\n    position: 'relative',\n    padding: '1rem 1.5rem',\n    fontSize: '16px',\n    background: '#161C24',\n    '::after': {\n      content: '\" \"',\n      marginLeft: '1rem',\n      width: '20px',\n      height: '20px',\n      display: 'inline-block',\n      color: 'white',\n      backgroundPosition: 'center',\n      backgroundSize: 'cover',\n    },\n  },\n  space: {\n    flex: '1',\n  },\n  hasOptions: {\n    '::after': {\n      backgroundImage: `url(${icon.arrowTop})`,\n      transform: 'rotate(90deg)',\n    },\n  },\n  selected: {\n    color: 'white',\n    fontWeight: 'bold',\n    '::after': {\n      backgroundColor: 'var(--setting-check-icon-color, var(--primary-highlight, white))',\n      maskImage: `url(${icon.check})`,\n    },\n  },\n}\n\n// TODO some of styles are for older version UI design, can be simplified\nconst desktopStyle = {\n  overlay: {\n    position: 'absolute',\n    zIndex: '1',\n    bottom: 'calc(5em + var(--bottom-spacing, 0rem))',\n    right: '1rem',\n    display: 'flex',\n    alignItems: 'flex-end',\n    width: '18rem',\n    height: 'calc(100% - 8rem - var(--bottom-spacing, 0rem))',\n    outline: 'none',\n    opacity: '1',\n    transform: 'translateY(-100vh)',\n  },\n  open: {\n    position: 'fixed',\n    opacity: '1',\n    transform: 'translateY(0)',\n    transition: 'opacity 0.2s ease, transform 0s',\n  },\n  container: {\n    ...mobileStyle.container,\n    flex: '0 var(--setting-ui-width, calc(100% - 2rem))',\n    maxHeight: '100%',\n  },\n  head: {\n    ...mobileStyle.head,\n    color: 'var(--setting-head-color, inherit)',\n    background: 'var(--setting-head-background, rgba(22, 28, 36, 0.8))',\n  },\n  row: {\n    ...mobileStyle.row,\n    background: 'rgba(22, 28, 36, 0.8)',\n    '::after': {\n      order: 'var(--setting-check-icon-order, 0)',\n      ...mobileStyle.row['::after'],\n      marginLeft: '0',\n      marginRight: '0.5em',\n    },\n  },\n  hasOptions: {\n    '::after': {\n      order: '0',\n      backgroundImage: `url(${icon.arrowTop})`,\n      transform: 'rotate(90deg)',\n    },\n  },\n  selected: {\n    color: 'white',\n    fontWeight: 'bold',\n    '::after': {\n      backgroundColor:\n        'var(--setting-check-icon-color, var(--primary-highlight, white))',\n      maskImage: `url(${icon.check})`,\n    },\n  },\n}\n\nconst MenuItemText = ({text = ''}) => (\n  <FormattedMessage\n    id={text}\n    defaultMessage={\n      <FormattedMessage id={`KKS.SETTING.${text}`} defaultMessage={text} />\n    }\n  />\n)\n\nconst LoopSwitch = ({style, checked, onChange}) => (\n  <li\n    role=\"menuitem\"\n    css={[style, {'::before': {display: 'none'}, '::after': {display: 'none'}}]}\n    onClick={() => {\n      onChange({name: 'loop', value: !checked, keepOpen: true})\n    }}\n  >\n    <MenuItemText text=\"KKS.PLAYER.LOOP\" />\n    <div css={mobileStyle.space} />\n    <Switch checked={checked} />\n  </li>\n)\n\nconst CloseButton = props => (\n  <button\n    type=\"button\"\n    aria-label=\"Close Settings\"\n    css={mobileStyle.dismiss}\n    {...props}\n  />\n)\n\nconst DesktopContainer = ({open, style, children, onClose, ...rest}) => (\n  <div css={[style, open && desktopStyle.open]} {...rest}>\n    {children}\n  </div>\n)\n\nconst matchValue = (a, b) => a === b || (a?.id ?? a) === (b?.id ?? b)\n\nconst Settings = ({\n  open,\n  values,\n  sections: originalSections,\n  type: uiType,\n  closeBy,\n  buttonPosition,\n  enabledSettingSections = {\n    audio: true,\n    subtitles: true,\n    quality: true,\n    speed: true,\n    loop: true,\n  }, \n  slots = {\n    root: uiType === 'desktop' ? DesktopContainer : SwipeableDrawer,\n  },\n  onChange,\n  onOpen,\n  onClose,\n}) => {\n  const sections = originalSections.filter(section => enabledSettingSections[section.name])\n  const Container = slots.root\n  const ButtonWrap =\n    buttonPosition === 'top-right' ? TitleBarExtension : FunctionBarExtension\n  const commonStyle = uiType === 'desktop' ? desktopStyle : mobileStyle\n  const [path, setPath] = useState('/')\n  useEffect(() => {\n    if (!open) {\n      setPath('/')\n    }\n  }, [open])\n\n  const ref = useOnclickOutside(\n    () => {\n      if (open && uiType === 'desktop') {\n        onClose()\n      }\n    },\n    {eventTypes: ['click']}\n  )\n  const currentSection = sections.find(it => path === `/${it.name}`)\n  const menu =\n    path === '/'\n      ? {\n          title: 'KKS.SETTING',\n          items: sections.map(({type, name, title, items = []}) => ({\n            type,\n            link: `/${name}`,\n            label: title,\n            value:\n              items.find(item => matchValue(item.value, values[name]))?.label ||\n              values[name],\n          })),\n        }\n      : {\n          title: currentSection.title,\n          items: currentSection.items.map(({value, label}) => ({\n            label,\n            checked: matchValue(values[currentSection.name], value),\n            data: value,\n          })),\n          previous: '/',\n        }\n  const navigate = dest => requestAnimationFrame(() => setPath(dest))\n  return (\n    sections.length > 0 && (\n    <>\n      <ButtonWrap position=\"right\">\n        <Button\n          startIcon=\"setting\"\n          style={{order: 2}}\n          title=\"KKS.SETTING\"\n            onClick={event =>\n              setTimeout(() => (open ? onClose(event) : onOpen(event)), 1)\n            }\n        />\n      </ButtonWrap>\n      <Container style={commonStyle.overlay} open={open} onClose={onClose}>\n        <ul role=\"menu\" ref={ref} css={commonStyle.container}>\n          <div css={commonStyle.head}>\n            {menu.previous ? (\n              <Button startIcon=\"back1\" onClick={() => navigate('/')} />\n            ) : (\n              uiType !== 'desktop' &&\n              closeBy === 'button' && <CloseButton onClick={onClose} />\n            )}\n            <FormattedMessage id={menu.title} />\n          </div>\n          {menu.items.map(({type, label, link, value, data, checked}) =>\n            type === 'switch' ? (\n              <LoopSwitch\n                style={commonStyle.row}\n                checked={values.loop}\n                onChange={() =>\n                    onChange({\n                      name: 'loop',\n                      value: !values.loop,\n                      keepOpen: true,\n                    })\n                }\n              />\n            ) : (\n              <li\n                role={link ? 'menuitem' : 'menuitemradio'}\n                aria-checked={checked}\n                css={[\n                  commonStyle.row,\n                  link && commonStyle.hasOptions,\n                  checked && commonStyle.selected,\n                ]}\n                onClick={() =>\n                  link\n                    ? navigate(link)\n                    : onChange({name: currentSection.name, value: data})\n                }\n              >\n                <MenuItemText text={label} />\n                <div css={mobileStyle.space} />\n                {value && <MenuItemText text={value.toString()} />}\n              </li>\n            )\n          )}\n        </ul>\n      </Container>\n    </>\n    )\n  )\n}\n\nexport default Settings\n"]} */"],
|
|
4778
4723
|
...rest,
|
|
4779
4724
|
children: children
|
|
4780
4725
|
});
|
|
@@ -4861,6 +4806,9 @@ const Settings = ({
|
|
|
4861
4806
|
position: "right",
|
|
4862
4807
|
children: jsx$1(Button, {
|
|
4863
4808
|
startIcon: "setting",
|
|
4809
|
+
style: {
|
|
4810
|
+
order: 2
|
|
4811
|
+
},
|
|
4864
4812
|
title: "KKS.SETTING",
|
|
4865
4813
|
onClick: event => setTimeout(() => open ? onClose(event) : onOpen(event), 1)
|
|
4866
4814
|
})
|
|
@@ -4900,7 +4848,7 @@ const Settings = ({
|
|
|
4900
4848
|
}) : jsxs("li", {
|
|
4901
4849
|
role: link ? 'menuitem' : 'menuitemradio',
|
|
4902
4850
|
"aria-checked": checked,
|
|
4903
|
-
css: [commonStyle.row, link && commonStyle.hasOptions, checked && commonStyle.selected, process.env.NODE_ENV === "production" ? "" : ";label:Settings;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Settings.js"],"names":[],"mappings":"AAmTgB","file":"Settings.js","sourcesContent":["/* @jsxImportSource @emotion/react */\n/* eslint-disable jsx-a11y/no-noninteractive-element-interactions */\nimport {useState, useEffect} from 'react'\nimport useOnclickOutside from 'react-cool-onclickoutside'\n\nimport icon from 'style/icon'\nimport {FormattedMessage} from 'context/I18n'\nimport {Button, Switch} from './buttons'\nimport SwipeableDrawer from './SwipeableDrawer'\nimport {FunctionBarExtension, TitleBarExtension} from './uiExtensions'\n\nconst ulReset = {\n  marginBlockStart: 0,\n  marginBlockEnd: 0,\n  paddingInlineStart: 0,\n}\n\nconst mobileStyle = {\n  head: {\n    position: 'sticky',\n    zIndex: '1',\n    top: '0',\n    display: 'flex',\n    alignItems: 'center',\n    padding: '1rem 1.5rem',\n    color: 'white',\n    background: '#161C24',\n    fontSize: '16px',\n    fontWeight: 'bold',\n    button: {\n      marginRight: '1rem',\n      padding: '0',\n      width: '1.5rem',\n      height: '1.5rem',\n      border: 'none',\n    },\n  },\n  overlay: {\n    '> div': {\n      // drawer container\n      background: 'var(--setting-ui-background, #161C24)',\n      maxHeight: 'calc(100% - var(--setting-ui-vertical-padding, 2rem))',\n    },\n  },\n  container: {\n    ...ulReset,\n    // TODO use dvh once we may drop iOS 14\n    maxHeight: 'calc(90vh - var(--setting-ui-vertical-padding, 2rem))',\n    color: '#ccc',\n    whiteSpace: 'nowrap',\n    borderRadius: '4px',\n    userSelect: 'none',\n    overflow: 'auto',\n    overscrollBehavior: 'contain',\n  },\n  title: {\n    padding: '12px 18px',\n  },\n  dismiss: {\n    background: `center / 1rem no-repeat url(${icon.close}), transparent`,\n  },\n  back: {\n    background: '#ccc',\n    maskImage: `var(--icon-setting-back, url(\"${icon.back}\"))`,\n    maskSize: 'contain',\n  },\n  row: {\n    cursor: 'pointer',\n    display: 'flex',\n    position: 'relative',\n    padding: '1rem 1.5rem',\n    fontSize: '16px',\n    background: '#161C24',\n    '::after': {\n      content: '\" \"',\n      marginLeft: '1rem',\n      width: '20px',\n      height: '20px',\n      display: 'inline-block',\n      color: 'white',\n      backgroundPosition: 'center',\n      backgroundSize: 'cover',\n    },\n  },\n  space: {\n    flex: '1',\n  },\n  hasOptions: {\n    '::after': {\n      backgroundImage: `url(${icon.arrowTop})`,\n      transform: 'rotate(90deg)',\n    },\n  },\n  selected: {\n    color: 'white',\n    fontWeight: 'bold',\n    '::after': {\n      backgroundColor: 'var(--setting-check-icon-color, var(--primary-highlight, white))',\n      maskImage: `url(${icon.check})`,\n    },\n  },\n}\n\n// TODO some of styles are for older version UI design, can be simplified\nconst desktopStyle = {\n  overlay: {\n    position: 'absolute',\n    zIndex: '1',\n    bottom: 'calc(5em + var(--bottom-spacing, 0rem))',\n    right: '1rem',\n    display: 'flex',\n    alignItems: 'flex-end',\n    width: '18rem',\n    height: 'calc(100% - 8rem - var(--bottom-spacing, 0rem))',\n    outline: 'none',\n    opacity: '1',\n    transform: 'translateY(-100vh)',\n  },\n  open: {\n    position: 'fixed',\n    opacity: '1',\n    transform: 'translateY(0)',\n    transition: 'opacity 0.2s ease, transform 0s',\n  },\n  container: {\n    ...mobileStyle.container,\n    flex: '0 var(--setting-ui-width, calc(100% - 2rem))',\n    maxHeight: '100%',\n  },\n  head: {\n    ...mobileStyle.head,\n    color: 'var(--setting-head-color, inherit)',\n    background: 'var(--setting-head-background, rgba(22, 28, 36, 0.8))',\n  },\n  row: {\n    ...mobileStyle.row,\n    background: 'rgba(22, 28, 36, 0.8)',\n    '::after': {\n      order: 'var(--setting-check-icon-order, 0)',\n      ...mobileStyle.row['::after'],\n      marginLeft: '0',\n      marginRight: '0.5em',\n    },\n  },\n  hasOptions: {\n    '::after': {\n      order: '0',\n      backgroundImage: `url(${icon.arrowTop})`,\n      transform: 'rotate(90deg)',\n    },\n  },\n  selected: {\n    color: 'white',\n    fontWeight: 'bold',\n    '::after': {\n      backgroundColor:\n        'var(--setting-check-icon-color, var(--primary-highlight, white))',\n      maskImage: `url(${icon.check})`,\n    },\n  },\n}\n\nconst MenuItemText = ({text = ''}) => (\n  <FormattedMessage\n    id={text}\n    defaultMessage={\n      <FormattedMessage id={`KKS.SETTING.${text}`} defaultMessage={text} />\n    }\n  />\n)\n\nconst LoopSwitch = ({style, checked, onChange}) => (\n  <li\n    role=\"menuitem\"\n    css={[style, {'::before': {display: 'none'}, '::after': {display: 'none'}}]}\n    onClick={() => {\n      onChange({name: 'loop', value: !checked, keepOpen: true})\n    }}\n  >\n    <MenuItemText text=\"KKS.PLAYER.LOOP\" />\n    <div css={mobileStyle.space} />\n    <Switch checked={checked} />\n  </li>\n)\n\nconst CloseButton = props => (\n  <button\n    type=\"button\"\n    aria-label=\"Close Settings\"\n    css={mobileStyle.dismiss}\n    {...props}\n  />\n)\n\nconst DesktopContainer = ({open, style, children, onClose, ...rest}) => (\n  <div css={[style, open && desktopStyle.open]} {...rest}>\n    {children}\n  </div>\n)\n\nconst matchValue = (a, b) => a === b || (a?.id ?? a) === (b?.id ?? b)\n\nconst Settings = ({\n  open,\n  values,\n  sections: originalSections,\n  type: uiType,\n  closeBy,\n  buttonPosition,\n  enabledSettingSections = {\n    audio: true,\n    subtitles: true,\n    quality: true,\n    speed: true,\n    loop: true,\n  }, \n  slots = {\n    root: uiType === 'desktop' ? DesktopContainer : SwipeableDrawer,\n  },\n  onChange,\n  onOpen,\n  onClose,\n}) => {\n  const sections = originalSections.filter(section => enabledSettingSections[section.name])\n  const Container = slots.root\n  const ButtonWrap =\n    buttonPosition === 'top-right' ? TitleBarExtension : FunctionBarExtension\n  const commonStyle = uiType === 'desktop' ? desktopStyle : mobileStyle\n  const [path, setPath] = useState('/')\n  useEffect(() => {\n    if (!open) {\n      setPath('/')\n    }\n  }, [open])\n\n  const ref = useOnclickOutside(\n    () => {\n      if (open && uiType === 'desktop') {\n        onClose()\n      }\n    },\n    {eventTypes: ['click']}\n  )\n  const currentSection = sections.find(it => path === `/${it.name}`)\n  const menu =\n    path === '/'\n      ? {\n          title: 'KKS.SETTING',\n          items: sections.map(({type, name, title, items = []}) => ({\n            type,\n            link: `/${name}`,\n            label: title,\n            value:\n              items.find(item => matchValue(item.value, values[name]))?.label ||\n              values[name],\n          })),\n        }\n      : {\n          title: currentSection.title,\n          items: currentSection.items.map(({value, label}) => ({\n            label,\n            checked: matchValue(values[currentSection.name], value),\n            data: value,\n          })),\n          previous: '/',\n        }\n  const navigate = dest => requestAnimationFrame(() => setPath(dest))\n  return (\n    sections.length > 0 && (\n    <>\n      <ButtonWrap position=\"right\">\n        <Button\n          startIcon=\"setting\"\n          title=\"KKS.SETTING\"\n            onClick={event =>\n              setTimeout(() => (open ? onClose(event) : onOpen(event)), 1)\n            }\n        />\n      </ButtonWrap>\n      <Container style={commonStyle.overlay} open={open} onClose={onClose}>\n        <ul role=\"menu\" ref={ref} css={commonStyle.container}>\n          <div css={commonStyle.head}>\n            {menu.previous ? (\n              <Button startIcon=\"back1\" onClick={() => navigate('/')} />\n            ) : (\n              uiType !== 'desktop' &&\n              closeBy === 'button' && <CloseButton onClick={onClose} />\n            )}\n            <FormattedMessage id={menu.title} />\n          </div>\n          {menu.items.map(({type, label, link, value, data, checked}) =>\n            type === 'switch' ? (\n              <LoopSwitch\n                style={commonStyle.row}\n                checked={values.loop}\n                onChange={() =>\n                    onChange({\n                      name: 'loop',\n                      value: !values.loop,\n                      keepOpen: true,\n                    })\n                }\n              />\n            ) : (\n              <li\n                role={link ? 'menuitem' : 'menuitemradio'}\n                aria-checked={checked}\n                css={[\n                  commonStyle.row,\n                  link && commonStyle.hasOptions,\n                  checked && commonStyle.selected,\n                ]}\n                onClick={() =>\n                  link\n                    ? navigate(link)\n                    : onChange({name: currentSection.name, value: data})\n                }\n              >\n                <MenuItemText text={label} />\n                <div css={mobileStyle.space} />\n                {value && <MenuItemText text={value.toString()} />}\n              </li>\n            )\n          )}\n        </ul>\n      </Container>\n    </>\n    )\n  )\n}\n\nexport default Settings\n"]} */"],
|
|
4851
|
+
css: [commonStyle.row, link && commonStyle.hasOptions, checked && commonStyle.selected, process.env.NODE_ENV === "production" ? "" : ";label:Settings;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Settings.js"],"names":[],"mappings":"AAoTgB","file":"Settings.js","sourcesContent":["/* @jsxImportSource @emotion/react */\n/* eslint-disable jsx-a11y/no-noninteractive-element-interactions */\nimport {useState, useEffect} from 'react'\nimport useOnclickOutside from 'react-cool-onclickoutside'\n\nimport icon from 'style/icon'\nimport {FormattedMessage} from 'context/I18n'\nimport {Button, Switch} from './buttons'\nimport SwipeableDrawer from './SwipeableDrawer'\nimport {FunctionBarExtension, TitleBarExtension} from './uiExtensions'\n\nconst ulReset = {\n  marginBlockStart: 0,\n  marginBlockEnd: 0,\n  paddingInlineStart: 0,\n}\n\nconst mobileStyle = {\n  head: {\n    position: 'sticky',\n    zIndex: '1',\n    top: '0',\n    display: 'flex',\n    alignItems: 'center',\n    padding: '1rem 1.5rem',\n    color: 'white',\n    background: '#161C24',\n    fontSize: '16px',\n    fontWeight: 'bold',\n    button: {\n      marginRight: '1rem',\n      padding: '0',\n      width: '1.5rem',\n      height: '1.5rem',\n      border: 'none',\n    },\n  },\n  overlay: {\n    '> div': {\n      // drawer container\n      background: 'var(--setting-ui-background, #161C24)',\n      maxHeight: 'calc(100% - var(--setting-ui-vertical-padding, 2rem))',\n    },\n  },\n  container: {\n    ...ulReset,\n    // TODO use dvh once we may drop iOS 14\n    maxHeight: 'calc(90vh - var(--setting-ui-vertical-padding, 2rem))',\n    color: '#ccc',\n    whiteSpace: 'nowrap',\n    borderRadius: '4px',\n    userSelect: 'none',\n    overflow: 'auto',\n    overscrollBehavior: 'contain',\n  },\n  title: {\n    padding: '12px 18px',\n  },\n  dismiss: {\n    background: `center / 1rem no-repeat url(${icon.close}), transparent`,\n  },\n  back: {\n    background: '#ccc',\n    maskImage: `var(--icon-setting-back, url(\"${icon.back}\"))`,\n    maskSize: 'contain',\n  },\n  row: {\n    cursor: 'pointer',\n    display: 'flex',\n    position: 'relative',\n    padding: '1rem 1.5rem',\n    fontSize: '16px',\n    background: '#161C24',\n    '::after': {\n      content: '\" \"',\n      marginLeft: '1rem',\n      width: '20px',\n      height: '20px',\n      display: 'inline-block',\n      color: 'white',\n      backgroundPosition: 'center',\n      backgroundSize: 'cover',\n    },\n  },\n  space: {\n    flex: '1',\n  },\n  hasOptions: {\n    '::after': {\n      backgroundImage: `url(${icon.arrowTop})`,\n      transform: 'rotate(90deg)',\n    },\n  },\n  selected: {\n    color: 'white',\n    fontWeight: 'bold',\n    '::after': {\n      backgroundColor: 'var(--setting-check-icon-color, var(--primary-highlight, white))',\n      maskImage: `url(${icon.check})`,\n    },\n  },\n}\n\n// TODO some of styles are for older version UI design, can be simplified\nconst desktopStyle = {\n  overlay: {\n    position: 'absolute',\n    zIndex: '1',\n    bottom: 'calc(5em + var(--bottom-spacing, 0rem))',\n    right: '1rem',\n    display: 'flex',\n    alignItems: 'flex-end',\n    width: '18rem',\n    height: 'calc(100% - 8rem - var(--bottom-spacing, 0rem))',\n    outline: 'none',\n    opacity: '1',\n    transform: 'translateY(-100vh)',\n  },\n  open: {\n    position: 'fixed',\n    opacity: '1',\n    transform: 'translateY(0)',\n    transition: 'opacity 0.2s ease, transform 0s',\n  },\n  container: {\n    ...mobileStyle.container,\n    flex: '0 var(--setting-ui-width, calc(100% - 2rem))',\n    maxHeight: '100%',\n  },\n  head: {\n    ...mobileStyle.head,\n    color: 'var(--setting-head-color, inherit)',\n    background: 'var(--setting-head-background, rgba(22, 28, 36, 0.8))',\n  },\n  row: {\n    ...mobileStyle.row,\n    background: 'rgba(22, 28, 36, 0.8)',\n    '::after': {\n      order: 'var(--setting-check-icon-order, 0)',\n      ...mobileStyle.row['::after'],\n      marginLeft: '0',\n      marginRight: '0.5em',\n    },\n  },\n  hasOptions: {\n    '::after': {\n      order: '0',\n      backgroundImage: `url(${icon.arrowTop})`,\n      transform: 'rotate(90deg)',\n    },\n  },\n  selected: {\n    color: 'white',\n    fontWeight: 'bold',\n    '::after': {\n      backgroundColor:\n        'var(--setting-check-icon-color, var(--primary-highlight, white))',\n      maskImage: `url(${icon.check})`,\n    },\n  },\n}\n\nconst MenuItemText = ({text = ''}) => (\n  <FormattedMessage\n    id={text}\n    defaultMessage={\n      <FormattedMessage id={`KKS.SETTING.${text}`} defaultMessage={text} />\n    }\n  />\n)\n\nconst LoopSwitch = ({style, checked, onChange}) => (\n  <li\n    role=\"menuitem\"\n    css={[style, {'::before': {display: 'none'}, '::after': {display: 'none'}}]}\n    onClick={() => {\n      onChange({name: 'loop', value: !checked, keepOpen: true})\n    }}\n  >\n    <MenuItemText text=\"KKS.PLAYER.LOOP\" />\n    <div css={mobileStyle.space} />\n    <Switch checked={checked} />\n  </li>\n)\n\nconst CloseButton = props => (\n  <button\n    type=\"button\"\n    aria-label=\"Close Settings\"\n    css={mobileStyle.dismiss}\n    {...props}\n  />\n)\n\nconst DesktopContainer = ({open, style, children, onClose, ...rest}) => (\n  <div css={[style, open && desktopStyle.open]} {...rest}>\n    {children}\n  </div>\n)\n\nconst matchValue = (a, b) => a === b || (a?.id ?? a) === (b?.id ?? b)\n\nconst Settings = ({\n  open,\n  values,\n  sections: originalSections,\n  type: uiType,\n  closeBy,\n  buttonPosition,\n  enabledSettingSections = {\n    audio: true,\n    subtitles: true,\n    quality: true,\n    speed: true,\n    loop: true,\n  }, \n  slots = {\n    root: uiType === 'desktop' ? DesktopContainer : SwipeableDrawer,\n  },\n  onChange,\n  onOpen,\n  onClose,\n}) => {\n  const sections = originalSections.filter(section => enabledSettingSections[section.name])\n  const Container = slots.root\n  const ButtonWrap =\n    buttonPosition === 'top-right' ? TitleBarExtension : FunctionBarExtension\n  const commonStyle = uiType === 'desktop' ? desktopStyle : mobileStyle\n  const [path, setPath] = useState('/')\n  useEffect(() => {\n    if (!open) {\n      setPath('/')\n    }\n  }, [open])\n\n  const ref = useOnclickOutside(\n    () => {\n      if (open && uiType === 'desktop') {\n        onClose()\n      }\n    },\n    {eventTypes: ['click']}\n  )\n  const currentSection = sections.find(it => path === `/${it.name}`)\n  const menu =\n    path === '/'\n      ? {\n          title: 'KKS.SETTING',\n          items: sections.map(({type, name, title, items = []}) => ({\n            type,\n            link: `/${name}`,\n            label: title,\n            value:\n              items.find(item => matchValue(item.value, values[name]))?.label ||\n              values[name],\n          })),\n        }\n      : {\n          title: currentSection.title,\n          items: currentSection.items.map(({value, label}) => ({\n            label,\n            checked: matchValue(values[currentSection.name], value),\n            data: value,\n          })),\n          previous: '/',\n        }\n  const navigate = dest => requestAnimationFrame(() => setPath(dest))\n  return (\n    sections.length > 0 && (\n    <>\n      <ButtonWrap position=\"right\">\n        <Button\n          startIcon=\"setting\"\n          style={{order: 2}}\n          title=\"KKS.SETTING\"\n            onClick={event =>\n              setTimeout(() => (open ? onClose(event) : onOpen(event)), 1)\n            }\n        />\n      </ButtonWrap>\n      <Container style={commonStyle.overlay} open={open} onClose={onClose}>\n        <ul role=\"menu\" ref={ref} css={commonStyle.container}>\n          <div css={commonStyle.head}>\n            {menu.previous ? (\n              <Button startIcon=\"back1\" onClick={() => navigate('/')} />\n            ) : (\n              uiType !== 'desktop' &&\n              closeBy === 'button' && <CloseButton onClick={onClose} />\n            )}\n            <FormattedMessage id={menu.title} />\n          </div>\n          {menu.items.map(({type, label, link, value, data, checked}) =>\n            type === 'switch' ? (\n              <LoopSwitch\n                style={commonStyle.row}\n                checked={values.loop}\n                onChange={() =>\n                    onChange({\n                      name: 'loop',\n                      value: !values.loop,\n                      keepOpen: true,\n                    })\n                }\n              />\n            ) : (\n              <li\n                role={link ? 'menuitem' : 'menuitemradio'}\n                aria-checked={checked}\n                css={[\n                  commonStyle.row,\n                  link && commonStyle.hasOptions,\n                  checked && commonStyle.selected,\n                ]}\n                onClick={() =>\n                  link\n                    ? navigate(link)\n                    : onChange({name: currentSection.name, value: data})\n                }\n              >\n                <MenuItemText text={label} />\n                <div css={mobileStyle.space} />\n                {value && <MenuItemText text={value.toString()} />}\n              </li>\n            )\n          )}\n        </ul>\n      </Container>\n    </>\n    )\n  )\n}\n\nexport default Settings\n"]} */"],
|
|
4904
4852
|
onClick: () => link ? navigate(link) : onChange({
|
|
4905
4853
|
name: currentSection.name,
|
|
4906
4854
|
value: data
|
|
@@ -4927,12 +4875,12 @@ const containerStyle$3 = {
|
|
|
4927
4875
|
height: '100%',
|
|
4928
4876
|
transform: 'translateY(-150vh)',
|
|
4929
4877
|
opacity: '0',
|
|
4930
|
-
transition: 'opacity 0.
|
|
4878
|
+
transition: 'opacity 0.1s ease, transform 0s 0.1s linear'
|
|
4931
4879
|
};
|
|
4932
4880
|
const openStyle$1 = {
|
|
4933
4881
|
transform: 'translateY(0)',
|
|
4934
4882
|
opacity: '1',
|
|
4935
|
-
transition: 'opacity 0.
|
|
4883
|
+
transition: 'opacity 0.2s ease'
|
|
4936
4884
|
};
|
|
4937
4885
|
|
|
4938
4886
|
const DefaultContainer = ({
|
|
@@ -4940,7 +4888,7 @@ const DefaultContainer = ({
|
|
|
4940
4888
|
containerRef,
|
|
4941
4889
|
children
|
|
4942
4890
|
}) => jsx$1("div", {
|
|
4943
|
-
css: [containerStyle$3, open && openStyle$1, process.env.NODE_ENV === "production" ? "" : ";label:DefaultContainer;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
4891
|
+
css: [containerStyle$3, open && openStyle$1, process.env.NODE_ENV === "production" ? "" : ";label:DefaultContainer;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIk92ZXJsYXlQYW5lbC5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUE0QkkiLCJmaWxlIjoiT3ZlcmxheVBhbmVsLmpzIiwic291cmNlc0NvbnRlbnQiOlsiLyogZXNsaW50LWRpc2FibGUganN4LWExMXkvbm8tc3RhdGljLWVsZW1lbnQtaW50ZXJhY3Rpb25zICovXG4vKiBAanN4SW1wb3J0U291cmNlIEBlbW90aW9uL3JlYWN0ICovXG5pbXBvcnQge3VzZUVmZmVjdCwgdXNlUmVmLCB1c2VTdGF0ZX0gZnJvbSAncmVhY3QnXG5pbXBvcnQgdXNlT25jbGlja091dHNpZGUgZnJvbSAncmVhY3QtY29vbC1vbmNsaWNrb3V0c2lkZSdcbmltcG9ydCB7b259IGZyb20gJ3V0aWwvZXZlbnRzJ1xuaW1wb3J0IHtpc0Rlc2t0b3B9IGZyb20gJ3V0aWwvZW52aXJvbm1lbnQnXG5pbXBvcnQge0J1dHRvbn0gZnJvbSAnLi9idXR0b25zJ1xuaW1wb3J0IHtGdW5jdGlvbkJhckV4dGVuc2lvbiwgVGl0bGVCYXJFeHRlbnNpb259IGZyb20gJy4vdWlFeHRlbnNpb25zJ1xuXG5jb25zdCBjb250YWluZXJTdHlsZSA9IHtcbiAgcG9zaXRpb246ICdhYnNvbHV0ZScsXG4gIHpJbmRleDogJzInLFxuICB0b3A6ICcwJyxcbiAgd2lkdGg6ICcxMDAlJyxcbiAgaGVpZ2h0OiAnMTAwJScsXG4gIHRyYW5zZm9ybTogJ3RyYW5zbGF0ZVkoLTE1MHZoKScsXG4gIG9wYWNpdHk6ICcwJyxcbiAgdHJhbnNpdGlvbjogJ29wYWNpdHkgMC4xcyBlYXNlLCB0cmFuc2Zvcm0gMHMgMC4xcyBsaW5lYXInLFxufVxuXG5jb25zdCBvcGVuU3R5bGUgPSB7XG4gIHRyYW5zZm9ybTogJ3RyYW5zbGF0ZVkoMCknLFxuICBvcGFjaXR5OiAnMScsXG4gIHRyYW5zaXRpb246ICdvcGFjaXR5IDAuMnMgZWFzZScsXG59XG5cbmNvbnN0IERlZmF1bHRDb250YWluZXIgPSAoe29wZW4sIGNvbnRhaW5lclJlZiwgY2hpbGRyZW59KSA9PiAoXG4gIDxkaXZcbiAgICBjc3M9e1tjb250YWluZXJTdHlsZSwgb3BlbiAmJiBvcGVuU3R5bGVdfVxuICAgIHJlZj17Y29udGFpbmVyUmVmfVxuICAgIG9uQ2xpY2s9e2V2ZW50ID0+IGV2ZW50LnN0b3BQcm9wYWdhdGlvbigpfVxuICA+XG4gICAge2NoaWxkcmVufVxuICA8L2Rpdj5cbilcblxuY29uc3QgZGVza3RvcENvbnRhaW5lclN0eWxlID0ge1xuICBwb3NpdGlvbjogJ2Fic29sdXRlJyxcbiAgekluZGV4OiAnMicsXG4gIGJvdHRvbTogJ2NhbGMoM2VtICsgdmFyKC0tYm90dG9tLXNwYWNpbmcsIDByZW0pKScsXG4gIHJpZ2h0OiAnY2FsYyh2YXIoLS1zcGFjaW5nLCAwLjc1ZW0pICsgMC41ZW0gKyAwLjVyZW0pJyxcbiAgYmFja2dyb3VuZDogJ3RyYW5zcGFyZW50JyxcbiAgb3V0bGluZTogJ25vbmUnLFxuICBvcGFjaXR5OiAnMCcsXG4gIHRyYW5zZm9ybTogJ3RyYW5zbGF0ZVkoLTEwMHZoKScsXG4gIHRyYW5zaXRpb246ICdvcGFjaXR5IDAuMnMgZWFzZSwgdHJhbnNmb3JtIDBzIGxpbmVhciAwLjJzJyxcbn1cblxuY29uc3QgRGVza3RvcE9wZW5TdHlsZSA9IHtcbiAgb3BhY2l0eTogJzEnLFxuICB0cmFuc2Zvcm06ICd0cmFuc2xhdGVZKDApJyxcbiAgdHJhbnNpdGlvbjogJ29wYWNpdHkgMC4ycyBlYXNlLCB0cmFuc2Zvcm0gMHMnLFxufVxuXG5jb25zdCBEZXNrdG9wQ29udGFpbmVyID0gKHtvcGVuLCBjb250YWluZXJSZWYsIGNoaWxkcmVuLCBvbkNsb3NlLCAuLi5yZXN0fSkgPT4gKFxuICA8ZGl2XG4gICAgY3NzPXtbZGVza3RvcENvbnRhaW5lclN0eWxlLCBvcGVuICYmIERlc2t0b3BPcGVuU3R5bGVdfVxuICAgIHJlZj17Y29udGFpbmVyUmVmfVxuICAgIG9uQ2xpY2s9e2V2ZW50ID0+IGV2ZW50LnN0b3BQcm9wYWdhdGlvbigpfVxuICAgIHsuLi5yZXN0fVxuICA+XG4gICAge2NoaWxkcmVufVxuICA8L2Rpdj5cbilcblxuY29uc3QgT3ZlcmxheVBhbmVsID0gKHtcbiAgYnV0dG9uSWNvbixcbiAgdGl0bGUsXG4gIHVpVHlwZSA9IGlzRGVza3RvcCgpID8gJ2Rlc2t0b3AnIDogJ21vYmlsZScsXG4gIGNoaWxkcmVuLFxuICBzbG90cyA9IHtcbiAgICByb290OiB1aVR5cGUgPT09ICdkZXNrdG9wJyA/IERlc2t0b3BDb250YWluZXIgOiBEZWZhdWx0Q29udGFpbmVyLFxuICB9LFxuICAuLi5yZXN0XG59KSA9PiB7XG4gIGNvbnN0IGNvbXBvbmVudHMgPSB7XG4gICAgcm9vdDogc2xvdHMucm9vdCxcbiAgfVxuICBjb25zdCBCdXR0b25XcmFwID1cbiAgICB1aVR5cGUgPT09ICdkZXNrdG9wJyA/IEZ1bmN0aW9uQmFyRXh0ZW5zaW9uIDogVGl0bGVCYXJFeHRlbnNpb25cbiAgY29uc3QgW29wZW4sIHNldE9wZW5dID0gdXNlU3RhdGUoZmFsc2UpXG4gIGNvbnN0IG9uT3BlbiA9IGV2ZW50ID0+IHtcbiAgICBzZXRUaW1lb3V0KCgpID0+IHNldE9wZW4odHJ1ZSksIDEpXG4gICAgZXZlbnQuY3VycmVudFRhcmdldC5kaXNwYXRjaEV2ZW50KFxuICAgICAgbmV3IEN1c3RvbUV2ZW50KCdmb2N1cy1tZW51Jywge2RldGFpbDoge3N0YXR1czogJ29wZW4nfSwgYnViYmxlczogdHJ1ZX0pXG4gICAgKVxuICB9XG4gIGNvbnN0IGNvbnRhaW5lclJlZiA9IHVzZVJlZigpXG4gIGNvbnN0IG9uQ2xvc2UgPSAoKSA9PiB7XG4gICAgc2V0T3BlbihmYWxzZSlcbiAgICBjb250YWluZXJSZWYuY3VycmVudC5kaXNwYXRjaEV2ZW50KFxuICAgICAgbmV3IEN1c3RvbUV2ZW50KCdmb2N1cy1tZW51Jywge2RldGFpbDoge3N0YXR1czogJ2Nsb3NlZCd9LCBidWJibGVzOiB0cnVlfSlcbiAgICApXG4gIH1cbiAgY29uc3QgcmVmID0gdXNlT25jbGlja091dHNpZGUoXG4gICAgKCkgPT4ge1xuICAgICAgaWYgKG9wZW4gJiYgdWlUeXBlID09PSAnZGVza3RvcCcpIHtcbiAgICAgICAgb25DbG9zZSgpXG4gICAgICB9XG4gICAgfSxcbiAgICB7ZXZlbnRUeXBlczogWydjbGljayddfVxuICApXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgcmVmKGNvbnRhaW5lclJlZi5jdXJyZW50KVxuICAgIHJldHVybiBvbihjb250YWluZXJSZWYuY3VycmVudCwgJ2Nsb3NlLW1lbnUnLCBvbkNsb3NlKVxuICB9LCBbXSlcblxuICByZXR1cm4gKFxuICAgIDw+XG4gICAgICA8QnV0dG9uV3JhcCBwb3NpdGlvbj1cInJpZ2h0XCI+XG4gICAgICAgIDxCdXR0b25cbiAgICAgICAgICB0aXRsZT17dGl0bGV9XG4gICAgICAgICAgc3RhcnRJY29uPXtidXR0b25JY29ufVxuICAgICAgICAgIG9uQ2xpY2s9e29wZW4gPyB1bmRlZmluZWQgOiBvbk9wZW59XG4gICAgICAgIC8+XG4gICAgICA8L0J1dHRvbldyYXA+XG4gICAgICA8Y29tcG9uZW50cy5yb290IGNvbnRhaW5lclJlZj17Y29udGFpbmVyUmVmfSBvcGVuPXtvcGVufSB7Li4ucmVzdH0+XG4gICAgICAgIHtjaGlsZHJlbn1cbiAgICAgIDwvY29tcG9uZW50cy5yb290PlxuICAgIDwvPlxuICApXG59XG5cbmV4cG9ydCBkZWZhdWx0IE92ZXJsYXlQYW5lbFxuIl19 */"],
|
|
4944
4892
|
ref: containerRef,
|
|
4945
4893
|
onClick: event => event.stopPropagation(),
|
|
4946
4894
|
children: children
|
|
@@ -4948,9 +4896,9 @@ const DefaultContainer = ({
|
|
|
4948
4896
|
|
|
4949
4897
|
const desktopContainerStyle$1 = {
|
|
4950
4898
|
position: 'absolute',
|
|
4951
|
-
zIndex: '
|
|
4952
|
-
bottom: 'calc(
|
|
4953
|
-
right: '
|
|
4899
|
+
zIndex: '2',
|
|
4900
|
+
bottom: 'calc(3em + var(--bottom-spacing, 0rem))',
|
|
4901
|
+
right: 'calc(var(--spacing, 0.75em) + 0.5em + 0.5rem)',
|
|
4954
4902
|
background: 'transparent',
|
|
4955
4903
|
outline: 'none',
|
|
4956
4904
|
opacity: '0',
|
|
@@ -4970,7 +4918,7 @@ const DesktopContainer = ({
|
|
|
4970
4918
|
onClose,
|
|
4971
4919
|
...rest
|
|
4972
4920
|
}) => jsx$1("div", {
|
|
4973
|
-
css: [desktopContainerStyle$1, open && DesktopOpenStyle, process.env.NODE_ENV === "production" ? "" : ";label:DesktopContainer;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
4921
|
+
css: [desktopContainerStyle$1, open && DesktopOpenStyle, process.env.NODE_ENV === "production" ? "" : ";label:DesktopContainer;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIk92ZXJsYXlQYW5lbC5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUF3REkiLCJmaWxlIjoiT3ZlcmxheVBhbmVsLmpzIiwic291cmNlc0NvbnRlbnQiOlsiLyogZXNsaW50LWRpc2FibGUganN4LWExMXkvbm8tc3RhdGljLWVsZW1lbnQtaW50ZXJhY3Rpb25zICovXG4vKiBAanN4SW1wb3J0U291cmNlIEBlbW90aW9uL3JlYWN0ICovXG5pbXBvcnQge3VzZUVmZmVjdCwgdXNlUmVmLCB1c2VTdGF0ZX0gZnJvbSAncmVhY3QnXG5pbXBvcnQgdXNlT25jbGlja091dHNpZGUgZnJvbSAncmVhY3QtY29vbC1vbmNsaWNrb3V0c2lkZSdcbmltcG9ydCB7b259IGZyb20gJ3V0aWwvZXZlbnRzJ1xuaW1wb3J0IHtpc0Rlc2t0b3B9IGZyb20gJ3V0aWwvZW52aXJvbm1lbnQnXG5pbXBvcnQge0J1dHRvbn0gZnJvbSAnLi9idXR0b25zJ1xuaW1wb3J0IHtGdW5jdGlvbkJhckV4dGVuc2lvbiwgVGl0bGVCYXJFeHRlbnNpb259IGZyb20gJy4vdWlFeHRlbnNpb25zJ1xuXG5jb25zdCBjb250YWluZXJTdHlsZSA9IHtcbiAgcG9zaXRpb246ICdhYnNvbHV0ZScsXG4gIHpJbmRleDogJzInLFxuICB0b3A6ICcwJyxcbiAgd2lkdGg6ICcxMDAlJyxcbiAgaGVpZ2h0OiAnMTAwJScsXG4gIHRyYW5zZm9ybTogJ3RyYW5zbGF0ZVkoLTE1MHZoKScsXG4gIG9wYWNpdHk6ICcwJyxcbiAgdHJhbnNpdGlvbjogJ29wYWNpdHkgMC4xcyBlYXNlLCB0cmFuc2Zvcm0gMHMgMC4xcyBsaW5lYXInLFxufVxuXG5jb25zdCBvcGVuU3R5bGUgPSB7XG4gIHRyYW5zZm9ybTogJ3RyYW5zbGF0ZVkoMCknLFxuICBvcGFjaXR5OiAnMScsXG4gIHRyYW5zaXRpb246ICdvcGFjaXR5IDAuMnMgZWFzZScsXG59XG5cbmNvbnN0IERlZmF1bHRDb250YWluZXIgPSAoe29wZW4sIGNvbnRhaW5lclJlZiwgY2hpbGRyZW59KSA9PiAoXG4gIDxkaXZcbiAgICBjc3M9e1tjb250YWluZXJTdHlsZSwgb3BlbiAmJiBvcGVuU3R5bGVdfVxuICAgIHJlZj17Y29udGFpbmVyUmVmfVxuICAgIG9uQ2xpY2s9e2V2ZW50ID0+IGV2ZW50LnN0b3BQcm9wYWdhdGlvbigpfVxuICA+XG4gICAge2NoaWxkcmVufVxuICA8L2Rpdj5cbilcblxuY29uc3QgZGVza3RvcENvbnRhaW5lclN0eWxlID0ge1xuICBwb3NpdGlvbjogJ2Fic29sdXRlJyxcbiAgekluZGV4OiAnMicsXG4gIGJvdHRvbTogJ2NhbGMoM2VtICsgdmFyKC0tYm90dG9tLXNwYWNpbmcsIDByZW0pKScsXG4gIHJpZ2h0OiAnY2FsYyh2YXIoLS1zcGFjaW5nLCAwLjc1ZW0pICsgMC41ZW0gKyAwLjVyZW0pJyxcbiAgYmFja2dyb3VuZDogJ3RyYW5zcGFyZW50JyxcbiAgb3V0bGluZTogJ25vbmUnLFxuICBvcGFjaXR5OiAnMCcsXG4gIHRyYW5zZm9ybTogJ3RyYW5zbGF0ZVkoLTEwMHZoKScsXG4gIHRyYW5zaXRpb246ICdvcGFjaXR5IDAuMnMgZWFzZSwgdHJhbnNmb3JtIDBzIGxpbmVhciAwLjJzJyxcbn1cblxuY29uc3QgRGVza3RvcE9wZW5TdHlsZSA9IHtcbiAgb3BhY2l0eTogJzEnLFxuICB0cmFuc2Zvcm06ICd0cmFuc2xhdGVZKDApJyxcbiAgdHJhbnNpdGlvbjogJ29wYWNpdHkgMC4ycyBlYXNlLCB0cmFuc2Zvcm0gMHMnLFxufVxuXG5jb25zdCBEZXNrdG9wQ29udGFpbmVyID0gKHtvcGVuLCBjb250YWluZXJSZWYsIGNoaWxkcmVuLCBvbkNsb3NlLCAuLi5yZXN0fSkgPT4gKFxuICA8ZGl2XG4gICAgY3NzPXtbZGVza3RvcENvbnRhaW5lclN0eWxlLCBvcGVuICYmIERlc2t0b3BPcGVuU3R5bGVdfVxuICAgIHJlZj17Y29udGFpbmVyUmVmfVxuICAgIG9uQ2xpY2s9e2V2ZW50ID0+IGV2ZW50LnN0b3BQcm9wYWdhdGlvbigpfVxuICAgIHsuLi5yZXN0fVxuICA+XG4gICAge2NoaWxkcmVufVxuICA8L2Rpdj5cbilcblxuY29uc3QgT3ZlcmxheVBhbmVsID0gKHtcbiAgYnV0dG9uSWNvbixcbiAgdGl0bGUsXG4gIHVpVHlwZSA9IGlzRGVza3RvcCgpID8gJ2Rlc2t0b3AnIDogJ21vYmlsZScsXG4gIGNoaWxkcmVuLFxuICBzbG90cyA9IHtcbiAgICByb290OiB1aVR5cGUgPT09ICdkZXNrdG9wJyA/IERlc2t0b3BDb250YWluZXIgOiBEZWZhdWx0Q29udGFpbmVyLFxuICB9LFxuICAuLi5yZXN0XG59KSA9PiB7XG4gIGNvbnN0IGNvbXBvbmVudHMgPSB7XG4gICAgcm9vdDogc2xvdHMucm9vdCxcbiAgfVxuICBjb25zdCBCdXR0b25XcmFwID1cbiAgICB1aVR5cGUgPT09ICdkZXNrdG9wJyA/IEZ1bmN0aW9uQmFyRXh0ZW5zaW9uIDogVGl0bGVCYXJFeHRlbnNpb25cbiAgY29uc3QgW29wZW4sIHNldE9wZW5dID0gdXNlU3RhdGUoZmFsc2UpXG4gIGNvbnN0IG9uT3BlbiA9IGV2ZW50ID0+IHtcbiAgICBzZXRUaW1lb3V0KCgpID0+IHNldE9wZW4odHJ1ZSksIDEpXG4gICAgZXZlbnQuY3VycmVudFRhcmdldC5kaXNwYXRjaEV2ZW50KFxuICAgICAgbmV3IEN1c3RvbUV2ZW50KCdmb2N1cy1tZW51Jywge2RldGFpbDoge3N0YXR1czogJ29wZW4nfSwgYnViYmxlczogdHJ1ZX0pXG4gICAgKVxuICB9XG4gIGNvbnN0IGNvbnRhaW5lclJlZiA9IHVzZVJlZigpXG4gIGNvbnN0IG9uQ2xvc2UgPSAoKSA9PiB7XG4gICAgc2V0T3BlbihmYWxzZSlcbiAgICBjb250YWluZXJSZWYuY3VycmVudC5kaXNwYXRjaEV2ZW50KFxuICAgICAgbmV3IEN1c3RvbUV2ZW50KCdmb2N1cy1tZW51Jywge2RldGFpbDoge3N0YXR1czogJ2Nsb3NlZCd9LCBidWJibGVzOiB0cnVlfSlcbiAgICApXG4gIH1cbiAgY29uc3QgcmVmID0gdXNlT25jbGlja091dHNpZGUoXG4gICAgKCkgPT4ge1xuICAgICAgaWYgKG9wZW4gJiYgdWlUeXBlID09PSAnZGVza3RvcCcpIHtcbiAgICAgICAgb25DbG9zZSgpXG4gICAgICB9XG4gICAgfSxcbiAgICB7ZXZlbnRUeXBlczogWydjbGljayddfVxuICApXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgcmVmKGNvbnRhaW5lclJlZi5jdXJyZW50KVxuICAgIHJldHVybiBvbihjb250YWluZXJSZWYuY3VycmVudCwgJ2Nsb3NlLW1lbnUnLCBvbkNsb3NlKVxuICB9LCBbXSlcblxuICByZXR1cm4gKFxuICAgIDw+XG4gICAgICA8QnV0dG9uV3JhcCBwb3NpdGlvbj1cInJpZ2h0XCI+XG4gICAgICAgIDxCdXR0b25cbiAgICAgICAgICB0aXRsZT17dGl0bGV9XG4gICAgICAgICAgc3RhcnRJY29uPXtidXR0b25JY29ufVxuICAgICAgICAgIG9uQ2xpY2s9e29wZW4gPyB1bmRlZmluZWQgOiBvbk9wZW59XG4gICAgICAgIC8+XG4gICAgICA8L0J1dHRvbldyYXA+XG4gICAgICA8Y29tcG9uZW50cy5yb290IGNvbnRhaW5lclJlZj17Y29udGFpbmVyUmVmfSBvcGVuPXtvcGVufSB7Li4ucmVzdH0+XG4gICAgICAgIHtjaGlsZHJlbn1cbiAgICAgIDwvY29tcG9uZW50cy5yb290PlxuICAgIDwvPlxuICApXG59XG5cbmV4cG9ydCBkZWZhdWx0IE92ZXJsYXlQYW5lbFxuIl19 */"],
|
|
4974
4922
|
ref: containerRef,
|
|
4975
4923
|
onClick: event => event.stopPropagation(),
|
|
4976
4924
|
...rest,
|
|
@@ -4979,6 +4927,7 @@ const DesktopContainer = ({
|
|
|
4979
4927
|
|
|
4980
4928
|
const OverlayPanel = ({
|
|
4981
4929
|
buttonIcon,
|
|
4930
|
+
title,
|
|
4982
4931
|
uiType = isDesktop() ? 'desktop' : 'mobile',
|
|
4983
4932
|
children,
|
|
4984
4933
|
slots = {
|
|
@@ -4992,11 +4941,28 @@ const OverlayPanel = ({
|
|
|
4992
4941
|
const ButtonWrap = uiType === 'desktop' ? FunctionBarExtension : TitleBarExtension;
|
|
4993
4942
|
const [open, setOpen] = useState(false);
|
|
4994
4943
|
|
|
4995
|
-
const
|
|
4996
|
-
|
|
4997
|
-
|
|
4944
|
+
const onOpen = event => {
|
|
4945
|
+
setTimeout(() => setOpen(true), 1);
|
|
4946
|
+
event.currentTarget.dispatchEvent(new CustomEvent('focus-menu', {
|
|
4947
|
+
detail: {
|
|
4948
|
+
status: 'open'
|
|
4949
|
+
},
|
|
4950
|
+
bubbles: true
|
|
4951
|
+
}));
|
|
4952
|
+
};
|
|
4998
4953
|
|
|
4999
4954
|
const containerRef = useRef();
|
|
4955
|
+
|
|
4956
|
+
const onClose = () => {
|
|
4957
|
+
setOpen(false);
|
|
4958
|
+
containerRef.current.dispatchEvent(new CustomEvent('focus-menu', {
|
|
4959
|
+
detail: {
|
|
4960
|
+
status: 'closed'
|
|
4961
|
+
},
|
|
4962
|
+
bubbles: true
|
|
4963
|
+
}));
|
|
4964
|
+
};
|
|
4965
|
+
|
|
5000
4966
|
const ref = useOnclickOutside(() => {
|
|
5001
4967
|
if (open && uiType === 'desktop') {
|
|
5002
4968
|
onClose();
|
|
@@ -5012,8 +4978,9 @@ const OverlayPanel = ({
|
|
|
5012
4978
|
children: [jsx$1(ButtonWrap, {
|
|
5013
4979
|
position: "right",
|
|
5014
4980
|
children: jsx$1(Button, {
|
|
4981
|
+
title: title,
|
|
5015
4982
|
startIcon: buttonIcon,
|
|
5016
|
-
onClick:
|
|
4983
|
+
onClick: open ? undefined : onOpen
|
|
5017
4984
|
})
|
|
5018
4985
|
}), jsx$1(components.root, {
|
|
5019
4986
|
containerRef: containerRef,
|
|
@@ -5024,7 +4991,7 @@ const OverlayPanel = ({
|
|
|
5024
4991
|
});
|
|
5025
4992
|
};
|
|
5026
4993
|
|
|
5027
|
-
/*
|
|
4994
|
+
/* @jsxImportSource @emotion/react */
|
|
5028
4995
|
const menuItemStyle = {
|
|
5029
4996
|
padding: 'var(--menu-item-padding, 0.5em 1em) ',
|
|
5030
4997
|
display: 'flex',
|
|
@@ -5058,10 +5025,13 @@ const MenuItem = ({
|
|
|
5058
5025
|
selected,
|
|
5059
5026
|
onClick
|
|
5060
5027
|
}) => jsx$1("div", {
|
|
5061
|
-
css: [menuItemStyle, selected && menuItemSelectedStyle, process.env.NODE_ENV === "production" ? "" : ";label:MenuItem;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
5028
|
+
css: [menuItemStyle, selected && menuItemSelectedStyle, process.env.NODE_ENV === "production" ? "" : ";label:MenuItem;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIk1lbnVJdGVtLmpzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQW9DSSIsImZpbGUiOiJNZW51SXRlbS5qcyIsInNvdXJjZXNDb250ZW50IjpbIi8qIEBqc3hJbXBvcnRTb3VyY2UgQGVtb3Rpb24vcmVhY3QgKi9cbi8qIGVzbGludC1kaXNhYmxlIGpzeC1hMTF5L25vLXN0YXRpYy1lbGVtZW50LWludGVyYWN0aW9ucyAqL1xuaW1wb3J0IGljb24gZnJvbSAnc3R5bGUvaWNvbidcbmltcG9ydCB7Rm9ybWF0dGVkTWVzc2FnZX0gZnJvbSAnY29udGV4dC9JMThuJ1xuXG5jb25zdCBtZW51SXRlbVN0eWxlID0ge1xuICBwYWRkaW5nOiAndmFyKC0tbWVudS1pdGVtLXBhZGRpbmcsIDAuNWVtIDFlbSkgJyxcbiAgZGlzcGxheTogJ2ZsZXgnLFxuICBhbGlnbkl0ZW1zOiAnY2VudGVyJyxcbiAgbWluV2lkdGg6IDAsXG4gIGNvbG9yOiAncmdiYSgyNTUsIDI1NSwgMjU1LCAwLjYpJyxcbiAgYmFja2dyb3VuZDogJ3ZhcigtLW1lbnUtaXRlbS1iYWNrZ3JvdW5kLCByZ2JhKDI1NSwgMjU1LCAyNTUsIDAuMDgpKScsXG4gICc+IGRpdic6IHtcbiAgICBvdmVyZmxvdzogJ2hpZGRlbicsXG4gICAgd2hpdGVTcGFjZTogJ25vd3JhcCcsXG4gICAgdGV4dE92ZXJmbG93OiAnZWxsaXBzaXMnLFxuICB9LFxufVxuXG5jb25zdCBtZW51SXRlbVNlbGVjdGVkU3R5bGUgPSB7XG4gIGNvbG9yOiAnI0ZGRicsXG4gIGZvbnRXZWlnaHQ6ICc2MDAnLFxuICAnOjphZnRlcic6IHtcbiAgICBjb250ZW50OiAnXCIgXCInLFxuICAgIGRpc3BsYXk6ICdpbmxpbmUtYmxvY2snLFxuICAgIGZsZXg6ICcwIDAgMWVtJyxcbiAgICBoZWlnaHQ6ICcxZW0nLFxuICAgIG1hcmdpbkxlZnQ6ICcwLjVlbScsXG4gICAgYmFja2dyb3VuZENvbG9yOiAndmFyKC0tc2V0dGluZy1jaGVjay1pY29uLWNvbG9yLCAjZmZmKScsXG4gICAgbWFza1NpemU6ICdjb250YWluJyxcbiAgICBtYXNrSW1hZ2U6IGB1cmwoJHtpY29uLmNoZWNrfSlgLFxuICB9LFxufVxuXG5jb25zdCBNZW51SXRlbSA9ICh7bGFiZWwsIHNlbGVjdGVkLCBvbkNsaWNrfSkgPT4gKFxuICA8ZGl2XG4gICAgY3NzPXtbbWVudUl0ZW1TdHlsZSwgc2VsZWN0ZWQgJiYgbWVudUl0ZW1TZWxlY3RlZFN0eWxlXX1cbiAgICBvbkNsaWNrPXtvbkNsaWNrfVxuICA+XG4gICAgPGRpdj5cbiAgICAgIDxGb3JtYXR0ZWRNZXNzYWdlIGlkPXtgS0tTLlNFVFRJTkcuJHtsYWJlbH1gfSBkZWZhdWx0TWVzc2FnZT17bGFiZWx9IC8+XG4gICAgPC9kaXY+XG4gIDwvZGl2PlxuKVxuXG5leHBvcnQgZGVmYXVsdCBNZW51SXRlbVxuIl19 */"],
|
|
5062
5029
|
onClick: onClick,
|
|
5063
5030
|
children: jsx$1("div", {
|
|
5064
|
-
children:
|
|
5031
|
+
children: jsx$1(FormattedMessage, {
|
|
5032
|
+
id: `KKS.SETTING.${label}`,
|
|
5033
|
+
defaultMessage: label
|
|
5034
|
+
})
|
|
5065
5035
|
})
|
|
5066
5036
|
});
|
|
5067
5037
|
|
|
@@ -5127,15 +5097,21 @@ const twoColumnStyle = {
|
|
|
5127
5097
|
const desktopContainerStyle = {
|
|
5128
5098
|
paddingBottom: '1.5em',
|
|
5129
5099
|
width: '34em',
|
|
5130
|
-
minHeight: '30.
|
|
5131
|
-
maxHeight: '63.
|
|
5100
|
+
minHeight: 'min(30.5vh, calc(100vh - 16em))',
|
|
5101
|
+
maxHeight: 'min(63.5vh, calc(100vh - 16em))',
|
|
5132
5102
|
borderRadius: '4px',
|
|
5103
|
+
overflowY: 'auto',
|
|
5104
|
+
fontWeight: '500',
|
|
5133
5105
|
background: 'rgba(0, 0, 0, 0.60)',
|
|
5134
5106
|
'--menu-item-background': 'transparent',
|
|
5135
|
-
'--menu-item-padding': '0.66em
|
|
5107
|
+
'--menu-item-padding': '0.66em 2.5em',
|
|
5136
5108
|
h3: {
|
|
5109
|
+
position: 'sticky',
|
|
5110
|
+
top: 0,
|
|
5111
|
+
zIndex: 1,
|
|
5137
5112
|
padding: '0.66em 1.5em',
|
|
5138
5113
|
fontSize: '16px',
|
|
5114
|
+
fontWeight: '500',
|
|
5139
5115
|
color: 'rgba(255, 255, 255, 0.60)',
|
|
5140
5116
|
background: 'rgba(0, 0, 0, 0.80)'
|
|
5141
5117
|
}
|
|
@@ -5147,13 +5123,13 @@ const desktop2ColumnStyle = {
|
|
|
5147
5123
|
borderLeft: '1px solid rgba(255, 255, 255, 0.20)'
|
|
5148
5124
|
},
|
|
5149
5125
|
'> div > div': {
|
|
5150
|
-
padding: '0
|
|
5126
|
+
padding: '0'
|
|
5151
5127
|
},
|
|
5152
5128
|
'> div > div > div:first-of-type': {
|
|
5153
|
-
padding: '0.66em
|
|
5129
|
+
padding: '0.66em 1.5em',
|
|
5154
5130
|
color: '#fff',
|
|
5155
5131
|
fontSize: '16px',
|
|
5156
|
-
fontWeight: '
|
|
5132
|
+
fontWeight: '700'
|
|
5157
5133
|
}
|
|
5158
5134
|
};
|
|
5159
5135
|
|
|
@@ -5168,7 +5144,7 @@ const LanguageMenu = ({
|
|
|
5168
5144
|
sectionOptions = [],
|
|
5169
5145
|
onChange
|
|
5170
5146
|
}) => jsxs("div", {
|
|
5171
|
-
css: [containerStyle$2, twoColumnStyle, uiType === 'desktop' && [desktopContainerStyle, desktop2ColumnStyle], process.env.NODE_ENV === "production" ? "" : ";label:LanguageMenu;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
5147
|
+
css: [containerStyle$2, twoColumnStyle, uiType === 'desktop' && [desktopContainerStyle, desktop2ColumnStyle], process.env.NODE_ENV === "production" ? "" : ";label:LanguageMenu;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIkxhbmd1YWdlTWVudS5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFvSEkiLCJmaWxlIjoiTGFuZ3VhZ2VNZW51LmpzIiwic291cmNlc0NvbnRlbnQiOlsiLyogQGpzeEltcG9ydFNvdXJjZSBAZW1vdGlvbi9yZWFjdCAqL1xuaW1wb3J0IHtCdXR0b259IGZyb20gJy4vYnV0dG9ucydcbmltcG9ydCBNZW51SXRlbSBmcm9tICcuL01lbnVJdGVtJ1xuXG5jb25zdCBsYW5kc2NhcGVTdHlsZSA9IHtcbiAgZGlzcGxheTogJ2ZsZXgnLFxuICBmbGV4V3JhcDogJ3dyYXAnLFxuICBqdXN0aWZ5Q29udGVudDogJ3NwYWNlLWJldHdlZW4nLFxuICAnPiBkaXYnOiB7XG4gICAgYm94U2l6aW5nOiAnYm9yZGVyLWJveCcsXG4gICAgZmxleDogJzAgMCA1MCUnLFxuICAgIG1pbldpZHRoOiAwLFxuICB9LFxuICAnPiBkaXY6bm90KDpmaXJzdC1vZi10eXBlKSc6IHtcbiAgICBtYXJnaW5Ub3A6IDAsXG4gICAgZmxleDogJzAgMCBjYWxjKDUwJSAtIDFweCknLFxuICB9LFxuICBocjoge1xuICAgIGhlaWdodDogJ2F1dG8nLFxuICAgIGJvcmRlckxlZnQ6ICcxcHggc29saWQgdHJhbnNwYXJlbnQnLFxuICB9LFxufVxuXG5jb25zdCBjb250YWluZXJTdHlsZSA9IHtcbiAgZm9udFNpemU6ICcxNnB4JyxcbiAgYmFja2dyb3VuZDogJyMyNjI2MjYnLFxuICBoZWlnaHQ6ICcxMDAlJyxcbiAgaDM6IHtcbiAgICBtYXJnaW46IDAsXG4gICAgJ21hcmdpbi1ibG9jay1zdGFydCc6IDAsXG4gICAgJ21hcmdpbi1ibG9jay1lbmQnOiAwLFxuICAgIHBhZGRpbmc6ICcwLjZlbSAxZW0nLFxuICAgIGRpc3BsYXk6ICdmbGV4JyxcbiAgICBhbGlnbkl0ZW1zOiAnY2VudGVyJyxcbiAgICBib3JkZXJSYWRpdXM6ICc0cHggNHB4IDAgMCcsXG4gICAgZm9udFNpemU6ICcxN3B4JyxcbiAgICBidXR0b246IHtcbiAgICAgIGZvbnRTaXplOiAnMC42ZW0nLFxuICAgICAgbWFyZ2luUmlnaHQ6ICcwLjVlbScsXG4gICAgfSxcbiAgfSxcbn1cblxuY29uc3QgdHdvQ29sdW1uU3R5bGUgPSB7XG4gIGhyOiB7XG4gICAgYm9yZGVyOiAnbm9uZScsXG4gIH0sXG4gICc+IGRpdiAnOiB7XG4gICAgJz4gZGl2Om5vdCg6Zmlyc3Qtb2YtdHlwZSknOiB7XG4gICAgICBtYXJnaW5Ub3A6ICcwLjY2ZW0nLFxuICAgIH0sXG4gICAgJ0BtZWRpYSAob3JpZW50YXRpb246IGxhbmRzY2FwZSknOiBsYW5kc2NhcGVTdHlsZSxcbiAgfSxcbiAgJz4gZGl2ID4gZGl2ID4gZGl2OmZpcnN0LW9mLXR5cGUnOiB7XG4gICAgcGFkZGluZzogJzAuNmVtIDEuMWVtJyxcbiAgICBmb250U2l6ZTogJzE0cHgnLFxuICAgIGNvbG9yOiAnI2FhYWViNScsXG4gICAgYmFja2dyb3VuZDogJ3ZhcigtLW1lbnUtaXRlbS1iYWNrZ3JvdW5kLCAjMjYyNjI2KScsXG4gICAgb3ZlcmZsb3c6ICdoaWRkZW4nLFxuICAgIHRleHRPdmVyZmxvdzogJ2VsbGlwc2lzJyxcbiAgfSxcbn1cblxuY29uc3QgZGVza3RvcENvbnRhaW5lclN0eWxlID0ge1xuICBwYWRkaW5nQm90dG9tOiAnMS41ZW0nLFxuICB3aWR0aDogJzM0ZW0nLFxuICBtaW5IZWlnaHQ6ICdtaW4oMzAuNXZoLCBjYWxjKDEwMHZoIC0gMTZlbSkpJyxcbiAgbWF4SGVpZ2h0OiAnbWluKDYzLjV2aCwgY2FsYygxMDB2aCAtIDE2ZW0pKScsXG4gIGJvcmRlclJhZGl1czogJzRweCcsXG4gIG92ZXJmbG93WTogJ2F1dG8nLFxuICBmb250V2VpZ2h0OiAnNTAwJyxcbiAgYmFja2dyb3VuZDogJ3JnYmEoMCwgMCwgMCwgMC42MCknLFxuICAnLS1tZW51LWl0ZW0tYmFja2dyb3VuZCc6ICd0cmFuc3BhcmVudCcsXG4gICctLW1lbnUtaXRlbS1wYWRkaW5nJzogJzAuNjZlbSAyLjVlbScsXG4gIGgzOiB7XG4gICAgcG9zaXRpb246ICdzdGlja3knLFxuICAgIHRvcDogMCxcbiAgICB6SW5kZXg6IDEsXG4gICAgcGFkZGluZzogJzAuNjZlbSAxLjVlbScsXG4gICAgZm9udFNpemU6ICcxNnB4JyxcbiAgICBmb250V2VpZ2h0OiAnNTAwJyxcbiAgICBjb2xvcjogJ3JnYmEoMjU1LCAyNTUsIDI1NSwgMC42MCknLFxuICAgIGJhY2tncm91bmQ6ICdyZ2JhKDAsIDAsIDAsIDAuODApJyxcbiAgfSxcbn1cblxuY29uc3QgZGVza3RvcDJDb2x1bW5TdHlsZSA9IHtcbiAgJz4gZGl2JzogbGFuZHNjYXBlU3R5bGUsXG4gICc+IGRpdiBocic6IHtcbiAgICBmb250U2l6ZTogJzE1MCUnLFxuICAgIGJvcmRlckxlZnQ6ICcxcHggc29saWQgIHJnYmEoMjU1LCAyNTUsIDI1NSwgMC4yMCknLFxuICB9LFxuICAnPiBkaXYgPiBkaXYnOiB7XG4gICAgcGFkZGluZzogJzAnLFxuICB9LFxuICAnPiBkaXYgPiBkaXYgPiBkaXY6Zmlyc3Qtb2YtdHlwZSc6IHtcbiAgICBwYWRkaW5nOiAnMC42NmVtIDEuNWVtJyxcbiAgICBjb2xvcjogJyNmZmYnLFxuICAgIGZvbnRTaXplOiAnMTZweCcsXG4gICAgZm9udFdlaWdodDogJzcwMCcsXG4gIH0sXG59XG5cbmNvbnN0IGNsb3NlTWVudSA9IGV2ZW50ID0+XG4gIGV2ZW50LmN1cnJlbnRUYXJnZXQuZGlzcGF0Y2hFdmVudChcbiAgICBuZXcgQ3VzdG9tRXZlbnQoJ2Nsb3NlLW1lbnUnLCB7YnViYmxlczogdHJ1ZX0pXG4gIClcblxuY29uc3QgTGFuZ3VhZ2VNZW51ID0gKHtcbiAgdWlUeXBlID0gJ21vYmlsZScsXG4gIHRpdGxlID0gJ+mfs+WjsCDjg7sg5a2X5bmVJyxcbiAgc2VjdGlvbnMgPSBbJ+mfs+WjsCcsICflrZfluZUnXSxcbiAgc2VjdGlvbk9wdGlvbnMgPSBbXSxcbiAgb25DaGFuZ2UsXG59KSA9PiAoXG4gIDxkaXZcbiAgICBjc3M9e1tcbiAgICAgIGNvbnRhaW5lclN0eWxlLFxuICAgICAgdHdvQ29sdW1uU3R5bGUsXG4gICAgICB1aVR5cGUgPT09ICdkZXNrdG9wJyAmJiBbZGVza3RvcENvbnRhaW5lclN0eWxlLCBkZXNrdG9wMkNvbHVtblN0eWxlXSxcbiAgICBdfVxuICA+XG4gICAgPGgzPlxuICAgICAge3VpVHlwZSAhPT0gJ2Rlc2t0b3AnICYmIDxCdXR0b24gc3RhcnRJY29uPVwiY2xvc2VcIiBvbkNsaWNrPXtjbG9zZU1lbnV9IC8+fVxuICAgICAge3RpdGxlfVxuICAgIDwvaDM+XG4gICAgPGRpdj5cbiAgICAgIDxkaXY+XG4gICAgICAgIDxkaXY+e3NlY3Rpb25zWzBdfTwvZGl2PlxuICAgICAgICB7c2VjdGlvbk9wdGlvbnNbMF0ubWFwKG9wdGlvbiA9PiAoXG4gICAgICAgICAgPE1lbnVJdGVtIHsuLi5vcHRpb259IG9uQ2xpY2s9e2V2ZW50ID0+IG9uQ2hhbmdlKGV2ZW50LCBvcHRpb24pfSAvPlxuICAgICAgICApKX1cbiAgICAgIDwvZGl2PlxuICAgICAgPGhyIC8+XG4gICAgICA8ZGl2PlxuICAgICAgICA8ZGl2PntzZWN0aW9uc1sxXX08L2Rpdj5cbiAgICAgICAge3NlY3Rpb25PcHRpb25zWzFdLm1hcChvcHRpb24gPT4gKFxuICAgICAgICAgIDxNZW51SXRlbSB7Li4ub3B0aW9ufSBvbkNsaWNrPXtldmVudCA9PiBvbkNoYW5nZShldmVudCwgb3B0aW9uKX0gLz5cbiAgICAgICAgKSl9XG4gICAgICA8L2Rpdj5cbiAgICA8L2Rpdj5cbiAgPC9kaXY+XG4pXG5cbkxhbmd1YWdlTWVudS5zdHlsZXMgPSBjb250YWluZXJTdHlsZVxuTGFuZ3VhZ2VNZW51LmRlc2t0b3BTdHlsZXMgPSBkZXNrdG9wQ29udGFpbmVyU3R5bGVcblxuZXhwb3J0IGRlZmF1bHQgTGFuZ3VhZ2VNZW51XG4iXX0= */"],
|
|
5172
5148
|
children: [jsxs("h3", {
|
|
5173
5149
|
children: [uiType !== 'desktop' && jsx$1(Button, {
|
|
5174
5150
|
startIcon: "close",
|
|
@@ -6048,7 +6024,7 @@ const makeResponse = (headers, data, status, uri, responseURL, requestType) => {
|
|
|
6048
6024
|
throw new shaka$1.util.Error(severity, shaka$1.util.Error.Category.NETWORK, shaka$1.util.Error.Code.BAD_HTTP_STATUS, uri, status, responseText, headers, requestType);
|
|
6049
6025
|
};
|
|
6050
6026
|
|
|
6051
|
-
const goog$
|
|
6027
|
+
const goog$2 = {
|
|
6052
6028
|
asserts: {
|
|
6053
6029
|
assert: () => {}
|
|
6054
6030
|
}
|
|
@@ -6194,7 +6170,7 @@ class HttpFetchPlugin {
|
|
|
6194
6170
|
}
|
|
6195
6171
|
|
|
6196
6172
|
if (readObj.done) {
|
|
6197
|
-
goog$
|
|
6173
|
+
goog$2.asserts.assert(!readObj.value, 'readObj should be unset when "done" is true.');
|
|
6198
6174
|
controller.close();
|
|
6199
6175
|
} else {
|
|
6200
6176
|
controller.enqueue(readObj.value);
|
|
@@ -6433,6 +6409,664 @@ const fixDashManifest = (data, {
|
|
|
6433
6409
|
return new XMLSerializer().serializeToString(doc);
|
|
6434
6410
|
};
|
|
6435
6411
|
|
|
6412
|
+
/* eslint-disable no-cond-assign */
|
|
6413
|
+
|
|
6414
|
+
/* eslint-disable prefer-destructuring */
|
|
6415
|
+
|
|
6416
|
+
/*! @license
|
|
6417
|
+
* Shaka Player
|
|
6418
|
+
* Copyright 2016 Google LLC
|
|
6419
|
+
* SPDX-License-Identifier: Apache-2.0
|
|
6420
|
+
*/
|
|
6421
|
+
const goog$1 = {
|
|
6422
|
+
asserts: {
|
|
6423
|
+
assert: (condition, message) => {
|
|
6424
|
+
if (!condition) {
|
|
6425
|
+
console.warn('GOOG Assert!', message);
|
|
6426
|
+
}
|
|
6427
|
+
}
|
|
6428
|
+
}
|
|
6429
|
+
};
|
|
6430
|
+
|
|
6431
|
+
const addDefaultTextColor = styles => {
|
|
6432
|
+
const textColor = shaka.text.Cue.defaultTextColor;
|
|
6433
|
+
|
|
6434
|
+
for (const [key, value] of Object.entries(textColor)) {
|
|
6435
|
+
const cue = new shaka.text.Cue(0, 0, '');
|
|
6436
|
+
cue.color = value;
|
|
6437
|
+
styles.set(`.${key}`, cue);
|
|
6438
|
+
}
|
|
6439
|
+
|
|
6440
|
+
const bgColor = shaka.text.Cue.defaultTextBackgroundColor;
|
|
6441
|
+
|
|
6442
|
+
for (const [key, value] of Object.entries(bgColor)) {
|
|
6443
|
+
const cue = new shaka.text.Cue(0, 0, '');
|
|
6444
|
+
cue.backgroundColor = value;
|
|
6445
|
+
styles.set(`.${key}`, cue);
|
|
6446
|
+
}
|
|
6447
|
+
};
|
|
6448
|
+
|
|
6449
|
+
const parseTime = text => {
|
|
6450
|
+
var _Array$from;
|
|
6451
|
+
|
|
6452
|
+
const timeFormat = /(?:(\d{1,}):)?(\d{2}):(\d{2})((\.(\d{1,3})))?/g;
|
|
6453
|
+
const results = (_Array$from = Array.from(text.matchAll(timeFormat))) === null || _Array$from === void 0 ? void 0 : _Array$from[0];
|
|
6454
|
+
|
|
6455
|
+
if (results == null) {
|
|
6456
|
+
return null;
|
|
6457
|
+
} // This capture is optional, but will still be in the array as undefined,
|
|
6458
|
+
// in which case it is 0.
|
|
6459
|
+
|
|
6460
|
+
|
|
6461
|
+
const hours = Number(results[1]) || 0;
|
|
6462
|
+
const minutes = Number(results[2]);
|
|
6463
|
+
const seconds = Number(results[3]);
|
|
6464
|
+
const milliseconds = Number(results[6]) || 0;
|
|
6465
|
+
|
|
6466
|
+
if (minutes > 59 || seconds > 59) {
|
|
6467
|
+
return null;
|
|
6468
|
+
}
|
|
6469
|
+
|
|
6470
|
+
return milliseconds / 1000 + seconds + minutes * 60 + hours * 3600;
|
|
6471
|
+
};
|
|
6472
|
+
|
|
6473
|
+
const parseRegion = block => {
|
|
6474
|
+
if (block[0].trim() !== 'REGION') {
|
|
6475
|
+
return [];
|
|
6476
|
+
}
|
|
6477
|
+
|
|
6478
|
+
const region = new shaka.text.CueRegion();
|
|
6479
|
+
block.slice(1).forEach(word => {
|
|
6480
|
+
let results = null;
|
|
6481
|
+
|
|
6482
|
+
if (results = /^id:(.*)$/.exec(word)) {
|
|
6483
|
+
region.id = results[1];
|
|
6484
|
+
} else if (results = /^width:(\d{1,2}|100)%$/.exec(word)) {
|
|
6485
|
+
region.width = Number(results[1]);
|
|
6486
|
+
} else if (results = /^lines:(\d+)$/.exec(word)) {
|
|
6487
|
+
region.height = Number(results[1]);
|
|
6488
|
+
region.heightUnits = shaka.text.CueRegion.units.LINES;
|
|
6489
|
+
} else if (results = /^regionanchor:(\d{1,2}|100)%,(\d{1,2}|100)%$/.exec(word)) {
|
|
6490
|
+
region.regionAnchorX = Number(results[1]);
|
|
6491
|
+
region.regionAnchorY = Number(results[2]);
|
|
6492
|
+
} else if (results = /^viewportanchor:(\d{1,2}|100)%,(\d{1,2}|100)%$/.exec(word)) {
|
|
6493
|
+
region.viewportAnchorX = Number(results[1]);
|
|
6494
|
+
region.viewportAnchorY = Number(results[2]);
|
|
6495
|
+
} else if (results = /^scroll:up$/.exec(word)) {
|
|
6496
|
+
region.scroll = shaka.text.CueRegion.scrollMode.UP;
|
|
6497
|
+
} else {
|
|
6498
|
+
shaka.log.warning('VTT parser encountered an invalid VTTRegion setting: ', word, ' The setting will be ignored.');
|
|
6499
|
+
}
|
|
6500
|
+
});
|
|
6501
|
+
return [region];
|
|
6502
|
+
};
|
|
6503
|
+
/**
|
|
6504
|
+
* @implements {shaka.extern.TextParser}
|
|
6505
|
+
* @export
|
|
6506
|
+
*/
|
|
6507
|
+
|
|
6508
|
+
|
|
6509
|
+
class VttTextParser {
|
|
6510
|
+
/** Constructs a VTT parser. */
|
|
6511
|
+
constructor() {
|
|
6512
|
+
/** @private {boolean} */
|
|
6513
|
+
this.sequenceMode_ = false;
|
|
6514
|
+
/** @private {string} */
|
|
6515
|
+
|
|
6516
|
+
this.manifestType_ = shaka.media.ManifestParser.UNKNOWN;
|
|
6517
|
+
}
|
|
6518
|
+
/**
|
|
6519
|
+
* @override
|
|
6520
|
+
* @export
|
|
6521
|
+
*/
|
|
6522
|
+
|
|
6523
|
+
|
|
6524
|
+
parseInit() {
|
|
6525
|
+
goog$1.asserts.assert(false, 'VTT does not have init segments');
|
|
6526
|
+
}
|
|
6527
|
+
/**
|
|
6528
|
+
* @override
|
|
6529
|
+
* @export
|
|
6530
|
+
*/
|
|
6531
|
+
|
|
6532
|
+
|
|
6533
|
+
setSequenceMode(sequenceMode) {
|
|
6534
|
+
this.sequenceMode_ = sequenceMode;
|
|
6535
|
+
}
|
|
6536
|
+
/**
|
|
6537
|
+
* @override
|
|
6538
|
+
* @export
|
|
6539
|
+
*/
|
|
6540
|
+
|
|
6541
|
+
|
|
6542
|
+
setManifestType(manifestType) {
|
|
6543
|
+
this.manifestType_ = manifestType;
|
|
6544
|
+
}
|
|
6545
|
+
/**
|
|
6546
|
+
* @override
|
|
6547
|
+
* @export
|
|
6548
|
+
*/
|
|
6549
|
+
|
|
6550
|
+
|
|
6551
|
+
parseMedia(data, time) {
|
|
6552
|
+
// Get the input as a string. Normalize newlines to \n.
|
|
6553
|
+
let str = shaka.util.StringUtils.fromUTF8(data);
|
|
6554
|
+
str = str.replace(/\r\n|\r(?=[^\n]|$)/gm, '\n');
|
|
6555
|
+
const blocks = str.split(/\n{2,}/m);
|
|
6556
|
+
|
|
6557
|
+
if (!/^WEBVTT($|[ \t\n])/m.test(blocks[0])) {
|
|
6558
|
+
throw new shaka.util.Error(shaka.util.Error.Severity.CRITICAL, shaka.util.Error.Category.TEXT, shaka.util.Error.Code.INVALID_TEXT_HEADER);
|
|
6559
|
+
} // Depending on "segmentRelativeVttTiming" configuration,
|
|
6560
|
+
// "vttOffset" will correspond to either "periodStart" (default)
|
|
6561
|
+
// or "segmentStart", for segmented VTT where timings are relative
|
|
6562
|
+
// to the beginning of each segment.
|
|
6563
|
+
// NOTE: "periodStart" is the timestamp offset applied via TextEngine.
|
|
6564
|
+
// It is no longer closely tied to periods, but the name stuck around.
|
|
6565
|
+
// NOTE: This offset and the flag choosing its meaning have no effect on
|
|
6566
|
+
// HLS content, which should use X-TIMESTAMP-MAP and periodStart instead.
|
|
6567
|
+
|
|
6568
|
+
|
|
6569
|
+
let offset = time.vttOffset;
|
|
6570
|
+
|
|
6571
|
+
if (this.manifestType_ == shaka.media.ManifestParser.HLS) {
|
|
6572
|
+
// Only use 'X-TIMESTAMP-MAP' with HLS.
|
|
6573
|
+
if (blocks[0].includes('X-TIMESTAMP-MAP')) {
|
|
6574
|
+
offset = this.computeHlsOffset_(blocks[0], time);
|
|
6575
|
+
} else if (time.periodStart && time.vttOffset == time.periodStart) {
|
|
6576
|
+
// In the case where X-TIMESTAMP-MAP is not used and it is HLS, we
|
|
6577
|
+
// should not use offset unless segment-relative times are used.
|
|
6578
|
+
offset = 0;
|
|
6579
|
+
}
|
|
6580
|
+
}
|
|
6581
|
+
|
|
6582
|
+
const regions = [];
|
|
6583
|
+
/** @type {!Map<string, !shaka.text.Cue>} */
|
|
6584
|
+
|
|
6585
|
+
const styles = new Map();
|
|
6586
|
+
addDefaultTextColor(styles); // Parse cues.
|
|
6587
|
+
|
|
6588
|
+
const ret = [];
|
|
6589
|
+
|
|
6590
|
+
for (const block of blocks.slice(1)) {
|
|
6591
|
+
const lines = block.split('\n');
|
|
6592
|
+
VttTextParser.parseStyle_(lines, styles);
|
|
6593
|
+
regions.push(...parseRegion(lines));
|
|
6594
|
+
const cue = VttTextParser.parseCue_(lines, offset, regions, styles);
|
|
6595
|
+
|
|
6596
|
+
if (cue) {
|
|
6597
|
+
ret.push(cue);
|
|
6598
|
+
}
|
|
6599
|
+
}
|
|
6600
|
+
|
|
6601
|
+
return ret;
|
|
6602
|
+
}
|
|
6603
|
+
/**
|
|
6604
|
+
* @param {string} headerBlock Contains X-TIMESTAMP-MAP.
|
|
6605
|
+
* @param {shaka.extern.TextParser.TimeContext} time
|
|
6606
|
+
* @return {number}
|
|
6607
|
+
* @private
|
|
6608
|
+
*/
|
|
6609
|
+
|
|
6610
|
+
|
|
6611
|
+
computeHlsOffset_(headerBlock, time) {
|
|
6612
|
+
// https://bit.ly/2K92l7y
|
|
6613
|
+
// The 'X-TIMESTAMP-MAP' header is used in HLS to align text with
|
|
6614
|
+
// the rest of the media.
|
|
6615
|
+
// The header format is 'X-TIMESTAMP-MAP=MPEGTS:n,LOCAL:m'
|
|
6616
|
+
// (the attributes can go in any order)
|
|
6617
|
+
// where n is MPEG-2 time and m is cue time it maps to.
|
|
6618
|
+
// For example 'X-TIMESTAMP-MAP=LOCAL:00:00:00.000,MPEGTS:900000'
|
|
6619
|
+
// means an offset of 10 seconds
|
|
6620
|
+
// 900000/MPEG_TIMESCALE - cue time.
|
|
6621
|
+
const cueTimeMatch = headerBlock.match(/LOCAL:((?:(\d{1,}):)?(\d{2}):(\d{2})\.(\d{3}))/m);
|
|
6622
|
+
const mpegTimeMatch = headerBlock.match(/MPEGTS:(\d+)/m);
|
|
6623
|
+
|
|
6624
|
+
if (!cueTimeMatch || !mpegTimeMatch) {
|
|
6625
|
+
throw new shaka.util.Error(shaka.util.Error.Severity.CRITICAL, shaka.util.Error.Category.TEXT, shaka.util.Error.Code.INVALID_TEXT_HEADER);
|
|
6626
|
+
}
|
|
6627
|
+
|
|
6628
|
+
const cueTime = parseTime(cueTimeMatch[1]);
|
|
6629
|
+
|
|
6630
|
+
if (cueTime == null) {
|
|
6631
|
+
throw new shaka.util.Error(shaka.util.Error.Severity.CRITICAL, shaka.util.Error.Category.TEXT, shaka.util.Error.Code.INVALID_TEXT_HEADER);
|
|
6632
|
+
}
|
|
6633
|
+
|
|
6634
|
+
let mpegTime = Number(mpegTimeMatch[1]);
|
|
6635
|
+
const mpegTimescale = VttTextParser.MPEG_TIMESCALE_;
|
|
6636
|
+
const rolloverSeconds = VttTextParser.TS_ROLLOVER_ / mpegTimescale;
|
|
6637
|
+
let segmentStart = time.segmentStart - time.periodStart;
|
|
6638
|
+
|
|
6639
|
+
while (segmentStart >= rolloverSeconds) {
|
|
6640
|
+
segmentStart -= rolloverSeconds;
|
|
6641
|
+
mpegTime += VttTextParser.TS_ROLLOVER_;
|
|
6642
|
+
}
|
|
6643
|
+
|
|
6644
|
+
return time.periodStart + mpegTime / mpegTimescale - cueTime;
|
|
6645
|
+
}
|
|
6646
|
+
/**
|
|
6647
|
+
* Parses a style block into a Cue object.
|
|
6648
|
+
*
|
|
6649
|
+
* @param {!Array<string>} text
|
|
6650
|
+
* @param {!Map<string, !shaka.text.Cue>} styles
|
|
6651
|
+
* @private
|
|
6652
|
+
*/
|
|
6653
|
+
|
|
6654
|
+
|
|
6655
|
+
static parseStyle_(text, styles) {
|
|
6656
|
+
// Skip empty blocks.
|
|
6657
|
+
if (text.length == 1 && !text[0]) {
|
|
6658
|
+
return;
|
|
6659
|
+
} // Skip comment blocks.
|
|
6660
|
+
|
|
6661
|
+
|
|
6662
|
+
if (/^NOTE($|[ \t])/.test(text[0])) {
|
|
6663
|
+
return;
|
|
6664
|
+
} // Only style block are allowed.
|
|
6665
|
+
|
|
6666
|
+
|
|
6667
|
+
if (text[0] != 'STYLE') {
|
|
6668
|
+
return;
|
|
6669
|
+
}
|
|
6670
|
+
/** @type {!Array<!Array<string>>} */
|
|
6671
|
+
|
|
6672
|
+
|
|
6673
|
+
const styleBlocks = [];
|
|
6674
|
+
let lastBlockIndex = -1;
|
|
6675
|
+
|
|
6676
|
+
for (let i = 1; i < text.length; i++) {
|
|
6677
|
+
if (text[i].includes('::cue')) {
|
|
6678
|
+
styleBlocks.push([]);
|
|
6679
|
+
lastBlockIndex = styleBlocks.length - 1;
|
|
6680
|
+
}
|
|
6681
|
+
|
|
6682
|
+
if (lastBlockIndex == -1) {
|
|
6683
|
+
continue;
|
|
6684
|
+
}
|
|
6685
|
+
|
|
6686
|
+
styleBlocks[lastBlockIndex].push(text[i]);
|
|
6687
|
+
|
|
6688
|
+
if (text[i].includes('}')) {
|
|
6689
|
+
lastBlockIndex = -1;
|
|
6690
|
+
}
|
|
6691
|
+
}
|
|
6692
|
+
|
|
6693
|
+
for (const styleBlock of styleBlocks) {
|
|
6694
|
+
let styleSelector = 'global'; // Look for what is within parentheses. For example:
|
|
6695
|
+
// <code>:: cue (b) {</code>, what we are looking for is <code>b</code>
|
|
6696
|
+
|
|
6697
|
+
const selector = styleBlock[0].match(/\((.*)\)/);
|
|
6698
|
+
|
|
6699
|
+
if (selector) {
|
|
6700
|
+
styleSelector = selector.pop();
|
|
6701
|
+
} // We start at 1 to avoid '::cue' and end earlier to avoid '}'
|
|
6702
|
+
|
|
6703
|
+
|
|
6704
|
+
let propertyLines = styleBlock.slice(1, -1);
|
|
6705
|
+
|
|
6706
|
+
if (styleBlock[0].includes('}')) {
|
|
6707
|
+
const payload = /\{(.*?)\}/.exec(styleBlock[0]);
|
|
6708
|
+
|
|
6709
|
+
if (payload) {
|
|
6710
|
+
propertyLines = payload[1].split(';');
|
|
6711
|
+
}
|
|
6712
|
+
} // Continue styles over multiple selectors if necessary.
|
|
6713
|
+
// For example,
|
|
6714
|
+
// ::cue(b) { background: white; } ::cue(b) { color: blue; }
|
|
6715
|
+
// should set both the background and foreground of bold tags.
|
|
6716
|
+
|
|
6717
|
+
|
|
6718
|
+
let cue = styles.get(styleSelector);
|
|
6719
|
+
|
|
6720
|
+
if (!cue) {
|
|
6721
|
+
cue = new shaka.text.Cue(0, 0, '');
|
|
6722
|
+
}
|
|
6723
|
+
|
|
6724
|
+
let validStyle = false;
|
|
6725
|
+
|
|
6726
|
+
for (let i = 0; i < propertyLines.length; i++) {
|
|
6727
|
+
// We look for CSS properties. As a general rule they are separated by
|
|
6728
|
+
// <code>:</code>. Eg: <code>color: red;</code>
|
|
6729
|
+
const lineParts = /^\s*([^:]+):\s*(.*)/.exec(propertyLines[i]);
|
|
6730
|
+
|
|
6731
|
+
if (lineParts) {
|
|
6732
|
+
const name = lineParts[1].trim();
|
|
6733
|
+
const value = lineParts[2].trim().replace(';', '');
|
|
6734
|
+
|
|
6735
|
+
switch (name) {
|
|
6736
|
+
case 'background-color':
|
|
6737
|
+
case 'background':
|
|
6738
|
+
validStyle = true;
|
|
6739
|
+
cue.backgroundColor = value;
|
|
6740
|
+
break;
|
|
6741
|
+
|
|
6742
|
+
case 'color':
|
|
6743
|
+
validStyle = true;
|
|
6744
|
+
cue.color = value;
|
|
6745
|
+
break;
|
|
6746
|
+
|
|
6747
|
+
case 'font-family':
|
|
6748
|
+
validStyle = true;
|
|
6749
|
+
cue.fontFamily = value;
|
|
6750
|
+
break;
|
|
6751
|
+
|
|
6752
|
+
case 'font-size':
|
|
6753
|
+
validStyle = true;
|
|
6754
|
+
cue.fontSize = value;
|
|
6755
|
+
break;
|
|
6756
|
+
|
|
6757
|
+
case 'font-weight':
|
|
6758
|
+
if (parseInt(value, 10) >= 700 || value == 'bold') {
|
|
6759
|
+
validStyle = true;
|
|
6760
|
+
cue.fontWeight = shaka.text.Cue.fontWeight.BOLD;
|
|
6761
|
+
}
|
|
6762
|
+
|
|
6763
|
+
break;
|
|
6764
|
+
|
|
6765
|
+
case 'font-style':
|
|
6766
|
+
switch (value) {
|
|
6767
|
+
case 'normal':
|
|
6768
|
+
validStyle = true;
|
|
6769
|
+
cue.fontStyle = shaka.text.Cue.fontStyle.NORMAL;
|
|
6770
|
+
break;
|
|
6771
|
+
|
|
6772
|
+
case 'italic':
|
|
6773
|
+
validStyle = true;
|
|
6774
|
+
cue.fontStyle = shaka.text.Cue.fontStyle.ITALIC;
|
|
6775
|
+
break;
|
|
6776
|
+
|
|
6777
|
+
case 'oblique':
|
|
6778
|
+
validStyle = true;
|
|
6779
|
+
cue.fontStyle = shaka.text.Cue.fontStyle.OBLIQUE;
|
|
6780
|
+
break;
|
|
6781
|
+
}
|
|
6782
|
+
|
|
6783
|
+
break;
|
|
6784
|
+
|
|
6785
|
+
case 'opacity':
|
|
6786
|
+
validStyle = true;
|
|
6787
|
+
cue.opacity = parseFloat(value);
|
|
6788
|
+
break;
|
|
6789
|
+
|
|
6790
|
+
case 'text-combine-upright':
|
|
6791
|
+
validStyle = true;
|
|
6792
|
+
cue.textCombineUpright = value;
|
|
6793
|
+
break;
|
|
6794
|
+
|
|
6795
|
+
case 'text-shadow':
|
|
6796
|
+
validStyle = true;
|
|
6797
|
+
cue.textShadow = value;
|
|
6798
|
+
break;
|
|
6799
|
+
|
|
6800
|
+
case 'white-space':
|
|
6801
|
+
validStyle = true;
|
|
6802
|
+
cue.wrapLine = value != 'noWrap';
|
|
6803
|
+
break;
|
|
6804
|
+
|
|
6805
|
+
default:
|
|
6806
|
+
shaka.log.warning('VTT parser encountered an unsupported style: ', lineParts);
|
|
6807
|
+
break;
|
|
6808
|
+
}
|
|
6809
|
+
}
|
|
6810
|
+
}
|
|
6811
|
+
|
|
6812
|
+
if (validStyle) {
|
|
6813
|
+
styles.set(styleSelector, cue);
|
|
6814
|
+
}
|
|
6815
|
+
}
|
|
6816
|
+
}
|
|
6817
|
+
/**
|
|
6818
|
+
* Parses a text block into a Cue object.
|
|
6819
|
+
*
|
|
6820
|
+
* @param {!Array<string>} text
|
|
6821
|
+
* @param {number} timeOffset
|
|
6822
|
+
* @param {!Array<!shaka.text.CueRegion>} regions
|
|
6823
|
+
* @param {!Map<string, !shaka.text.Cue>} styles
|
|
6824
|
+
* @return {shaka.text.Cue}
|
|
6825
|
+
* @private
|
|
6826
|
+
*/
|
|
6827
|
+
|
|
6828
|
+
|
|
6829
|
+
static parseCue_(text, timeOffset, regions, styles) {
|
|
6830
|
+
// Skip empty blocks.
|
|
6831
|
+
if (text.length == 1 && !text[0]) {
|
|
6832
|
+
return null;
|
|
6833
|
+
} // Skip comment blocks.
|
|
6834
|
+
|
|
6835
|
+
|
|
6836
|
+
if (/^NOTE($|[ \t])/.test(text[0])) {
|
|
6837
|
+
return null;
|
|
6838
|
+
} // Skip style and region blocks.
|
|
6839
|
+
|
|
6840
|
+
|
|
6841
|
+
if (text[0] == 'STYLE' || text[0] == 'REGION') {
|
|
6842
|
+
return null;
|
|
6843
|
+
}
|
|
6844
|
+
|
|
6845
|
+
const skipIndex = text.findIndex(line => /^#/.test(line.trim()));
|
|
6846
|
+
|
|
6847
|
+
if (skipIndex > 0) {
|
|
6848
|
+
text = text.slice(skipIndex);
|
|
6849
|
+
}
|
|
6850
|
+
|
|
6851
|
+
if (text.length < 2) {
|
|
6852
|
+
return;
|
|
6853
|
+
}
|
|
6854
|
+
|
|
6855
|
+
let id = null;
|
|
6856
|
+
|
|
6857
|
+
if (!text[0].includes('-->')) {
|
|
6858
|
+
id = text[0];
|
|
6859
|
+
text.splice(0, 1);
|
|
6860
|
+
} // Parse the times.
|
|
6861
|
+
|
|
6862
|
+
|
|
6863
|
+
let [start, end] = text[0].split('-->').map(part => parseTime(part.trim()));
|
|
6864
|
+
|
|
6865
|
+
if (start == null || end == null) {
|
|
6866
|
+
shaka.log.alwaysWarn('Failed to parse VTT time code. Cue skipped:', id, text);
|
|
6867
|
+
return null;
|
|
6868
|
+
}
|
|
6869
|
+
|
|
6870
|
+
start += timeOffset;
|
|
6871
|
+
end += timeOffset; // Get the payload.
|
|
6872
|
+
|
|
6873
|
+
const payload = text.slice(1).join('\n').trim();
|
|
6874
|
+
let cue = null;
|
|
6875
|
+
|
|
6876
|
+
if (styles.has('global')) {
|
|
6877
|
+
cue = styles.get('global').clone();
|
|
6878
|
+
cue.startTime = start;
|
|
6879
|
+
cue.endTime = end;
|
|
6880
|
+
cue.payload = payload;
|
|
6881
|
+
} else {
|
|
6882
|
+
cue = new shaka.text.Cue(start, end, payload);
|
|
6883
|
+
} // Parse optional settings.
|
|
6884
|
+
|
|
6885
|
+
|
|
6886
|
+
text[0].split(/\s+/g).slice(3).forEach(word => {
|
|
6887
|
+
if (!word.trim()) {
|
|
6888
|
+
return;
|
|
6889
|
+
}
|
|
6890
|
+
|
|
6891
|
+
if (!VttTextParser.parseCueSetting(cue, word, regions)) {
|
|
6892
|
+
shaka.log.warning('VTT parser encountered an invalid VTT setting: ', word, ' The setting will be ignored.');
|
|
6893
|
+
}
|
|
6894
|
+
});
|
|
6895
|
+
shaka.text.Cue.parseCuePayload(cue, styles);
|
|
6896
|
+
|
|
6897
|
+
if (id != null) {
|
|
6898
|
+
cue.id = id;
|
|
6899
|
+
}
|
|
6900
|
+
|
|
6901
|
+
return cue;
|
|
6902
|
+
}
|
|
6903
|
+
/**
|
|
6904
|
+
* Parses a WebVTT setting from the given word.
|
|
6905
|
+
*
|
|
6906
|
+
* @param {!shaka.text.Cue} cue
|
|
6907
|
+
* @param {string} word
|
|
6908
|
+
* @param {!Array<!shaka.text.CueRegion>} regions
|
|
6909
|
+
* @return {boolean} True on success.
|
|
6910
|
+
*/
|
|
6911
|
+
|
|
6912
|
+
|
|
6913
|
+
static parseCueSetting(cue, word, regions) {
|
|
6914
|
+
let results = null;
|
|
6915
|
+
|
|
6916
|
+
if (results = /^align:(start|middle|center|end|left|right)$/.exec(word)) {
|
|
6917
|
+
VttTextParser.setTextAlign_(cue, results[1]);
|
|
6918
|
+
} else if (results = /^vertical:(lr|rl)$/.exec(word)) {
|
|
6919
|
+
VttTextParser.setVerticalWritingMode_(cue, results[1]);
|
|
6920
|
+
} else if (results = /^size:([\d.]+)%$/.exec(word)) {
|
|
6921
|
+
cue.size = Number(results[1]);
|
|
6922
|
+
} else if (results = /^position:([\d.]+)%(?:,(line-left|line-right|middle|center|start|end|auto))?$/.exec(word)) {
|
|
6923
|
+
cue.position = Number(results[1]);
|
|
6924
|
+
|
|
6925
|
+
if (results[2]) {
|
|
6926
|
+
VttTextParser.setPositionAlign_(cue, results[2]);
|
|
6927
|
+
}
|
|
6928
|
+
} else if (results = /^region:(.*)$/.exec(word)) {
|
|
6929
|
+
const region = VttTextParser.getRegionById_(regions, results[1]);
|
|
6930
|
+
|
|
6931
|
+
if (region) {
|
|
6932
|
+
cue.region = region;
|
|
6933
|
+
}
|
|
6934
|
+
} else {
|
|
6935
|
+
return VttTextParser.parsedLineValueAndInterpretation_(cue, word);
|
|
6936
|
+
}
|
|
6937
|
+
|
|
6938
|
+
return true;
|
|
6939
|
+
}
|
|
6940
|
+
/**
|
|
6941
|
+
*
|
|
6942
|
+
* @param {!Array<!shaka.text.CueRegion>} regions
|
|
6943
|
+
* @param {string} id
|
|
6944
|
+
* @return {?shaka.text.CueRegion}
|
|
6945
|
+
* @private
|
|
6946
|
+
*/
|
|
6947
|
+
|
|
6948
|
+
|
|
6949
|
+
static getRegionById_(regions, id) {
|
|
6950
|
+
const regionsWithId = regions.filter(region => region.id == id);
|
|
6951
|
+
|
|
6952
|
+
if (!regionsWithId.length) {
|
|
6953
|
+
shaka.log.warning('VTT parser could not find a region with id: ', id, ' The region will be ignored.');
|
|
6954
|
+
return null;
|
|
6955
|
+
}
|
|
6956
|
+
|
|
6957
|
+
goog$1.asserts.assert(regionsWithId.length == 1, 'VTTRegion ids should be unique!');
|
|
6958
|
+
return regionsWithId[0];
|
|
6959
|
+
}
|
|
6960
|
+
/**
|
|
6961
|
+
* @param {!shaka.text.Cue} cue
|
|
6962
|
+
* @param {string} align
|
|
6963
|
+
* @private
|
|
6964
|
+
*/
|
|
6965
|
+
|
|
6966
|
+
|
|
6967
|
+
static setTextAlign_(cue, align) {
|
|
6968
|
+
const Cue = shaka.text.Cue;
|
|
6969
|
+
|
|
6970
|
+
if (align == 'middle') {
|
|
6971
|
+
cue.textAlign = Cue.textAlign.CENTER;
|
|
6972
|
+
} else {
|
|
6973
|
+
goog$1.asserts.assert(align.toUpperCase() in Cue.textAlign, `${align.toUpperCase()} Should be in Cue.textAlign values!`);
|
|
6974
|
+
cue.textAlign = Cue.textAlign[align.toUpperCase()];
|
|
6975
|
+
}
|
|
6976
|
+
}
|
|
6977
|
+
/**
|
|
6978
|
+
* @param {!shaka.text.Cue} cue
|
|
6979
|
+
* @param {string} align
|
|
6980
|
+
* @private
|
|
6981
|
+
*/
|
|
6982
|
+
|
|
6983
|
+
|
|
6984
|
+
static setPositionAlign_(cue, align) {
|
|
6985
|
+
const Cue = shaka.text.Cue;
|
|
6986
|
+
|
|
6987
|
+
if (align == 'line-left' || align == 'start') {
|
|
6988
|
+
cue.positionAlign = Cue.positionAlign.LEFT;
|
|
6989
|
+
} else if (align == 'line-right' || align == 'end') {
|
|
6990
|
+
cue.positionAlign = Cue.positionAlign.RIGHT;
|
|
6991
|
+
} else if (align == 'center' || align == 'middle') {
|
|
6992
|
+
cue.positionAlign = Cue.positionAlign.CENTER;
|
|
6993
|
+
} else {
|
|
6994
|
+
cue.positionAlign = Cue.positionAlign.AUTO;
|
|
6995
|
+
}
|
|
6996
|
+
}
|
|
6997
|
+
/**
|
|
6998
|
+
* @param {!shaka.text.Cue} cue
|
|
6999
|
+
* @param {string} value
|
|
7000
|
+
* @private
|
|
7001
|
+
*/
|
|
7002
|
+
|
|
7003
|
+
|
|
7004
|
+
static setVerticalWritingMode_(cue, value) {
|
|
7005
|
+
const Cue = shaka.text.Cue;
|
|
7006
|
+
|
|
7007
|
+
if (value == 'lr') {
|
|
7008
|
+
cue.writingMode = Cue.writingMode.VERTICAL_LEFT_TO_RIGHT;
|
|
7009
|
+
} else {
|
|
7010
|
+
cue.writingMode = Cue.writingMode.VERTICAL_RIGHT_TO_LEFT;
|
|
7011
|
+
}
|
|
7012
|
+
}
|
|
7013
|
+
/**
|
|
7014
|
+
* @param {!shaka.text.Cue} cue
|
|
7015
|
+
* @param {string} word
|
|
7016
|
+
* @return {boolean}
|
|
7017
|
+
* @private
|
|
7018
|
+
*/
|
|
7019
|
+
|
|
7020
|
+
|
|
7021
|
+
static parsedLineValueAndInterpretation_(cue, word) {
|
|
7022
|
+
const Cue = shaka.text.Cue;
|
|
7023
|
+
let results = null;
|
|
7024
|
+
|
|
7025
|
+
if (results = /^line:([\d.]+)%(?:,(start|end|center))?$/.exec(word)) {
|
|
7026
|
+
cue.lineInterpretation = Cue.lineInterpretation.PERCENTAGE;
|
|
7027
|
+
cue.line = Number(results[1]);
|
|
7028
|
+
|
|
7029
|
+
if (results[2]) {
|
|
7030
|
+
goog$1.asserts.assert(results[2].toUpperCase() in Cue.lineAlign, `${results[2].toUpperCase()} Should be in Cue.lineAlign values!`);
|
|
7031
|
+
cue.lineAlign = Cue.lineAlign[results[2].toUpperCase()];
|
|
7032
|
+
}
|
|
7033
|
+
} else if (results = /^line:(-?\d+)(?:,(start|end|center))?$/.exec(word)) {
|
|
7034
|
+
cue.lineInterpretation = Cue.lineInterpretation.LINE_NUMBER;
|
|
7035
|
+
cue.line = Number(results[1]);
|
|
7036
|
+
|
|
7037
|
+
if (results[2]) {
|
|
7038
|
+
goog$1.asserts.assert(results[2].toUpperCase() in Cue.lineAlign, `${results[2].toUpperCase()} Should be in Cue.lineAlign values!`);
|
|
7039
|
+
cue.lineAlign = Cue.lineAlign[results[2].toUpperCase()];
|
|
7040
|
+
}
|
|
7041
|
+
} else {
|
|
7042
|
+
return false;
|
|
7043
|
+
}
|
|
7044
|
+
|
|
7045
|
+
return true;
|
|
7046
|
+
}
|
|
7047
|
+
|
|
7048
|
+
}
|
|
7049
|
+
/**
|
|
7050
|
+
* @const {number}
|
|
7051
|
+
* @private
|
|
7052
|
+
*/
|
|
7053
|
+
|
|
7054
|
+
|
|
7055
|
+
VttTextParser.MPEG_TIMESCALE_ = 90000;
|
|
7056
|
+
/**
|
|
7057
|
+
* At this value, timestamps roll over in TS content.
|
|
7058
|
+
* @const {number}
|
|
7059
|
+
* @private
|
|
7060
|
+
*/
|
|
7061
|
+
|
|
7062
|
+
VttTextParser.TS_ROLLOVER_ = 0x200000000;
|
|
7063
|
+
|
|
7064
|
+
VttTextParser.register = shakaNameSpace => {
|
|
7065
|
+
shakaNameSpace.text.TextEngine.registerParser('text/vtt', () => new VttTextParser());
|
|
7066
|
+
shakaNameSpace.text.TextEngine.registerParser('text/vtt; codecs="vtt"', () => new VttTextParser());
|
|
7067
|
+
shakaNameSpace.text.TextEngine.registerParser('text/vtt; codecs="wvtt"', () => new VttTextParser());
|
|
7068
|
+
};
|
|
7069
|
+
|
|
6436
7070
|
/*! @license
|
|
6437
7071
|
* Shaka Player
|
|
6438
7072
|
* Copyright 2016 Google LLC
|
|
@@ -7322,9 +7956,13 @@ class UITextDisplayer {
|
|
|
7322
7956
|
} else if (cue.textAlign == Cue.textAlign.RIGHT || cue.textAlign == Cue.textAlign.END) {
|
|
7323
7957
|
style.width = '100%';
|
|
7324
7958
|
style.alignItems = 'end';
|
|
7325
|
-
}
|
|
7959
|
+
} // in VTT displayAlign can't be set, it defaults to 'after',
|
|
7960
|
+
// but for vertical, it should be 'before'
|
|
7961
|
+
|
|
7326
7962
|
|
|
7327
|
-
if (cue.
|
|
7963
|
+
if (/vertical/.test(cue.writingMode)) {
|
|
7964
|
+
style.justifyContent = 'flex-start';
|
|
7965
|
+
} else if (cue.displayAlign == Cue.displayAlign.BEFORE) {
|
|
7328
7966
|
style.justifyContent = 'flex-start';
|
|
7329
7967
|
} else if (cue.displayAlign == Cue.displayAlign.CENTER) {
|
|
7330
7968
|
style.justifyContent = 'center';
|
|
@@ -7697,6 +8335,7 @@ const loadShaka = async (videoElement, config = {}, options = {}) => {
|
|
|
7697
8335
|
return getQualityItem(activeTrack);
|
|
7698
8336
|
};
|
|
7699
8337
|
|
|
8338
|
+
VttTextParser.register(shaka);
|
|
7700
8339
|
HttpFetchPlugin.register(shaka);
|
|
7701
8340
|
const networkEngine = player.getNetworkingEngine();
|
|
7702
8341
|
networkEngine.registerRequestFilter((type, request) => extensionOptions.requestHandlers.reduce((merged, handler) => handler(type, merged, player, extensionOptions), request));
|
|
@@ -7756,7 +8395,7 @@ const loadPlayer = async (videoElement, {
|
|
|
7756
8395
|
// TODO unsubscribe to release the video element
|
|
7757
8396
|
// when resuming from background, video may play without no media events
|
|
7758
8397
|
// on some iOS devices, pause again to workaround
|
|
7759
|
-
if (
|
|
8398
|
+
if (isIOS()) {
|
|
7760
8399
|
on(document, 'visibilitychange', () => setTimeout(() => videoElement.pause(), 50));
|
|
7761
8400
|
}
|
|
7762
8401
|
|
|
@@ -7793,6 +8432,16 @@ const loadPlayer = async (videoElement, {
|
|
|
7793
8432
|
return player;
|
|
7794
8433
|
};
|
|
7795
8434
|
|
|
8435
|
+
/* eslint-disable no-param-reassign */
|
|
8436
|
+
|
|
8437
|
+
const selectVideoResolution = (_media, {
|
|
8438
|
+
player
|
|
8439
|
+
}, restrictions) => {
|
|
8440
|
+
player.preferredSettings.videoResolution = restrictions;
|
|
8441
|
+
player.configure('abr.restrictions', restrictions);
|
|
8442
|
+
};
|
|
8443
|
+
// for unit test
|
|
8444
|
+
|
|
7796
8445
|
const baseVideoStyle = {
|
|
7797
8446
|
objectFit: 'contain',
|
|
7798
8447
|
width: '100%',
|
|
@@ -7818,9 +8467,9 @@ const Video = ({
|
|
|
7818
8467
|
currentTime: targetTime,
|
|
7819
8468
|
playbackRate,
|
|
7820
8469
|
videoResolution,
|
|
8470
|
+
audioTrack = {},
|
|
7821
8471
|
textTracks = emptyArray,
|
|
7822
8472
|
textTrack,
|
|
7823
|
-
audio = {},
|
|
7824
8473
|
liveResume,
|
|
7825
8474
|
iOSFullscreenDetectThreshold,
|
|
7826
8475
|
// TODO organize props
|
|
@@ -7837,7 +8486,7 @@ const Video = ({
|
|
|
7837
8486
|
onPlaylogFired,
|
|
7838
8487
|
...videoAttributes
|
|
7839
8488
|
}) => {
|
|
7840
|
-
var _videoElement$current;
|
|
8489
|
+
var _videoElement$current, _videoElement$current2;
|
|
7841
8490
|
|
|
7842
8491
|
const handlers = useRef();
|
|
7843
8492
|
handlers.current = {
|
|
@@ -7938,11 +8587,9 @@ const Video = ({
|
|
|
7938
8587
|
}, videoResolution);
|
|
7939
8588
|
}
|
|
7940
8589
|
}, [videoResolution, player]);
|
|
7941
|
-
useEffect(() => {
|
|
7942
|
-
|
|
7943
|
-
|
|
7944
|
-
}, audio);
|
|
7945
|
-
}, [(audio === null || audio === void 0 ? void 0 : audio.id) || `${audio.language || 'unk'}:${audio.label || ''}`, player]);
|
|
8590
|
+
useEffect(() => setAudioTrack(videoElement.current, {
|
|
8591
|
+
player
|
|
8592
|
+
}, audioTrack), [((_videoElement$current = videoElement.current) === null || _videoElement$current === void 0 ? void 0 : _videoElement$current.readyState) >= 2 && audioTrack.label, audioTrack.language]);
|
|
7946
8593
|
useEffect(() => {
|
|
7947
8594
|
if ((textTracks === null || textTracks === void 0 ? void 0 : textTracks.length) > 0 || textTrack) {
|
|
7948
8595
|
return syncTextTrack(videoElement.current, {
|
|
@@ -7951,12 +8598,12 @@ const Video = ({
|
|
|
7951
8598
|
textTracks
|
|
7952
8599
|
} : textTrack);
|
|
7953
8600
|
}
|
|
7954
|
-
}, [((_videoElement$
|
|
8601
|
+
}, [((_videoElement$current2 = videoElement.current) === null || _videoElement$current2 === void 0 ? void 0 : _videoElement$current2.readyState) >= 2 && textTracks, textTrack]);
|
|
7955
8602
|
return jsx$1("div", {
|
|
7956
8603
|
ref: videoContainer,
|
|
7957
8604
|
css: /*#__PURE__*/css({
|
|
7958
8605
|
video: baseVideoStyle
|
|
7959
|
-
}, process.env.NODE_ENV === "production" ? "" : ";label:Video;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
8606
|
+
}, process.env.NODE_ENV === "production" ? "" : ";label:Video;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIlZpZGVvLmpzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQW1LOEIiLCJmaWxlIjoiVmlkZW8uanMiLCJzb3VyY2VzQ29udGVudCI6WyIvKiBlc2xpbnQtZGlzYWJsZSBuby1wYXJhbS1yZWFzc2lnbiAqL1xuLyogQGpzeEltcG9ydFNvdXJjZSBAZW1vdGlvbi9yZWFjdCAqL1xuaW1wb3J0IHt1c2VFZmZlY3QsIHVzZUxheW91dEVmZmVjdCwgdXNlUmVmLCB1c2VTdGF0ZX0gZnJvbSAncmVhY3QnXG5cbmltcG9ydCBtdWx0aVJlZiBmcm9tICd1dGlsL211bHRpUmVmJ1xuaW1wb3J0IGxvYWRQbGF5ZXIgZnJvbSAncGxheWVyQ29yZS9sb2FkUGxheWVyJ1xuaW1wb3J0IHtcbiAgZ2V0TWVkaWFUaW1lLFxuICBzdWJzY3JpYmVQbGF5YmFja1N0YXRlLFxuICBsb2FkLFxuICBzeW5jUGxheWJhY2tTdGF0ZSxcbiAgc2VlayxcbiAgc2V0UGxheWJhY2tSYXRlLFxuICBzZXRBdWRpb1RyYWNrLFxuICBzeW5jVGV4dFRyYWNrLFxufSBmcm9tICdwbGF5ZXJDb3JlL21lZGlhQmluZGluZ3MnXG5pbXBvcnQge3NlbGVjdFZpZGVvUmVzb2x1dGlvbn0gZnJvbSAncGxheWVyQ29yZS9hZGFwdGF0aW9uJ1xuXG5jb25zdCBiYXNlVmlkZW9TdHlsZSA9IHtcbiAgb2JqZWN0Rml0OiAnY29udGFpbicsXG4gIHdpZHRoOiAnMTAwJScsXG4gIG1heEhlaWdodDogJzEwMCUnLFxuICAnJjpmdWxsc2NyZWVuJzoge1xuICAgIHBvc2l0aW9uOiAnYWJzb2x1dGUnLFxuICAgIHRvcDogMCxcbiAgICBsZWZ0OiAwLFxuICAgIGJvdHRvbTogMCxcbiAgICByaWdodDogMCxcbiAgfSxcbiAgJ34gLnNoYWthLXRleHQtY29udGFpbmVyJzoge1xuICAgIHBvc2l0aW9uOiAnYWJzb2x1dGUnLFxuICAgIGhlaWdodDogJzEwMCUnLFxuICAgIHdpZHRoOiAnMTAwJScsXG4gIH0sXG59XG5cbmNvbnN0IGVtcHR5QXJyYXkgPSBbXVxuXG5jb25zdCBWaWRlbyA9ICh7XG4gIHNvdXJjZSxcbiAgcGxheWJhY2tTdGF0ZTogdGFyZ2V0U3RhdGUsXG4gIGN1cnJlbnRUaW1lOiB0YXJnZXRUaW1lLFxuICBwbGF5YmFja1JhdGUsXG4gIHZpZGVvUmVzb2x1dGlvbixcbiAgYXVkaW9UcmFjayA9IHt9LFxuICB0ZXh0VHJhY2tzID0gZW1wdHlBcnJheSxcbiAgdGV4dFRyYWNrLFxuICBsaXZlUmVzdW1lLFxuICBpT1NGdWxsc2NyZWVuRGV0ZWN0VGhyZXNob2xkLCAvLyBUT0RPIG9yZ2FuaXplIHByb3BzXG4gIHBsdWdpbnMgPSBbXSxcbiAgc2hha2EsXG4gIGJpdG1vdmluLFxuICBsaWNlbnNlLFxuICB2YWxpZGF0aW9uSG9zdCxcbiAgdmlkZW9SZWYsXG4gIHBsYXllclJlZixcbiAgb25QbGF5ZXJMb2FkZWQsXG4gIG9uUGxheWJhY2tTdGF0ZUNoYW5nZSxcbiAgb25CbG9ja2VkQXV0b3BsYXksXG4gIG9uUGxheWxvZ0ZpcmVkLFxuICAuLi52aWRlb0F0dHJpYnV0ZXNcbn0pID0+IHtcbiAgY29uc3QgaGFuZGxlcnMgPSB1c2VSZWYoKVxuICBoYW5kbGVycy5jdXJyZW50ID0ge29uUGxheWJhY2tTdGF0ZUNoYW5nZSwgb25CbG9ja2VkQXV0b3BsYXl9XG4gIGNvbnN0IHZpZGVvQ29udGFpbmVyID0gdXNlUmVmKClcbiAgY29uc3QgdmlkZW9FbGVtZW50ID0gdXNlUmVmKClcbiAgY29uc3QgW3BsYXliYWNrU3RhdGUsIHNldFBsYXliYWNrU3RhdGVdID0gdXNlU3RhdGUoJycpXG4gIGNvbnN0IFtwbGF5ZXIsIHNldFBsYXllcl0gPSB1c2VTdGF0ZSgpXG5cbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBjb25zdCBsb2FkVGFzayA9IGxvYWRQbGF5ZXIodmlkZW9FbGVtZW50LmN1cnJlbnQsIHtcbiAgICAgIGNvbnRhaW5lcjogdmlkZW9Db250YWluZXIuY3VycmVudCxcbiAgICAgIHNvdXJjZSxcbiAgICAgIHNoYWthLFxuICAgICAgYml0bW92aW4sXG4gICAgfSkudGhlbihiYXNlUGxheWVyID0+IHtcbiAgICAgIHNldFBsYXllcihiYXNlUGxheWVyKVxuICAgICAgb25QbGF5ZXJMb2FkZWQ/LihiYXNlUGxheWVyKVxuICAgICAgaWYgKHBsYXllclJlZikge1xuICAgICAgICBwbGF5ZXJSZWYuY3VycmV0ID0gYmFzZVBsYXllclxuICAgICAgfVxuICAgICAgcmV0dXJuIGJhc2VQbGF5ZXJcbiAgICB9KVxuICAgIGNvbnN0IG9mZlBsYXliYWNrU3RhdGUgPSBzdWJzY3JpYmVQbGF5YmFja1N0YXRlKFxuICAgICAgdmlkZW9FbGVtZW50LmN1cnJlbnQsXG4gICAgICAoZXZlbnQsIHN0YXRlKSA9PiB7XG4gICAgICAgIGhhbmRsZXJzLmN1cnJlbnQub25QbGF5YmFja1N0YXRlQ2hhbmdlPy4oZXZlbnQsIHN0YXRlKVxuICAgICAgICAvLyBleHRlcm5hbCBsb2dpYyBtYXkgd2FudCB0byBjaGFuZ2UgdGFyZ2V0U3RhdGUsIGhvbGQgcGxheWJhY2tTdGF0ZSB1cGRhdGVcbiAgICAgICAgLy8gdG8gcHJldmVudCB1bndhbnRlZCBzeW5jUGxheWJhY2tTdGF0ZVxuICAgICAgICByZXF1ZXN0QW5pbWF0aW9uRnJhbWUoKCkgPT4gc2V0UGxheWJhY2tTdGF0ZShzdGF0ZSkpXG4gICAgICB9LFxuICAgICAge2lPU0Z1bGxzY3JlZW5EZXRlY3RUaHJlc2hvbGR9XG4gICAgKVxuICAgIHJldHVybiAoKSA9PiB7XG4gICAgICBvZmZQbGF5YmFja1N0YXRlKClcbiAgICAgIGxvYWRUYXNrLnRoZW4oY3VycmVudFBsYXllciA9PiBjdXJyZW50UGxheWVyPy5kZXN0cm95KCkpXG4gICAgfVxuICB9LCBbXSlcbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBpZiAoXG4gICAgICBzb3VyY2UgJiZcbiAgICAgIChzb3VyY2UubGVuZ3RoID4gMCB8fCBzb3VyY2Uuc3JjIHx8IHNvdXJjZS5obHMgfHwgc291cmNlLmRhc2gpICYmXG4gICAgICBwbGF5ZXI/LmlzQWxpdmUoKVxuICAgICkge1xuICAgICAgUHJvbWlzZS5yZXNvbHZlKGZhbHNlKVxuICAgICAgICAudGhlbigoKSA9PlxuICAgICAgICAgIGxvYWQoXG4gICAgICAgICAgICB2aWRlb0VsZW1lbnQuY3VycmVudCxcbiAgICAgICAgICAgIHtwbGF5ZXIsIHBsdWdpbnMsIHN0YXJ0VGltZTogdGFyZ2V0VGltZX0sXG4gICAgICAgICAgICBzb3VyY2VcbiAgICAgICAgICApXG4gICAgICAgIClcbiAgICAgICAgLmNhdGNoKGVycm9yID0+IGNvbnNvbGUud2FybihlcnJvcikpXG4gICAgfVxuICAgIGlmICgoIXNvdXJjZSB8fCBzb3VyY2UubGVuZ3RoID09PSAwKSAmJiBwbGF5ZXIpIHtcbiAgICAgIHBsYXllci51bmxvYWQoKVxuICAgICAgcGxheWVyLmxhc3RTcmMgPSAnJ1xuICAgIH1cbiAgfSwgW3BsYXllciwgc291cmNlXSlcblxuICAvLyB1c2VFZmZlY3QgaXMgdG9vIGxhdGUgdG8gdW5sb2NrIHBsYXkgb24gU2FmYXJpXG4gIC8vIFRPRE8gY2hlY2sgaWYgdGhpcyB3b3JrIGFmdGVyIHVwZ3JhZGluZyBSZWFjdCAxOFxuICB1c2VMYXlvdXRFZmZlY3QoKCkgPT4ge1xuICAgIGlmIChwbGF5ZXI/LmlzQWxpdmUoKSkge1xuICAgICAgc3luY1BsYXliYWNrU3RhdGUoXG4gICAgICAgIHZpZGVvRWxlbWVudC5jdXJyZW50LFxuICAgICAgICB7cGxheWVyLCBwbHVnaW5zLCBsaXZlUmVzdW1lfSxcbiAgICAgICAgdGFyZ2V0U3RhdGVcbiAgICAgICk/LmNhdGNoKGVycm9yID0+IHtcbiAgICAgICAgaGFuZGxlcnMuY3VycmVudC5vbkJsb2NrZWRBdXRvcGxheT8uKGVycm9yKVxuICAgICAgfSlcbiAgICB9XG4gIH0sIFtwbGF5ZXIgJiYgdGFyZ2V0U3RhdGVdKVxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGNvbnN0IHtjdXJyZW50VGltZX0gPSBnZXRNZWRpYVRpbWUodmlkZW9FbGVtZW50LmN1cnJlbnQsIHtwbGF5ZXIsIHBsdWdpbnN9KVxuICAgIGlmIChwbGF5ZXI/LmlzQWxpdmUoKSAmJiBNYXRoLmFicyhjdXJyZW50VGltZSAtIHRhcmdldFRpbWUpID4gMC41KSB7XG4gICAgICAvLyBzZWVraW5nIHVuYXZhaWxhYmxlIGNhc2VzIGFyZSBoYW5kbGVkIGJ5IHNlZWsgZnVuY3Rpb25cbiAgICAgIHNlZWsodmlkZW9FbGVtZW50LmN1cnJlbnQsIHtwbGF5ZXIsIHBsdWdpbnN9LCB0YXJnZXRUaW1lKVxuICAgIH1cbiAgfSwgW3BsYXllciAmJiB0YXJnZXRUaW1lXSlcbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBzZXRQbGF5YmFja1JhdGUodmlkZW9FbGVtZW50LmN1cnJlbnQsIHtwbGF5ZXJ9LCBwbGF5YmFja1JhdGUpXG4gIH0sIFtwbGF5YmFja1JhdGUsIHBsYXliYWNrU3RhdGUgPT09ICdwbGF5aW5nJ10pXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKHBsYXllcikge1xuICAgICAgc2VsZWN0VmlkZW9SZXNvbHV0aW9uKHZpZGVvRWxlbWVudC5jdXJyZW50LCB7cGxheWVyfSwgdmlkZW9SZXNvbHV0aW9uKVxuICAgIH1cbiAgfSwgW3ZpZGVvUmVzb2x1dGlvbiwgcGxheWVyXSlcbiAgdXNlRWZmZWN0KFxuICAgICgpID0+IHNldEF1ZGlvVHJhY2sodmlkZW9FbGVtZW50LmN1cnJlbnQsIHtwbGF5ZXJ9LCBhdWRpb1RyYWNrKSxcbiAgICBbdmlkZW9FbGVtZW50LmN1cnJlbnQ/LnJlYWR5U3RhdGUgPj0gMiAmJiAgYXVkaW9UcmFjay5sYWJlbCwgYXVkaW9UcmFjay5sYW5ndWFnZV1cbiAgKVxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGlmICh0ZXh0VHJhY2tzPy5sZW5ndGggPiAwIHx8IHRleHRUcmFjaykge1xuICAgICAgcmV0dXJuIHN5bmNUZXh0VHJhY2soXG4gICAgICAgIHZpZGVvRWxlbWVudC5jdXJyZW50LFxuICAgICAgICB7cGxheWVyfSxcbiAgICAgICAgdGV4dFRyYWNrcz8ubGVuZ3RoID4gMCA/IHt0ZXh0VHJhY2tzfSA6IHRleHRUcmFja1xuICAgICAgKVxuICAgIH1cbiAgfSwgW3ZpZGVvRWxlbWVudC5jdXJyZW50Py5yZWFkeVN0YXRlID49IDIgJiYgdGV4dFRyYWNrcywgdGV4dFRyYWNrXSlcblxuICByZXR1cm4gKFxuICAgIDxkaXYgcmVmPXt2aWRlb0NvbnRhaW5lcn0gY3NzPXt7dmlkZW86IGJhc2VWaWRlb1N0eWxlfX0+XG4gICAgICA8dmlkZW8gLy8gZXNsaW50LWRpc2FibGUtbGluZSBqc3gtYTExeS9tZWRpYS1oYXMtY2FwdGlvblxuICAgICAgICByZWY9e211bHRpUmVmKHZpZGVvUmVmLCB2aWRlb0VsZW1lbnQpfVxuICAgICAgICBwbGF5c0lubGluZVxuICAgICAgICB7Li4udmlkZW9BdHRyaWJ1dGVzfVxuICAgICAgLz5cbiAgICA8L2Rpdj5cbiAgKVxufVxuXG5leHBvcnQgZGVmYXVsdCBWaWRlb1xuIl19 */"),
|
|
7960
8607
|
children: jsx$1("video", {
|
|
7961
8608
|
// eslint-disable-line jsx-a11y/media-has-caption
|
|
7962
8609
|
ref: multiRef(videoRef, videoElement),
|
|
@@ -8149,14 +8796,14 @@ const shouldPause = ({
|
|
|
8149
8796
|
|
|
8150
8797
|
|
|
8151
8798
|
const menuClasses = {
|
|
8152
|
-
default: /*#__PURE__*/css$1(LanguageMenu.styles, process.env.NODE_ENV === "production" ? "" : ";label:default;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["PremiumPlayer.js"],"names":[],"mappings":"AAuEW","file":"PremiumPlayer.js","sourcesContent":["/* eslint-disable no-nested-ternary */\nimport {useEffect, useMemo, useRef, useState} from 'react'\nimport {css} from '@emotion/css'\nimport useDimensions from 'react-cool-dimensions'\nimport {ResizeObserver} from '@juggle/resize-observer'\n\nimport {handleError} from 'playerCore/errors'\nimport {on} from 'util/events'\nimport {havePointer, isDesktop} from 'util/environment'\nimport multiRef from 'util/multiRef'\nimport {onViewModeChange, toggleFullscreen} from 'util/viewModes'\nimport {useAutoHide} from 'hooks'\nimport {IntlProvider} from 'context/I18n'\nimport {getMediaTime, isBuffered} from 'playerCore/mediaBindings'\nimport Error from 'playerUi/Error'\nimport Layout from 'playerUi/DefaultLayout'\nimport {\n  Button,\n  LiveButton,\n  FullscreenButton,\n  PlayButton,\n  ForwardButton,\n} from 'playerUi/buttons'\nimport Seekbar from 'playerUi/Seekbar'\nimport DisplayTime from 'playerUi/DisplayTime'\nimport LoadingSpinner from 'playerUi/LoadingSpinner'\nimport Backdrop from 'playerUi/Backdrop'\nimport Settings from 'playerUi/Settings'\nimport OverlayPanel from 'playerUi/OverlayPanel'\nimport LanguageMenu from 'playerUi/LanguageMenu'\nimport PlayPanel from 'playerUi/PlayPane'\nimport VolumeControl from 'component/VolumeControl'\nimport CastUi from 'cast/CastUi'\nimport {linkCast} from 'cast/framework'\nimport {CastState} from 'Enum'\nimport {dispatchChapterEvents} from './timeline'\nimport Video from '../Video'\nimport {getLanguageOptions, getQualityOptions, getSettingsData} from './settings'\nimport {linkMediaVolume, syncVolume} from './volume'\nimport SeekPreview from './SeekPreview'\n\nconst sizes = {\n  'small-embed': 200,\n  embed: 400,\n  'tablet-portrait': 600,\n  'tablet-landscape': 900,\n  desktop: 1200,\n}\n\nconst useLinkState = (request, dependencies = []) => {\n  const [state, setState] = useState()\n  useEffect(() => {\n    request(setState)\n  }, dependencies)\n  return state\n}\n\nconst flipState = state => (state === 'playing' ? 'paused' : 'playing')\n\n// FIXME: too few lines to split a file, looking a better place\nconst getThumbnailsUrl = source =>\n  [].concat(source).find(item => item.type === 'thumbnail')?.src\n\nconst defaultSettings = {}\n\nconst shouldPause = ({userFocus, uiType}) =>\n  (uiType === 'mobile' && /settings|chapter-list/.test(userFocus)) ||\n  /blocking|seekbar|share-inner/.test(userFocus)\n\n// TODO extract to somewhere else\nconst menuClasses = {\n  default: css(LanguageMenu.styles),\n  desktop: css(LanguageMenu.styles, LanguageMenu.desktopStyles),\n}\n\nconst LanguageSettings = ({\n  uiType,\n  audioTracks = [],\n  subtitleTracks = [],\n  getPlayer,\n  onChange,\n  slots = {LanguageMenu},\n  ...rest\n}) => {\n  const options = getLanguageOptions(getPlayer(), {audioTracks, subtitleTracks})\n  return (\n    <OverlayPanel buttonIcon=\"subtitle\">\n      <slots.LanguageMenu\n        uiType={uiType}\n        classes={menuClasses}\n        sectionOptions={[options.audioTracks, options.textTracks]}\n        onChange={onChange}\n        {...rest}\n      />\n    </OverlayPanel>\n  )\n}\n\nconst PremiumPlayer = ({\n  source,\n  startTime,\n  quality = {},\n  title,\n  metadata,\n  channelTitle,\n  chapters,\n  playbackState: appPlaybackState,\n  currentTime: appCurrentTime,\n  playbackRate: appPlaybackRate,\n  loop,\n  volume: appVolume,\n  thumbnailsUrl,\n  controls = {autohide: 3000},\n  marks = [],\n  // TODO sectionId\n  intl,\n  settings: appSettings = defaultSettings,\n  blocking,\n  plugins = [],\n  modulesConfig,\n\n  uiType = isDesktop() ? 'desktop' : 'mobile',\n  style,\n  children,\n  uiMode = 'standalone',\n  cast: castOptions = 'compatible',\n  slots,\n  slotProps = {},\n  onError,\n  onPlaybackStateChange,\n  onBack,\n  onCast,\n  onChangeNext,\n  onChangePrevious,\n  onOpenSettings,\n  onChangeSettings,\n  onPlayerLoaded,\n  onBlockedAutoplay,\n  onPlaylogFired,\n  ...videoProps\n}) => {\n  const components = {\n    VolumeControl,\n    CastUi,\n    DisplayTime,\n    LiveButton,\n    Seekbar,\n    Settings,\n    ...slots,\n  }\n  const videoRef = useRef()\n  const containerRef = useRef()\n  const playerRef = useRef()\n  const adContainerRef = useRef()\n  const refs = useRef({})\n  // TODO move RWD related to Layout\n  const {\n    currentBreakpoint: size,\n    width,\n    observe,\n  } = useDimensions({\n    polyfill: ResizeObserver,\n    breakpoints: sizes,\n  })\n  const [isUserActive, setIsUserActive] = useState(\n    uiMode === 'standalone' ? true : videoProps.autoplay\n  )\n  const [userFocus, setUserFocus] = useState('')\n  const [targetState, setTargetState] = useState(() => ({\n    playbackState:\n      appPlaybackState || (videoProps.autoplay ? 'playing' : 'paused'),\n    currentTime: startTime,\n  }))\n  const [playbackTime, setPlaybackTime] = useState({\n    currentTime: 0,\n    bufferTime: 0,\n  })\n  const togglePlay = (overrideState, {triggeredBy} = {}) => {\n    if (targetState.playbackState !== overrideState) {\n      setTargetState(state => ({\n        ...state,\n        playbackState: overrideState || flipState(state.playbackState),\n        animation:\n          triggeredBy === 'user-action' && flipState(state.playbackState),\n      }))\n    }\n  }\n\n  useEffect(() => {\n    if (appPlaybackState) {\n      togglePlay(appPlaybackState)\n    }\n  }, [appPlaybackState])\n  useEffect(() => {\n    if (uiType === 'mobile') {\n      on(document, 'visibilitychange', () => togglePlay('paused'))\n    }\n  }, [uiType, targetState.playbackState])\n  const updatePlaybackTime = event =>\n    requestAnimationFrame(\n      () =>\n        (event?.type !== 'timeupdate' || isBuffered(videoRef.current)) &&\n        setPlaybackTime(state => ({\n          ...state,\n          ...getMediaTime(videoRef.current, {\n            player: playerRef.current,\n            plugins,\n          }),\n          ...(event?.type === 'durationchange' && {\n            currentTime: state.currentTime,\n          }),\n        }))\n    )\n  const setTargetTime = time => {\n    const trimmed = Math.min(\n      time,\n      videoRef.current?.initialDuration || Infinity\n    )\n    setTargetState(state => ({\n      ...state,\n      // seek to 0 repeatedly edge case\n      currentTime: state.currentTime !== trimmed ? trimmed : trimmed + 0.01,\n    }))\n    updatePlaybackTime()\n  }\n  const [playbackState, setPlaybackState] = useState('init')\n  const [castState, setCastState] = useState('')\n  useEffect(() => {\n    if (typeof appCurrentTime === 'number') setTargetTime(appCurrentTime || 0)\n    if (typeof appCurrentTime === 'object')\n      setTargetTime(appCurrentTime?.value || 0)\n  }, [appCurrentTime])\n  const [errorData, setErrorData] = useState()\n  const errorHandler = reactEvent => {\n    onError?.(reactEvent.nativeEvent)\n    handleError(reactEvent, {\n      media: videoRef.current,\n      displayError: data => {\n        setPlaybackState('error')\n        setErrorData(current => (current?.code ? current : data))\n      },\n    })\n  }\n  const isLive = playbackTime.streamType === 'live'\n  const [settings, setSettings] = useState(() => ({\n    sections: [],\n    values: {speed: 1},\n  }))\n  const fetchSettings = () => {\n    if (!playerRef.current) {\n      return\n    }\n    const speedItems = appSettings.speedItems || (isLive ? [] : undefined)\n    setSettings(current => {\n      const {values, sections} = getSettingsData({\n        media: videoRef.current,\n        player: playerRef.current,\n        source,\n        quality,\n        speedItems,\n        loop,\n        preferred: current.preferred,\n      })\n      return {preferred: current.preferred, values, sections}\n    })\n  }\n  const lastState = useRef(playbackState)\n  const handlePlaybackStateChange = (event, state) => {\n    if (lastState.current === 'error') {\n      return\n    }\n    onPlaybackStateChange?.(event, state)\n    // previously UI playback state was synced only when\n    // exiting iOS video only fullscreen(webkitendfullscreen)\n    if (\n      /playing|paused/.test(state) &&\n      !(uiType === 'mobile' && blocking) &&\n      !shouldPause({userFocus, uiType})\n    ) {\n      togglePlay(state)\n    }\n    if (state === 'ended') {\n      togglePlay('paused')\n    }\n    if (state === 'loading' && lastState.current !== 'init') {\n      setTimeout(() => togglePlay('playing'), 1)\n    }\n    if (state === 'playing') {\n      setIsUserActive(true)\n    }\n    if (lastState.current === 'loading') {\n      fetchSettings()\n    }\n    lastState.current = state\n    setPlaybackState(state)\n  }\n\n  const changeSettings = (name, value, {keepOpen} = {}) => {\n    // TODO consider merge into useReducer?\n    onChangeSettings?.({name, value})\n    setTargetTime(playbackTime.currentTime)\n    setSettings(current => ({\n      ...current,\n      values: {...current.values, [name]: value},\n      preferred: {...current.preferred, [name]: value},\n    }))\n    if (!keepOpen) {\n      setUserFocus('')\n    }\n  }\n  const openSettings = event => {\n    const animationFrame =\n      userFocus !== 'settings' &&\n      requestAnimationFrame(() => {\n        onOpenSettings?.(event, settings)\n        // In iOS Safari, we need to update settings data\n        fetchSettings()\n      })\n    setUserFocus(current => (current === 'settings' ? '' : 'settings'))\n    return animationFrame\n  }\n  useEffect(() => {\n    if (appPlaybackRate > 0) {\n      setSettings(current => ({\n        ...current,\n        values: {...current.values, speed: appPlaybackRate},\n      }))\n    }\n  }, [appPlaybackRate])\n\n  const qualityOptions = useMemo(\n    () => getQualityOptions(settings),\n    [settings.values.quality]\n  )\n  const viewMode = useLinkState(update =>\n    onViewModeChange(videoRef.current, update)\n  )\n  const sourceOverride = useLinkState(\n    async update => {\n      const result =\n        source && (await quality.rewriteManifest?.(source, qualityOptions))\n      update(result || source)\n    },\n    [source, qualityOptions]\n  )\n  const waiting = /emptied|loading|buffering/.test(playbackState)\n  const activePlayback =\n    playbackState === 'playing' || playbackState === 'waiting'\n  const {\n    mode: autoHideMode,\n    onClick,\n    onMouseMove,\n  } = useAutoHide({\n    pinned: !controls.autohide || waiting || !activePlayback || userFocus,\n    tapToHide: uiType === 'mobile',\n    hideTimeMs: controls.autohide,\n  })\n  const mode =\n    userFocus === 'seekbar'\n      ? 'hidden'\n      : controls.autohide\n      ? autoHideMode\n      : controls\n      ? 'shown'\n      : 'hidden'\n  const controlsDisplay =\n    userFocus === 'seekbar'\n      ? 'seekbar-only'\n      : controls === 'title-only' || playbackState === 'emptied'\n      ? 'hidden'\n      : mode\n  const shouldHidePanels =\n    (controls === 'no-panel' || controlsDisplay === 'hidden') && userFocus\n  const havePlayPanel =\n    !blocking &&\n    havePointer() &&\n    uiType === 'desktop' &&\n    !waiting &&\n    !/volume|''/.test(userFocus) &&\n    !/title-only|no-panel/.test(controls) &&\n    (controls.autohide || mode === 'shown')\n\n  useEffect(() => {\n    if (shouldHidePanels) {\n      setUserFocus('')\n    }\n  }, [shouldHidePanels])\n  const {subscribe, onChange, toggleMute} = linkMediaVolume(() => ({\n    video: videoRef.current,\n    getPlayer: () => playerRef.current,\n  }))\n  const changePrevious = event => {\n    onChangePrevious(event)\n    togglePlay('paused')\n    videoRef.current.dispatchEvent(new CustomEvent('loadstart'))\n  }\n  const changeNext = event => {\n    onChangeNext(event)\n    togglePlay('paused')\n    videoRef.current.dispatchEvent(new CustomEvent('loadstart'))\n  }\n\n  useEffect(fetchSettings, [appSettings])\n  useEffect(() => {\n    // The adContainer should be set before `load` because ImaDai.load needs it.\n    plugins.forEach(plugin => plugin.setAdContainer?.(adContainerRef.current))\n  }, [])\n\n  useEffect(\n    () =>\n      dispatchChapterEvents({\n        media: videoRef.current,\n        chapters,\n        getTime: () => videoRef.current.currentTime,\n      }),\n    [chapters]\n  )\n\n  const isEnd = playbackState === 'ended'\n  const canSeek = !isEnd && playbackTime.duration > 0\n  const derivedPlaybackState =\n    (uiType === 'mobile' && blocking) || shouldPause({userFocus, uiType})\n      ? 'paused'\n      : targetState.playbackState\n  const activeThumbnailsUrl =\n    (!userFocus || userFocus === 'seekbar') &&\n    (getThumbnailsUrl(source) || thumbnailsUrl)\n  const cssVariables = {\n    '--playing': playbackState === 'playing' ? '1' : '0',\n  }\n  const onRewind =\n    (!isLive || playbackTime.currentTime < 0) &&\n    !isEnd &&\n    (() => setTargetTime(playbackTime.currentTime - 10, 'rewind'))\n  const onForward =\n    (!isLive || playbackTime.currentTime < 0) &&\n    canSeek &&\n    (() => setTargetTime(playbackTime.currentTime + 10, 'forward'))\n  const onClickBlank =\n    havePlayPanel && (() => togglePlay('', {triggeredBy: 'user-action'}))\n\n  const uiHandlers = {\n    onPlay: () => {\n      togglePlay('', {triggeredBy: 'user-action'})\n      setIsUserActive(true)\n    },\n    onClickLive: isLive && (() => setTargetTime(0, 'seekToLive')),\n    onClickSeekbar: () => {\n      refs.current.deferedSeekFocus = setTimeout(\n        () => setUserFocus('seekbar'),\n        66\n      )\n    },\n    onSeek: time => {\n      setTargetTime(time, 'seek')\n      clearTimeout(refs.current.deferedSeekFocus)\n      setTimeout(() => setUserFocus(''), 66)\n    },\n    onChangeSettings: ({name, value, keepOpen}) =>\n      changeSettings(name, value, {keepOpen}),\n    onCloseSettings: event => {\n      setUserFocus('')\n      slotProps?.settings?.onClose?.(event)\n    },\n  }\n\n  const uiElements = {\n    liveButton: uiHandlers.onClickLive && (\n      <components.LiveButton\n        usingStartOver={playbackTime.currentTime < 0}\n        onClick={uiHandlers.onClickLive}\n      />\n    ),\n    controlButtons: {\n      playButton: (\n        <PlayButton\n          playbackState={derivedPlaybackState}\n          ended={isEnd}\n          hidden={\n            uiType === 'mobile' && (waiting || /loading/.test(playbackState))\n          }\n          variant={!isUserActive && 'firstplay'}\n          onClick={uiHandlers.onPlay}\n        />\n      ),\n      rewindButton: onRewind && (\n        <Button\n          startIcon=\"rewind10\"\n          title=\"KKS.PLAYER.REWIND\"\n          disabled={isEnd}\n          onClick={onRewind}\n        />\n      ),\n      forwardButton: onForward && (\n        <ForwardButton\n          startIcon=\"forward10\"\n          title=\"KKS.PLAYER.FORWARD\"\n          disabled={!canSeek}\n          onClick={onForward}\n        />\n      ),\n      nextEpisodeButton: (\n        <Button\n          startIcon=\"next\"\n          title=\"KKS.PLAYER.NEXT\"\n          disabled={!onChangeNext}\n          onClick={changeNext}\n        />\n      ),\n      previousEpisodeButton: (\n        <Button\n          startIcon=\"previous\"\n          title=\"KKS.PLAYER.PREVIOUS\"\n          disabled={!onChangePrevious}\n          onClick={changePrevious}\n        />\n      ),\n    },\n    seekbar: (\n      <components.Seekbar\n        style={/volume/.test(userFocus) && {opacity: '0'}}\n        startTime={playbackTime.startTime}\n        currentTime={playbackTime.currentTime}\n        bufferTime={playbackTime.bufferTime}\n        duration={playbackTime.duration}\n        chapters={chapters}\n        onChange={uiHandlers.onClickSeekbar}\n        onChangeCommitted={uiHandlers.onSeek}\n        marks={marks}\n        plugins={plugins}\n        {...slotProps.seekbar}\n      >\n        {activeThumbnailsUrl && (\n          <SeekPreview\n            thumbnailsUrl={activeThumbnailsUrl}\n            duration={playbackTime.duration}\n            chapters={chapters}\n          />\n        )}\n      </components.Seekbar>\n    ),\n    displayTime: <components.DisplayTime {...playbackTime} />,\n    fullscreenButton: (\n      <FullscreenButton\n        viewMode={viewMode}\n        onClick={() => toggleFullscreen(containerRef.current)}\n      />\n    ),\n    volumeControl: width >= sizes['small-embed'] && (\n      <components.VolumeControl\n        {...{subscribe, onChange, toggleMute}}\n        onMouseOver={() => setUserFocus('volume')}\n        onMouseOut={() => setUserFocus('')}\n        {...slotProps?.volumeControl}\n      />\n    ),\n    backItems: (\n      <PlayPanel animation={targetState.animation} onClick={onClickBlank} />\n    ),\n  }\n  const compatibilityCastProps = castOptions === 'compatible' && {\n    source,\n    values: {title, ...settings.values, ...intl, modulesConfig},\n    onLoad: onCast,\n  }\n\n  useEffect(() => {\n    if (castOptions && castOptions !== 'compatible') {\n      return linkCast({source, ...castOptions})\n    } // TODO playlist, audio / subtitle setting\n  }, [source, castOptions.contentId, castOptions.customData])\n\n  return (\n    <IntlProvider {...intl}>\n      <Layout\n        style={{...style, cssVariables}}\n        type={uiType}\n        display={mode}\n        controlsDisplay={controlsDisplay}\n        size={size}\n        video={\n          <Video\n            {...videoProps}\n            videoRef={multiRef(videoRef, videoProps.videoRef)}\n            source={\n              castState !== CastState.CONNECTED &&\n              playbackState !== 'error' &&\n              sourceOverride\n            }\n            playbackState={derivedPlaybackState}\n            currentTime={targetState.currentTime}\n            {...(appVolume >= 0 && {volume: appVolume, muted: appVolume <= 0})}\n            playbackRate={settings.values.speed}\n            videoResolution={qualityOptions}\n            audio={settings.values.audio}\n            textTrack={settings.values.subtitles}\n            loop={settings.values.loop}\n            plugins={plugins}\n            modulesConfig={modulesConfig}\n            onPlaylogFired={onPlaylogFired}\n            onError={errorHandler}\n            onPlaybackStateChange={handlePlaybackStateChange}\n            onBlockedAutoplay={error => onBlockedAutoplay?.(error)}\n            onCanPlay={event => {\n              updatePlaybackTime(event)\n              videoProps.onCanPlay?.(event)\n            }}\n            onTimeUpdate={updatePlaybackTime}\n            onDurationChange={updatePlaybackTime}\n            onPlayerLoaded={player => {\n              playerRef.current = player\n              onPlayerLoaded?.(player)\n              syncVolume(videoRef.current, videoProps.muted ? 0 : appVolume)\n            }}\n          />\n        }\n        containerRef={element => {\n          containerRef.current = element\n          observe(element)\n        }}\n        backRef={adContainerRef}\n        title={title}\n        channelTitle={channelTitle}\n        backButton={onBack && <Button startIcon=\"back\" onClick={onBack} />}\n        {...(isUserActive && uiElements)}\n        onClick={onClick}\n        onMouseMove={onMouseMove}\n      >\n        <LoadingSpinner active={waiting} />\n        {errorData && <Error error={errorData} onBack={onBack} />}\n        {children}\n        {/* TODO sync Cast last played time back */}\n        {isUserActive && (\n          <components.CastUi\n            onBack={onBack}\n            onStateChange={setCastState}\n            onLoad={onCast}\n            {...compatibilityCastProps}\n            {...slotProps.castUi}\n          />\n        )}\n        {isUserActive && ( // TODO: Design <Extenstion /> flag\n          <components.Settings\n            type={uiType}\n            sections={settings.sections}\n            // TODO hasBottomPanel bottom: 8em\n            open={userFocus === 'settings'}\n            values={settings.values}\n            onOpen={openSettings}\n            onChange={uiHandlers.onChangeSettings}\n            {...slotProps?.settings}\n            onClose={uiHandlers.onCloseSettings}\n          />\n        )}\n        <LanguageSettings\n          uiType={uiType}\n          getPlayer={() => playerRef.current}\n          onChange={(_event, item) => changeSettings(item.type, item.value)}\n          {...slotProps.languageSettings}\n        />\n        <Backdrop open={!playbackState || playbackState === 'loading'}>\n          <LoadingSpinner />\n        </Backdrop>\n      </Layout>\n    </IntlProvider>\n  )\n}\n\nexport default PremiumPlayer\n"]} */"),
|
|
8153
|
-
desktop: /*#__PURE__*/css$1(LanguageMenu.styles, LanguageMenu.desktopStyles, process.env.NODE_ENV === "production" ? "" : ";label:desktop;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["PremiumPlayer.js"],"names":[],"mappings":"AAwEW","file":"PremiumPlayer.js","sourcesContent":["/* eslint-disable no-nested-ternary */\nimport {useEffect, useMemo, useRef, useState} from 'react'\nimport {css} from '@emotion/css'\nimport useDimensions from 'react-cool-dimensions'\nimport {ResizeObserver} from '@juggle/resize-observer'\n\nimport {handleError} from 'playerCore/errors'\nimport {on} from 'util/events'\nimport {havePointer, isDesktop} from 'util/environment'\nimport multiRef from 'util/multiRef'\nimport {onViewModeChange, toggleFullscreen} from 'util/viewModes'\nimport {useAutoHide} from 'hooks'\nimport {IntlProvider} from 'context/I18n'\nimport {getMediaTime, isBuffered} from 'playerCore/mediaBindings'\nimport Error from 'playerUi/Error'\nimport Layout from 'playerUi/DefaultLayout'\nimport {\n  Button,\n  LiveButton,\n  FullscreenButton,\n  PlayButton,\n  ForwardButton,\n} from 'playerUi/buttons'\nimport Seekbar from 'playerUi/Seekbar'\nimport DisplayTime from 'playerUi/DisplayTime'\nimport LoadingSpinner from 'playerUi/LoadingSpinner'\nimport Backdrop from 'playerUi/Backdrop'\nimport Settings from 'playerUi/Settings'\nimport OverlayPanel from 'playerUi/OverlayPanel'\nimport LanguageMenu from 'playerUi/LanguageMenu'\nimport PlayPanel from 'playerUi/PlayPane'\nimport VolumeControl from 'component/VolumeControl'\nimport CastUi from 'cast/CastUi'\nimport {linkCast} from 'cast/framework'\nimport {CastState} from 'Enum'\nimport {dispatchChapterEvents} from './timeline'\nimport Video from '../Video'\nimport {getLanguageOptions, getQualityOptions, getSettingsData} from './settings'\nimport {linkMediaVolume, syncVolume} from './volume'\nimport SeekPreview from './SeekPreview'\n\nconst sizes = {\n  'small-embed': 200,\n  embed: 400,\n  'tablet-portrait': 600,\n  'tablet-landscape': 900,\n  desktop: 1200,\n}\n\nconst useLinkState = (request, dependencies = []) => {\n  const [state, setState] = useState()\n  useEffect(() => {\n    request(setState)\n  }, dependencies)\n  return state\n}\n\nconst flipState = state => (state === 'playing' ? 'paused' : 'playing')\n\n// FIXME: too few lines to split a file, looking a better place\nconst getThumbnailsUrl = source =>\n  [].concat(source).find(item => item.type === 'thumbnail')?.src\n\nconst defaultSettings = {}\n\nconst shouldPause = ({userFocus, uiType}) =>\n  (uiType === 'mobile' && /settings|chapter-list/.test(userFocus)) ||\n  /blocking|seekbar|share-inner/.test(userFocus)\n\n// TODO extract to somewhere else\nconst menuClasses = {\n  default: css(LanguageMenu.styles),\n  desktop: css(LanguageMenu.styles, LanguageMenu.desktopStyles),\n}\n\nconst LanguageSettings = ({\n  uiType,\n  audioTracks = [],\n  subtitleTracks = [],\n  getPlayer,\n  onChange,\n  slots = {LanguageMenu},\n  ...rest\n}) => {\n  const options = getLanguageOptions(getPlayer(), {audioTracks, subtitleTracks})\n  return (\n    <OverlayPanel buttonIcon=\"subtitle\">\n      <slots.LanguageMenu\n        uiType={uiType}\n        classes={menuClasses}\n        sectionOptions={[options.audioTracks, options.textTracks]}\n        onChange={onChange}\n        {...rest}\n      />\n    </OverlayPanel>\n  )\n}\n\nconst PremiumPlayer = ({\n  source,\n  startTime,\n  quality = {},\n  title,\n  metadata,\n  channelTitle,\n  chapters,\n  playbackState: appPlaybackState,\n  currentTime: appCurrentTime,\n  playbackRate: appPlaybackRate,\n  loop,\n  volume: appVolume,\n  thumbnailsUrl,\n  controls = {autohide: 3000},\n  marks = [],\n  // TODO sectionId\n  intl,\n  settings: appSettings = defaultSettings,\n  blocking,\n  plugins = [],\n  modulesConfig,\n\n  uiType = isDesktop() ? 'desktop' : 'mobile',\n  style,\n  children,\n  uiMode = 'standalone',\n  cast: castOptions = 'compatible',\n  slots,\n  slotProps = {},\n  onError,\n  onPlaybackStateChange,\n  onBack,\n  onCast,\n  onChangeNext,\n  onChangePrevious,\n  onOpenSettings,\n  onChangeSettings,\n  onPlayerLoaded,\n  onBlockedAutoplay,\n  onPlaylogFired,\n  ...videoProps\n}) => {\n  const components = {\n    VolumeControl,\n    CastUi,\n    DisplayTime,\n    LiveButton,\n    Seekbar,\n    Settings,\n    ...slots,\n  }\n  const videoRef = useRef()\n  const containerRef = useRef()\n  const playerRef = useRef()\n  const adContainerRef = useRef()\n  const refs = useRef({})\n  // TODO move RWD related to Layout\n  const {\n    currentBreakpoint: size,\n    width,\n    observe,\n  } = useDimensions({\n    polyfill: ResizeObserver,\n    breakpoints: sizes,\n  })\n  const [isUserActive, setIsUserActive] = useState(\n    uiMode === 'standalone' ? true : videoProps.autoplay\n  )\n  const [userFocus, setUserFocus] = useState('')\n  const [targetState, setTargetState] = useState(() => ({\n    playbackState:\n      appPlaybackState || (videoProps.autoplay ? 'playing' : 'paused'),\n    currentTime: startTime,\n  }))\n  const [playbackTime, setPlaybackTime] = useState({\n    currentTime: 0,\n    bufferTime: 0,\n  })\n  const togglePlay = (overrideState, {triggeredBy} = {}) => {\n    if (targetState.playbackState !== overrideState) {\n      setTargetState(state => ({\n        ...state,\n        playbackState: overrideState || flipState(state.playbackState),\n        animation:\n          triggeredBy === 'user-action' && flipState(state.playbackState),\n      }))\n    }\n  }\n\n  useEffect(() => {\n    if (appPlaybackState) {\n      togglePlay(appPlaybackState)\n    }\n  }, [appPlaybackState])\n  useEffect(() => {\n    if (uiType === 'mobile') {\n      on(document, 'visibilitychange', () => togglePlay('paused'))\n    }\n  }, [uiType, targetState.playbackState])\n  const updatePlaybackTime = event =>\n    requestAnimationFrame(\n      () =>\n        (event?.type !== 'timeupdate' || isBuffered(videoRef.current)) &&\n        setPlaybackTime(state => ({\n          ...state,\n          ...getMediaTime(videoRef.current, {\n            player: playerRef.current,\n            plugins,\n          }),\n          ...(event?.type === 'durationchange' && {\n            currentTime: state.currentTime,\n          }),\n        }))\n    )\n  const setTargetTime = time => {\n    const trimmed = Math.min(\n      time,\n      videoRef.current?.initialDuration || Infinity\n    )\n    setTargetState(state => ({\n      ...state,\n      // seek to 0 repeatedly edge case\n      currentTime: state.currentTime !== trimmed ? trimmed : trimmed + 0.01,\n    }))\n    updatePlaybackTime()\n  }\n  const [playbackState, setPlaybackState] = useState('init')\n  const [castState, setCastState] = useState('')\n  useEffect(() => {\n    if (typeof appCurrentTime === 'number') setTargetTime(appCurrentTime || 0)\n    if (typeof appCurrentTime === 'object')\n      setTargetTime(appCurrentTime?.value || 0)\n  }, [appCurrentTime])\n  const [errorData, setErrorData] = useState()\n  const errorHandler = reactEvent => {\n    onError?.(reactEvent.nativeEvent)\n    handleError(reactEvent, {\n      media: videoRef.current,\n      displayError: data => {\n        setPlaybackState('error')\n        setErrorData(current => (current?.code ? current : data))\n      },\n    })\n  }\n  const isLive = playbackTime.streamType === 'live'\n  const [settings, setSettings] = useState(() => ({\n    sections: [],\n    values: {speed: 1},\n  }))\n  const fetchSettings = () => {\n    if (!playerRef.current) {\n      return\n    }\n    const speedItems = appSettings.speedItems || (isLive ? [] : undefined)\n    setSettings(current => {\n      const {values, sections} = getSettingsData({\n        media: videoRef.current,\n        player: playerRef.current,\n        source,\n        quality,\n        speedItems,\n        loop,\n        preferred: current.preferred,\n      })\n      return {preferred: current.preferred, values, sections}\n    })\n  }\n  const lastState = useRef(playbackState)\n  const handlePlaybackStateChange = (event, state) => {\n    if (lastState.current === 'error') {\n      return\n    }\n    onPlaybackStateChange?.(event, state)\n    // previously UI playback state was synced only when\n    // exiting iOS video only fullscreen(webkitendfullscreen)\n    if (\n      /playing|paused/.test(state) &&\n      !(uiType === 'mobile' && blocking) &&\n      !shouldPause({userFocus, uiType})\n    ) {\n      togglePlay(state)\n    }\n    if (state === 'ended') {\n      togglePlay('paused')\n    }\n    if (state === 'loading' && lastState.current !== 'init') {\n      setTimeout(() => togglePlay('playing'), 1)\n    }\n    if (state === 'playing') {\n      setIsUserActive(true)\n    }\n    if (lastState.current === 'loading') {\n      fetchSettings()\n    }\n    lastState.current = state\n    setPlaybackState(state)\n  }\n\n  const changeSettings = (name, value, {keepOpen} = {}) => {\n    // TODO consider merge into useReducer?\n    onChangeSettings?.({name, value})\n    setTargetTime(playbackTime.currentTime)\n    setSettings(current => ({\n      ...current,\n      values: {...current.values, [name]: value},\n      preferred: {...current.preferred, [name]: value},\n    }))\n    if (!keepOpen) {\n      setUserFocus('')\n    }\n  }\n  const openSettings = event => {\n    const animationFrame =\n      userFocus !== 'settings' &&\n      requestAnimationFrame(() => {\n        onOpenSettings?.(event, settings)\n        // In iOS Safari, we need to update settings data\n        fetchSettings()\n      })\n    setUserFocus(current => (current === 'settings' ? '' : 'settings'))\n    return animationFrame\n  }\n  useEffect(() => {\n    if (appPlaybackRate > 0) {\n      setSettings(current => ({\n        ...current,\n        values: {...current.values, speed: appPlaybackRate},\n      }))\n    }\n  }, [appPlaybackRate])\n\n  const qualityOptions = useMemo(\n    () => getQualityOptions(settings),\n    [settings.values.quality]\n  )\n  const viewMode = useLinkState(update =>\n    onViewModeChange(videoRef.current, update)\n  )\n  const sourceOverride = useLinkState(\n    async update => {\n      const result =\n        source && (await quality.rewriteManifest?.(source, qualityOptions))\n      update(result || source)\n    },\n    [source, qualityOptions]\n  )\n  const waiting = /emptied|loading|buffering/.test(playbackState)\n  const activePlayback =\n    playbackState === 'playing' || playbackState === 'waiting'\n  const {\n    mode: autoHideMode,\n    onClick,\n    onMouseMove,\n  } = useAutoHide({\n    pinned: !controls.autohide || waiting || !activePlayback || userFocus,\n    tapToHide: uiType === 'mobile',\n    hideTimeMs: controls.autohide,\n  })\n  const mode =\n    userFocus === 'seekbar'\n      ? 'hidden'\n      : controls.autohide\n      ? autoHideMode\n      : controls\n      ? 'shown'\n      : 'hidden'\n  const controlsDisplay =\n    userFocus === 'seekbar'\n      ? 'seekbar-only'\n      : controls === 'title-only' || playbackState === 'emptied'\n      ? 'hidden'\n      : mode\n  const shouldHidePanels =\n    (controls === 'no-panel' || controlsDisplay === 'hidden') && userFocus\n  const havePlayPanel =\n    !blocking &&\n    havePointer() &&\n    uiType === 'desktop' &&\n    !waiting &&\n    !/volume|''/.test(userFocus) &&\n    !/title-only|no-panel/.test(controls) &&\n    (controls.autohide || mode === 'shown')\n\n  useEffect(() => {\n    if (shouldHidePanels) {\n      setUserFocus('')\n    }\n  }, [shouldHidePanels])\n  const {subscribe, onChange, toggleMute} = linkMediaVolume(() => ({\n    video: videoRef.current,\n    getPlayer: () => playerRef.current,\n  }))\n  const changePrevious = event => {\n    onChangePrevious(event)\n    togglePlay('paused')\n    videoRef.current.dispatchEvent(new CustomEvent('loadstart'))\n  }\n  const changeNext = event => {\n    onChangeNext(event)\n    togglePlay('paused')\n    videoRef.current.dispatchEvent(new CustomEvent('loadstart'))\n  }\n\n  useEffect(fetchSettings, [appSettings])\n  useEffect(() => {\n    // The adContainer should be set before `load` because ImaDai.load needs it.\n    plugins.forEach(plugin => plugin.setAdContainer?.(adContainerRef.current))\n  }, [])\n\n  useEffect(\n    () =>\n      dispatchChapterEvents({\n        media: videoRef.current,\n        chapters,\n        getTime: () => videoRef.current.currentTime,\n      }),\n    [chapters]\n  )\n\n  const isEnd = playbackState === 'ended'\n  const canSeek = !isEnd && playbackTime.duration > 0\n  const derivedPlaybackState =\n    (uiType === 'mobile' && blocking) || shouldPause({userFocus, uiType})\n      ? 'paused'\n      : targetState.playbackState\n  const activeThumbnailsUrl =\n    (!userFocus || userFocus === 'seekbar') &&\n    (getThumbnailsUrl(source) || thumbnailsUrl)\n  const cssVariables = {\n    '--playing': playbackState === 'playing' ? '1' : '0',\n  }\n  const onRewind =\n    (!isLive || playbackTime.currentTime < 0) &&\n    !isEnd &&\n    (() => setTargetTime(playbackTime.currentTime - 10, 'rewind'))\n  const onForward =\n    (!isLive || playbackTime.currentTime < 0) &&\n    canSeek &&\n    (() => setTargetTime(playbackTime.currentTime + 10, 'forward'))\n  const onClickBlank =\n    havePlayPanel && (() => togglePlay('', {triggeredBy: 'user-action'}))\n\n  const uiHandlers = {\n    onPlay: () => {\n      togglePlay('', {triggeredBy: 'user-action'})\n      setIsUserActive(true)\n    },\n    onClickLive: isLive && (() => setTargetTime(0, 'seekToLive')),\n    onClickSeekbar: () => {\n      refs.current.deferedSeekFocus = setTimeout(\n        () => setUserFocus('seekbar'),\n        66\n      )\n    },\n    onSeek: time => {\n      setTargetTime(time, 'seek')\n      clearTimeout(refs.current.deferedSeekFocus)\n      setTimeout(() => setUserFocus(''), 66)\n    },\n    onChangeSettings: ({name, value, keepOpen}) =>\n      changeSettings(name, value, {keepOpen}),\n    onCloseSettings: event => {\n      setUserFocus('')\n      slotProps?.settings?.onClose?.(event)\n    },\n  }\n\n  const uiElements = {\n    liveButton: uiHandlers.onClickLive && (\n      <components.LiveButton\n        usingStartOver={playbackTime.currentTime < 0}\n        onClick={uiHandlers.onClickLive}\n      />\n    ),\n    controlButtons: {\n      playButton: (\n        <PlayButton\n          playbackState={derivedPlaybackState}\n          ended={isEnd}\n          hidden={\n            uiType === 'mobile' && (waiting || /loading/.test(playbackState))\n          }\n          variant={!isUserActive && 'firstplay'}\n          onClick={uiHandlers.onPlay}\n        />\n      ),\n      rewindButton: onRewind && (\n        <Button\n          startIcon=\"rewind10\"\n          title=\"KKS.PLAYER.REWIND\"\n          disabled={isEnd}\n          onClick={onRewind}\n        />\n      ),\n      forwardButton: onForward && (\n        <ForwardButton\n          startIcon=\"forward10\"\n          title=\"KKS.PLAYER.FORWARD\"\n          disabled={!canSeek}\n          onClick={onForward}\n        />\n      ),\n      nextEpisodeButton: (\n        <Button\n          startIcon=\"next\"\n          title=\"KKS.PLAYER.NEXT\"\n          disabled={!onChangeNext}\n          onClick={changeNext}\n        />\n      ),\n      previousEpisodeButton: (\n        <Button\n          startIcon=\"previous\"\n          title=\"KKS.PLAYER.PREVIOUS\"\n          disabled={!onChangePrevious}\n          onClick={changePrevious}\n        />\n      ),\n    },\n    seekbar: (\n      <components.Seekbar\n        style={/volume/.test(userFocus) && {opacity: '0'}}\n        startTime={playbackTime.startTime}\n        currentTime={playbackTime.currentTime}\n        bufferTime={playbackTime.bufferTime}\n        duration={playbackTime.duration}\n        chapters={chapters}\n        onChange={uiHandlers.onClickSeekbar}\n        onChangeCommitted={uiHandlers.onSeek}\n        marks={marks}\n        plugins={plugins}\n        {...slotProps.seekbar}\n      >\n        {activeThumbnailsUrl && (\n          <SeekPreview\n            thumbnailsUrl={activeThumbnailsUrl}\n            duration={playbackTime.duration}\n            chapters={chapters}\n          />\n        )}\n      </components.Seekbar>\n    ),\n    displayTime: <components.DisplayTime {...playbackTime} />,\n    fullscreenButton: (\n      <FullscreenButton\n        viewMode={viewMode}\n        onClick={() => toggleFullscreen(containerRef.current)}\n      />\n    ),\n    volumeControl: width >= sizes['small-embed'] && (\n      <components.VolumeControl\n        {...{subscribe, onChange, toggleMute}}\n        onMouseOver={() => setUserFocus('volume')}\n        onMouseOut={() => setUserFocus('')}\n        {...slotProps?.volumeControl}\n      />\n    ),\n    backItems: (\n      <PlayPanel animation={targetState.animation} onClick={onClickBlank} />\n    ),\n  }\n  const compatibilityCastProps = castOptions === 'compatible' && {\n    source,\n    values: {title, ...settings.values, ...intl, modulesConfig},\n    onLoad: onCast,\n  }\n\n  useEffect(() => {\n    if (castOptions && castOptions !== 'compatible') {\n      return linkCast({source, ...castOptions})\n    } // TODO playlist, audio / subtitle setting\n  }, [source, castOptions.contentId, castOptions.customData])\n\n  return (\n    <IntlProvider {...intl}>\n      <Layout\n        style={{...style, cssVariables}}\n        type={uiType}\n        display={mode}\n        controlsDisplay={controlsDisplay}\n        size={size}\n        video={\n          <Video\n            {...videoProps}\n            videoRef={multiRef(videoRef, videoProps.videoRef)}\n            source={\n              castState !== CastState.CONNECTED &&\n              playbackState !== 'error' &&\n              sourceOverride\n            }\n            playbackState={derivedPlaybackState}\n            currentTime={targetState.currentTime}\n            {...(appVolume >= 0 && {volume: appVolume, muted: appVolume <= 0})}\n            playbackRate={settings.values.speed}\n            videoResolution={qualityOptions}\n            audio={settings.values.audio}\n            textTrack={settings.values.subtitles}\n            loop={settings.values.loop}\n            plugins={plugins}\n            modulesConfig={modulesConfig}\n            onPlaylogFired={onPlaylogFired}\n            onError={errorHandler}\n            onPlaybackStateChange={handlePlaybackStateChange}\n            onBlockedAutoplay={error => onBlockedAutoplay?.(error)}\n            onCanPlay={event => {\n              updatePlaybackTime(event)\n              videoProps.onCanPlay?.(event)\n            }}\n            onTimeUpdate={updatePlaybackTime}\n            onDurationChange={updatePlaybackTime}\n            onPlayerLoaded={player => {\n              playerRef.current = player\n              onPlayerLoaded?.(player)\n              syncVolume(videoRef.current, videoProps.muted ? 0 : appVolume)\n            }}\n          />\n        }\n        containerRef={element => {\n          containerRef.current = element\n          observe(element)\n        }}\n        backRef={adContainerRef}\n        title={title}\n        channelTitle={channelTitle}\n        backButton={onBack && <Button startIcon=\"back\" onClick={onBack} />}\n        {...(isUserActive && uiElements)}\n        onClick={onClick}\n        onMouseMove={onMouseMove}\n      >\n        <LoadingSpinner active={waiting} />\n        {errorData && <Error error={errorData} onBack={onBack} />}\n        {children}\n        {/* TODO sync Cast last played time back */}\n        {isUserActive && (\n          <components.CastUi\n            onBack={onBack}\n            onStateChange={setCastState}\n            onLoad={onCast}\n            {...compatibilityCastProps}\n            {...slotProps.castUi}\n          />\n        )}\n        {isUserActive && ( // TODO: Design <Extenstion /> flag\n          <components.Settings\n            type={uiType}\n            sections={settings.sections}\n            // TODO hasBottomPanel bottom: 8em\n            open={userFocus === 'settings'}\n            values={settings.values}\n            onOpen={openSettings}\n            onChange={uiHandlers.onChangeSettings}\n            {...slotProps?.settings}\n            onClose={uiHandlers.onCloseSettings}\n          />\n        )}\n        <LanguageSettings\n          uiType={uiType}\n          getPlayer={() => playerRef.current}\n          onChange={(_event, item) => changeSettings(item.type, item.value)}\n          {...slotProps.languageSettings}\n        />\n        <Backdrop open={!playbackState || playbackState === 'loading'}>\n          <LoadingSpinner />\n        </Backdrop>\n      </Layout>\n    </IntlProvider>\n  )\n}\n\nexport default PremiumPlayer\n"]} */")
|
|
8799
|
+
default: /*#__PURE__*/css$1(LanguageMenu.styles, process.env.NODE_ENV === "production" ? "" : ";label:default;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["PremiumPlayer.js"],"names":[],"mappings":"AAuEW","file":"PremiumPlayer.js","sourcesContent":["/* eslint-disable no-nested-ternary */\nimport {useEffect, useMemo, useRef, useState} from 'react'\nimport {css} from '@emotion/css'\nimport useDimensions from 'react-cool-dimensions'\nimport {ResizeObserver} from '@juggle/resize-observer'\n\nimport {handleError} from 'playerCore/errors'\nimport {on} from 'util/events'\nimport {isDesktop} from 'util/environment'\nimport multiRef from 'util/multiRef'\nimport {onViewModeChange, toggleFullscreen} from 'util/viewModes'\nimport {useAutoHide} from 'hooks'\nimport {IntlProvider} from 'context/I18n'\nimport {getMediaTime, isBuffered} from 'playerCore/mediaBindings'\nimport Error from 'playerUi/Error'\nimport Layout from 'playerUi/DefaultLayout'\nimport {\n  Button,\n  LiveButton,\n  FullscreenButton,\n  PlayButton,\n  ForwardButton,\n} from 'playerUi/buttons'\nimport Seekbar from 'playerUi/Seekbar'\nimport DisplayTime from 'playerUi/DisplayTime'\nimport LoadingSpinner from 'playerUi/LoadingSpinner'\nimport Backdrop from 'playerUi/Backdrop'\nimport Settings from 'playerUi/Settings'\nimport OverlayPanel from 'playerUi/OverlayPanel'\nimport LanguageMenu from 'playerUi/LanguageMenu'\nimport PlayPanel from 'playerUi/PlayPane'\nimport VolumeControl from 'component/VolumeControl'\nimport CastUi from 'cast/CastUi'\nimport {linkCast} from 'cast/framework'\nimport {CastState} from 'Enum'\nimport {dispatchChapterEvents} from './timeline'\nimport Video from '../Video'\nimport {getLanguageOptions, getQualityOptions, getSettingsData} from './settings'\nimport {linkMediaVolume, syncVolume} from './volume'\nimport SeekPreview from './SeekPreview'\n\nconst sizes = {\n  'small-embed': 200,\n  embed: 400,\n  'tablet-portrait': 600,\n  'tablet-landscape': 900,\n  desktop: 1200,\n}\n\nconst useLinkState = (request, dependencies = []) => {\n  const [state, setState] = useState()\n  useEffect(() => {\n    request(setState)\n  }, dependencies)\n  return state\n}\n\nconst flipState = state => (state === 'playing' ? 'paused' : 'playing')\n\n// FIXME: too few lines to split a file, looking a better place\nconst getThumbnailsUrl = source =>\n  [].concat(source).find(item => item.type === 'thumbnail')?.src\n\nconst defaultSettings = {}\n\nconst shouldPause = ({userFocus, uiType}) =>\n  (uiType === 'mobile' && /settings|chapter-list/.test(userFocus)) ||\n  /blocking|seekbar|share-inner/.test(userFocus)\n\n// TODO extract to somewhere else\nconst menuClasses = {\n  default: css(LanguageMenu.styles),\n  desktop: css(LanguageMenu.styles, LanguageMenu.desktopStyles),\n}\n\nconst LanguageSettings = ({\n  uiType,\n  audioTracks = [],\n  textTracks = [],\n  getPlayer,\n  onChange,\n  slots = {LanguageMenu},\n  ...rest\n}) => {\n  const options = getLanguageOptions(getPlayer(), {audioTracks, textTracks})\n  return (\n    audioTracks.length > 0 &&\n    textTracks.length > 0 && (\n    <OverlayPanel title=\"字幕・音声を切り替える\" buttonIcon=\"subtitle\">\n      <slots.LanguageMenu\n        uiType={uiType}\n        classes={menuClasses}\n        sectionOptions={[options.audioTracks, options.textTracks]}\n        onChange={onChange}\n        {...rest}\n      />\n    </OverlayPanel>\n    )\n  )\n}\n\nconst PremiumPlayer = ({\n  source,\n  startTime,\n  quality = {},\n  title,\n  metadata,\n  channelTitle,\n  chapters,\n  playbackState: appPlaybackState,\n  currentTime: appCurrentTime,\n  playbackRate: appPlaybackRate,\n  loop,\n  volume: appVolume,\n  audioTrack: appAudioTrack,\n  textTrack: appTextTrack,\n  thumbnailsUrl,\n  controls = {autohide: 3000},\n  marks = [],\n  // TODO sectionId\n  intl,\n  settings: appSettings = defaultSettings,\n  blocking,\n  plugins = [],\n  modulesConfig,\n\n  uiType = isDesktop() ? 'desktop' : 'mobile',\n  style,\n  children,\n  uiMode = 'standalone',\n  cast: castOptions = 'compatible',\n  slots,\n  slotProps = {},\n  onError,\n  onPlaybackStateChange,\n  onBack,\n  onCast,\n  onChangeNext,\n  onChangePrevious,\n  onOpenSettings,\n  onChangeSettings,\n  onPlayerLoaded,\n  onBlockedAutoplay,\n  onPlaylogFired,\n  ...videoProps\n}) => {\n  const components = {\n    VolumeControl,\n    CastUi,\n    DisplayTime,\n    LiveButton,\n    Seekbar,\n    Settings,\n    ...slots,\n  }\n  const videoRef = useRef()\n  const containerRef = useRef()\n  const playerRef = useRef()\n  const adContainerRef = useRef()\n  const refs = useRef({})\n  // TODO move RWD related to Layout\n  const {\n    currentBreakpoint: size,\n    width,\n    observe,\n  } = useDimensions({\n    polyfill: ResizeObserver,\n    breakpoints: sizes,\n  })\n  const [isUserActive, setIsUserActive] = useState(\n    uiMode === 'standalone' ? true : videoProps.autoplay\n  )\n  const [userFocus, setUserFocus] = useState('')\n  const [targetState, setTargetState] = useState(() => ({\n    playbackState:\n      appPlaybackState || (videoProps.autoplay ? 'playing' : 'paused'),\n    currentTime: startTime,\n  }))\n  const [playbackTime, setPlaybackTime] = useState({\n    currentTime: 0,\n    bufferTime: 0,\n  })\n  const togglePlay = (overrideState, {triggeredBy} = {}) => {\n    if (targetState.playbackState !== overrideState) {\n      setTargetState(state => ({\n        ...state,\n        playbackState: overrideState || flipState(state.playbackState),\n        animation:\n          triggeredBy === 'user-action' && flipState(state.playbackState),\n      }))\n    }\n  }\n\n  useEffect(() => {\n    if (appPlaybackState) {\n      togglePlay(appPlaybackState)\n    }\n  }, [appPlaybackState])\n  useEffect(() => {\n    if (uiType === 'mobile') {\n      on(document, 'visibilitychange', () => togglePlay('paused'))\n    }\n  }, [uiType, targetState.playbackState])\n  const updatePlaybackTime = event =>\n    requestAnimationFrame(\n      () =>\n        (event?.type !== 'timeupdate' || isBuffered(videoRef.current)) &&\n        setPlaybackTime(state => ({\n          ...state,\n          ...getMediaTime(videoRef.current, {\n            player: playerRef.current,\n            plugins,\n          }),\n          ...(event?.type === 'durationchange' && {\n            currentTime: state.currentTime,\n          }),\n        }))\n    )\n  const setTargetTime = time => {\n    const trimmed = Math.min(\n      time,\n      videoRef.current?.initialDuration || Infinity\n    )\n    setTargetState(state => ({\n      ...state,\n      // seek to 0 repeatedly edge case\n      currentTime: state.currentTime !== trimmed ? trimmed : trimmed + 0.01,\n    }))\n    updatePlaybackTime()\n  }\n  const [playbackState, setPlaybackState] = useState('init')\n  const [castState, setCastState] = useState('')\n  useEffect(() => {\n    if (typeof appCurrentTime === 'number') setTargetTime(appCurrentTime || 0)\n    if (typeof appCurrentTime === 'object')\n      setTargetTime(appCurrentTime?.value || 0)\n  }, [appCurrentTime])\n  const [errorData, setErrorData] = useState()\n  const errorHandler = reactEvent => {\n    onError?.(reactEvent.nativeEvent)\n    handleError(reactEvent, {\n      media: videoRef.current,\n      displayError: data => {\n        setPlaybackState('error')\n        setErrorData(current => (current?.code ? current : data))\n      },\n    })\n  }\n  const isLive = playbackTime.streamType === 'live'\n  const [settings, setSettings] = useState(() => ({\n    sections: [],\n    values: {speed: 1},\n  }))\n  const fetchSettings = () => {\n    if (!playerRef.current) {\n      return\n    }\n    const speedItems = appSettings.speedItems || (isLive ? [] : undefined)\n    setSettings(current => {\n      const {values, sections} = getSettingsData({\n        media: videoRef.current,\n        player: playerRef.current,\n        source,\n        quality,\n        speedItems,\n        loop,\n        preferred: current.preferred,\n        otherSections: appSettings.sections,\n      })\n      return {preferred: current.preferred, values, sections}\n    })\n  }\n  const lastState = useRef(playbackState)\n  const handlePlaybackStateChange = (event, state) => {\n    if (lastState.current === 'error') {\n      return\n    }\n    onPlaybackStateChange?.(event, state)\n    // previously UI playback state was synced only when\n    // exiting iOS video only fullscreen(webkitendfullscreen)\n    if (\n      /playing|paused/.test(state) &&\n      !(uiType === 'mobile' && blocking) &&\n      !shouldPause({userFocus, uiType})\n    ) {\n      togglePlay(state)\n    }\n    if (state === 'ended') {\n      togglePlay('paused')\n    }\n    if (state === 'loading' && lastState.current !== 'init') {\n      setTimeout(() => togglePlay('playing'), 1)\n    }\n    if (state === 'playing') {\n      setIsUserActive(true)\n    }\n    if (lastState.current === 'loading') {\n      fetchSettings()\n    }\n    lastState.current = state\n    setPlaybackState(state)\n  }\n\n  const changeSettings = (name, value, {keepOpen} = {}) => {\n    // TODO consider merge into useReducer?\n    onChangeSettings?.({name, value})\n    setTargetTime(playbackTime.currentTime)\n    setSettings(current => ({\n      ...current,\n      values: {...current.values, [name]: value},\n      preferred: {...current.preferred, [name]: value},\n    }))\n    if (!keepOpen) {\n      setUserFocus('')\n    }\n  }\n  const openSettings = event => {\n    const animationFrame =\n      userFocus !== 'settings' &&\n      requestAnimationFrame(() => {\n        onOpenSettings?.(event, settings)\n        // In iOS Safari, we need to update settings data\n        fetchSettings()\n      })\n    setUserFocus(current => (current === 'settings' ? '' : 'settings'))\n    return animationFrame\n  }\n  useEffect(() => {\n    if (appPlaybackRate > 0) {\n      setSettings(current => ({\n        ...current,\n        values: {...current.values, speed: appPlaybackRate},\n      }))\n    }\n  }, [appPlaybackRate])\n\n  const qualityOptions = useMemo(\n    () => getQualityOptions(settings),\n    [settings.values.quality]\n  )\n  const viewMode = useLinkState(update =>\n    onViewModeChange(videoRef.current, update)\n  )\n  const sourceOverride = useLinkState(\n    async update => {\n      const result =\n        source && (await quality.rewriteManifest?.(source, qualityOptions))\n      update(result || source)\n    },\n    [source, qualityOptions]\n  )\n  const waiting = /emptied|loading|buffering/.test(playbackState)\n  const activePlayback =\n    playbackState === 'playing' || playbackState === 'waiting'\n  const {\n    mode: autoHideMode,\n    onClick,\n    onMouseMove,\n  } = useAutoHide({\n    pinned: !controls.autohide || waiting || !activePlayback || userFocus,\n    tapToHide: uiType === 'mobile',\n    hideTimeMs: controls.autohide,\n  })\n  const mode =\n    userFocus === 'seekbar'\n      ? 'hidden'\n      : controls.autohide\n      ? autoHideMode\n      : controls\n      ? 'shown'\n      : 'hidden'\n  const controlsDisplay =\n    userFocus === 'seekbar'\n      ? 'seekbar-only'\n      : controls === 'title-only' || playbackState === 'emptied'\n      ? 'hidden'\n      : mode\n  const shouldHidePanels =\n    (controls === 'no-panel' || controlsDisplay === 'hidden') && userFocus\n  const havePlayPanel =\n    !blocking &&\n    uiType === 'desktop' &&\n    !waiting &&\n    !/volume|''/.test(userFocus) &&\n    !/title-only|no-panel/.test(controls) &&\n    (controls.autohide || mode === 'shown')\n\n  useEffect(() => {\n    if (shouldHidePanels) {\n      setUserFocus('')\n    }\n  }, [shouldHidePanels])\n  useEffect(() => \n    on(containerRef.current, 'focus-menu', event => {\n      if (event.detail?.status === 'open') {\n        setUserFocus('menu')\n      }\n      if (event.detail?.status === 'closed') {\n        setUserFocus(current => (current === 'menu' ? '' : current))\n      }\n    })\n  , [])\n  const {subscribe, onChange, toggleMute} = linkMediaVolume(() => ({\n    video: videoRef.current,\n    getPlayer: () => playerRef.current,\n  }))\n  const changePrevious = event => {\n    onChangePrevious(event)\n    togglePlay('paused')\n    videoRef.current.dispatchEvent(new CustomEvent('loadstart'))\n  }\n  const changeNext = event => {\n    onChangeNext(event)\n    togglePlay('paused')\n    videoRef.current.dispatchEvent(new CustomEvent('loadstart'))\n  }\n\n  useEffect(fetchSettings, [appSettings])\n  useEffect(() => {\n    // The adContainer should be set before `load` because ImaDai.load needs it.\n    plugins.forEach(plugin => plugin.setAdContainer?.(adContainerRef.current))\n  }, [])\n\n  useEffect(\n    () =>\n      dispatchChapterEvents({\n        media: videoRef.current,\n        chapters,\n        getTime: () => videoRef.current.currentTime,\n      }),\n    [chapters]\n  )\n\n  const isEnd = playbackState === 'ended'\n  const canSeek = !isEnd && playbackTime.duration > 0\n  const derivedPlaybackState =\n    (uiType === 'mobile' && blocking) || shouldPause({userFocus, uiType})\n      ? 'paused'\n      : targetState.playbackState\n  const activeThumbnailsUrl =\n    (!userFocus || userFocus === 'seekbar') &&\n    (getThumbnailsUrl(source) || thumbnailsUrl)\n  const cssVariables = {\n    '--playing': playbackState === 'playing' ? '1' : '0',\n  }\n  const onRewind =\n    (!isLive || playbackTime.currentTime < 0) &&\n    canSeek &&\n    (() => setTargetTime(playbackTime.currentTime - 10, 'rewind'))\n  const onForward =\n    (!isLive || playbackTime.currentTime < 0) &&\n    canSeek &&\n    (() => setTargetTime(playbackTime.currentTime + 10, 'forward'))\n  const onClickBlank =\n    havePlayPanel && (() => togglePlay('', {triggeredBy: 'user-action'}))\n\n  const uiHandlers = {\n    onPlay: () => {\n      togglePlay('', {triggeredBy: 'user-action'})\n      setIsUserActive(true)\n    },\n    onClickLive: isLive && (() => setTargetTime(0, 'seekToLive')),\n    onClickSeekbar: () => {\n      refs.current.deferedSeekFocus = setTimeout(\n        () => setUserFocus('seekbar'),\n        66\n      )\n    },\n    onSeek: time => {\n      setTargetTime(time, 'seek')\n      clearTimeout(refs.current.deferedSeekFocus)\n      setTimeout(() => setUserFocus(''), 66)\n    },\n    onChangeSettings: ({name, value, keepOpen}) =>\n      changeSettings(name, value, {keepOpen}),\n    onCloseSettings: event => {\n      setUserFocus('')\n      slotProps?.settings?.onClose?.(event)\n    },\n  }\n\n  const uiElements = {\n    liveButton: uiHandlers.onClickLive && (\n      <components.LiveButton\n        usingStartOver={playbackTime.currentTime < 0}\n        onClick={uiHandlers.onClickLive}\n      />\n    ),\n    controlButtons: {\n      playButton: (\n        <PlayButton\n          playbackState={derivedPlaybackState}\n          ended={isEnd}\n          hidden={\n            uiType === 'mobile' && (waiting || /loading/.test(playbackState))\n          }\n          variant={!isUserActive && 'firstplay'}\n          onClick={uiHandlers.onPlay}\n        />\n      ),\n      rewindButton: onRewind && (\n        <Button\n          startIcon=\"rewind10\"\n          title=\"KKS.PLAYER.REWIND\"\n          disabled={isEnd}\n          onClick={onRewind}\n        />\n      ),\n      forwardButton: onForward && (\n        <ForwardButton\n          startIcon=\"forward10\"\n          title=\"KKS.PLAYER.FORWARD\"\n          disabled={!canSeek}\n          onClick={onForward}\n        />\n      ),\n      nextEpisodeButton: (\n        <Button\n          startIcon=\"next\"\n          title=\"KKS.PLAYER.NEXT\"\n          disabled={!onChangeNext}\n          onClick={changeNext}\n        />\n      ),\n      previousEpisodeButton: (\n        <Button\n          startIcon=\"previous\"\n          title=\"KKS.PLAYER.PREVIOUS\"\n          disabled={!onChangePrevious}\n          onClick={changePrevious}\n        />\n      ),\n    },\n    seekbar: (\n      <components.Seekbar\n        style={/volume/.test(userFocus) && {opacity: '0'}}\n        startTime={playbackTime.startTime}\n        currentTime={playbackTime.currentTime}\n        bufferTime={playbackTime.bufferTime}\n        duration={playbackTime.duration}\n        chapters={chapters}\n        onChange={uiHandlers.onClickSeekbar}\n        onChangeCommitted={uiHandlers.onSeek}\n        marks={marks}\n        plugins={plugins}\n        {...slotProps.seekbar}\n      >\n        {activeThumbnailsUrl && (\n          <SeekPreview\n            thumbnailsUrl={activeThumbnailsUrl}\n            duration={playbackTime.duration}\n            chapters={chapters}\n          />\n        )}\n      </components.Seekbar>\n    ),\n    displayTime: <components.DisplayTime {...playbackTime} />,\n    fullscreenButton: (\n      <FullscreenButton\n        viewMode={viewMode}\n        onClick={() => toggleFullscreen(containerRef.current)}\n      />\n    ),\n    volumeControl: width >= sizes['small-embed'] && (\n      <components.VolumeControl\n        {...{subscribe, onChange, toggleMute}}\n        onMouseOver={() => setUserFocus('volume')}\n        onMouseOut={() => setUserFocus('')}\n        {...slotProps?.volumeControl}\n      />\n    ),\n    backItems: (\n      <PlayPanel animation={targetState.animation} onClick={onClickBlank} />\n    ),\n  }\n  const compatibilityCastProps = castOptions === 'compatible' && {\n    source,\n    values: {title, ...settings.values, ...intl, modulesConfig},\n    onLoad: onCast,\n  }\n\n  useEffect(() => {\n    if (castOptions && castOptions !== 'compatible') {\n      return linkCast({source, ...castOptions})\n    } // TODO playlist, audio / subtitle setting\n  }, [source, castOptions.contentId, castOptions.customData])\n  const selectedAudioTrack = settings.preferred?.audio || appAudioTrack\n  const selectedTextTrack = settings.preferred?.subtitles || appTextTrack\n\n  return (\n    <IntlProvider {...intl}>\n      <Layout\n        style={{...style, cssVariables}}\n        type={uiType}\n        display={mode}\n        controlsDisplay={controlsDisplay}\n        size={size}\n        video={\n          <Video\n            {...videoProps}\n            videoRef={multiRef(videoRef, videoProps.videoRef)}\n            source={\n              castState !== CastState.CONNECTED &&\n              playbackState !== 'error' &&\n              sourceOverride\n            }\n            playbackState={derivedPlaybackState}\n            currentTime={targetState.currentTime}\n            {...(appVolume >= 0 && {volume: appVolume, muted: appVolume <= 0})}\n            playbackRate={settings.values.speed}\n            videoResolution={qualityOptions}\n            audioTrack={selectedAudioTrack}\n            textTrack={selectedTextTrack}\n            loop={settings.values.loop}\n            plugins={plugins}\n            modulesConfig={modulesConfig}\n            onPlaylogFired={onPlaylogFired}\n            onError={errorHandler}\n            onPlaybackStateChange={handlePlaybackStateChange}\n            onBlockedAutoplay={error => onBlockedAutoplay?.(error)}\n            onCanPlay={event => {\n              updatePlaybackTime(event)\n              videoProps.onCanPlay?.(event)\n            }}\n            onTimeUpdate={updatePlaybackTime}\n            onDurationChange={updatePlaybackTime}\n            onPlayerLoaded={player => {\n              playerRef.current = player\n              onPlayerLoaded?.(player)\n              syncVolume(videoRef.current, videoProps.muted ? 0 : appVolume)\n            }}\n          />\n        }\n        containerRef={element => {\n          containerRef.current = element\n          observe(element)\n        }}\n        backRef={adContainerRef}\n        title={title}\n        channelTitle={channelTitle}\n        backButton={onBack && <Button startIcon=\"back\" onClick={onBack} />}\n        {...(isUserActive && uiElements)}\n        onClick={onClick}\n        onMouseMove={onMouseMove}\n      >\n        <LoadingSpinner active={waiting} />\n        {errorData && <Error error={errorData} onBack={onBack} />}\n        {children}\n        {/* TODO sync Cast last played time back */}\n        {isUserActive && (\n          <components.CastUi\n            onBack={onBack}\n            onStateChange={setCastState}\n            onLoad={onCast}\n            {...compatibilityCastProps}\n            {...slotProps.castUi}\n          />\n        )}\n        {isUserActive && ( // TODO: Design <Extenstion /> flag\n          <components.Settings\n            type={uiType}\n            sections={settings.sections}\n            // TODO hasBottomPanel bottom: 8em\n            open={userFocus === 'settings'}\n            values={settings.values}\n            onOpen={openSettings}\n            onChange={uiHandlers.onChangeSettings}\n            {...slotProps?.settings}\n            onClose={uiHandlers.onCloseSettings}\n          />\n        )}\n        <LanguageSettings\n          uiType={uiType}\n          getPlayer={() => playerRef.current}\n          onChange={(_event, item) =>\n            uiHandlers.onChangeSettings({\n              name: item.type,\n              value: item.value,\n              keepOpen: true,\n            })\n          }\n          {...slotProps.languageSettings}\n        />\n        <Backdrop open={!playbackState || playbackState === 'loading'}>\n          <LoadingSpinner />\n        </Backdrop>\n      </Layout>\n    </IntlProvider>\n  )\n}\n\nexport default PremiumPlayer\n"]} */"),
|
|
8800
|
+
desktop: /*#__PURE__*/css$1(LanguageMenu.styles, LanguageMenu.desktopStyles, process.env.NODE_ENV === "production" ? "" : ";label:desktop;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["PremiumPlayer.js"],"names":[],"mappings":"AAwEW","file":"PremiumPlayer.js","sourcesContent":["/* eslint-disable no-nested-ternary */\nimport {useEffect, useMemo, useRef, useState} from 'react'\nimport {css} from '@emotion/css'\nimport useDimensions from 'react-cool-dimensions'\nimport {ResizeObserver} from '@juggle/resize-observer'\n\nimport {handleError} from 'playerCore/errors'\nimport {on} from 'util/events'\nimport {isDesktop} from 'util/environment'\nimport multiRef from 'util/multiRef'\nimport {onViewModeChange, toggleFullscreen} from 'util/viewModes'\nimport {useAutoHide} from 'hooks'\nimport {IntlProvider} from 'context/I18n'\nimport {getMediaTime, isBuffered} from 'playerCore/mediaBindings'\nimport Error from 'playerUi/Error'\nimport Layout from 'playerUi/DefaultLayout'\nimport {\n  Button,\n  LiveButton,\n  FullscreenButton,\n  PlayButton,\n  ForwardButton,\n} from 'playerUi/buttons'\nimport Seekbar from 'playerUi/Seekbar'\nimport DisplayTime from 'playerUi/DisplayTime'\nimport LoadingSpinner from 'playerUi/LoadingSpinner'\nimport Backdrop from 'playerUi/Backdrop'\nimport Settings from 'playerUi/Settings'\nimport OverlayPanel from 'playerUi/OverlayPanel'\nimport LanguageMenu from 'playerUi/LanguageMenu'\nimport PlayPanel from 'playerUi/PlayPane'\nimport VolumeControl from 'component/VolumeControl'\nimport CastUi from 'cast/CastUi'\nimport {linkCast} from 'cast/framework'\nimport {CastState} from 'Enum'\nimport {dispatchChapterEvents} from './timeline'\nimport Video from '../Video'\nimport {getLanguageOptions, getQualityOptions, getSettingsData} from './settings'\nimport {linkMediaVolume, syncVolume} from './volume'\nimport SeekPreview from './SeekPreview'\n\nconst sizes = {\n  'small-embed': 200,\n  embed: 400,\n  'tablet-portrait': 600,\n  'tablet-landscape': 900,\n  desktop: 1200,\n}\n\nconst useLinkState = (request, dependencies = []) => {\n  const [state, setState] = useState()\n  useEffect(() => {\n    request(setState)\n  }, dependencies)\n  return state\n}\n\nconst flipState = state => (state === 'playing' ? 'paused' : 'playing')\n\n// FIXME: too few lines to split a file, looking a better place\nconst getThumbnailsUrl = source =>\n  [].concat(source).find(item => item.type === 'thumbnail')?.src\n\nconst defaultSettings = {}\n\nconst shouldPause = ({userFocus, uiType}) =>\n  (uiType === 'mobile' && /settings|chapter-list/.test(userFocus)) ||\n  /blocking|seekbar|share-inner/.test(userFocus)\n\n// TODO extract to somewhere else\nconst menuClasses = {\n  default: css(LanguageMenu.styles),\n  desktop: css(LanguageMenu.styles, LanguageMenu.desktopStyles),\n}\n\nconst LanguageSettings = ({\n  uiType,\n  audioTracks = [],\n  textTracks = [],\n  getPlayer,\n  onChange,\n  slots = {LanguageMenu},\n  ...rest\n}) => {\n  const options = getLanguageOptions(getPlayer(), {audioTracks, textTracks})\n  return (\n    audioTracks.length > 0 &&\n    textTracks.length > 0 && (\n    <OverlayPanel title=\"字幕・音声を切り替える\" buttonIcon=\"subtitle\">\n      <slots.LanguageMenu\n        uiType={uiType}\n        classes={menuClasses}\n        sectionOptions={[options.audioTracks, options.textTracks]}\n        onChange={onChange}\n        {...rest}\n      />\n    </OverlayPanel>\n    )\n  )\n}\n\nconst PremiumPlayer = ({\n  source,\n  startTime,\n  quality = {},\n  title,\n  metadata,\n  channelTitle,\n  chapters,\n  playbackState: appPlaybackState,\n  currentTime: appCurrentTime,\n  playbackRate: appPlaybackRate,\n  loop,\n  volume: appVolume,\n  audioTrack: appAudioTrack,\n  textTrack: appTextTrack,\n  thumbnailsUrl,\n  controls = {autohide: 3000},\n  marks = [],\n  // TODO sectionId\n  intl,\n  settings: appSettings = defaultSettings,\n  blocking,\n  plugins = [],\n  modulesConfig,\n\n  uiType = isDesktop() ? 'desktop' : 'mobile',\n  style,\n  children,\n  uiMode = 'standalone',\n  cast: castOptions = 'compatible',\n  slots,\n  slotProps = {},\n  onError,\n  onPlaybackStateChange,\n  onBack,\n  onCast,\n  onChangeNext,\n  onChangePrevious,\n  onOpenSettings,\n  onChangeSettings,\n  onPlayerLoaded,\n  onBlockedAutoplay,\n  onPlaylogFired,\n  ...videoProps\n}) => {\n  const components = {\n    VolumeControl,\n    CastUi,\n    DisplayTime,\n    LiveButton,\n    Seekbar,\n    Settings,\n    ...slots,\n  }\n  const videoRef = useRef()\n  const containerRef = useRef()\n  const playerRef = useRef()\n  const adContainerRef = useRef()\n  const refs = useRef({})\n  // TODO move RWD related to Layout\n  const {\n    currentBreakpoint: size,\n    width,\n    observe,\n  } = useDimensions({\n    polyfill: ResizeObserver,\n    breakpoints: sizes,\n  })\n  const [isUserActive, setIsUserActive] = useState(\n    uiMode === 'standalone' ? true : videoProps.autoplay\n  )\n  const [userFocus, setUserFocus] = useState('')\n  const [targetState, setTargetState] = useState(() => ({\n    playbackState:\n      appPlaybackState || (videoProps.autoplay ? 'playing' : 'paused'),\n    currentTime: startTime,\n  }))\n  const [playbackTime, setPlaybackTime] = useState({\n    currentTime: 0,\n    bufferTime: 0,\n  })\n  const togglePlay = (overrideState, {triggeredBy} = {}) => {\n    if (targetState.playbackState !== overrideState) {\n      setTargetState(state => ({\n        ...state,\n        playbackState: overrideState || flipState(state.playbackState),\n        animation:\n          triggeredBy === 'user-action' && flipState(state.playbackState),\n      }))\n    }\n  }\n\n  useEffect(() => {\n    if (appPlaybackState) {\n      togglePlay(appPlaybackState)\n    }\n  }, [appPlaybackState])\n  useEffect(() => {\n    if (uiType === 'mobile') {\n      on(document, 'visibilitychange', () => togglePlay('paused'))\n    }\n  }, [uiType, targetState.playbackState])\n  const updatePlaybackTime = event =>\n    requestAnimationFrame(\n      () =>\n        (event?.type !== 'timeupdate' || isBuffered(videoRef.current)) &&\n        setPlaybackTime(state => ({\n          ...state,\n          ...getMediaTime(videoRef.current, {\n            player: playerRef.current,\n            plugins,\n          }),\n          ...(event?.type === 'durationchange' && {\n            currentTime: state.currentTime,\n          }),\n        }))\n    )\n  const setTargetTime = time => {\n    const trimmed = Math.min(\n      time,\n      videoRef.current?.initialDuration || Infinity\n    )\n    setTargetState(state => ({\n      ...state,\n      // seek to 0 repeatedly edge case\n      currentTime: state.currentTime !== trimmed ? trimmed : trimmed + 0.01,\n    }))\n    updatePlaybackTime()\n  }\n  const [playbackState, setPlaybackState] = useState('init')\n  const [castState, setCastState] = useState('')\n  useEffect(() => {\n    if (typeof appCurrentTime === 'number') setTargetTime(appCurrentTime || 0)\n    if (typeof appCurrentTime === 'object')\n      setTargetTime(appCurrentTime?.value || 0)\n  }, [appCurrentTime])\n  const [errorData, setErrorData] = useState()\n  const errorHandler = reactEvent => {\n    onError?.(reactEvent.nativeEvent)\n    handleError(reactEvent, {\n      media: videoRef.current,\n      displayError: data => {\n        setPlaybackState('error')\n        setErrorData(current => (current?.code ? current : data))\n      },\n    })\n  }\n  const isLive = playbackTime.streamType === 'live'\n  const [settings, setSettings] = useState(() => ({\n    sections: [],\n    values: {speed: 1},\n  }))\n  const fetchSettings = () => {\n    if (!playerRef.current) {\n      return\n    }\n    const speedItems = appSettings.speedItems || (isLive ? [] : undefined)\n    setSettings(current => {\n      const {values, sections} = getSettingsData({\n        media: videoRef.current,\n        player: playerRef.current,\n        source,\n        quality,\n        speedItems,\n        loop,\n        preferred: current.preferred,\n        otherSections: appSettings.sections,\n      })\n      return {preferred: current.preferred, values, sections}\n    })\n  }\n  const lastState = useRef(playbackState)\n  const handlePlaybackStateChange = (event, state) => {\n    if (lastState.current === 'error') {\n      return\n    }\n    onPlaybackStateChange?.(event, state)\n    // previously UI playback state was synced only when\n    // exiting iOS video only fullscreen(webkitendfullscreen)\n    if (\n      /playing|paused/.test(state) &&\n      !(uiType === 'mobile' && blocking) &&\n      !shouldPause({userFocus, uiType})\n    ) {\n      togglePlay(state)\n    }\n    if (state === 'ended') {\n      togglePlay('paused')\n    }\n    if (state === 'loading' && lastState.current !== 'init') {\n      setTimeout(() => togglePlay('playing'), 1)\n    }\n    if (state === 'playing') {\n      setIsUserActive(true)\n    }\n    if (lastState.current === 'loading') {\n      fetchSettings()\n    }\n    lastState.current = state\n    setPlaybackState(state)\n  }\n\n  const changeSettings = (name, value, {keepOpen} = {}) => {\n    // TODO consider merge into useReducer?\n    onChangeSettings?.({name, value})\n    setTargetTime(playbackTime.currentTime)\n    setSettings(current => ({\n      ...current,\n      values: {...current.values, [name]: value},\n      preferred: {...current.preferred, [name]: value},\n    }))\n    if (!keepOpen) {\n      setUserFocus('')\n    }\n  }\n  const openSettings = event => {\n    const animationFrame =\n      userFocus !== 'settings' &&\n      requestAnimationFrame(() => {\n        onOpenSettings?.(event, settings)\n        // In iOS Safari, we need to update settings data\n        fetchSettings()\n      })\n    setUserFocus(current => (current === 'settings' ? '' : 'settings'))\n    return animationFrame\n  }\n  useEffect(() => {\n    if (appPlaybackRate > 0) {\n      setSettings(current => ({\n        ...current,\n        values: {...current.values, speed: appPlaybackRate},\n      }))\n    }\n  }, [appPlaybackRate])\n\n  const qualityOptions = useMemo(\n    () => getQualityOptions(settings),\n    [settings.values.quality]\n  )\n  const viewMode = useLinkState(update =>\n    onViewModeChange(videoRef.current, update)\n  )\n  const sourceOverride = useLinkState(\n    async update => {\n      const result =\n        source && (await quality.rewriteManifest?.(source, qualityOptions))\n      update(result || source)\n    },\n    [source, qualityOptions]\n  )\n  const waiting = /emptied|loading|buffering/.test(playbackState)\n  const activePlayback =\n    playbackState === 'playing' || playbackState === 'waiting'\n  const {\n    mode: autoHideMode,\n    onClick,\n    onMouseMove,\n  } = useAutoHide({\n    pinned: !controls.autohide || waiting || !activePlayback || userFocus,\n    tapToHide: uiType === 'mobile',\n    hideTimeMs: controls.autohide,\n  })\n  const mode =\n    userFocus === 'seekbar'\n      ? 'hidden'\n      : controls.autohide\n      ? autoHideMode\n      : controls\n      ? 'shown'\n      : 'hidden'\n  const controlsDisplay =\n    userFocus === 'seekbar'\n      ? 'seekbar-only'\n      : controls === 'title-only' || playbackState === 'emptied'\n      ? 'hidden'\n      : mode\n  const shouldHidePanels =\n    (controls === 'no-panel' || controlsDisplay === 'hidden') && userFocus\n  const havePlayPanel =\n    !blocking &&\n    uiType === 'desktop' &&\n    !waiting &&\n    !/volume|''/.test(userFocus) &&\n    !/title-only|no-panel/.test(controls) &&\n    (controls.autohide || mode === 'shown')\n\n  useEffect(() => {\n    if (shouldHidePanels) {\n      setUserFocus('')\n    }\n  }, [shouldHidePanels])\n  useEffect(() => \n    on(containerRef.current, 'focus-menu', event => {\n      if (event.detail?.status === 'open') {\n        setUserFocus('menu')\n      }\n      if (event.detail?.status === 'closed') {\n        setUserFocus(current => (current === 'menu' ? '' : current))\n      }\n    })\n  , [])\n  const {subscribe, onChange, toggleMute} = linkMediaVolume(() => ({\n    video: videoRef.current,\n    getPlayer: () => playerRef.current,\n  }))\n  const changePrevious = event => {\n    onChangePrevious(event)\n    togglePlay('paused')\n    videoRef.current.dispatchEvent(new CustomEvent('loadstart'))\n  }\n  const changeNext = event => {\n    onChangeNext(event)\n    togglePlay('paused')\n    videoRef.current.dispatchEvent(new CustomEvent('loadstart'))\n  }\n\n  useEffect(fetchSettings, [appSettings])\n  useEffect(() => {\n    // The adContainer should be set before `load` because ImaDai.load needs it.\n    plugins.forEach(plugin => plugin.setAdContainer?.(adContainerRef.current))\n  }, [])\n\n  useEffect(\n    () =>\n      dispatchChapterEvents({\n        media: videoRef.current,\n        chapters,\n        getTime: () => videoRef.current.currentTime,\n      }),\n    [chapters]\n  )\n\n  const isEnd = playbackState === 'ended'\n  const canSeek = !isEnd && playbackTime.duration > 0\n  const derivedPlaybackState =\n    (uiType === 'mobile' && blocking) || shouldPause({userFocus, uiType})\n      ? 'paused'\n      : targetState.playbackState\n  const activeThumbnailsUrl =\n    (!userFocus || userFocus === 'seekbar') &&\n    (getThumbnailsUrl(source) || thumbnailsUrl)\n  const cssVariables = {\n    '--playing': playbackState === 'playing' ? '1' : '0',\n  }\n  const onRewind =\n    (!isLive || playbackTime.currentTime < 0) &&\n    canSeek &&\n    (() => setTargetTime(playbackTime.currentTime - 10, 'rewind'))\n  const onForward =\n    (!isLive || playbackTime.currentTime < 0) &&\n    canSeek &&\n    (() => setTargetTime(playbackTime.currentTime + 10, 'forward'))\n  const onClickBlank =\n    havePlayPanel && (() => togglePlay('', {triggeredBy: 'user-action'}))\n\n  const uiHandlers = {\n    onPlay: () => {\n      togglePlay('', {triggeredBy: 'user-action'})\n      setIsUserActive(true)\n    },\n    onClickLive: isLive && (() => setTargetTime(0, 'seekToLive')),\n    onClickSeekbar: () => {\n      refs.current.deferedSeekFocus = setTimeout(\n        () => setUserFocus('seekbar'),\n        66\n      )\n    },\n    onSeek: time => {\n      setTargetTime(time, 'seek')\n      clearTimeout(refs.current.deferedSeekFocus)\n      setTimeout(() => setUserFocus(''), 66)\n    },\n    onChangeSettings: ({name, value, keepOpen}) =>\n      changeSettings(name, value, {keepOpen}),\n    onCloseSettings: event => {\n      setUserFocus('')\n      slotProps?.settings?.onClose?.(event)\n    },\n  }\n\n  const uiElements = {\n    liveButton: uiHandlers.onClickLive && (\n      <components.LiveButton\n        usingStartOver={playbackTime.currentTime < 0}\n        onClick={uiHandlers.onClickLive}\n      />\n    ),\n    controlButtons: {\n      playButton: (\n        <PlayButton\n          playbackState={derivedPlaybackState}\n          ended={isEnd}\n          hidden={\n            uiType === 'mobile' && (waiting || /loading/.test(playbackState))\n          }\n          variant={!isUserActive && 'firstplay'}\n          onClick={uiHandlers.onPlay}\n        />\n      ),\n      rewindButton: onRewind && (\n        <Button\n          startIcon=\"rewind10\"\n          title=\"KKS.PLAYER.REWIND\"\n          disabled={isEnd}\n          onClick={onRewind}\n        />\n      ),\n      forwardButton: onForward && (\n        <ForwardButton\n          startIcon=\"forward10\"\n          title=\"KKS.PLAYER.FORWARD\"\n          disabled={!canSeek}\n          onClick={onForward}\n        />\n      ),\n      nextEpisodeButton: (\n        <Button\n          startIcon=\"next\"\n          title=\"KKS.PLAYER.NEXT\"\n          disabled={!onChangeNext}\n          onClick={changeNext}\n        />\n      ),\n      previousEpisodeButton: (\n        <Button\n          startIcon=\"previous\"\n          title=\"KKS.PLAYER.PREVIOUS\"\n          disabled={!onChangePrevious}\n          onClick={changePrevious}\n        />\n      ),\n    },\n    seekbar: (\n      <components.Seekbar\n        style={/volume/.test(userFocus) && {opacity: '0'}}\n        startTime={playbackTime.startTime}\n        currentTime={playbackTime.currentTime}\n        bufferTime={playbackTime.bufferTime}\n        duration={playbackTime.duration}\n        chapters={chapters}\n        onChange={uiHandlers.onClickSeekbar}\n        onChangeCommitted={uiHandlers.onSeek}\n        marks={marks}\n        plugins={plugins}\n        {...slotProps.seekbar}\n      >\n        {activeThumbnailsUrl && (\n          <SeekPreview\n            thumbnailsUrl={activeThumbnailsUrl}\n            duration={playbackTime.duration}\n            chapters={chapters}\n          />\n        )}\n      </components.Seekbar>\n    ),\n    displayTime: <components.DisplayTime {...playbackTime} />,\n    fullscreenButton: (\n      <FullscreenButton\n        viewMode={viewMode}\n        onClick={() => toggleFullscreen(containerRef.current)}\n      />\n    ),\n    volumeControl: width >= sizes['small-embed'] && (\n      <components.VolumeControl\n        {...{subscribe, onChange, toggleMute}}\n        onMouseOver={() => setUserFocus('volume')}\n        onMouseOut={() => setUserFocus('')}\n        {...slotProps?.volumeControl}\n      />\n    ),\n    backItems: (\n      <PlayPanel animation={targetState.animation} onClick={onClickBlank} />\n    ),\n  }\n  const compatibilityCastProps = castOptions === 'compatible' && {\n    source,\n    values: {title, ...settings.values, ...intl, modulesConfig},\n    onLoad: onCast,\n  }\n\n  useEffect(() => {\n    if (castOptions && castOptions !== 'compatible') {\n      return linkCast({source, ...castOptions})\n    } // TODO playlist, audio / subtitle setting\n  }, [source, castOptions.contentId, castOptions.customData])\n  const selectedAudioTrack = settings.preferred?.audio || appAudioTrack\n  const selectedTextTrack = settings.preferred?.subtitles || appTextTrack\n\n  return (\n    <IntlProvider {...intl}>\n      <Layout\n        style={{...style, cssVariables}}\n        type={uiType}\n        display={mode}\n        controlsDisplay={controlsDisplay}\n        size={size}\n        video={\n          <Video\n            {...videoProps}\n            videoRef={multiRef(videoRef, videoProps.videoRef)}\n            source={\n              castState !== CastState.CONNECTED &&\n              playbackState !== 'error' &&\n              sourceOverride\n            }\n            playbackState={derivedPlaybackState}\n            currentTime={targetState.currentTime}\n            {...(appVolume >= 0 && {volume: appVolume, muted: appVolume <= 0})}\n            playbackRate={settings.values.speed}\n            videoResolution={qualityOptions}\n            audioTrack={selectedAudioTrack}\n            textTrack={selectedTextTrack}\n            loop={settings.values.loop}\n            plugins={plugins}\n            modulesConfig={modulesConfig}\n            onPlaylogFired={onPlaylogFired}\n            onError={errorHandler}\n            onPlaybackStateChange={handlePlaybackStateChange}\n            onBlockedAutoplay={error => onBlockedAutoplay?.(error)}\n            onCanPlay={event => {\n              updatePlaybackTime(event)\n              videoProps.onCanPlay?.(event)\n            }}\n            onTimeUpdate={updatePlaybackTime}\n            onDurationChange={updatePlaybackTime}\n            onPlayerLoaded={player => {\n              playerRef.current = player\n              onPlayerLoaded?.(player)\n              syncVolume(videoRef.current, videoProps.muted ? 0 : appVolume)\n            }}\n          />\n        }\n        containerRef={element => {\n          containerRef.current = element\n          observe(element)\n        }}\n        backRef={adContainerRef}\n        title={title}\n        channelTitle={channelTitle}\n        backButton={onBack && <Button startIcon=\"back\" onClick={onBack} />}\n        {...(isUserActive && uiElements)}\n        onClick={onClick}\n        onMouseMove={onMouseMove}\n      >\n        <LoadingSpinner active={waiting} />\n        {errorData && <Error error={errorData} onBack={onBack} />}\n        {children}\n        {/* TODO sync Cast last played time back */}\n        {isUserActive && (\n          <components.CastUi\n            onBack={onBack}\n            onStateChange={setCastState}\n            onLoad={onCast}\n            {...compatibilityCastProps}\n            {...slotProps.castUi}\n          />\n        )}\n        {isUserActive && ( // TODO: Design <Extenstion /> flag\n          <components.Settings\n            type={uiType}\n            sections={settings.sections}\n            // TODO hasBottomPanel bottom: 8em\n            open={userFocus === 'settings'}\n            values={settings.values}\n            onOpen={openSettings}\n            onChange={uiHandlers.onChangeSettings}\n            {...slotProps?.settings}\n            onClose={uiHandlers.onCloseSettings}\n          />\n        )}\n        <LanguageSettings\n          uiType={uiType}\n          getPlayer={() => playerRef.current}\n          onChange={(_event, item) =>\n            uiHandlers.onChangeSettings({\n              name: item.type,\n              value: item.value,\n              keepOpen: true,\n            })\n          }\n          {...slotProps.languageSettings}\n        />\n        <Backdrop open={!playbackState || playbackState === 'loading'}>\n          <LoadingSpinner />\n        </Backdrop>\n      </Layout>\n    </IntlProvider>\n  )\n}\n\nexport default PremiumPlayer\n"]} */")
|
|
8154
8801
|
};
|
|
8155
8802
|
|
|
8156
8803
|
const LanguageSettings = ({
|
|
8157
8804
|
uiType,
|
|
8158
8805
|
audioTracks = [],
|
|
8159
|
-
|
|
8806
|
+
textTracks = [],
|
|
8160
8807
|
getPlayer,
|
|
8161
8808
|
onChange,
|
|
8162
8809
|
slots = {
|
|
@@ -8166,9 +8813,10 @@ const LanguageSettings = ({
|
|
|
8166
8813
|
}) => {
|
|
8167
8814
|
const options = getLanguageOptions(getPlayer(), {
|
|
8168
8815
|
audioTracks,
|
|
8169
|
-
|
|
8816
|
+
textTracks
|
|
8170
8817
|
});
|
|
8171
|
-
return /*#__PURE__*/jsx(OverlayPanel, {
|
|
8818
|
+
return audioTracks.length > 0 && textTracks.length > 0 && /*#__PURE__*/jsx(OverlayPanel, {
|
|
8819
|
+
title: "\u5B57\u5E55\u30FB\u97F3\u58F0\u3092\u5207\u308A\u66FF\u3048\u308B",
|
|
8172
8820
|
buttonIcon: "subtitle",
|
|
8173
8821
|
children: /*#__PURE__*/jsx(slots.LanguageMenu, {
|
|
8174
8822
|
uiType: uiType,
|
|
@@ -8193,6 +8841,8 @@ const PremiumPlayer = ({
|
|
|
8193
8841
|
playbackRate: appPlaybackRate,
|
|
8194
8842
|
loop,
|
|
8195
8843
|
volume: appVolume,
|
|
8844
|
+
audioTrack: appAudioTrack,
|
|
8845
|
+
textTrack: appTextTrack,
|
|
8196
8846
|
thumbnailsUrl,
|
|
8197
8847
|
controls = {
|
|
8198
8848
|
autohide: 3000
|
|
@@ -8224,6 +8874,8 @@ const PremiumPlayer = ({
|
|
|
8224
8874
|
onPlaylogFired,
|
|
8225
8875
|
...videoProps
|
|
8226
8876
|
}) => {
|
|
8877
|
+
var _settings$preferred, _settings$preferred2;
|
|
8878
|
+
|
|
8227
8879
|
const components = {
|
|
8228
8880
|
VolumeControl,
|
|
8229
8881
|
CastUi,
|
|
@@ -8345,7 +8997,8 @@ const PremiumPlayer = ({
|
|
|
8345
8997
|
quality,
|
|
8346
8998
|
speedItems,
|
|
8347
8999
|
loop,
|
|
8348
|
-
preferred: current.preferred
|
|
9000
|
+
preferred: current.preferred,
|
|
9001
|
+
otherSections: appSettings.sections
|
|
8349
9002
|
});
|
|
8350
9003
|
return {
|
|
8351
9004
|
preferred: current.preferred,
|
|
@@ -8456,12 +9109,23 @@ const PremiumPlayer = ({
|
|
|
8456
9109
|
const mode = userFocus === 'seekbar' ? 'hidden' : controls.autohide ? autoHideMode : controls ? 'shown' : 'hidden';
|
|
8457
9110
|
const controlsDisplay = userFocus === 'seekbar' ? 'seekbar-only' : controls === 'title-only' || playbackState === 'emptied' ? 'hidden' : mode;
|
|
8458
9111
|
const shouldHidePanels = (controls === 'no-panel' || controlsDisplay === 'hidden') && userFocus;
|
|
8459
|
-
const havePlayPanel = !blocking &&
|
|
9112
|
+
const havePlayPanel = !blocking && uiType === 'desktop' && !waiting && !/volume|''/.test(userFocus) && !/title-only|no-panel/.test(controls) && (controls.autohide || mode === 'shown');
|
|
8460
9113
|
useEffect(() => {
|
|
8461
9114
|
if (shouldHidePanels) {
|
|
8462
9115
|
setUserFocus('');
|
|
8463
9116
|
}
|
|
8464
9117
|
}, [shouldHidePanels]);
|
|
9118
|
+
useEffect(() => on(containerRef.current, 'focus-menu', event => {
|
|
9119
|
+
var _event$detail, _event$detail2;
|
|
9120
|
+
|
|
9121
|
+
if (((_event$detail = event.detail) === null || _event$detail === void 0 ? void 0 : _event$detail.status) === 'open') {
|
|
9122
|
+
setUserFocus('menu');
|
|
9123
|
+
}
|
|
9124
|
+
|
|
9125
|
+
if (((_event$detail2 = event.detail) === null || _event$detail2 === void 0 ? void 0 : _event$detail2.status) === 'closed') {
|
|
9126
|
+
setUserFocus(current => current === 'menu' ? '' : current);
|
|
9127
|
+
}
|
|
9128
|
+
}), []);
|
|
8465
9129
|
const {
|
|
8466
9130
|
subscribe,
|
|
8467
9131
|
onChange,
|
|
@@ -8508,7 +9172,7 @@ const PremiumPlayer = ({
|
|
|
8508
9172
|
'--playing': playbackState === 'playing' ? '1' : '0'
|
|
8509
9173
|
};
|
|
8510
9174
|
|
|
8511
|
-
const onRewind = (!isLive || playbackTime.currentTime < 0) &&
|
|
9175
|
+
const onRewind = (!isLive || playbackTime.currentTime < 0) && canSeek && (() => setTargetTime(playbackTime.currentTime - 10));
|
|
8512
9176
|
|
|
8513
9177
|
const onForward = (!isLive || playbackTime.currentTime < 0) && canSeek && (() => setTargetTime(playbackTime.currentTime + 10));
|
|
8514
9178
|
|
|
@@ -8642,6 +9306,8 @@ const PremiumPlayer = ({
|
|
|
8642
9306
|
} // TODO playlist, audio / subtitle setting
|
|
8643
9307
|
|
|
8644
9308
|
}, [source, castOptions.contentId, castOptions.customData]);
|
|
9309
|
+
const selectedAudioTrack = ((_settings$preferred = settings.preferred) === null || _settings$preferred === void 0 ? void 0 : _settings$preferred.audio) || appAudioTrack;
|
|
9310
|
+
const selectedTextTrack = ((_settings$preferred2 = settings.preferred) === null || _settings$preferred2 === void 0 ? void 0 : _settings$preferred2.subtitles) || appTextTrack;
|
|
8645
9311
|
return /*#__PURE__*/jsx(IntlProvider, { ...intl,
|
|
8646
9312
|
children: /*#__PURE__*/jsxs$1(DefaultLayout, {
|
|
8647
9313
|
style: { ...style,
|
|
@@ -8662,8 +9328,8 @@ const PremiumPlayer = ({
|
|
|
8662
9328
|
}),
|
|
8663
9329
|
playbackRate: settings.values.speed,
|
|
8664
9330
|
videoResolution: qualityOptions,
|
|
8665
|
-
|
|
8666
|
-
textTrack:
|
|
9331
|
+
audioTrack: selectedAudioTrack,
|
|
9332
|
+
textTrack: selectedTextTrack,
|
|
8667
9333
|
loop: settings.values.loop,
|
|
8668
9334
|
plugins: plugins,
|
|
8669
9335
|
modulesConfig: modulesConfig,
|
|
@@ -8726,7 +9392,11 @@ const PremiumPlayer = ({
|
|
|
8726
9392
|
}), /*#__PURE__*/jsx(LanguageSettings, {
|
|
8727
9393
|
uiType: uiType,
|
|
8728
9394
|
getPlayer: () => playerRef.current,
|
|
8729
|
-
onChange: (_event, item) =>
|
|
9395
|
+
onChange: (_event, item) => uiHandlers.onChangeSettings({
|
|
9396
|
+
name: item.type,
|
|
9397
|
+
value: item.value,
|
|
9398
|
+
keepOpen: true
|
|
9399
|
+
}),
|
|
8730
9400
|
...slotProps.languageSettings
|
|
8731
9401
|
}), /*#__PURE__*/jsx(Backdrop, {
|
|
8732
9402
|
open: !playbackState || playbackState === 'loading',
|