@kkcompany/player 2.25.0-canary.1 → 2.25.0-canary.11

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/dist/index.mjs CHANGED
@@ -3,6 +3,7 @@ import React, { useRef, useState, useEffect, useContext, cloneElement, createCon
3
3
  import ReactDom, { createPortal } from 'react-dom';
4
4
  import mitt from 'mitt';
5
5
  import UAParser from 'ua-parser-js';
6
+ import { keyframes as keyframes$1, css as css$1 } from '@emotion/css';
6
7
  import useDimensions from 'react-cool-dimensions';
7
8
  import { ResizeObserver as ResizeObserver$1 } from '@juggle/resize-observer';
8
9
  import require$$0 from 'react-is';
@@ -10,7 +11,6 @@ import { jsx, jsxs as jsxs$1, Fragment as Fragment$1 } from 'react/jsx-runtime';
10
11
  import { css, keyframes } from '@emotion/react';
11
12
  import { jsx as jsx$1, jsxs, Fragment } from '@emotion/react/jsx-runtime';
12
13
  import useOnclickOutside from 'react-cool-onclickoutside';
13
- import { keyframes as keyframes$1 } from '@emotion/css';
14
14
  import { selectHlsQualities } from 'playcraft/modules';
15
15
  import { CastSender } from 'playcraft/react';
16
16
 
@@ -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,82 +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.1";
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
- const matchAll = (input, pattern) => {
325
- const flags = [pattern.global && 'g', pattern.ignoreCase && 'i', pattern.multiline && 'm'].filter(Boolean).join('');
326
- const clone = new RegExp(pattern, flags);
327
- return Array.from(function* () {
328
- let matched = true;
329
-
330
- while (1) {
331
- matched = clone.exec(input);
332
-
333
- if (!matched) {
334
- return;
335
- }
336
-
337
- yield matched;
338
- }
339
- }());
340
- };
341
-
342
- const meetRestriction = (quality, {
343
- minHeight,
344
- maxHeight
345
- } = {}) => !(quality.height < minHeight || quality.height > maxHeight);
346
-
347
- const getHlsQualityOptions = manifest => {
348
- const resolutionList = matchAll(manifest, /RESOLUTION=\d+x(\d+)/g);
349
- return Array.from(new Set(resolutionList.map(([, height]) => ({
350
- height: +height
351
- })))).sort((a, b) => b.height - a.height);
352
- };
353
-
354
- const selectRestrictedQuality = (availableQualities, {
355
- suggested,
356
- restrictions
357
- }) => {
358
- if (meetRestriction(suggested, restrictions)) {
359
- return suggested.id;
360
- }
361
-
362
- const allowed = availableQualities.filter(quality => meetRestriction(quality, restrictions));
363
- return (nearest(allowed, item => item.height - suggested.height) || suggested).id;
364
- };
365
- // for unit test
366
-
367
251
  /* eslint-disable no-param-reassign */
368
252
  // when the gap is small enough, we consider it is on edge.
369
253
  // The magic number 10s comes from observation of YouTube
@@ -377,7 +261,7 @@ const isLiveDuration = duration => duration >= SHAKA_LIVE_DURATION;
377
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.
378
262
 
379
263
 
380
- const isBuffered = media => needNativeHls() || Array.from({
264
+ const isBuffered = media => isSafari() || Array.from({
381
265
  length: media.buffered.length
382
266
  }, (_, index) => ({
383
267
  start: media.buffered.start(index),
@@ -533,22 +417,6 @@ const subscribePlaybackState = (media, updateState, {
533
417
  return () => registered.forEach(off => off());
534
418
  };
535
419
 
536
- const tryPatchHlsVideoQualities = async (player, hlsUrl) => {
537
- if (/(^data)|(mp4$)/.test(hlsUrl)) {
538
- return;
539
- } // filtered manifest comes with data URI and should be ignored
540
-
541
-
542
- const manifest = await retryRequest(hlsUrl, {}, {
543
- responseType: 'text'
544
- }).catch(e => console.warn('Failed to get HLS video qualities', e));
545
- const videoQualities = getHlsQualityOptions(manifest);
546
-
547
- if (videoQualities) {
548
- player.getAvailableVideoQualities = () => videoQualities;
549
- }
550
- };
551
-
552
420
  const seek = async (media, {
553
421
  player,
554
422
  plugins = []
@@ -698,10 +566,6 @@ const load = async (media, {
698
566
  loadStartTime = merged.startTime;
699
567
  }
700
568
 
701
- if (merged.type === 'application/x-mpegurl') {
702
- await tryPatchHlsVideoQualities(player, merged.src);
703
- }
704
-
705
569
  return player.unload().then(() => player.load(merged.src, loadStartTime, merged.type)).then(loadResult => {
706
570
  getSourceText(source).forEach(({
707
571
  src,
@@ -827,51 +691,40 @@ const setPlaybackRate = (media, {
827
691
  media.playbackRate = rate;
828
692
  };
829
693
 
830
- const setQuality = (_, {
831
- player
832
- }, restrictions) => {
833
- var _player$setAdaptation, _player$setQuality;
834
-
835
- // For Bitmovin
836
- (_player$setAdaptation = player.setAdaptationHandler) === null || _player$setAdaptation === void 0 ? void 0 : _player$setAdaptation.call(player, ({
837
- availableQualities,
838
- suggested
839
- }) => selectRestrictedQuality(availableQualities, {
840
- suggested,
841
- restrictions
842
- })); // For Shaka
843
-
844
- (_player$setQuality = player.setQuality) === null || _player$setQuality === void 0 ? void 0 : _player$setQuality.call(player, restrictions);
845
- };
846
-
847
694
  const getTextTracks = (_, {
848
695
  player
849
696
  }) => {
850
- var _player$getSubtitles;
697
+ var _player$getTextTracks;
851
698
 
852
699
  if (!player) return [];
700
+ return (((_player$getTextTracks = player.getTextTracks) === null || _player$getTextTracks === void 0 ? void 0 : _player$getTextTracks.call(player)) || []).map(track => {
701
+ var _track$label;
853
702
 
854
- if (player.getTextTracks) {
855
- return player.getTextTracks().map(track => {
856
- var _track$label;
857
-
858
- return {
859
- label: ((_track$label = track.label) === null || _track$label === void 0 ? void 0 : _track$label.trim()) || track.language,
860
- language: track.language,
861
- enabled: player.isTextTrackVisible() ? track.active : false,
862
- id: track.id
863
- };
864
- }).filter(track => track.label !== 'unknown' && track.language && !track.language.startsWith('unk'));
865
- }
703
+ const label = ((_track$label = track.label) === null || _track$label === void 0 ? void 0 : _track$label.trim()) || track.language;
704
+ const id = `${track.language}:${label}`;
705
+ return { ...track,
706
+ label,
707
+ type: 'subtitles',
708
+ selected: player.isTextTrackVisible() ? track.active : false,
709
+ value: {
710
+ id,
711
+ label,
712
+ language: track.language
713
+ }
714
+ };
715
+ }).filter(track => track.label !== 'unknown' && track.language && !/^un/i.test(track.language));
716
+ };
866
717
 
867
- return (player === null || player === void 0 ? void 0 : (_player$getSubtitles = player.getSubtitles()) === null || _player$getSubtitles === void 0 ? void 0 : _player$getSubtitles.filter(track => !/un(d|k)/i.test(track.value || '') && track.label)) || [];
718
+ const textTrackOptionOff = {
719
+ label: 'none',
720
+ language: 'none'
868
721
  };
869
722
 
870
723
  const getCurrentSubtitle = (_, {
871
724
  player
872
725
  }) => getTextTracks({}, {
873
726
  player
874
- }).find(track => track.enabled) || 'off';
727
+ }).find(track => track.selected) || textTrackOptionOff.label;
875
728
 
876
729
  const textTrackLabel = 'playcraft-text-track';
877
730
  /** @param media HTMLMediaElement */
@@ -889,9 +742,7 @@ const syncTextTrack = (media, {
889
742
  label,
890
743
  language,
891
744
  textTracks = []
892
- } = selected === 'off' ? {
893
- language: 'off'
894
- } : selected;
745
+ } = selected === textTrackOptionOff.label ? textTrackOptionOff : selected;
895
746
  const cues = (_textTracks$find = textTracks.find(track => {
896
747
  var _track$cues;
897
748
 
@@ -911,7 +762,7 @@ const syncTextTrack = (media, {
911
762
 
912
763
  Array.from(cues).forEach(cue => customTextTrack.addCue(new VTTCue(cue.startTime, cue.endTime, cue.text)));
913
764
  customTextTrack.mode = 'showing';
914
- } else if (language === 'off') {
765
+ } else if (language === textTrackOptionOff.language) {
915
766
  player.setTextTrackVisibility(false);
916
767
  } else {
917
768
  if (customTextTrack) {
@@ -931,10 +782,119 @@ const syncTextTrack = (media, {
931
782
  }
932
783
 
933
784
  return ((_media$textTracks = media.textTracks) === null || _media$textTracks === void 0 ? void 0 : _media$textTracks.addEventListener) && on(media.textTracks, 'change', () => {
934
- if (language === 'off') {
785
+ if (language === textTrackOptionOff.language) {
935
786
  player.setTextTrackVisibility(false);
936
787
  }
937
788
  });
789
+ }; // assume unique labels as defined in spec
790
+
791
+
792
+ const setAudioTrack = (_, {
793
+ player
794
+ }, next) => {
795
+ if (!next || !player) return;
796
+
797
+ try {
798
+ player.selectAudioLanguage(next.language);
799
+
800
+ if (next.label) {
801
+ player.selectVariantsByLabel(next.label);
802
+ }
803
+ } catch (error) {
804
+ console.warn('Unable to set audio', error, next);
805
+ }
806
+ };
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.11";
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
+
841
+ /* eslint-disable no-param-reassign */
842
+
843
+ const getSpeedItems = items => items.map(value => ({
844
+ label: `${value}x`,
845
+ value
846
+ }));
847
+
848
+ const getQualityOptions = ({
849
+ sections,
850
+ values: {
851
+ quality
852
+ }
853
+ }) => {
854
+ var _sections$find, _sections$find$items$;
855
+
856
+ return (_sections$find = sections.find(item => item.name === 'quality')) === null || _sections$find === void 0 ? void 0 : (_sections$find$items$ = _sections$find.items.find(item => item.value === quality)) === null || _sections$find$items$ === void 0 ? void 0 : _sections$find$items$.options;
857
+ };
858
+
859
+ const autoQualityOption = {
860
+ label: `Auto`,
861
+ options: {
862
+ maxHeight: Infinity,
863
+ minHeight: 0
864
+ },
865
+ value: 'auto'
866
+ };
867
+
868
+ const getQualityItemsFromManifest = player => {
869
+ var _player$getAvailableV;
870
+
871
+ return [autoQualityOption].concat(((player === null || player === void 0 ? void 0 : (_player$getAvailableV = player.getAvailableVideoQualities) === null || _player$getAvailableV === void 0 ? void 0 : _player$getAvailableV.call(player)) || []).filter(q => q.height > 0).sort((a, b) => b.height - a.height).map(q => ({
872
+ label: `${q.height}p`,
873
+ // Set the min/max height to the same value to fix the quality.
874
+ options: {
875
+ maxHeight: q.height,
876
+ minHeight: q.height
877
+ },
878
+ value: q.height
879
+ })));
880
+ };
881
+
882
+ const getQualitySettings = (options, player) => {
883
+ // With native HLS, manifest rewrite is required to enable quality setting
884
+ // TODO let this covered by test, maybe refactor?
885
+ const items = isSafari() && !options.rewriteManifest ? [] : options.items || getQualityItemsFromManifest(player);
886
+
887
+ const getDefault = (preferred = options.default || items[0].value) => {
888
+ const maxHeight = preferred || items[0].value;
889
+ return (nearest(items.filter(item => (item.height || item.value) <= maxHeight), item => (item.height || item.value) - maxHeight) || items[0]).value;
890
+ };
891
+
892
+ return items.length > 0 && items[0] && {
893
+ name: 'quality',
894
+ title: 'KKS.QUALITY',
895
+ items,
896
+ getDefault
897
+ };
938
898
  };
939
899
 
940
900
  const getAudioTracks = (_, {
@@ -942,51 +902,223 @@ const getAudioTracks = (_, {
942
902
  }) => {
943
903
  if (!player) return []; // Bitmovin support is omitted as it is not needed in near future
944
904
 
945
- const all = player.getVariantTracks().reduce((result, track) => {
946
- var _track$label2, _result$id;
905
+ const all = player.getAudioTracks().reduce((result, track) => {
906
+ var _track$label;
947
907
 
948
- const label = ((_track$label2 = track.label) === null || _track$label2 === void 0 ? void 0 : _track$label2.trim()) || track.language;
908
+ const label = ((_track$label = track.label) === null || _track$label === void 0 ? void 0 : _track$label.trim()) || track.language || 'unknown';
949
909
  const id = `${track.language}:${label}`;
950
- result[id] = { ...track,
951
- language: track.language,
910
+ result[id] = result[id] || { ...track,
911
+ type: 'audio',
952
912
  label,
953
- enabled: ((_result$id = result[id]) === null || _result$id === void 0 ? void 0 : _result$id.enabled) || track.active
913
+ value: {
914
+ id,
915
+ label,
916
+ language: track.language
917
+ }
954
918
  };
919
+
920
+ if (track.active) {
921
+ result[id].selected = true;
922
+ }
923
+
955
924
  return result;
956
925
  }, {});
957
- return Object.values(all);
926
+ const allTracks = Object.values(all);
927
+ const namedTracks = allTracks.filter(track => !/^un/i.test(track.language || ''));
928
+ return Object.values(namedTracks.length > 0 ? namedTracks : allTracks);
958
929
  };
959
930
 
960
931
  const getCurrentAudioTrack = (media, {
961
932
  player
962
933
  }) => getAudioTracks(media, {
963
934
  player
964
- }).find(track => track.enabled); // assume unique labels as defined in spec
935
+ }).find(track => track.selected);
965
936
 
937
+ const getSelectedAudioName = player => {
938
+ var _currentAudio$value;
966
939
 
967
- const setAudioTrack = (_, {
940
+ const track = getCurrentAudioTrack({}, {
941
+ player
942
+ });
943
+ /*
944
+ Sometimes, HLS manifest doesn't describe the default audio track.
945
+ Get current audio track information is undefined even though the player still has audio streaming.
946
+ For this case, we select first audio track.
947
+ More detail please refer to OTP-3450.
948
+ */
949
+
950
+ const audioList = getAudioTracks({}, {
951
+ player
952
+ });
953
+ const currentAudio = track !== undefined ? track : audioList === null || audioList === void 0 ? void 0 : audioList[0];
954
+ return currentAudio && { ...currentAudio,
955
+ id: ((_currentAudio$value = currentAudio.value) === null || _currentAudio$value === void 0 ? void 0 : _currentAudio$value.id) || [currentAudio.language, currentAudio.label].join(':')
956
+ };
957
+ };
958
+
959
+ const getDefault = (section, {
960
+ preferred,
968
961
  player
969
- }, next) => {
970
- if (!next) return;
962
+ }) => {
963
+ if (typeof section.getDefault === 'function') {
964
+ return section.getDefault(preferred);
965
+ }
971
966
 
972
- try {
973
- const tracks = getAudioTracks({}, {
967
+ if (section.name === 'audio') {
968
+ return preferred !== null && preferred !== void 0 ? preferred : getSelectedAudioName(player);
969
+ }
970
+
971
+ if (section.name === 'subtitles') {
972
+ var _ref;
973
+
974
+ return ((_ref = section.items.find(track => {
975
+ var _track$value, _track$value2;
976
+
977
+ return ((_track$value = track.value) === null || _track$value === void 0 ? void 0 : _track$value.label) === (preferred === null || preferred === void 0 ? void 0 : preferred.label) && ((_track$value2 = track.value) === null || _track$value2 === void 0 ? void 0 : _track$value2.language) === (preferred === null || preferred === void 0 ? void 0 : preferred.language);
978
+ }) || section.items.find(track => {
979
+ var _track$value3;
980
+
981
+ return ((_track$value3 = track.value) === null || _track$value3 === void 0 ? void 0 : _track$value3.language) === (preferred === null || preferred === void 0 ? void 0 : preferred.language);
982
+ })) === null || _ref === void 0 ? void 0 : _ref.value) || {
983
+ language: 'off',
984
+ id: 'off'
985
+ };
986
+ }
987
+
988
+ if (section.name === 'speed') {
989
+ return preferred !== null && preferred !== void 0 ? preferred : 1;
990
+ }
991
+ };
992
+
993
+ const shouldProvideSettingSection = ({
994
+ name,
995
+ items
996
+ }, {
997
+ player,
998
+ loop
999
+ }) => {
1000
+ if (name === 'loop') {
1001
+ return !(player !== null && player !== void 0 && player.isLive()) && loop !== 'disabled';
1002
+ }
1003
+
1004
+ if (name === 'speed') {
1005
+ return !(player !== null && player !== void 0 && player.isLive());
1006
+ }
1007
+
1008
+ if (name === 'subtitles' || name === 'audio') {
1009
+ return true;
1010
+ }
1011
+
1012
+ return (items === null || items === void 0 ? void 0 : items.length) > 1;
1013
+ };
1014
+
1015
+ const getSettingsData = ({
1016
+ media,
1017
+ player,
1018
+ source = [],
1019
+ quality = {},
1020
+ speedItems = [0.25, 0.5, 0.75, 1, 1.25, 1.5, 1.75, 2],
1021
+ loop,
1022
+ preferred = {},
1023
+ otherSections = []
1024
+ }) => {
1025
+ // TODO extract base player specific things
1026
+ const audioItems = getAudioTracks({}, {
1027
+ player
1028
+ });
1029
+ const selectedSource = getSource(source, {
1030
+ preferManifestType: 'platform'
1031
+ }) || {};
1032
+ const currentSpeedItems = getSpeedItems(speedItems);
1033
+ const sections = [].concat(otherSections, [quality && getQualitySettings({ ...quality,
1034
+ items: selectedSource.qualityOptions
1035
+ }, player), {
1036
+ name: 'subtitles',
1037
+ title: 'KKS.SUBTITLES',
1038
+ items: [].concat(getTextTracks({}, {
974
1039
  player
1040
+ }), {
1041
+ label: 'OFF',
1042
+ value: {
1043
+ language: 'off',
1044
+ id: 'off'
1045
+ }
1046
+ })
1047
+ }, {
1048
+ name: 'audio',
1049
+ title: 'KKS.AUDIO',
1050
+ items: audioItems
1051
+ }, {
1052
+ name: 'speed',
1053
+ title: 'KKS.SETTING.SPEED',
1054
+ items: currentSpeedItems
1055
+ }, {
1056
+ type: 'switch',
1057
+ name: 'loop',
1058
+ title: 'KKS.SETTING.LOOP'
1059
+ }]).filter(section => shouldProvideSettingSection(section, {
1060
+ player,
1061
+ loop,
1062
+ quality
1063
+ }));
1064
+ const values = sections.reduce((result, section) => {
1065
+ // TODO take fallback option if preferred is not available
1066
+ // eslint-disable-next-line no-param-reassign
1067
+ result[section.name] = getDefault(section, {
1068
+ preferred: preferred[section.name],
1069
+ player,
1070
+ media
1071
+ }) || preferred[section.name];
1072
+ return result;
1073
+ }, // TODO forced subtitles?
1074
+ {
1075
+ audio: getSelectedAudioName(player),
1076
+ subtitles: 'off'
1077
+ });
1078
+
1079
+ if (!values.speed && (media === null || media === void 0 ? void 0 : media.playbackRate) > 0) {
1080
+ values.speed = media.playbackRate;
1081
+ }
1082
+
1083
+ if (typeof preferred.loop === 'boolean') {
1084
+ values.loop = preferred.loop;
1085
+ } else {
1086
+ values.loop = loop === 'disabled' ? false : Boolean(loop);
1087
+ dispatchCustomEvent(media, 'loopChange', {
1088
+ loop: values.loop
975
1089
  });
976
- const activeTrack = tracks.find(track => track.language === next.language && track.label === next.label) || tracks.find(track => track.language === next.language && track.label === next.language);
1090
+ }
977
1091
 
978
- if (activeTrack !== null && activeTrack !== void 0 && activeTrack.enabled) {
979
- return;
980
- }
1092
+ return {
1093
+ sections,
1094
+ values
1095
+ };
1096
+ };
981
1097
 
982
- player.selectAudioLanguage(activeTrack.language);
1098
+ const matchTracks = (tracks, sorted) => (sorted === null || sorted === void 0 ? void 0 : sorted.map(track => tracks.find(t => t.language === track.language)).filter(Boolean)) || tracks;
983
1099
 
984
- if (activeTrack.label) {
985
- player.selectVariantsByLabel(activeTrack.label);
1100
+ const getLanguageOptions = (player, sorted) => {
1101
+ const audioTracks = player ? getAudioTracks({}, {
1102
+ player
1103
+ }) : [];
1104
+ const textTracks = getTextTracks({}, {
1105
+ player
1106
+ });
1107
+ const textOptions = [].concat(textTracks, {
1108
+ type: 'subtitles',
1109
+ label: 'none',
1110
+ language: 'none',
1111
+ selected: !textTracks.some(track => track.selected),
1112
+ value: {
1113
+ label: 'none',
1114
+ language: 'none',
1115
+ id: 'off'
986
1116
  }
987
- } catch (error) {
988
- console.warn('Unable to set audio', error, next);
989
- }
1117
+ });
1118
+ return {
1119
+ audioTracks: matchTracks(audioTracks, sorted.audioTracks),
1120
+ textTracks: matchTracks(textOptions, sorted.textTracks)
1121
+ };
990
1122
  };
991
1123
 
992
1124
  const vendors = {
@@ -2183,6 +2315,7 @@ var ja = {
2183
2315
  "KKS.QUALITY": "解像度",
2184
2316
  "KKS.AUDIO": "音声",
2185
2317
  "KKS.SETTING.OFF": "オフ",
2318
+ "KKS.SETTING.none": "オフ",
2186
2319
  "KKS.SUBTITLES": "字幕",
2187
2320
  "KKS.SETTING": "設定",
2188
2321
  "KKS.SETTING.CONFLICT": "ご視聴する動画は、デフォルト画質設定に一致しません。",
@@ -2500,6 +2633,7 @@ var iconUrls = {
2500
2633
  volumeLow: 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0naHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmcnIHdpZHRoPSczNicgaGVpZ2h0PSczNicgdmlld0JveD0nMCAwIDM2IDM2Jz4gPGcgZmlsbD0nbm9uZScgZmlsbC1ydWxlPSdldmVub2RkJz4gPHBhdGggZD0nTTAgMGgzNnYzNkgweicgb3BhY2l0eT0nLjUnLz4gPHBhdGggZmlsbD0nI0ZGRicgZD0nTTI2LjUyMyAxNy41NDdjMCAxLjY2Mi0uNDIzIDMuMTY1LTEuMjcgNC41MWE4LjQ1MyA4LjQ1MyAwIDAgMS0zLjQ0NSAzLjEwNlY5LjkzYzEuNDUuNzI1IDIuNiAxLjc2IDMuNDQ1IDMuMTA1Ljg0NyAxLjM0NSAxLjI3IDIuODQ4IDEuMjcgNC41MXpNMSAxMS44OGg3LjUyNUwxOCAyLjQwNXYzMC4yODNsLTkuNDc1LTkuNDc1SDF6Jy8+IDwvZz4gPC9zdmc+',
2501
2634
  mute: 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0naHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmcnIHdpZHRoPSczNicgaGVpZ2h0PSczNicgdmlld0JveD0nMCAwIDM2IDM2Jz4gPGcgZmlsbD0nbm9uZScgZmlsbC1ydWxlPSdldmVub2RkJz4gPHBhdGggZD0nTTAgMGgzNnYzNkgweicgb3BhY2l0eT0nLjUnLz4gPHBhdGggZmlsbD0nI0ZGRicgZD0nTTEgMTEuODhoNy41MjVMMTggMi40MDV2MzAuMjgzbC05LjQ3NS05LjQ3NUgxeicvPiA8ZyBmaWxsLXJ1bGU9J25vbnplcm8nPiA8cGF0aCBkPSdNMjAgMTFoMTR2MTRIMjB6Jy8+IDxwYXRoIGZpbGw9JyNGRkYnIGQ9J00zMS45NCAxMC45NGwyLjEyIDIuMTItMTIgMTItMi4xMi0yLjEyeicvPiA8cGF0aCBmaWxsPScjRkZGJyBkPSdNMTkuOTQgMTMuMDZsMi4xMi0yLjEyIDEyIDEyLTIuMTIgMi4xMnonLz4gPC9nPiA8L2c+IDwvc3ZnPg==',
2502
2635
  setting: 'data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iNDgiIGhlaWdodD0iNDgiIHZpZXdCb3g9IjAgMCA0OCA0OCIgZmlsbD0ibm9uZSIgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIj4KPHBhdGggZD0iTTI0LjM5MjQgNDYuMDJDMjIuNzU5NSA0Ni4wMiAyMS4xMTY5IDQ1LjkzMiAxOS40OTM3IDQ1Ljc1NkMxOC45MDcxIDQ1LjY5NzMgMTguNDI3OSA0NS4yODY3IDE4LjI1MTkgNDQuNzI5M0wxNi42Nzc3IDM5LjY1NDZDMTYuMzM1NSAzOS40ODg0IDE2LjAwMyAzOS4zMTI0IDE1LjY4MDQgMzkuMTE2OEMxNS42NjA4IDM5LjEwNyAxNS42NDEyIDM5LjA5NzMgMTUuNjIxNyAzOS4wNzc3QzE1LjMxODYgMzguOTAxNyAxNS4wMTU1IDM4LjcxNTkgMTQuNzIyMSAzOC41MTA2TDkuNTMwMDcgMzkuNjgzOUM4Ljk2Mjk1IDM5LjgxMTEgOC4zNjY1IDM5LjU4NjIgOC4wMjQyNyAzOS4xMTY4QzYuMDk4MDMgMzYuNDc2OCA0LjQ0NTU3IDMzLjYyMTcgMy4xMjU1NiAzMC42Mjk2QzIuODkwODkgMzAuMDkxOCAyLjk4ODY3IDI5LjQ2NjEgMy4zODk1NiAyOS4wNDU2TDcuMDE3MTUgMjUuMTI0N0M2Ljk2ODI2IDI0LjM4MTYgNi45NjgyNiAyMy42Mzg0IDcuMDE3MTUgMjIuODk1M0wzLjM5OTM0IDE4Ljk3NDRDMi45OTg0NCAxOC41NDQyIDIuOTAwNjcgMTcuOTE4NCAzLjEzNTM0IDE3LjM4MDZDNC40NjUxMyAxNC4zODg2IDYuMTA3ODEgMTEuNTMzNCA4LjAzNDA1IDguOTAzMThDOC4zNzYyOCA4LjQzMzg0IDguOTcyNzMgOC4yMDg5NSA5LjUzOTg0IDguMzM2MDZMMTQuNzUxNSA5LjUwOTQxQzE1LjM2NzUgOS4wOTg3NCAxNi4wMTI4IDguNzI3MTggMTYuNjg3NSA4LjM5NDczTDE4LjI3MTUgMy4yOTA2OEMxOC40NDc1IDIuNzMzMzQgMTguOTM2NCAyLjMzMjQ1IDE5LjUxMzMgMi4yNjRDMjIuNzY5MyAxLjkxMiAyNi4wNjQ1IDEuOTEyIDI5LjMxMDcgMi4yNjRDMjkuODk3NCAyLjMyMjY3IDMwLjM3NjUgMi43MzMzNCAzMC41NTI1IDMuMjkwNjhMMzIuMTI2NyA4LjM2NTRDMzIuODAxNCA4LjY5Nzg1IDMzLjQ2NjMgOS4wNzkxOCAzNC4wOTIxIDkuNDk5NjNMMzkuMjc0NCA4LjMyNjI5QzM5Ljg1MTMgOC4xOTkxNyA0MC40Mzc5IDguNDI0MDYgNDAuNzgwMiA4Ljg5MzRDNDIuNzA2NCAxMS41MzM0IDQ0LjM1ODkgMTQuMzg4NiA0NS42Nzg5IDE3LjM4MDZDNDUuOTEzNSAxNy45MTg0IDQ1LjgxNTggMTguNTM0NCA0NS40MTQ5IDE4Ljk2NDZMNDEuNzg3MyAyMi44ODU1QzQxLjgzNjIgMjMuNjI4NyA0MS44MzYyIDI0LjM3MTggNDEuNzg3MyAyNS4xMTQ5TDQ1LjQwNTEgMjkuMDM1OEM0NS44MDYgMjkuNDY2MSA0NS45MDM4IDMwLjA5MTggNDUuNjY5MSAzMC42Mjk2QzQ1LjAwNDIgMzIuMTE1OSA0NC4yNjExIDMzLjU5MjMgNDMuNDQ5NSAzNS4wMDAzQzQyLjYzOCAzNi40MDgzIDQxLjczODQgMzcuNzk2OCA0MC43NzA0IDM5LjEwN0M0MC40MjgyIDM5LjU3NjQgMzkuODMxNyAzOS44MDEzIDM5LjI2NDYgMzkuNjc0MkwzNC4wNjI4IDM4LjUwMDhDMzMuNDQ2OCAzOC45MTE1IDMyLjgwMTQgMzkuMjgzIDMyLjEyNjcgMzkuNjE1NUwzMC41NDI3IDQ0LjcwOThDMzAuMzY2NyA0NS4yNjcxIDI5Ljg3NzggNDUuNjY4IDI5LjMwMDkgNDUuNzM2NEMyNy42Nzc4IDQ1LjkxMjQgMjYuMDM1MSA0Ni4wMDA0IDI0LjQwMjIgNDYuMDAwNEwyNC4zOTI0IDQ2LjAyWk0yMC43NjQ4IDQyLjkzMDJDMjMuMTggNDMuMTM1NSAyNS42MTQ3IDQzLjEzNTUgMjguMDI5OCA0Mi45MzAyTDI5LjUxNiAzOC4xNDg4QzI5LjY0MzIgMzcuNzQ3OSAyOS45MzY1IDM3LjQxNTUgMzAuMzE3OCAzNy4yNDkyQzMxLjIxNzQgMzYuODQ4NCAzMi4wODc2IDM2LjM0OTcgMzIuODg5NCAzNS43NjNDMzMuMjMxNiAzNS41MTg2IDMzLjY2MTkgMzUuNDMwNiAzNC4wNzI1IDM1LjUxODZMMzguOTUxNyAzNi42MjM1QzM5LjY0NTkgMzUuNjI2MSA0MC4zMDEgMzQuNTk5NCA0MC44OTc1IDMzLjU1MzJDNDEuNTAzNyAzMi41MDcgNDIuMDYxMSAzMS40MzE0IDQyLjU3OTMgMzAuMzM2M0wzOS4xODY0IDI2LjY1OThDMzguOTAyOCAyNi4zNDY5IDM4Ljc2NTkgMjUuOTM2MiAzOC44MDUgMjUuNTE1OEMzOC45MDI4IDI0LjUyODIgMzguOTAyOCAyMy41MjExIDM4LjgwNSAyMi41NDMzQzM4Ljc2NTkgMjIuMTIyOSAzOC45MDI4IDIxLjcwMjQgMzkuMTg2NCAyMS4zOTkzTDQyLjU4OTEgMTcuNzEzQzQxLjU2MjQgMTUuNTIyOCA0MC4zNDAyIDEzLjQxMDggMzguOTYxNSAxMS40MjU5TDM0LjEwMTkgMTIuNTIxQzMzLjY5MTIgMTIuNjE4OCAzMy4yNjEgMTIuNTIxIDMyLjkxODcgMTIuMjc2NUMzMi4xMDcyIDExLjY4OTkgMzEuMjI3MiAxMS4xODE0IDMwLjMxNzggMTAuNzcwOEMyOS45MzY1IDEwLjU5NDggMjkuNjQzMiAxMC4yNzIxIDI5LjUxNiA5Ljg2MTQxTDI4LjAzOTYgNS4wOTk1OUMyNS42MjQ1IDQuODk0MjUgMjMuMTg5OCA0Ljg5NDI1IDIwLjc3NDYgNS4wOTk1OUwxOS4yODg0IDkuODkwNzVDMTkuMTYxMyAxMC4yOTE2IDE4Ljg2NzkgMTAuNjI0MSAxOC40ODY2IDEwLjc5MDNDMTcuNTc3MyAxMS4xOTEyIDE2LjcxNjggMTEuNjg5OSAxNS45MTUgMTIuMjc2NUMxNS41NzI4IDEyLjUyMSAxNS4xNDI2IDEyLjYwOSAxNC43MzE5IDEyLjUyMUw5Ljg0Mjk2IDExLjQxNjFDOC40NjQyOCAxMy40MDEgNy4yNDIwNCAxNS41MTMgNi4yMTUzNyAxNy43MDMzTDkuNjA4MjkgMjEuMzc5OEM5Ljg5MTg1IDIxLjY5MjYgMTAuMDI4NyAyMi4xMDMzIDkuOTg5NjMgMjIuNTIzOEM5Ljg5MTg1IDIzLjUxMTMgOS44OTE4NSAyNC41MDg3IDkuOTg5NjMgMjUuNDk2MkMxMC4wMjg3IDI1LjkxNjcgOS44OTE4NSAyNi4zMzcxIDkuNjA4MjkgMjYuNjVMNi4yMDU1OSAzMC4zMzYzQzcuMjMyMjcgMzIuNTI2NSA4LjQ1NDUgMzQuNjM4NiA5LjgzMzE4IDM2LjYyMzVMMTQuNzAyNiAzNS41MjgzQzE1LjExMzIgMzUuNDMwNiAxNS41NDM1IDM1LjUyODMgMTUuODg1NyAzNS43NzI4QzE2LjI5NjQgMzYuMDY2MSAxNi43MjY2IDM2LjM0OTcgMTcuMTU2OCAzNi41OTQxQzE3LjE3NjQgMzYuNjAzOSAxNy4xOTU5IDM2LjYxMzcgMTcuMjA1NyAzNi42MjM1QzE3LjYyNjEgMzYuODU4MSAxOC4wNTY0IDM3LjA4MyAxOC40OTY0IDM3LjI3ODZDMTguODc3NyAzNy40NTQ2IDE5LjE3MTEgMzcuNzc3MyAxOS4yOTgyIDM4LjE3ODFMMjAuNzc0NiA0Mi45NEwyMC43NjQ4IDQyLjkzMDJaTTI0LjM4MjcgMzIuMzMxQzIyLjk0NTMgMzIuMzMxIDIxLjUxNzcgMzEuOTQ5NiAyMC4yMzY4IDMxLjIxNjNDMTYuMjg2NiAyOC45MzgxIDE0LjkxNzcgMjMuODkyNyAxNy4xNTY4IDE5LjkzMjZDMTcuMTY2NiAxOS45MTMxIDE3LjE3NjQgMTkuODgzNyAxNy4xOTU5IDE5Ljg2NDJDMTguMzAwOCAxNy45Mzc5IDIwLjEgMTYuNTY5IDIyLjI0MTMgMTUuOTkyMUMyNC4zODI3IDE1LjQxNTIgMjYuNjIxOCAxNS43MDg2IDI4LjU0OCAxNi44MjMzQzMwLjQ3NDMgMTcuOTM3OSAzMS44NDMyIDE5LjcyNzMgMzIuNDIwMSAyMS44Njg2QzMyLjk5NyAyNC4wMSAzMi43MDM2IDI2LjI0OTEgMzEuNTg5IDI4LjE3NTRDMzAuNDg0MSAzMC4xMDE2IDI4LjY4NDkgMzEuNDcwNSAyNi41NDM2IDMyLjA0NzRDMjUuODI5OCAzMi4yNDMgMjUuMTA2MiAzMi4zMzEgMjQuMzgyNyAzMi4zMzFaTTE5Ljc2NzUgMjEuMjcyMkMxOS43Njc1IDIxLjI3MjIgMTkuNzQ3OSAyMS4zMTEzIDE5LjczODIgMjEuMzMwOUMxOC4yNTE5IDIzLjkwMjQgMTkuMTQxNyAyNy4xOTc2IDIxLjcwMzUgMjguNjc0QzIyLjk0NTMgMjkuMzg3OCAyNC4zOTI0IDI5LjU4MzQgMjUuNzgwOSAyOS4yMTE4QzI3LjE2OTQgMjguODQwMyAyOC4zMzI5IDI3Ljk1MDUgMjkuMDQ2NyAyNi43MDg3QzI5Ljc2MDUgMjUuNDY2OSAyOS45NTYxIDI0LjAxOTggMjkuNTg0NSAyMi42MzEzQzI5LjIxMjkgMjEuMjQyOSAyOC4zMjMxIDIwLjA4OTEgMjcuMDgxNCAxOS4zNjU1QzI1LjgzOTYgMTguNjQxOSAyNC4zODI3IDE4LjQ1NjIgMjMuMDA0IDE4LjgyNzdDMjEuNjM1MSAxOS4xODk1IDIwLjQ5MTEgMjAuMDU5NyAxOS43Njc1IDIxLjI4MlYyMS4yNzIyWiIgZmlsbD0id2hpdGUiLz4KPC9zdmc+Cg==',
2636
+ subtitle: 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHdpZHRoPSIzNiIgaGVpZ2h0PSIzNiIgZmlsbD0ibm9uZSI+CiAgPHBhdGggZmlsbD0iI2ZmZiIgZmlsbC1vcGFjaXR5PSIuOCIgZD0iTTExLjMgMzIuOWMuOCAwIDEuMzQtLjQxIDIuMzItMS4zbDQuODYtNC4zNUgyN2M0LjQ3IDAgNi44LTIuNDEgNi44LTYuOFY5LjljMC00LjQtMi4zMy02LjgtNi44LTYuOEg5Yy00LjQ2IDAtNi44IDIuNC02LjggNi44djEwLjU1YzAgNC4zOSAyLjM0IDYuOCA2LjggNi44aC41MnYzLjZjMCAxLjI2LjY0IDIuMDYgMS43OCAyLjA2bS42Ny0yLjk3di00LjA4YzAtLjg4LS4zMi0xLjIxLTEuMjEtMS4yMUg5Yy0yLjkyIDAtNC4xOS0xLjQtNC4xOS00LjE5VjkuOWMwLTIuNzggMS4yNy00LjE3IDQuMTktNC4xN2gxOGMyLjkyIDAgNC4yIDEuNCA0LjIgNC4xN3YxMC41NWMwIDIuOC0xLjI4IDQuMTktNC4yIDQuMTloLTguNjZjLS44NyAwLTEuMzMuMTMtMS45Ljc0eiIvPgogIDxwYXRoIGZpbGw9IiNmZmYiIGZpbGwtb3BhY2l0eT0iLjgiIGZpbGwtcnVsZT0iZXZlbm9kZCIgZD0iTTEzLjc1IDMxLjc1Yy0uOTcuODgtMS41NyAxLjM1LTIuNDUgMS4zNXEtLjk0IDAtMS40Ny0uNjItLjUtLjYyLS41LTEuNjJ2LTMuNDFIOXEtMy40MS0uMDEtNS4yMi0xLjgyUTIgMjMuODEgMiAyMC40NVY5LjljMC0yLjIzLjYtMy45OSAxLjc4LTUuMTlRNS41OCAyLjkxIDkgMi45aDE4cTMuNDEgMCA1LjIyIDEuODFDMzMuNCA1LjkxIDM0IDcuNjcgMzQgOS45djEwLjU1YzAgMi4yMy0uNiAzLjk5LTEuNzggNS4xOHEtMS44IDEuODEtNS4yMiAxLjgyaC04LjQ0em0tLjEzLS4xNWMtLjk4Ljg5LTEuNTIgMS4zLTIuMzIgMS4zLTEuMTQgMC0xLjc4LS43OS0xLjc4LTIuMDR2LTMuNjFIOWMtNC40NiAwLTYuOC0yLjQxLTYuOC02LjhWOS45YzAtNC40IDIuMzQtNi44IDYuOC02LjhoMThjNC40NiAwIDYuOCAyLjQgNi44IDYuOHYxMC41NWMwIDQuMzktMi4zNCA2LjgtNi44IDYuOGgtOC41MnptLTEuNjUtNS43NWMwLS44OC0uMzMtMS4yMS0xLjIxLTEuMjFIOWMtMi45MiAwLTQuMi0xLjQtNC4yLTQuMTlWOS45YzAtMi43OCAxLjI4LTQuMTcgNC4yLTQuMTdoMThjMi45MiAwIDQuMiAxLjQgNC4yIDQuMTd2MTAuNTVjMCAyLjgtMS4yOCA0LjE5LTQuMiA0LjE5aC04LjY2Yy0uODcgMC0xLjMzLjEzLTEuOS43NGwtNC40NyA0LjU1em0uMiAzLjZ2LTMuNmMwLS40Ni0uMDgtLjgzLS4zMy0xLjA4cy0uNjItLjMzLTEuMDgtLjMzSDljLTEuNDMgMC0yLjQyLS4zNC0zLjA1LS45OXEtLjk0LS45NS0uOTQtM1Y5LjlxLS4wMS0yLjAzLjk0LTMgLjkyLS45NyAzLjA1LS45OWgxOHEyLjEzLjAyIDMuMDUuOTkuOTQuOTYuOTQgMi45OXYxMC41NXEuMDEgMi4wNS0uOTQgMy0uOTMuOTktMy4wNSAxaC04LjY2cS0uNjYtLjAyLTEuMTMuMTQtLjQ5LjE4LS45Mi42NnoiIGNsaXAtcnVsZT0iZXZlbm9kZCIvPgogIDxwYXRoIGZpbGw9IiNmZmYiIGZpbGwtb3BhY2l0eT0iLjgiIGQ9Ik05LjI5IDE2LjgyaDUuNjVhLjkuOSAwIDAgMCAuOTEtLjkyLjkuOSAwIDAgMC0uOS0uOUg5LjI4YS45LjkgMCAwIDAtLjkxLjljMCAuNTIuNC45Mi45LjkybTguNDggMGg4LjkyYS45LjkgMCAwIDAgLjkzLS45MmMwLS41LS40LS45LS45My0uOWgtOC45MmEuOS45IDAgMCAwLS45LjljMCAuNTIuNC45Mi45LjkyTTkuMyAyMC43aDIuMzdjLjUgMCAuOS0uNDIuOS0uOTNhLjkuOSAwIDAgMC0uOS0uOTJIOS4zYS45LjkgMCAwIDAtLjkxLjkyYzAgLjUuNC45My45LjkzbTUuMiAwaDYuOThjLjUgMCAuOS0uNDIuOS0uOTNhLjkuOSAwIDAgMC0uOS0uOTJIMTQuNWMtLjUgMC0uOS40LS45LjkyIDAgLjUuNC45My45LjkzbTkuOCAwaDIuNGMuNTMgMCAuOTMtLjQyLjkzLS45M2EuOS45IDAgMCAwLS45My0uOTJoLTIuNGEuOS45IDAgMCAwLS45Mi45MmMwIC41LjQuOTMuOTEuOTMiLz4KPC9zdmc+',
2503
2637
  check: 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0naHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmcnIHdpZHRoPScyMCcgaGVpZ2h0PScyMCcgdmlld0JveD0nMCAwIDIwIDIwJz48cGF0aCBmaWxsPScjRkZGJyBmaWxsLXJ1bGU9J25vbnplcm8nIGQ9J00yLjkwMiA5LjI2MWExLjA4NyAxLjA4NyAwIDAgMC0xLjU3NiAwIDEuMTgzIDEuMTgzIDAgMCAwIDAgMS42MzJsNS41NzEgNS43NjljLjQzNS40NSAxLjE0LjQ1IDEuNTc2IDBsMTEuMi0xMS42OTJhMS4xODMgMS4xODMgMCAwIDAgMC0xLjYzMiAxLjA4NyAxLjA4NyAwIDAgMC0xLjU3NSAwTDcuNjg1IDE0LjIxNCAyLjkwMiA5LjI2MXonLz48L3N2Zz4K',
2504
2638
  fullscreen: 'data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iNDgiIGhlaWdodD0iNDgiIHZpZXdCb3g9IjAgMCA0OCA0OCIgZmlsbD0ibm9uZSIgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIj4KPGcgY2xpcC1wYXRoPSJ1cmwoI2NsaXAwXzExNTlfMTE4KSI+CjxwYXRoIGQ9Ik0xMC4wMDAyIDYuMDAwMkM3LjgxNDA3IDYuMDAwMiA2LjAwMDIgNy44MTQwNyA2LjAwMDIgMTAuMDAwMlYxNi4wMDAyQzUuOTk2NDUgMTYuMjY1MiA2LjA0NTQxIDE2LjUyODMgNi4xNDQyNCAxNi43NzQzQzYuMjQzMDcgMTcuMDIwMiA2LjM4OTc5IDE3LjI0NDEgNi41NzU4OCAxNy40MzI4QzYuNzYxOTcgMTcuNjIxNSA2Ljk4MzcxIDE3Ljc3MTQgNy4yMjgyMyAxNy44NzM3QzcuNDcyNzQgMTcuOTc2IDcuNzM1MTUgMTguMDI4NyA4LjAwMDIgMTguMDI4N0M4LjI2NTI1IDE4LjAyODcgOC41Mjc2NiAxNy45NzYgOC43NzIxNyAxNy44NzM3QzkuMDE2NjkgMTcuNzcxNCA5LjIzODQzIDE3LjYyMTUgOS40MjQ1MiAxNy40MzI4QzkuNjEwNjEgMTcuMjQ0MSA5Ljc1NzMzIDE3LjAyMDIgOS44NTYxNiAxNi43NzQzQzkuOTU0OTkgMTYuNTI4MyAxMC4wMDM5IDE2LjI2NTIgMTAuMDAwMiAxNi4wMDAyVjEwLjAwMDJIMTYuMDAwMkMxNi4yNjUyIDEwLjAwMzkgMTYuNTI4MyA5Ljk1NDk5IDE2Ljc3NDMgOS44NTYxNkMxNy4wMjAyIDkuNzU3MzMgMTcuMjQ0MSA5LjYxMDYxIDE3LjQzMjggOS40MjQ1MkMxNy42MjE1IDkuMjM4NDMgMTcuNzcxNCA5LjAxNjY5IDE3Ljg3MzcgOC43NzIxN0MxNy45NzYgOC41Mjc2NiAxOC4wMjg3IDguMjY1MjUgMTguMDI4NyA4LjAwMDJDMTguMDI4NyA3LjczNTE1IDE3Ljk3NiA3LjQ3Mjc0IDE3Ljg3MzcgNy4yMjgyM0MxNy43NzE0IDYuOTgzNzEgMTcuNjIxNSA2Ljc2MTk3IDE3LjQzMjggNi41NzU4OEMxNy4yNDQxIDYuMzg5NzkgMTcuMDIwMiA2LjI0MzA3IDE2Ljc3NDMgNi4xNDQyNEMxNi41MjgzIDYuMDQ1NDEgMTYuMjY1MiA1Ljk5NjQ1IDE2LjAwMDIgNi4wMDAySDEwLjAwMDJaTTMyLjAwMDIgNi4wMDAyQzMxLjczNTIgNS45OTY0NSAzMS40NzIxIDYuMDQ1NDEgMzEuMjI2MSA2LjE0NDI0QzMwLjk4MDIgNi4yNDMwNyAzMC43NTYzIDYuMzg5NzkgMzAuNTY3NiA2LjU3NTg4QzMwLjM3ODkgNi43NjE5NyAzMC4yMjkgNi45ODM3MSAzMC4xMjY3IDcuMjI4MjNDMzAuMDI0NCA3LjQ3Mjc0IDI5Ljk3MTcgNy43MzUxNSAyOS45NzE3IDguMDAwMkMyOS45NzE3IDguMjY1MjUgMzAuMDI0NCA4LjUyNzY2IDMwLjEyNjcgOC43NzIxN0MzMC4yMjkgOS4wMTY2OSAzMC4zNzg5IDkuMjM4NDMgMzAuNTY3NiA5LjQyNDUyQzMwLjc1NjMgOS42MTA2MSAzMC45ODAyIDkuNzU3MzMgMzEuMjI2MSA5Ljg1NjE2QzMxLjQ3MjEgOS45NTQ5OSAzMS43MzUyIDEwLjAwMzkgMzIuMDAwMiAxMC4wMDAySDM4LjAwMDJWMTYuMDAwMkMzNy45OTY1IDE2LjI2NTIgMzguMDQ1NCAxNi41MjgzIDM4LjE0NDIgMTYuNzc0M0MzOC4yNDMxIDE3LjAyMDIgMzguMzg5OCAxNy4yNDQxIDM4LjU3NTkgMTcuNDMyOEMzOC43NjIgMTcuNjIxNSAzOC45ODM3IDE3Ljc3MTQgMzkuMjI4MiAxNy44NzM3QzM5LjQ3MjcgMTcuOTc2IDM5LjczNTEgMTguMDI4NyA0MC4wMDAyIDE4LjAyODdDNDAuMjY1MiAxOC4wMjg3IDQwLjUyNzcgMTcuOTc2IDQwLjc3MjIgMTcuODczN0M0MS4wMTY3IDE3Ljc3MTQgNDEuMjM4NCAxNy42MjE1IDQxLjQyNDUgMTcuNDMyOEM0MS42MTA2IDE3LjI0NDEgNDEuNzU3MyAxNy4wMjAyIDQxLjg1NjIgMTYuNzc0M0M0MS45NTUgMTYuNTI4MyA0Mi4wMDM5IDE2LjI2NTIgNDIuMDAwMiAxNi4wMDAyVjEwLjAwMDJDNDIuMDAwMiA3LjgxNDA3IDQwLjE4NjMgNi4wMDAyIDM4LjAwMDIgNi4wMDAySDMyLjAwMDJaTTcuOTY4OTUgMjkuOTcyOUM3LjQzOTIgMjkuOTgxMSA2LjkzNDM4IDMwLjE5OTMgNi41NjUyNyAzMC41NzkzQzYuMTk2MTcgMzAuOTU5NCA1Ljk5Mjk1IDMxLjQ3MDQgNi4wMDAyIDMyLjAwMDJWMzguMDAwMkM2LjAwMDIgNDAuMTg2MyA3LjgxNDA3IDQyLjAwMDIgMTAuMDAwMiA0Mi4wMDAySDE2LjAwMDJDMTYuMjY1MiA0Mi4wMDM5IDE2LjUyODMgNDEuOTU1IDE2Ljc3NDMgNDEuODU2MkMxNy4wMjAyIDQxLjc1NzMgMTcuMjQ0MSA0MS42MTA2IDE3LjQzMjggNDEuNDI0NUMxNy42MjE1IDQxLjIzODQgMTcuNzcxNCA0MS4wMTY3IDE3Ljg3MzcgNDAuNzcyMkMxNy45NzYgNDAuNTI3NyAxOC4wMjg3IDQwLjI2NTIgMTguMDI4NyA0MC4wMDAyQzE4LjAyODcgMzkuNzM1MSAxNy45NzYgMzkuNDcyNyAxNy44NzM3IDM5LjIyODJDMTcuNzcxNCAzOC45ODM3IDE3LjYyMTUgMzguNzYyIDE3LjQzMjggMzguNTc1OUMxNy4yNDQxIDM4LjM4OTggMTcuMDIwMiAzOC4yNDMxIDE2Ljc3NDMgMzguMTQ0MkMxNi41MjgzIDM4LjA0NTQgMTYuMjY1MiAzNy45OTY1IDE2LjAwMDIgMzguMDAwMkgxMC4wMDAyVjMyLjAwMDJDMTAuMDAzOSAzMS43MzI2IDkuOTUzOCAzMS40NjcgOS44NTI5NyAzMS4yMTkxQzkuNzUyMTMgMzAuOTcxMyA5LjYwMjU5IDMwLjc0NjEgOS40MTMxOCAzMC41NTcxQzkuMjIzNzcgMzAuMzY4MSA4Ljk5ODM2IDMwLjIxODkgOC43NTAyOCAzMC4xMTg2QzguNTAyMjEgMzAuMDE4MiA4LjIzNjUyIDI5Ljk2ODcgNy45Njg5NSAyOS45NzI5Wk0zOS45Njg5IDI5Ljk3MjlDMzkuNDM5MiAyOS45ODExIDM4LjkzNDQgMzAuMTk5MyAzOC41NjUzIDMwLjU3OTNDMzguMTk2MiAzMC45NTk0IDM3Ljk5MjkgMzEuNDcwNCAzOC4wMDAyIDMyLjAwMDJWMzguMDAwMkgzMi4wMDAyQzMxLjczNTIgMzcuOTk2NSAzMS40NzIxIDM4LjA0NTQgMzEuMjI2MSAzOC4xNDQyQzMwLjk4MDIgMzguMjQzMSAzMC43NTYzIDM4LjM4OTggMzAuNTY3NiAzOC41NzU5QzMwLjM3ODkgMzguNzYyIDMwLjIyOSAzOC45ODM3IDMwLjEyNjcgMzkuMjI4MkMzMC4wMjQ0IDM5LjQ3MjcgMjkuOTcxNyAzOS43MzUxIDI5Ljk3MTcgNDAuMDAwMkMyOS45NzE3IDQwLjI2NTIgMzAuMDI0NCA0MC41Mjc3IDMwLjEyNjcgNDAuNzcyMkMzMC4yMjkgNDEuMDE2NyAzMC4zNzg5IDQxLjIzODQgMzAuNTY3NiA0MS40MjQ1QzMwLjc1NjMgNDEuNjEwNiAzMC45ODAyIDQxLjc1NzMgMzEuMjI2MSA0MS44NTYyQzMxLjQ3MjEgNDEuOTU1IDMxLjczNTIgNDIuMDAzOSAzMi4wMDAyIDQyLjAwMDJIMzguMDAwMkM0MC4xODYzIDQyLjAwMDIgNDIuMDAwMiA0MC4xODYzIDQyLjAwMDIgMzguMDAwMlYzMi4wMDAyQzQyLjAwMzkgMzEuNzMyNiA0MS45NTM4IDMxLjQ2NyA0MS44NTMgMzEuMjE5MUM0MS43NTIxIDMwLjk3MTMgNDEuNjAyNiAzMC43NDYxIDQxLjQxMzIgMzAuNTU3MUM0MS4yMjM4IDMwLjM2ODEgNDAuOTk4NCAzMC4yMTg5IDQwLjc1MDMgMzAuMTE4NkM0MC41MDIyIDMwLjAxODIgNDAuMjM2NSAyOS45Njg3IDM5Ljk2ODkgMjkuOTcyOVoiIGZpbGw9IndoaXRlIi8+CjwvZz4KPGRlZnM+CjxjbGlwUGF0aCBpZD0iY2xpcDBfMTE1OV8xMTgiPgo8cmVjdCB3aWR0aD0iNDgiIGhlaWdodD0iNDgiIGZpbGw9IndoaXRlIi8+CjwvY2xpcFBhdGg+CjwvZGVmcz4KPC9zdmc+Cg==',
2505
2639
  'exit-fullscreen': 'data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iNDgiIGhlaWdodD0iNDgiIHZpZXdCb3g9IjAgMCA0OCA0OCIgZmlsbD0ibm9uZSIgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIj4KPGcgY2xpcC1wYXRoPSJ1cmwoI2NsaXAwXzExNTlfMTMyKSI+CjxwYXRoIGQ9Ik0xNi45NzA3IDUuOTcyNjZDMTYuNDQwNiA1Ljk4MDQyIDE1LjkzNTMgNi4xOTgzMiAxNS41NjU3IDYuNTc4NDZDMTUuMTk2MiA2Ljk1ODYxIDE0Ljk5MjcgNy40Njk5IDE1IDhMMTQuOTk4IDEzLjVDMTQuOTk4IDE0LjM0NTQgMTQuMzQzNCAxNSAxMy40OTggMTVINy45OTgwM0M3LjczMyAxNC45OTYzIDcuNDY5ODggMTUuMDQ1MiA3LjIyMzk0IDE1LjE0NEM2Ljk3ODAxIDE1LjI0MjkgNi43NTQxNiAxNS4zODk2IDYuNTY1NDIgMTUuNTc1N0M2LjM3NjY4IDE1Ljc2MTggNi4yMjY4MSAxNS45ODM1IDYuMTI0NTIgMTYuMjI4QzYuMDIyMjIgMTYuNDcyNSA1Ljk2OTU0IDE2LjczNDkgNS45Njk1NCAxN0M1Ljk2OTU0IDE3LjI2NSA2LjAyMjIyIDE3LjUyNzUgNi4xMjQ1MiAxNy43NzJDNi4yMjY4MSAxOC4wMTY1IDYuMzc2NjggMTguMjM4MiA2LjU2NTQyIDE4LjQyNDNDNi43NTQxNiAxOC42MTA0IDYuOTc4MDEgMTguNzU3MSA3LjIyMzk0IDE4Ljg1NkM3LjQ2OTg4IDE4Ljk1NDggNy43MzMgMTkuMDAzNyA3Ljk5ODAzIDE5SDEzLjQ5OEMxNi41MTI3IDE5IDE4Ljk5OCAxNi41MTQ2IDE4Ljk5OCAxMy41TDE5IDhDMTkuMDAzNiA3LjczMjU5IDE4Ljk1MzYgNy40NjcxNiAxOC44NTI5IDcuMjE5NDFDMTguNzUyMiA2Ljk3MTY2IDE4LjYwMjggNi43NDY2MiAxOC40MTM2IDYuNTU3NjFDMTguMjI0NCA2LjM2ODU5IDE3Ljk5OTMgNi4yMTk0NCAxNy43NTE0IDYuMTE4OTZDMTcuNTAzNiA2LjAxODQ5IDE3LjIzODEgNS45Njg3NCAxNi45NzA3IDUuOTcyNjZaTTMwLjk3MDcgNS45NzI2NkMzMC40NDA5IDUuOTgwOTQgMjkuOTM2MSA2LjE5OTA2IDI5LjU2NyA2LjU3OTE0QzI5LjE5NzkgNi45NTkyMyAyOC45OTQ3IDcuNDcwMjMgMjkuMDAxOSA4TDI5IDEzLjVDMjkgMTYuNTE0NiAzMS40ODU0IDE5IDM0LjUgMTlINDBDNDAuMjY1IDE5LjAwMzcgNDAuNTI4MSAxOC45NTQ4IDQwLjc3NDEgMTguODU2QzQxLjAyIDE4Ljc1NzEgNDEuMjQzOSAxOC42MTA0IDQxLjQzMjYgMTguNDI0M0M0MS42MjEzIDE4LjIzODIgNDEuNzcxMiAxOC4wMTY1IDQxLjg3MzUgMTcuNzcyQzQxLjk3NTggMTcuNTI3NSA0Mi4wMjg1IDE3LjI2NSA0Mi4wMjg1IDE3QzQyLjAyODUgMTYuNzM0OSA0MS45NzU4IDE2LjQ3MjUgNDEuODczNSAxNi4yMjhDNDEuNzcxMiAxNS45ODM1IDQxLjYyMTMgMTUuNzYxOCA0MS40MzI2IDE1LjU3NTdDNDEuMjQzOSAxNS4zODk2IDQxLjAyIDE1LjI0MjkgNDAuNzc0MSAxNS4xNDRDNDAuNTI4MSAxNS4wNDUyIDQwLjI2NSAxNC45OTYzIDQwIDE1SDM0LjVDMzMuNjU0NiAxNSAzMyAxNC4zNDU0IDMzIDEzLjVMMzMuMDAxOSA4QzMzLjAwNTYgNy43MzI0MiAzMi45NTU1IDcuNDY2ODMgMzIuODU0NyA3LjIxODk1QzMyLjc1MzkgNi45NzEwNyAzMi42MDQzIDYuNzQ1OTQgMzIuNDE0OSA2LjU1NjlDMzIuMjI1NSA2LjM2Nzg1IDMyLjAwMDEgNi4yMTg3NCAzMS43NTIgNi4xMTgzOUMzMS41MDM5IDYuMDE4MDMgMzEuMjM4MyA1Ljk2ODQ4IDMwLjk3MDcgNS45NzI2NlpNNy45OTk5OCAyOUM3LjczNDk2IDI4Ljk5NjIgNy40NzE4MyAyOS4wNDUyIDcuMjI1OSAyOS4xNDRDNi45Nzk5NiAyOS4yNDI5IDYuNzU2MTIgMjkuMzg5NiA2LjU2NzM4IDI5LjU3NTdDNi4zNzg2NCAyOS43NjE4IDYuMjI4NzcgMjkuOTgzNSA2LjEyNjQ3IDMwLjIyOEM2LjAyNDE3IDMwLjQ3MjUgNS45NzE1IDMwLjczNDkgNS45NzE1IDMxQzUuOTcxNSAzMS4yNjUgNi4wMjQxNyAzMS41Mjc1IDYuMTI2NDcgMzEuNzcyQzYuMjI4NzcgMzIuMDE2NSA2LjM3ODY0IDMyLjIzODIgNi41NjczOCAzMi40MjQzQzYuNzU2MTIgMzIuNjEwNCA2Ljk3OTk2IDMyLjc1NzEgNy4yMjU5IDMyLjg1NkM3LjQ3MTgzIDMyLjk1NDggNy43MzQ5NiAzMy4wMDM3IDcuOTk5OTggMzNIMTMuNUMxNC4zNDU0IDMzIDE1IDMzLjY1NDYgMTUgMzQuNVY0MEMxNC45OTYyIDQwLjI2NSAxNS4wNDUyIDQwLjUyODEgMTUuMTQ0IDQwLjc3NDFDMTUuMjQyOSA0MS4wMiAxNS4zODk2IDQxLjI0MzkgMTUuNTc1NyA0MS40MzI2QzE1Ljc2MTggNDEuNjIxMyAxNS45ODM1IDQxLjc3MTIgMTYuMjI4IDQxLjg3MzVDMTYuNDcyNSA0MS45NzU4IDE2LjczNDkgNDIuMDI4NSAxNyA0Mi4wMjg1QzE3LjI2NSA0Mi4wMjg1IDE3LjUyNzQgNDEuOTc1OCAxNy43NzIgNDEuODczNUMxOC4wMTY1IDQxLjc3MTIgMTguMjM4MiA0MS42MjEzIDE4LjQyNDMgNDEuNDMyNkMxOC42MTA0IDQxLjI0MzkgMTguNzU3MSA0MS4wMiAxOC44NTU5IDQwLjc3NDFDMTguOTU0OCA0MC41MjgxIDE5LjAwMzcgNDAuMjY1IDE5IDQwVjM0LjVDMTkgMzEuNDg1NCAxNi41MTQ2IDI5IDEzLjUgMjlINy45OTk5OFpNMzQuNSAyOUMzMS40ODU0IDI5IDI5IDMxLjQ4NTQgMjkgMzQuNVY0MEMyOC45OTYyIDQwLjI2NSAyOS4wNDUyIDQwLjUyODEgMjkuMTQ0IDQwLjc3NDFDMjkuMjQyOSA0MS4wMiAyOS4zODk2IDQxLjI0MzkgMjkuNTc1NyA0MS40MzI2QzI5Ljc2MTggNDEuNjIxMyAyOS45ODM1IDQxLjc3MTIgMzAuMjI4IDQxLjg3MzVDMzAuNDcyNSA0MS45NzU4IDMwLjczNDkgNDIuMDI4NSAzMSA0Mi4wMjg1QzMxLjI2NSA0Mi4wMjg1IDMxLjUyNzQgNDEuOTc1OCAzMS43NzIgNDEuODczNUMzMi4wMTY1IDQxLjc3MTIgMzIuMjM4MiA0MS42MjEzIDMyLjQyNDMgNDEuNDMyNkMzMi42MTA0IDQxLjI0MzkgMzIuNzU3MSA0MS4wMiAzMi44NTU5IDQwLjc3NDFDMzIuOTU0OCA0MC41MjgxIDMzLjAwMzcgNDAuMjY1IDMzIDQwVjM0LjVDMzMgMzMuNjU0NiAzMy42NTQ2IDMzIDM0LjUgMzNINDBDNDAuMjY1IDMzLjAwMzcgNDAuNTI4MSAzMi45NTQ4IDQwLjc3NDEgMzIuODU2QzQxLjAyIDMyLjc1NzEgNDEuMjQzOSAzMi42MTA0IDQxLjQzMjYgMzIuNDI0M0M0MS42MjEzIDMyLjIzODIgNDEuNzcxMiAzMi4wMTY1IDQxLjg3MzUgMzEuNzcyQzQxLjk3NTggMzEuNTI3NSA0Mi4wMjg1IDMxLjI2NSA0Mi4wMjg1IDMxQzQyLjAyODUgMzAuNzM0OSA0MS45NzU4IDMwLjQ3MjUgNDEuODczNSAzMC4yMjhDNDEuNzcxMiAyOS45ODM1IDQxLjYyMTMgMjkuNzYxOCA0MS40MzI2IDI5LjU3NTdDNDEuMjQzOSAyOS4zODk2IDQxLjAyIDI5LjI0MjkgNDAuNzc0MSAyOS4xNDRDNDAuNTI4MSAyOS4wNDUyIDQwLjI2NSAyOC45OTYyIDQwIDI5SDM0LjVaIiBmaWxsPSJ3aGl0ZSIvPgo8L2c+CjxkZWZzPgo8Y2xpcFBhdGggaWQ9ImNsaXAwXzExNTlfMTMyIj4KPHJlY3Qgd2lkdGg9IjQ4IiBoZWlnaHQ9IjQ4IiBmaWxsPSJ3aGl0ZSIvPgo8L2NsaXBQYXRoPgo8L2RlZnM+Cjwvc3ZnPgo=',
@@ -3012,7 +3146,7 @@ const invisible = hiddenArea => ({
3012
3146
  visibility: `var(${hiddenArea}, visible)`
3013
3147
  });
3014
3148
 
3015
- const containerStyle$3 = {
3149
+ const containerStyle$5 = {
3016
3150
  '--spacing': '0.5em',
3017
3151
  '--center-pointer-events': 'auto',
3018
3152
  width: '100%',
@@ -3298,7 +3432,7 @@ var _ref$1 = process.env.NODE_ENV === "production" ? {
3298
3432
  } : {
3299
3433
  name: "9zffum-DefaultLayout",
3300
3434
  styles: "&:not(:empty){flex:1;margin-left:calc(var(--spacing, 0.75em) - 0.5em + 0.5rem);};label:DefaultLayout;",
3301
- 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"]} */",
3435
+ 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"]} */",
3302
3436
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__$2
3303
3437
  };
3304
3438
 
@@ -3335,20 +3469,20 @@ const DefaultLayout = ({
3335
3469
  '--player-bottom-ui-top': controlsDisplay === 'shown' ? `-${bottomUiTop(containerRef.current)}px` : '-8px',
3336
3470
  '--player-bottom-ui-height': type === 'desktop' && controlsDisplay === 'shown' ? '3.5rem' : '0'
3337
3471
  },
3338
- css: [containerStyle$3, videoContainerStyle, responsiveStyles[size], type === 'desktop' && desktopStyle$2, adStatus && {
3472
+ css: [containerStyle$5, videoContainerStyle, responsiveStyles[size], type === 'desktop' && desktopStyle$2, adStatus && {
3339
3473
  '--bottom-spacing': 0
3340
- }, 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"]} */"],
3474
+ }, 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"]} */"],
3341
3475
  ref: containerRef,
3342
3476
  ...rest,
3343
3477
  children: [video, jsxs("div", {
3344
3478
  ref: backRef,
3345
- 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"]} */"],
3479
+ 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"]} */"],
3346
3480
  children: [type !== 'mobile' && backItems, adSkipButton && jsx$1("div", {
3347
3481
  css: skipStyle,
3348
3482
  children: adSkipButton
3349
3483
  })]
3350
3484
  }), jsxs("div", {
3351
- 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"]} */"],
3485
+ 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"]} */"],
3352
3486
  children: [backButton, jsx$1("div", {
3353
3487
  ref: element => {
3354
3488
  slotRef.current.titleBarLeft = element;
@@ -3373,9 +3507,9 @@ const DefaultLayout = ({
3373
3507
  // see https://kkvideo.atlassian.net/browse/CPT-4535 for detail
3374
3508
  adStatus && {
3375
3509
  margin: 'calc(var(--spacing) * 6) 0'
3376
- }, 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"]} */"],
3510
+ }, 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"]} */"],
3377
3511
  children: type === 'mobile' && jsx$1("div", {
3378
- 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"]} */"],
3512
+ 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"]} */"],
3379
3513
  children: jsx$1(ControlsBlock, {
3380
3514
  order: "mobile",
3381
3515
  ...controlButtons
@@ -3388,13 +3522,13 @@ const DefaultLayout = ({
3388
3522
  paddingBottom: 'calc(1em - var(--bottom-spacing, 0) / 5)',
3389
3523
  flexWrap: 'wrap',
3390
3524
  '> div:first-of-type': seekbarStyles
3391
- }, 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"]} */"],
3525
+ }, 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"]} */"],
3392
3526
  children: [seekbar || jsx$1("div", {}), type === 'desktop' && jsx$1(ControlsBlock, {
3393
3527
  order: "desktop",
3394
3528
  liveButton: liveButton,
3395
3529
  ...controlButtons
3396
3530
  }), type === 'desktop' ? jsx$1("div", {
3397
- 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"]} */"],
3531
+ 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"]} */"],
3398
3532
  children: displayTime
3399
3533
  }) : jsxs(Fragment, {
3400
3534
  children: [liveButton, jsx$1("div", {
@@ -3403,7 +3537,7 @@ const DefaultLayout = ({
3403
3537
  })]
3404
3538
  }), adStatus && jsx$1("div", {
3405
3539
  className: "pinned",
3406
- 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"]} */"],
3540
+ 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"]} */"],
3407
3541
  children: adStatus
3408
3542
  }), jsx$1("div", {
3409
3543
  css: !adStatus && [infoBarSlotStyle, type === 'desktop' && desktopInfoBarStyle],
@@ -3411,7 +3545,9 @@ const DefaultLayout = ({
3411
3545
  slotRef.current.infoBar = element;
3412
3546
  }
3413
3547
  }), volumeControl, jsx$1("div", {
3414
- css: adStatus && hidden,
3548
+ css: [adStatus ? hidden : {
3549
+ display: 'flex'
3550
+ }, 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"]} */"],
3415
3551
  ref: element => {
3416
3552
  slotRef.current.functionBar = element;
3417
3553
  }
@@ -3570,7 +3706,7 @@ const pointerHandlers = ({
3570
3706
  };
3571
3707
 
3572
3708
  /* eslint-disable no-param-reassign */
3573
- const containerStyle$2 = {
3709
+ const containerStyle$4 = {
3574
3710
  position: 'relative',
3575
3711
  width: '100%',
3576
3712
  height: '100%',
@@ -3819,7 +3955,7 @@ const SimpleSlider = ({
3819
3955
  };
3820
3956
  return jsxs("div", {
3821
3957
  className: className,
3822
- css: [containerStyle$2, disabled && disabledStyle, process.env.NODE_ENV === "production" ? "" : ";label:SimpleSlider;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SimpleSlider.jsx"],"names":[],"mappings":"AAwMM","file":"SimpleSlider.jsx","sourcesContent":["/* eslint-disable no-param-reassign */\n/* eslint-disable jsx-a11y/no-static-element-interactions */\n/* eslint-disable jsx-a11y/click-events-have-key-events */\n/* @jsxImportSource @emotion/react */\nimport {useState, useRef} from 'react'\n\nimport {getPointerData} from 'util/gestures'\n\nconst containerStyle = {\n  position: 'relative',\n  width: '100%',\n  height: '100%',\n  display: 'flex',\n  alignItems: 'center',\n  cursor: 'pointer',\n  userSelect: 'none',\n  touchAction: 'none',\n}\n\nconst disabledStyle = {\n  pointerEvents: 'none',\n}\n\nconst railStyle = {\n  flex: '100%',\n  background: 'rgba(255, 255, 255, 0.2)',\n}\n\nconst trackStyle = {\n  position: 'absolute',\n  width: '4px',\n  height: '100%',\n  backgroundColor: 'var(--sender-seekbar-background, #fff)',\n}\n\nconst markStyle = {\n  position: 'absolute',\n  height: railStyle.height,\n  width: '4px',\n  transform: 'translateX(-50%)',\n  backgroundColor: '#ff9835',\n}\n\nconst thumbStyle = {\n  position: 'absolute',\n  height: '0.66em',\n  width: '0.66em',\n  borderRadius: '100%',\n  backgroundColor: '#fff',\n  boxShadow: '0 2px 2px 0 rgba(0, 0, 0, 0.5)',\n  transform: 'translateY(calc(var(--slider-thumb-y) - 0.35em))',\n}\n\nconst getPointerValue = ({orientation, x, y, left, bottom, height, width}) =>\n  Math.max(\n    0,\n    orientation === 'vertical'\n      ? Math.min((bottom - y) / height, 1)\n      : Math.min((x - left) / width, 1),\n  )\n\nconst debouncedPointerHandlers = ({state, onMove, onLeave}) => {\n  const emit = () => {\n    if (!state.scheduled) {\n      return\n    }\n    if (state.type === 'leave') {\n      onLeave?.(state.event, state)\n    } else {\n      onMove(state.event, state)\n    }\n    state.scheduled = false\n  }\n  const schedule = () => {\n    if (state.scheduled) {\n      return\n    }\n    state.scheduled = true\n    requestAnimationFrame(emit)\n  }\n\n  return {\n    onPointerMove: event => {\n      const type =\n        event.buttons > 0 || event.touches?.length > 0 ? 'change' : 'move'\n      Object.assign(state, {event, type, ...getPointerData(event)})\n      schedule()\n    },\n    onPointerLeave: event => {\n      const type = 'leave'\n      Object.assign(state, {event, type})\n      schedule()\n    },\n    emit,\n  }\n}\n\nconst eventHandlers = ({\n  onPointerDown,\n  onPointerMove,\n  onPointerLeave,\n  onPointerUp,\n}) => ({\n  onPointerDown,\n  onPointerMove,\n  onPointerLeave,\n  onPointerUp,\n  onTouchStart: onPointerDown,\n  onTouchMove: onPointerMove,\n  onTouchEnd: event => {\n    onPointerLeave(event)\n    onPointerUp(event)\n  },\n})\n\nconst SliderRail = ({orientation, style, ...rest}) => (\n  <div\n    css={[style, orientation !== 'horizontal' && {borderRadius: '0.2em', alignSelf: 'normal'}]}\n    {...rest}\n  />\n)\n\nconst SliderTrack = ({value, orientation, style, ...rest}) => (\n  <div\n    css={[\n      style,\n      orientation === 'vertical'\n        ? {bottom: '0', height: `${value * 100}%`}\n        : {width: `${value * 100}%`},\n    ]}\n    {...rest}\n  />\n)\n\nconst defaultSlots = {\n  Rail: SliderRail,\n  Track: SliderTrack,\n}\n\n// TODO align with material ui more, move special handling of pointer events\nconst SimpleSlider = ({\n  min = 0,\n  max = 100,\n  value,\n  secondaryTrackValue, // TODO a better name\n  marks = [],\n  className = '',\n  classes = {},\n  disabled,\n  onPointerMove,\n  onPointerLeave,\n  onPointerDown,\n  onChange,\n  onChangeCommitted,\n  orientation = 'horizontal',\n  slots = defaultSlots,\n  slotProps = {},\n}) => {\n  const pointerState = useRef({})\n  const [focusValue, setFocusValue] = useState(-Infinity)\n  const thumbPosition =\n    ((focusValue >= min ? focusValue : value) - min) / (max - min)\n  const subTrackPosition = (secondaryTrackValue - min) / (max - min)\n  const pointerHandlers = debouncedPointerHandlers({\n    state: pointerState.current,\n    onMove: (event, {type, x, y, width, left, height, bottom}) => {\n      const pointerData = {orientation, x, y, left, bottom, width, height}\n      const pointerValue = min + (max - min) * getPointerValue(pointerData)\n\n      onPointerMove?.(event, {value: pointerValue, x, y})\n      if (type === 'change') {\n        setFocusValue(pointerValue)\n        onChange?.(event, {value: pointerValue, x, y})\n      }\n    },\n    onLeave: () => onPointerLeave?.(),\n  })\n  const handlePointerUp = event => {\n    if (event.pointerId) {\n      event.currentTarget.releasePointerCapture(event.pointerId)\n    }\n    const {x, y, width, left, height, bottom} = getPointerData(event)\n    const pointerValue =\n      min +\n      (max - min) *\n        getPointerValue({orientation, x, y, left, bottom, width, height})\n\n    pointerHandlers.emit()\n    onChangeCommitted?.(event, {value: pointerValue})\n    setFocusValue()\n  }\n\n  const thumbPositionStyle = {\n    bottom: `calc(var(--slider-thumb-y) - 0.35em)`,\n    left: `calc(var(--slider-thumb-x) - 0.25em)`,\n  }\n\n  return (\n    <div\n      className={className}\n      css={[containerStyle, disabled && disabledStyle]}\n      style={{\n        ...(orientation === 'vertical' && {\n          flexDirection: 'column',\n          '--slider-thumb-y': `${thumbPosition * 100}%`,\n        }),\n        ...(orientation === 'horizontal' && {\n          '--slider-thumb-x': `${thumbPosition * 100}%`,\n        }),\n      }}\n      onClick={event => event.stopPropagation()}\n      {...eventHandlers({\n        onPointerDown: event => {\n          if (event.type === 'pointerdown') {\n            event.currentTarget.setPointerCapture(event.pointerId)\n            onPointerDown?.()\n          }\n          pointerHandlers.onPointerMove(event)\n        },\n        ...pointerHandlers,\n        onPointerUp: handlePointerUp,\n      })}\n    >\n      <slots.Rail style={railStyle} className={classes.rail} {...slotProps.rail} />\n      {secondaryTrackValue && (\n        <slots.Track\n          style={{...trackStyle, backgroundColor: 'rgba(255, 255, 255, 0.3)'}}\n          orientation={orientation}\n          value={subTrackPosition}\n          {...slotProps.track}\n        />\n      )}\n      <slots.Track\n        className={classes.track}\n        style={trackStyle}\n        orientation={orientation}\n        value={thumbPosition}\n        {...slotProps.track}\n      />\n      {marks.map(position => (\n        <div\n          key={position}\n          css={markStyle}\n          className={classes.marked}\n          style={{left: `${(position / max) * 100}%`}}\n        />\n      ))}\n      {onChange && !disabled ? (\n        <div\n          css={thumbStyle}\n          className={classes.thumb}\n          style={thumbPositionStyle}\n        />\n      ) : (\n        <div />\n      )}\n    </div>\n  )\n}\n\nexport default SimpleSlider\n"]} */"],
3958
+ css: [containerStyle$4, disabled && disabledStyle, process.env.NODE_ENV === "production" ? "" : ";label:SimpleSlider;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SimpleSlider.jsx"],"names":[],"mappings":"AAwMM","file":"SimpleSlider.jsx","sourcesContent":["/* eslint-disable no-param-reassign */\n/* eslint-disable jsx-a11y/no-static-element-interactions */\n/* eslint-disable jsx-a11y/click-events-have-key-events */\n/* @jsxImportSource @emotion/react */\nimport {useState, useRef} from 'react'\n\nimport {getPointerData} from 'util/gestures'\n\nconst containerStyle = {\n  position: 'relative',\n  width: '100%',\n  height: '100%',\n  display: 'flex',\n  alignItems: 'center',\n  cursor: 'pointer',\n  userSelect: 'none',\n  touchAction: 'none',\n}\n\nconst disabledStyle = {\n  pointerEvents: 'none',\n}\n\nconst railStyle = {\n  flex: '100%',\n  background: 'rgba(255, 255, 255, 0.2)',\n}\n\nconst trackStyle = {\n  position: 'absolute',\n  width: '4px',\n  height: '100%',\n  backgroundColor: 'var(--sender-seekbar-background, #fff)',\n}\n\nconst markStyle = {\n  position: 'absolute',\n  height: railStyle.height,\n  width: '4px',\n  transform: 'translateX(-50%)',\n  backgroundColor: '#ff9835',\n}\n\nconst thumbStyle = {\n  position: 'absolute',\n  height: '0.66em',\n  width: '0.66em',\n  borderRadius: '100%',\n  backgroundColor: '#fff',\n  boxShadow: '0 2px 2px 0 rgba(0, 0, 0, 0.5)',\n  transform: 'translateY(calc(var(--slider-thumb-y) - 0.35em))',\n}\n\nconst getPointerValue = ({orientation, x, y, left, bottom, height, width}) =>\n  Math.max(\n    0,\n    orientation === 'vertical'\n      ? Math.min((bottom - y) / height, 1)\n      : Math.min((x - left) / width, 1),\n  )\n\nconst debouncedPointerHandlers = ({state, onMove, onLeave}) => {\n  const emit = () => {\n    if (!state.scheduled) {\n      return\n    }\n    if (state.type === 'leave') {\n      onLeave?.(state.event, state)\n    } else {\n      onMove(state.event, state)\n    }\n    state.scheduled = false\n  }\n  const schedule = () => {\n    if (state.scheduled) {\n      return\n    }\n    state.scheduled = true\n    requestAnimationFrame(emit)\n  }\n\n  return {\n    onPointerMove: event => {\n      const type =\n        event.buttons > 0 || event.touches?.length > 0 ? 'change' : 'move'\n      Object.assign(state, {event, type, ...getPointerData(event)})\n      schedule()\n    },\n    onPointerLeave: event => {\n      const type = 'leave'\n      Object.assign(state, {event, type})\n      schedule()\n    },\n    emit,\n  }\n}\n\nconst eventHandlers = ({\n  onPointerDown,\n  onPointerMove,\n  onPointerLeave,\n  onPointerUp,\n}) => ({\n  onPointerDown,\n  onPointerMove,\n  onPointerLeave,\n  onPointerUp,\n  onTouchStart: onPointerDown,\n  onTouchMove: onPointerMove,\n  onTouchEnd: event => {\n    onPointerLeave(event)\n    onPointerUp(event)\n  },\n})\n\nconst SliderRail = ({orientation, style, ...rest}) => (\n  <div\n    css={[style, orientation !== 'horizontal' && {borderRadius: '0.2em', alignSelf: 'normal'}]}\n    {...rest}\n  />\n)\n\nconst SliderTrack = ({value, orientation, style, ...rest}) => (\n  <div\n    css={[\n      style,\n      orientation === 'vertical'\n        ? {bottom: '0', height: `${value * 100}%`}\n        : {width: `${value * 100}%`},\n    ]}\n    {...rest}\n  />\n)\n\nconst defaultSlots = {\n  Rail: SliderRail,\n  Track: SliderTrack,\n}\n\n// TODO align with material ui more, move special handling of pointer events\nconst SimpleSlider = ({\n  min = 0,\n  max = 100,\n  value,\n  secondaryTrackValue, // TODO a better name\n  marks = [],\n  className = '',\n  classes = {},\n  disabled,\n  onPointerMove,\n  onPointerLeave,\n  onPointerDown,\n  onChange,\n  onChangeCommitted,\n  orientation = 'horizontal',\n  slots = defaultSlots,\n  slotProps = {},\n}) => {\n  const pointerState = useRef({})\n  const [focusValue, setFocusValue] = useState(-Infinity)\n  const thumbPosition =\n    ((focusValue >= min ? focusValue : value) - min) / (max - min)\n  const subTrackPosition = (secondaryTrackValue - min) / (max - min)\n  const pointerHandlers = debouncedPointerHandlers({\n    state: pointerState.current,\n    onMove: (event, {type, x, y, width, left, height, bottom}) => {\n      const pointerData = {orientation, x, y, left, bottom, width, height}\n      const pointerValue = min + (max - min) * getPointerValue(pointerData)\n\n      onPointerMove?.(event, {value: pointerValue, x, y})\n      if (type === 'change') {\n        setFocusValue(pointerValue)\n        onChange?.(event, {value: pointerValue, x, y})\n      }\n    },\n    onLeave: () => onPointerLeave?.(),\n  })\n  const handlePointerUp = event => {\n    if (event.pointerId) {\n      event.currentTarget.releasePointerCapture(event.pointerId)\n    }\n    const {x, y, width, left, height, bottom} = getPointerData(event)\n    const pointerValue =\n      min +\n      (max - min) *\n        getPointerValue({orientation, x, y, left, bottom, width, height})\n\n    pointerHandlers.emit()\n    onChangeCommitted?.(event, {value: pointerValue})\n    setFocusValue()\n  }\n\n  const thumbPositionStyle = {\n    bottom: `calc(var(--slider-thumb-y) - 0.35em)`,\n    left: `calc(var(--slider-thumb-x) - 0.25em)`,\n  }\n\n  return (\n    <div\n      className={className}\n      css={[containerStyle, disabled && disabledStyle]}\n      style={{\n        ...(orientation === 'vertical' && {\n          flexDirection: 'column',\n          '--slider-thumb-y': `${thumbPosition * 100}%`,\n        }),\n        ...(orientation === 'horizontal' && {\n          '--slider-thumb-x': `${thumbPosition * 100}%`,\n        }),\n      }}\n      onClick={event => event.stopPropagation()}\n      {...eventHandlers({\n        onPointerDown: event => {\n          if (event.type === 'pointerdown') {\n            event.currentTarget.setPointerCapture(event.pointerId)\n            onPointerDown?.()\n          }\n          pointerHandlers.onPointerMove(event)\n        },\n        ...pointerHandlers,\n        onPointerUp: handlePointerUp,\n      })}\n    >\n      <slots.Rail style={railStyle} className={classes.rail} {...slotProps.rail} />\n      {secondaryTrackValue && (\n        <slots.Track\n          style={{...trackStyle, backgroundColor: 'rgba(255, 255, 255, 0.3)'}}\n          orientation={orientation}\n          value={subTrackPosition}\n          {...slotProps.track}\n        />\n      )}\n      <slots.Track\n        className={classes.track}\n        style={trackStyle}\n        orientation={orientation}\n        value={thumbPosition}\n        {...slotProps.track}\n      />\n      {marks.map(position => (\n        <div\n          key={position}\n          css={markStyle}\n          className={classes.marked}\n          style={{left: `${(position / max) * 100}%`}}\n        />\n      ))}\n      {onChange && !disabled ? (\n        <div\n          css={thumbStyle}\n          className={classes.thumb}\n          style={thumbPositionStyle}\n        />\n      ) : (\n        <div />\n      )}\n    </div>\n  )\n}\n\nexport default SimpleSlider\n"]} */"],
3823
3959
  style: { ...(orientation === 'vertical' && {
3824
3960
  flexDirection: 'column',
3825
3961
  '--slider-thumb-y': `${thumbPosition * 100}%`
@@ -4002,6 +4138,7 @@ const reducePointer = (state, {
4002
4138
  const SeekbarRail = () => '';
4003
4139
 
4004
4140
  const segmentedTrackStyle = {
4141
+ boxSizing: 'border-box',
4005
4142
  position: 'relative',
4006
4143
  margin: '0 2px',
4007
4144
  height: '24px',
@@ -4035,7 +4172,7 @@ const SeekbarTrack = ({
4035
4172
  current,
4036
4173
  buffered
4037
4174
  }) => jsxs("div", {
4038
- css: [style, segmentedTrackStyle, process.env.NODE_ENV === "production" ? "" : ";label:SeekbarTrack;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIlNlZWtiYXIuanMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBb0ZNIiwiZmlsZSI6IlNlZWtiYXIuanMiLCJzb3VyY2VzQ29udGVudCI6WyIvKiBAanN4SW1wb3J0U291cmNlIEBlbW90aW9uL3JlYWN0ICovXG5pbXBvcnQge3VzZVJlZHVjZXIsIHVzZVJlZiwgY2xvbmVFbGVtZW50fSBmcm9tICdyZWFjdCdcbmltcG9ydCB1c2VEaW1lbnNpb25zIGZyb20gJ3JlYWN0LWNvb2wtZGltZW5zaW9ucydcbmltcG9ydCBTaW1wbGVTbGlkZXIgZnJvbSAncGxheWVyVWkvU2ltcGxlU2xpZGVyJ1xuaW1wb3J0IHtGb3JtYXR0ZWRUaW1lfSBmcm9tICdjb250ZXh0L0kxOG4nXG5pbXBvcnQge2dldFRpbWVsaW5lU2VnbWVudHN9IGZyb20gJ3ByZW1pdW0vdGltZWxpbmUnXG5cbmNvbnN0IHNlZWtiYXJTdHlsZSA9IHtcbiAgcG9zaXRpb246ICdyZWxhdGl2ZScsXG4gIGRpc3BsYXk6ICdmbGV4JyxcbiAgYWxpZ25JdGVtczogJ2NlbnRlcicsXG4gIG1pbldpZHRoOiAnMCcsXG4gIGhlaWdodDogJzI0cHgnLFxuICBmb250U2l6ZTogJzc1JScsXG4gIGxldHRlclNwYWNpbmc6ICcxcHgnLFxuICBjb2xvcjogJyNmZmYnLFxufVxuXG5jb25zdCBzbGlkZXJTdHlsZSA9IHtcbiAgbWluV2lkdGg6ICcwJyxcbiAgZmxleDogMSxcbiAgJ0BtZWRpYSAoaG92ZXI6IGhvdmVyKSwgc2NyZWVuIGFuZCAoLW1zLWhpZ2gtY29udHJhc3Q6IGFjdGl2ZSksICgtbXMtaGlnaC1jb250cmFzdDogbm9uZSknOlxuICAgIHtcbiAgICAgICc+IGRpdjpsYXN0LW9mLXR5cGUnOiB7XG4gICAgICAgIG9wYWNpdHk6IDAsXG4gICAgICAgIHRyYW5zaXRpb246ICdvcGFjaXR5IDAuMnMgZWFzZS1vdXQnLFxuICAgICAgfSxcbiAgICB9LFxuICAnPiBkaXY6bGFzdC1vZi10eXBlJzoge1xuICAgIHdpZHRoOiAnMS4yZW0nLFxuICAgIGhlaWdodDogJzEuMmVtJyxcbiAgICBiYWNrZ3JvdW5kOiAndmFyKC0tdGhlbWUtY29sb3IsIHJlZCknLFxuICB9LFxuICAnJjpob3ZlciA+IGRpdjpsYXN0LW9mLXR5cGUnOiB7XG4gICAgb3BhY2l0eTogMSxcbiAgfSxcbn1cblxuY29uc3QgcmVkdWNlUG9pbnRlciA9IChzdGF0ZSwge3R5cGUsIHZhbHVlLCB4fSkgPT4ge1xuICBzd2l0Y2ggKHR5cGUpIHtcbiAgICBjYXNlICdtb3ZlJzpcbiAgICAgIHJldHVybiB7Li4uc3RhdGUsIGhvdmVyOiB0cnVlLCB2YWx1ZSwgeH1cbiAgICBjYXNlICdjaGFuZ2UnOlxuICAgICAgcmV0dXJuIHsuLi5zdGF0ZSwgZm9jdXNlZDogdHJ1ZSwgdmFsdWV9XG4gICAgY2FzZSAncmVsZWFzZSc6XG4gICAgICByZXR1cm4gey4uLnN0YXRlLCBmb2N1c2VkOiBmYWxzZSwgdmFsdWV9XG4gICAgY2FzZSAnbGVhdmUnOlxuICAgICAgcmV0dXJuIHsuLi5zdGF0ZSwgaG92ZXI6IGZhbHNlfVxuICAgIGRlZmF1bHQ6XG4gICAgICByZXR1cm4gc3RhdGVcbiAgfVxufVxuXG5jb25zdCBTZWVrYmFyUmFpbCA9ICgpID0+ICcnXG5cbmNvbnN0IHNlZ21lbnRlZFRyYWNrU3R5bGUgPSB7XG4gIHBvc2l0aW9uOiAncmVsYXRpdmUnLFxuICBtYXJnaW46ICcwIDJweCcsXG4gIGhlaWdodDogJzI0cHgnLFxuICBib3JkZXJUb3A6ICdjYWxjKDEycHggLSAwLjNlbSkgc29saWQgdHJhbnNwYXJlbnQnLFxuICBib3JkZXJCb3R0b206ICdjYWxjKDEycHggLSAwLjNlbSkgc29saWQgdHJhbnNwYXJlbnQnLFxuICBiYWNrZ3JvdW5kOiAndHJhbnNwYXJlbnQnLFxuICAnJjpob3Zlcic6IHtcbiAgICB0cmFuc2Zvcm06ICdzY2FsZSgxLCAxLjUpJyxcbiAgfSxcbiAgJz4gZGl2Jzoge1xuICAgIGhlaWdodDogJzAuNmVtJyxcbiAgICBiYWNrZ3JvdW5kQ29sb3I6ICdyZ2JhKDI1NSwgMjU1LCAyNTUsIDAuMiknLFxuICAgICcmOm5vdCg6Zmlyc3Qtb2YtdHlwZSknOiB7XG4gICAgICBwb3NpdGlvbjogJ2Fic29sdXRlJyxcbiAgICAgIHRvcDogJzAnLFxuICAgIH0sXG4gIH0sXG4gICc+IGRpdjpudGgtb2YtdHlwZSgyKSc6IHtcbiAgICBiYWNrZ3JvdW5kQ29sb3I6ICdyZ2JhKDI1NSwgMjU1LCAyNTUsIDAuMyknLFxuICB9LFxuICAnPiBkaXY6bnRoLW9mLXR5cGUoMyknOiB7XG4gICAgYmFja2dyb3VuZENvbG9yOiAndmFyKC0tdGhlbWUtY29sb3IsIHJlZCknLFxuICB9LFxufVxuXG5jb25zdCBTZWVrYmFyVHJhY2sgPSAoe3NlZ21lbnRzLCBzdHlsZX0pID0+XG4gIHNlZ21lbnRzLm1hcCgoe2xlbmd0aCwgY3VycmVudCwgYnVmZmVyZWR9KSA9PiAoXG4gICAgPGRpdlxuICAgICAgY3NzPXtbc3R5bGUsIHNlZ21lbnRlZFRyYWNrU3R5bGVdfVxuICAgICAgc3R5bGU9e3tcbiAgICAgICAgZmxleDogYDAgJHtsZW5ndGggKiAxMDB9JWAsXG4gICAgICAgIHdpZHRoOiBgY2FsYygke2xlbmd0aCAqIDEwMH0lIC0gNHB4KWAsXG4gICAgICB9fVxuICAgID5cbiAgICAgIDxkaXYgc3R5bGU9e3t3aWR0aDogJzEwMCUnfX0gLz5cbiAgICAgIDxkaXYgc3R5bGU9e3t3aWR0aDogYCR7YnVmZmVyZWQgKiAxMDB9JWB9fSAvPlxuICAgICAgPGRpdiBzdHlsZT17e3dpZHRoOiBgJHtjdXJyZW50ICogMTAwfSVgfX0gLz5cbiAgICA8L2Rpdj5cbiAgKSlcblxuLy8gVE9ETyB1c2UgY2xhc3NOYW1lIGluc3RlYWQgb2YgY2xhc3NlcyA/XG5jb25zdCBTZWVrYmFyID0gKHtcbiAgc3R5bGUsXG4gIGNsYXNzZXMsXG4gIHN0YXJ0VGltZSA9IDAsXG4gIGN1cnJlbnRUaW1lLFxuICBidWZmZXJUaW1lLFxuICBkdXJhdGlvbixcbiAgY2hhcHRlcnMgPSBbXSxcbiAgbWFya3MsXG4gIHRpbWVEaXNwbGF5ID0gZmFsc2UsIC8vIFRPRE8gbW9yZSBzY2FsYWJsZSB3YXk/XG4gIG9uQ2hhbmdlLFxuICBvbkNoYW5nZUNvbW1pdHRlZCxcbiAgY2hpbGRyZW4sXG4gIG9uUG9pbnRlck1vdmUsXG4gIG9uUG9pbnRlckxlYXZlLFxuICAuLi5yZXN0XG59KSA9PiB7XG4gIGNvbnN0IFtwb2ludGVyU3RhdGUsIGRpc3BhdGNoUG9pbnRlcl0gPSB1c2VSZWR1Y2VyKHJlZHVjZVBvaW50ZXIsIHt9KVxuICBjb25zdCBwb2ludGVyQWN0aXZlID0gcG9pbnRlclN0YXRlLmhvdmVyIHx8IHBvaW50ZXJTdGF0ZS5mb2N1c2VkXG4gIC8vIHRvIHJlZmxlY3QgYm91bmRhcnkgd2hlbiBjb250YWluZXIgcmVzaXplZFxuICBjb25zdCB7b2JzZXJ2ZX0gPSB1c2VEaW1lbnNpb25zKClcbiAgY29uc3QgcmVmID0gdXNlUmVmKClcbiAgY29uc3QgcmVjdCA9IHJlZi5jdXJyZW50Py5nZXRCb3VuZGluZ0NsaWVudFJlY3QoKVxuICBjb25zdCBoYW5kbGVycyA9IG9uQ2hhbmdlQ29tbWl0dGVkICYmIHtcbiAgICBvblBvaW50ZXJNb3ZlOiAoZXZlbnQsIHt2YWx1ZSwgeH0pID0+IHtcbiAgICAgIG9uUG9pbnRlck1vdmU/LihldmVudCwge3ZhbHVlLCB4fSlcbiAgICAgIGRpc3BhdGNoUG9pbnRlcih7dHlwZTogJ21vdmUnLCB2YWx1ZSwgeH0pXG4gICAgfSxcbiAgICBvblBvaW50ZXJMZWF2ZTogZXZlbnQgPT4ge1xuICAgICAgb25Qb2ludGVyTGVhdmU/LihldmVudClcbiAgICAgIGRpc3BhdGNoUG9pbnRlcih7dHlwZTogJ2xlYXZlJ30pXG4gICAgfSxcbiAgICBvbkNoYW5nZTogKF8sIHt2YWx1ZX0pID0+IHtcbiAgICAgIG9uQ2hhbmdlPy4odmFsdWUpXG4gICAgICBkaXNwYXRjaFBvaW50ZXIoe3R5cGU6ICdjaGFuZ2UnLCB2YWx1ZX0pXG4gICAgfSxcbiAgICBvbkNoYW5nZUNvbW1pdHRlZDogKF8sIHt2YWx1ZX0pID0+IHtcbiAgICAgIGRpc3BhdGNoUG9pbnRlcih7dHlwZTogJ3JlbGVhc2UnLCB2YWx1ZX0pXG4gICAgICBvbkNoYW5nZUNvbW1pdHRlZD8uKHZhbHVlKVxuICAgIH0sXG4gIH1cbiAgY29uc3QgZW5kVGltZSA9IHN0YXJ0VGltZSArIGR1cmF0aW9uXG4gIGNvbnN0IHNlZ21lbnRzID0gZ2V0VGltZWxpbmVTZWdtZW50cyhjaGFwdGVycywge1xuICAgIHN0YXJ0VGltZSxcbiAgICBjdXJyZW50OiBwb2ludGVyU3RhdGUuZm9jdXNlZCA/IHBvaW50ZXJTdGF0ZS52YWx1ZSA6IGN1cnJlbnRUaW1lLFxuICAgIGJ1ZmZlcmVkOiBidWZmZXJUaW1lLFxuICAgIGR1cmF0aW9uLFxuICB9KVxuXG4gIHJldHVybiAhKGR1cmF0aW9uID4gMCkgPyAoXG4gICAgPGRpdiAvPlxuICApIDogKFxuICAgIDxkaXZcbiAgICAgIHJlZj17ZWxlbWVudCA9PiB7XG4gICAgICAgIG9ic2VydmUoZWxlbWVudClcbiAgICAgICAgcmVmLmN1cnJlbnQgPSBlbGVtZW50XG4gICAgICB9fVxuICAgICAgY3NzPXtbc2Vla2JhclN0eWxlLCBzdHlsZV19XG4gICAgICBzdHlsZT17XG4gICAgICAgIHJlY3QgJiYge1xuICAgICAgICAgICctLXNlZWtiYXItbGVmdCc6IGAke3JlY3QubGVmdH1weGAsXG4gICAgICAgICAgJy0tc2Vla2Jhci1yaWdodCc6IGAke3JlY3QucmlnaHR9cHhgLFxuICAgICAgICAgICctLXBvaW50ZXIteCc6IGAke3BvaW50ZXJTdGF0ZS54fXB4YCxcbiAgICAgICAgfVxuICAgICAgfVxuICAgID5cbiAgICAgIHshdGltZURpc3BsYXkgPyAoXG4gICAgICAgICcnXG4gICAgICApIDogcG9pbnRlckFjdGl2ZSA/IChcbiAgICAgICAgPEZvcm1hdHRlZFRpbWUgdGltZT17cG9pbnRlclN0YXRlLnZhbHVlfSAvPlxuICAgICAgKSA6IChcbiAgICAgICAgPEZvcm1hdHRlZFRpbWUgdGltZT17TWF0aC5taW4oTWF0aC5tYXgoMCwgY3VycmVudFRpbWUpLCBkdXJhdGlvbil9IC8+XG4gICAgICApfVxuICAgICAgPFNpbXBsZVNsaWRlclxuICAgICAgICBjc3M9e1tzbGlkZXJTdHlsZSwgdGltZURpc3BsYXkgJiYge21hcmdpbjogJzAgMWVtJ31dfVxuICAgICAgICBjbGFzc2VzPXtjbGFzc2VzfVxuICAgICAgICBkaXNhYmxlZD17IW9uQ2hhbmdlQ29tbWl0dGVkfVxuICAgICAgICAvLyBkaXNwbGF5IGZpbGxlZCB3aGVuIHNlZWsgaGFuZGxlciBpcyBub3QgcHJvdmlkZWQsIGZyb20gUGxheUJveSBiZWhhdmlvclxuICAgICAgICB2YWx1ZT17b25DaGFuZ2VDb21taXR0ZWQgPyBjdXJyZW50VGltZSA6IGVuZFRpbWV9XG4gICAgICAgIG1pbj17c3RhcnRUaW1lfVxuICAgICAgICBtYXg9e2VuZFRpbWV9XG4gICAgICAgIG1hcmtzPXttYXJrc31cbiAgICAgICAgey4uLmhhbmRsZXJzfVxuICAgICAgICB7Li4ucmVzdH1cbiAgICAgICAgc2xvdHM9e3tSYWlsOiBTZWVrYmFyUmFpbCwgVHJhY2s6IFNlZWtiYXJUcmFja319XG4gICAgICAgIHNsb3RQcm9wcz17e3RyYWNrOiB7c2VnbWVudHN9fX1cbiAgICAgIC8+XG4gICAgICB7dGltZURpc3BsYXkgJiYgPEZvcm1hdHRlZFRpbWUgdGltZT17ZHVyYXRpb259IC8+fVxuICAgICAge2NoaWxkcmVuICYmXG4gICAgICAgIFtdXG4gICAgICAgICAgLmNvbmNhdChjaGlsZHJlbilcbiAgICAgICAgICAubWFwKGNoaWxkID0+XG4gICAgICAgICAgICBjbG9uZUVsZW1lbnQoY2hpbGQsIHt0aW1lOiBwb2ludGVyQWN0aXZlICYmIHBvaW50ZXJTdGF0ZS52YWx1ZX0pXG4gICAgICAgICAgKX1cbiAgICA8L2Rpdj5cbiAgKVxufVxuXG5leHBvcnQgZGVmYXVsdCBTZWVrYmFyXG4iXX0= */"],
4175
+ css: [style, segmentedTrackStyle, process.env.NODE_ENV === "production" ? "" : ";label:SeekbarTrack;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIlNlZWtiYXIuanMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBcUZNIiwiZmlsZSI6IlNlZWtiYXIuanMiLCJzb3VyY2VzQ29udGVudCI6WyIvKiBAanN4SW1wb3J0U291cmNlIEBlbW90aW9uL3JlYWN0ICovXG5pbXBvcnQge3VzZVJlZHVjZXIsIHVzZVJlZiwgY2xvbmVFbGVtZW50fSBmcm9tICdyZWFjdCdcbmltcG9ydCB1c2VEaW1lbnNpb25zIGZyb20gJ3JlYWN0LWNvb2wtZGltZW5zaW9ucydcbmltcG9ydCBTaW1wbGVTbGlkZXIgZnJvbSAncGxheWVyVWkvU2ltcGxlU2xpZGVyJ1xuaW1wb3J0IHtGb3JtYXR0ZWRUaW1lfSBmcm9tICdjb250ZXh0L0kxOG4nXG5pbXBvcnQge2dldFRpbWVsaW5lU2VnbWVudHN9IGZyb20gJ3ByZW1pdW0vdGltZWxpbmUnXG5cbmNvbnN0IHNlZWtiYXJTdHlsZSA9IHtcbiAgcG9zaXRpb246ICdyZWxhdGl2ZScsXG4gIGRpc3BsYXk6ICdmbGV4JyxcbiAgYWxpZ25JdGVtczogJ2NlbnRlcicsXG4gIG1pbldpZHRoOiAnMCcsXG4gIGhlaWdodDogJzI0cHgnLFxuICBmb250U2l6ZTogJzc1JScsXG4gIGxldHRlclNwYWNpbmc6ICcxcHgnLFxuICBjb2xvcjogJyNmZmYnLFxufVxuXG5jb25zdCBzbGlkZXJTdHlsZSA9IHtcbiAgbWluV2lkdGg6ICcwJyxcbiAgZmxleDogMSxcbiAgJ0BtZWRpYSAoaG92ZXI6IGhvdmVyKSwgc2NyZWVuIGFuZCAoLW1zLWhpZ2gtY29udHJhc3Q6IGFjdGl2ZSksICgtbXMtaGlnaC1jb250cmFzdDogbm9uZSknOlxuICAgIHtcbiAgICAgICc+IGRpdjpsYXN0LW9mLXR5cGUnOiB7XG4gICAgICAgIG9wYWNpdHk6IDAsXG4gICAgICAgIHRyYW5zaXRpb246ICdvcGFjaXR5IDAuMnMgZWFzZS1vdXQnLFxuICAgICAgfSxcbiAgICB9LFxuICAnPiBkaXY6bGFzdC1vZi10eXBlJzoge1xuICAgIHdpZHRoOiAnMS4yZW0nLFxuICAgIGhlaWdodDogJzEuMmVtJyxcbiAgICBiYWNrZ3JvdW5kOiAndmFyKC0tdGhlbWUtY29sb3IsIHJlZCknLFxuICB9LFxuICAnJjpob3ZlciA+IGRpdjpsYXN0LW9mLXR5cGUnOiB7XG4gICAgb3BhY2l0eTogMSxcbiAgfSxcbn1cblxuY29uc3QgcmVkdWNlUG9pbnRlciA9IChzdGF0ZSwge3R5cGUsIHZhbHVlLCB4fSkgPT4ge1xuICBzd2l0Y2ggKHR5cGUpIHtcbiAgICBjYXNlICdtb3ZlJzpcbiAgICAgIHJldHVybiB7Li4uc3RhdGUsIGhvdmVyOiB0cnVlLCB2YWx1ZSwgeH1cbiAgICBjYXNlICdjaGFuZ2UnOlxuICAgICAgcmV0dXJuIHsuLi5zdGF0ZSwgZm9jdXNlZDogdHJ1ZSwgdmFsdWV9XG4gICAgY2FzZSAncmVsZWFzZSc6XG4gICAgICByZXR1cm4gey4uLnN0YXRlLCBmb2N1c2VkOiBmYWxzZSwgdmFsdWV9XG4gICAgY2FzZSAnbGVhdmUnOlxuICAgICAgcmV0dXJuIHsuLi5zdGF0ZSwgaG92ZXI6IGZhbHNlfVxuICAgIGRlZmF1bHQ6XG4gICAgICByZXR1cm4gc3RhdGVcbiAgfVxufVxuXG5jb25zdCBTZWVrYmFyUmFpbCA9ICgpID0+ICcnXG5cbmNvbnN0IHNlZ21lbnRlZFRyYWNrU3R5bGUgPSB7XG4gIGJveFNpemluZzogJ2JvcmRlci1ib3gnLFxuICBwb3NpdGlvbjogJ3JlbGF0aXZlJyxcbiAgbWFyZ2luOiAnMCAycHgnLFxuICBoZWlnaHQ6ICcyNHB4JyxcbiAgYm9yZGVyVG9wOiAnY2FsYygxMnB4IC0gMC4zZW0pIHNvbGlkIHRyYW5zcGFyZW50JyxcbiAgYm9yZGVyQm90dG9tOiAnY2FsYygxMnB4IC0gMC4zZW0pIHNvbGlkIHRyYW5zcGFyZW50JyxcbiAgYmFja2dyb3VuZDogJ3RyYW5zcGFyZW50JyxcbiAgJyY6aG92ZXInOiB7XG4gICAgdHJhbnNmb3JtOiAnc2NhbGUoMSwgMS41KScsXG4gIH0sXG4gICc+IGRpdic6IHtcbiAgICBoZWlnaHQ6ICcwLjZlbScsXG4gICAgYmFja2dyb3VuZENvbG9yOiAncmdiYSgyNTUsIDI1NSwgMjU1LCAwLjIpJyxcbiAgICAnJjpub3QoOmZpcnN0LW9mLXR5cGUpJzoge1xuICAgICAgcG9zaXRpb246ICdhYnNvbHV0ZScsXG4gICAgICB0b3A6ICcwJyxcbiAgICB9LFxuICB9LFxuICAnPiBkaXY6bnRoLW9mLXR5cGUoMiknOiB7XG4gICAgYmFja2dyb3VuZENvbG9yOiAncmdiYSgyNTUsIDI1NSwgMjU1LCAwLjMpJyxcbiAgfSxcbiAgJz4gZGl2Om50aC1vZi10eXBlKDMpJzoge1xuICAgIGJhY2tncm91bmRDb2xvcjogJ3ZhcigtLXRoZW1lLWNvbG9yLCByZWQpJyxcbiAgfSxcbn1cblxuY29uc3QgU2Vla2JhclRyYWNrID0gKHtzZWdtZW50cywgc3R5bGV9KSA9PlxuICBzZWdtZW50cy5tYXAoKHtsZW5ndGgsIGN1cnJlbnQsIGJ1ZmZlcmVkfSkgPT4gKFxuICAgIDxkaXZcbiAgICAgIGNzcz17W3N0eWxlLCBzZWdtZW50ZWRUcmFja1N0eWxlXX1cbiAgICAgIHN0eWxlPXt7XG4gICAgICAgIGZsZXg6IGAwICR7bGVuZ3RoICogMTAwfSVgLFxuICAgICAgICB3aWR0aDogYGNhbGMoJHtsZW5ndGggKiAxMDB9JSAtIDRweClgLFxuICAgICAgfX1cbiAgICA+XG4gICAgICA8ZGl2IHN0eWxlPXt7d2lkdGg6ICcxMDAlJ319IC8+XG4gICAgICA8ZGl2IHN0eWxlPXt7d2lkdGg6IGAke2J1ZmZlcmVkICogMTAwfSVgfX0gLz5cbiAgICAgIDxkaXYgc3R5bGU9e3t3aWR0aDogYCR7Y3VycmVudCAqIDEwMH0lYH19IC8+XG4gICAgPC9kaXY+XG4gICkpXG5cbi8vIFRPRE8gdXNlIGNsYXNzTmFtZSBpbnN0ZWFkIG9mIGNsYXNzZXMgP1xuY29uc3QgU2Vla2JhciA9ICh7XG4gIHN0eWxlLFxuICBjbGFzc2VzLFxuICBzdGFydFRpbWUgPSAwLFxuICBjdXJyZW50VGltZSxcbiAgYnVmZmVyVGltZSxcbiAgZHVyYXRpb24sXG4gIGNoYXB0ZXJzID0gW10sXG4gIG1hcmtzLFxuICB0aW1lRGlzcGxheSA9IGZhbHNlLCAvLyBUT0RPIG1vcmUgc2NhbGFibGUgd2F5P1xuICBvbkNoYW5nZSxcbiAgb25DaGFuZ2VDb21taXR0ZWQsXG4gIGNoaWxkcmVuLFxuICBvblBvaW50ZXJNb3ZlLFxuICBvblBvaW50ZXJMZWF2ZSxcbiAgLi4ucmVzdFxufSkgPT4ge1xuICBjb25zdCBbcG9pbnRlclN0YXRlLCBkaXNwYXRjaFBvaW50ZXJdID0gdXNlUmVkdWNlcihyZWR1Y2VQb2ludGVyLCB7fSlcbiAgY29uc3QgcG9pbnRlckFjdGl2ZSA9IHBvaW50ZXJTdGF0ZS5ob3ZlciB8fCBwb2ludGVyU3RhdGUuZm9jdXNlZFxuICAvLyB0byByZWZsZWN0IGJvdW5kYXJ5IHdoZW4gY29udGFpbmVyIHJlc2l6ZWRcbiAgY29uc3Qge29ic2VydmV9ID0gdXNlRGltZW5zaW9ucygpXG4gIGNvbnN0IHJlZiA9IHVzZVJlZigpXG4gIGNvbnN0IHJlY3QgPSByZWYuY3VycmVudD8uZ2V0Qm91bmRpbmdDbGllbnRSZWN0KClcbiAgY29uc3QgaGFuZGxlcnMgPSBvbkNoYW5nZUNvbW1pdHRlZCAmJiB7XG4gICAgb25Qb2ludGVyTW92ZTogKGV2ZW50LCB7dmFsdWUsIHh9KSA9PiB7XG4gICAgICBvblBvaW50ZXJNb3ZlPy4oZXZlbnQsIHt2YWx1ZSwgeH0pXG4gICAgICBkaXNwYXRjaFBvaW50ZXIoe3R5cGU6ICdtb3ZlJywgdmFsdWUsIHh9KVxuICAgIH0sXG4gICAgb25Qb2ludGVyTGVhdmU6IGV2ZW50ID0+IHtcbiAgICAgIG9uUG9pbnRlckxlYXZlPy4oZXZlbnQpXG4gICAgICBkaXNwYXRjaFBvaW50ZXIoe3R5cGU6ICdsZWF2ZSd9KVxuICAgIH0sXG4gICAgb25DaGFuZ2U6IChfLCB7dmFsdWV9KSA9PiB7XG4gICAgICBvbkNoYW5nZT8uKHZhbHVlKVxuICAgICAgZGlzcGF0Y2hQb2ludGVyKHt0eXBlOiAnY2hhbmdlJywgdmFsdWV9KVxuICAgIH0sXG4gICAgb25DaGFuZ2VDb21taXR0ZWQ6IChfLCB7dmFsdWV9KSA9PiB7XG4gICAgICBkaXNwYXRjaFBvaW50ZXIoe3R5cGU6ICdyZWxlYXNlJywgdmFsdWV9KVxuICAgICAgb25DaGFuZ2VDb21taXR0ZWQ/Lih2YWx1ZSlcbiAgICB9LFxuICB9XG4gIGNvbnN0IGVuZFRpbWUgPSBzdGFydFRpbWUgKyBkdXJhdGlvblxuICBjb25zdCBzZWdtZW50cyA9IGdldFRpbWVsaW5lU2VnbWVudHMoY2hhcHRlcnMsIHtcbiAgICBzdGFydFRpbWUsXG4gICAgY3VycmVudDogcG9pbnRlclN0YXRlLmZvY3VzZWQgPyBwb2ludGVyU3RhdGUudmFsdWUgOiBjdXJyZW50VGltZSxcbiAgICBidWZmZXJlZDogYnVmZmVyVGltZSxcbiAgICBkdXJhdGlvbixcbiAgfSlcblxuICByZXR1cm4gIShkdXJhdGlvbiA+IDApID8gKFxuICAgIDxkaXYgLz5cbiAgKSA6IChcbiAgICA8ZGl2XG4gICAgICByZWY9e2VsZW1lbnQgPT4ge1xuICAgICAgICBvYnNlcnZlKGVsZW1lbnQpXG4gICAgICAgIHJlZi5jdXJyZW50ID0gZWxlbWVudFxuICAgICAgfX1cbiAgICAgIGNzcz17W3NlZWtiYXJTdHlsZSwgc3R5bGVdfVxuICAgICAgc3R5bGU9e1xuICAgICAgICByZWN0ICYmIHtcbiAgICAgICAgICAnLS1zZWVrYmFyLWxlZnQnOiBgJHtyZWN0LmxlZnR9cHhgLFxuICAgICAgICAgICctLXNlZWtiYXItcmlnaHQnOiBgJHtyZWN0LnJpZ2h0fXB4YCxcbiAgICAgICAgICAnLS1wb2ludGVyLXgnOiBgJHtwb2ludGVyU3RhdGUueH1weGAsXG4gICAgICAgIH1cbiAgICAgIH1cbiAgICA+XG4gICAgICB7IXRpbWVEaXNwbGF5ID8gKFxuICAgICAgICAnJ1xuICAgICAgKSA6IHBvaW50ZXJBY3RpdmUgPyAoXG4gICAgICAgIDxGb3JtYXR0ZWRUaW1lIHRpbWU9e3BvaW50ZXJTdGF0ZS52YWx1ZX0gLz5cbiAgICAgICkgOiAoXG4gICAgICAgIDxGb3JtYXR0ZWRUaW1lIHRpbWU9e01hdGgubWluKE1hdGgubWF4KDAsIGN1cnJlbnRUaW1lKSwgZHVyYXRpb24pfSAvPlxuICAgICAgKX1cbiAgICAgIDxTaW1wbGVTbGlkZXJcbiAgICAgICAgY3NzPXtbc2xpZGVyU3R5bGUsIHRpbWVEaXNwbGF5ICYmIHttYXJnaW46ICcwIDFlbSd9XX1cbiAgICAgICAgY2xhc3Nlcz17Y2xhc3Nlc31cbiAgICAgICAgZGlzYWJsZWQ9eyFvbkNoYW5nZUNvbW1pdHRlZH1cbiAgICAgICAgLy8gZGlzcGxheSBmaWxsZWQgd2hlbiBzZWVrIGhhbmRsZXIgaXMgbm90IHByb3ZpZGVkLCBmcm9tIFBsYXlCb3kgYmVoYXZpb3JcbiAgICAgICAgdmFsdWU9e29uQ2hhbmdlQ29tbWl0dGVkID8gY3VycmVudFRpbWUgOiBlbmRUaW1lfVxuICAgICAgICBtaW49e3N0YXJ0VGltZX1cbiAgICAgICAgbWF4PXtlbmRUaW1lfVxuICAgICAgICBtYXJrcz17bWFya3N9XG4gICAgICAgIHsuLi5oYW5kbGVyc31cbiAgICAgICAgey4uLnJlc3R9XG4gICAgICAgIHNsb3RzPXt7UmFpbDogU2Vla2JhclJhaWwsIFRyYWNrOiBTZWVrYmFyVHJhY2t9fVxuICAgICAgICBzbG90UHJvcHM9e3t0cmFjazoge3NlZ21lbnRzfX19XG4gICAgICAvPlxuICAgICAge3RpbWVEaXNwbGF5ICYmIDxGb3JtYXR0ZWRUaW1lIHRpbWU9e2R1cmF0aW9ufSAvPn1cbiAgICAgIHtjaGlsZHJlbiAmJlxuICAgICAgICBbXVxuICAgICAgICAgIC5jb25jYXQoY2hpbGRyZW4pXG4gICAgICAgICAgLm1hcChjaGlsZCA9PlxuICAgICAgICAgICAgY2xvbmVFbGVtZW50KGNoaWxkLCB7dGltZTogcG9pbnRlckFjdGl2ZSAmJiBwb2ludGVyU3RhdGUudmFsdWV9KVxuICAgICAgICAgICl9XG4gICAgPC9kaXY+XG4gIClcbn1cblxuZXhwb3J0IGRlZmF1bHQgU2Vla2JhclxuIl19 */"],
4039
4176
  style: {
4040
4177
  flex: `0 ${length * 100}%`,
4041
4178
  width: `calc(${length * 100}% - 4px)`
@@ -4136,7 +4273,7 @@ const Seekbar = ({
4136
4273
  observe(element);
4137
4274
  ref.current = element;
4138
4275
  },
4139
- css: [seekbarStyle, style, process.env.NODE_ENV === "production" ? "" : ";label:Seekbar;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIlNlZWtiYXIuanMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBMEpNIiwiZmlsZSI6IlNlZWtiYXIuanMiLCJzb3VyY2VzQ29udGVudCI6WyIvKiBAanN4SW1wb3J0U291cmNlIEBlbW90aW9uL3JlYWN0ICovXG5pbXBvcnQge3VzZVJlZHVjZXIsIHVzZVJlZiwgY2xvbmVFbGVtZW50fSBmcm9tICdyZWFjdCdcbmltcG9ydCB1c2VEaW1lbnNpb25zIGZyb20gJ3JlYWN0LWNvb2wtZGltZW5zaW9ucydcbmltcG9ydCBTaW1wbGVTbGlkZXIgZnJvbSAncGxheWVyVWkvU2ltcGxlU2xpZGVyJ1xuaW1wb3J0IHtGb3JtYXR0ZWRUaW1lfSBmcm9tICdjb250ZXh0L0kxOG4nXG5pbXBvcnQge2dldFRpbWVsaW5lU2VnbWVudHN9IGZyb20gJ3ByZW1pdW0vdGltZWxpbmUnXG5cbmNvbnN0IHNlZWtiYXJTdHlsZSA9IHtcbiAgcG9zaXRpb246ICdyZWxhdGl2ZScsXG4gIGRpc3BsYXk6ICdmbGV4JyxcbiAgYWxpZ25JdGVtczogJ2NlbnRlcicsXG4gIG1pbldpZHRoOiAnMCcsXG4gIGhlaWdodDogJzI0cHgnLFxuICBmb250U2l6ZTogJzc1JScsXG4gIGxldHRlclNwYWNpbmc6ICcxcHgnLFxuICBjb2xvcjogJyNmZmYnLFxufVxuXG5jb25zdCBzbGlkZXJTdHlsZSA9IHtcbiAgbWluV2lkdGg6ICcwJyxcbiAgZmxleDogMSxcbiAgJ0BtZWRpYSAoaG92ZXI6IGhvdmVyKSwgc2NyZWVuIGFuZCAoLW1zLWhpZ2gtY29udHJhc3Q6IGFjdGl2ZSksICgtbXMtaGlnaC1jb250cmFzdDogbm9uZSknOlxuICAgIHtcbiAgICAgICc+IGRpdjpsYXN0LW9mLXR5cGUnOiB7XG4gICAgICAgIG9wYWNpdHk6IDAsXG4gICAgICAgIHRyYW5zaXRpb246ICdvcGFjaXR5IDAuMnMgZWFzZS1vdXQnLFxuICAgICAgfSxcbiAgICB9LFxuICAnPiBkaXY6bGFzdC1vZi10eXBlJzoge1xuICAgIHdpZHRoOiAnMS4yZW0nLFxuICAgIGhlaWdodDogJzEuMmVtJyxcbiAgICBiYWNrZ3JvdW5kOiAndmFyKC0tdGhlbWUtY29sb3IsIHJlZCknLFxuICB9LFxuICAnJjpob3ZlciA+IGRpdjpsYXN0LW9mLXR5cGUnOiB7XG4gICAgb3BhY2l0eTogMSxcbiAgfSxcbn1cblxuY29uc3QgcmVkdWNlUG9pbnRlciA9IChzdGF0ZSwge3R5cGUsIHZhbHVlLCB4fSkgPT4ge1xuICBzd2l0Y2ggKHR5cGUpIHtcbiAgICBjYXNlICdtb3ZlJzpcbiAgICAgIHJldHVybiB7Li4uc3RhdGUsIGhvdmVyOiB0cnVlLCB2YWx1ZSwgeH1cbiAgICBjYXNlICdjaGFuZ2UnOlxuICAgICAgcmV0dXJuIHsuLi5zdGF0ZSwgZm9jdXNlZDogdHJ1ZSwgdmFsdWV9XG4gICAgY2FzZSAncmVsZWFzZSc6XG4gICAgICByZXR1cm4gey4uLnN0YXRlLCBmb2N1c2VkOiBmYWxzZSwgdmFsdWV9XG4gICAgY2FzZSAnbGVhdmUnOlxuICAgICAgcmV0dXJuIHsuLi5zdGF0ZSwgaG92ZXI6IGZhbHNlfVxuICAgIGRlZmF1bHQ6XG4gICAgICByZXR1cm4gc3RhdGVcbiAgfVxufVxuXG5jb25zdCBTZWVrYmFyUmFpbCA9ICgpID0+ICcnXG5cbmNvbnN0IHNlZ21lbnRlZFRyYWNrU3R5bGUgPSB7XG4gIHBvc2l0aW9uOiAncmVsYXRpdmUnLFxuICBtYXJnaW46ICcwIDJweCcsXG4gIGhlaWdodDogJzI0cHgnLFxuICBib3JkZXJUb3A6ICdjYWxjKDEycHggLSAwLjNlbSkgc29saWQgdHJhbnNwYXJlbnQnLFxuICBib3JkZXJCb3R0b206ICdjYWxjKDEycHggLSAwLjNlbSkgc29saWQgdHJhbnNwYXJlbnQnLFxuICBiYWNrZ3JvdW5kOiAndHJhbnNwYXJlbnQnLFxuICAnJjpob3Zlcic6IHtcbiAgICB0cmFuc2Zvcm06ICdzY2FsZSgxLCAxLjUpJyxcbiAgfSxcbiAgJz4gZGl2Jzoge1xuICAgIGhlaWdodDogJzAuNmVtJyxcbiAgICBiYWNrZ3JvdW5kQ29sb3I6ICdyZ2JhKDI1NSwgMjU1LCAyNTUsIDAuMiknLFxuICAgICcmOm5vdCg6Zmlyc3Qtb2YtdHlwZSknOiB7XG4gICAgICBwb3NpdGlvbjogJ2Fic29sdXRlJyxcbiAgICAgIHRvcDogJzAnLFxuICAgIH0sXG4gIH0sXG4gICc+IGRpdjpudGgtb2YtdHlwZSgyKSc6IHtcbiAgICBiYWNrZ3JvdW5kQ29sb3I6ICdyZ2JhKDI1NSwgMjU1LCAyNTUsIDAuMyknLFxuICB9LFxuICAnPiBkaXY6bnRoLW9mLXR5cGUoMyknOiB7XG4gICAgYmFja2dyb3VuZENvbG9yOiAndmFyKC0tdGhlbWUtY29sb3IsIHJlZCknLFxuICB9LFxufVxuXG5jb25zdCBTZWVrYmFyVHJhY2sgPSAoe3NlZ21lbnRzLCBzdHlsZX0pID0+XG4gIHNlZ21lbnRzLm1hcCgoe2xlbmd0aCwgY3VycmVudCwgYnVmZmVyZWR9KSA9PiAoXG4gICAgPGRpdlxuICAgICAgY3NzPXtbc3R5bGUsIHNlZ21lbnRlZFRyYWNrU3R5bGVdfVxuICAgICAgc3R5bGU9e3tcbiAgICAgICAgZmxleDogYDAgJHtsZW5ndGggKiAxMDB9JWAsXG4gICAgICAgIHdpZHRoOiBgY2FsYygke2xlbmd0aCAqIDEwMH0lIC0gNHB4KWAsXG4gICAgICB9fVxuICAgID5cbiAgICAgIDxkaXYgc3R5bGU9e3t3aWR0aDogJzEwMCUnfX0gLz5cbiAgICAgIDxkaXYgc3R5bGU9e3t3aWR0aDogYCR7YnVmZmVyZWQgKiAxMDB9JWB9fSAvPlxuICAgICAgPGRpdiBzdHlsZT17e3dpZHRoOiBgJHtjdXJyZW50ICogMTAwfSVgfX0gLz5cbiAgICA8L2Rpdj5cbiAgKSlcblxuLy8gVE9ETyB1c2UgY2xhc3NOYW1lIGluc3RlYWQgb2YgY2xhc3NlcyA/XG5jb25zdCBTZWVrYmFyID0gKHtcbiAgc3R5bGUsXG4gIGNsYXNzZXMsXG4gIHN0YXJ0VGltZSA9IDAsXG4gIGN1cnJlbnRUaW1lLFxuICBidWZmZXJUaW1lLFxuICBkdXJhdGlvbixcbiAgY2hhcHRlcnMgPSBbXSxcbiAgbWFya3MsXG4gIHRpbWVEaXNwbGF5ID0gZmFsc2UsIC8vIFRPRE8gbW9yZSBzY2FsYWJsZSB3YXk/XG4gIG9uQ2hhbmdlLFxuICBvbkNoYW5nZUNvbW1pdHRlZCxcbiAgY2hpbGRyZW4sXG4gIG9uUG9pbnRlck1vdmUsXG4gIG9uUG9pbnRlckxlYXZlLFxuICAuLi5yZXN0XG59KSA9PiB7XG4gIGNvbnN0IFtwb2ludGVyU3RhdGUsIGRpc3BhdGNoUG9pbnRlcl0gPSB1c2VSZWR1Y2VyKHJlZHVjZVBvaW50ZXIsIHt9KVxuICBjb25zdCBwb2ludGVyQWN0aXZlID0gcG9pbnRlclN0YXRlLmhvdmVyIHx8IHBvaW50ZXJTdGF0ZS5mb2N1c2VkXG4gIC8vIHRvIHJlZmxlY3QgYm91bmRhcnkgd2hlbiBjb250YWluZXIgcmVzaXplZFxuICBjb25zdCB7b2JzZXJ2ZX0gPSB1c2VEaW1lbnNpb25zKClcbiAgY29uc3QgcmVmID0gdXNlUmVmKClcbiAgY29uc3QgcmVjdCA9IHJlZi5jdXJyZW50Py5nZXRCb3VuZGluZ0NsaWVudFJlY3QoKVxuICBjb25zdCBoYW5kbGVycyA9IG9uQ2hhbmdlQ29tbWl0dGVkICYmIHtcbiAgICBvblBvaW50ZXJNb3ZlOiAoZXZlbnQsIHt2YWx1ZSwgeH0pID0+IHtcbiAgICAgIG9uUG9pbnRlck1vdmU/LihldmVudCwge3ZhbHVlLCB4fSlcbiAgICAgIGRpc3BhdGNoUG9pbnRlcih7dHlwZTogJ21vdmUnLCB2YWx1ZSwgeH0pXG4gICAgfSxcbiAgICBvblBvaW50ZXJMZWF2ZTogZXZlbnQgPT4ge1xuICAgICAgb25Qb2ludGVyTGVhdmU/LihldmVudClcbiAgICAgIGRpc3BhdGNoUG9pbnRlcih7dHlwZTogJ2xlYXZlJ30pXG4gICAgfSxcbiAgICBvbkNoYW5nZTogKF8sIHt2YWx1ZX0pID0+IHtcbiAgICAgIG9uQ2hhbmdlPy4odmFsdWUpXG4gICAgICBkaXNwYXRjaFBvaW50ZXIoe3R5cGU6ICdjaGFuZ2UnLCB2YWx1ZX0pXG4gICAgfSxcbiAgICBvbkNoYW5nZUNvbW1pdHRlZDogKF8sIHt2YWx1ZX0pID0+IHtcbiAgICAgIGRpc3BhdGNoUG9pbnRlcih7dHlwZTogJ3JlbGVhc2UnLCB2YWx1ZX0pXG4gICAgICBvbkNoYW5nZUNvbW1pdHRlZD8uKHZhbHVlKVxuICAgIH0sXG4gIH1cbiAgY29uc3QgZW5kVGltZSA9IHN0YXJ0VGltZSArIGR1cmF0aW9uXG4gIGNvbnN0IHNlZ21lbnRzID0gZ2V0VGltZWxpbmVTZWdtZW50cyhjaGFwdGVycywge1xuICAgIHN0YXJ0VGltZSxcbiAgICBjdXJyZW50OiBwb2ludGVyU3RhdGUuZm9jdXNlZCA/IHBvaW50ZXJTdGF0ZS52YWx1ZSA6IGN1cnJlbnRUaW1lLFxuICAgIGJ1ZmZlcmVkOiBidWZmZXJUaW1lLFxuICAgIGR1cmF0aW9uLFxuICB9KVxuXG4gIHJldHVybiAhKGR1cmF0aW9uID4gMCkgPyAoXG4gICAgPGRpdiAvPlxuICApIDogKFxuICAgIDxkaXZcbiAgICAgIHJlZj17ZWxlbWVudCA9PiB7XG4gICAgICAgIG9ic2VydmUoZWxlbWVudClcbiAgICAgICAgcmVmLmN1cnJlbnQgPSBlbGVtZW50XG4gICAgICB9fVxuICAgICAgY3NzPXtbc2Vla2JhclN0eWxlLCBzdHlsZV19XG4gICAgICBzdHlsZT17XG4gICAgICAgIHJlY3QgJiYge1xuICAgICAgICAgICctLXNlZWtiYXItbGVmdCc6IGAke3JlY3QubGVmdH1weGAsXG4gICAgICAgICAgJy0tc2Vla2Jhci1yaWdodCc6IGAke3JlY3QucmlnaHR9cHhgLFxuICAgICAgICAgICctLXBvaW50ZXIteCc6IGAke3BvaW50ZXJTdGF0ZS54fXB4YCxcbiAgICAgICAgfVxuICAgICAgfVxuICAgID5cbiAgICAgIHshdGltZURpc3BsYXkgPyAoXG4gICAgICAgICcnXG4gICAgICApIDogcG9pbnRlckFjdGl2ZSA/IChcbiAgICAgICAgPEZvcm1hdHRlZFRpbWUgdGltZT17cG9pbnRlclN0YXRlLnZhbHVlfSAvPlxuICAgICAgKSA6IChcbiAgICAgICAgPEZvcm1hdHRlZFRpbWUgdGltZT17TWF0aC5taW4oTWF0aC5tYXgoMCwgY3VycmVudFRpbWUpLCBkdXJhdGlvbil9IC8+XG4gICAgICApfVxuICAgICAgPFNpbXBsZVNsaWRlclxuICAgICAgICBjc3M9e1tzbGlkZXJTdHlsZSwgdGltZURpc3BsYXkgJiYge21hcmdpbjogJzAgMWVtJ31dfVxuICAgICAgICBjbGFzc2VzPXtjbGFzc2VzfVxuICAgICAgICBkaXNhYmxlZD17IW9uQ2hhbmdlQ29tbWl0dGVkfVxuICAgICAgICAvLyBkaXNwbGF5IGZpbGxlZCB3aGVuIHNlZWsgaGFuZGxlciBpcyBub3QgcHJvdmlkZWQsIGZyb20gUGxheUJveSBiZWhhdmlvclxuICAgICAgICB2YWx1ZT17b25DaGFuZ2VDb21taXR0ZWQgPyBjdXJyZW50VGltZSA6IGVuZFRpbWV9XG4gICAgICAgIG1pbj17c3RhcnRUaW1lfVxuICAgICAgICBtYXg9e2VuZFRpbWV9XG4gICAgICAgIG1hcmtzPXttYXJrc31cbiAgICAgICAgey4uLmhhbmRsZXJzfVxuICAgICAgICB7Li4ucmVzdH1cbiAgICAgICAgc2xvdHM9e3tSYWlsOiBTZWVrYmFyUmFpbCwgVHJhY2s6IFNlZWtiYXJUcmFja319XG4gICAgICAgIHNsb3RQcm9wcz17e3RyYWNrOiB7c2VnbWVudHN9fX1cbiAgICAgIC8+XG4gICAgICB7dGltZURpc3BsYXkgJiYgPEZvcm1hdHRlZFRpbWUgdGltZT17ZHVyYXRpb259IC8+fVxuICAgICAge2NoaWxkcmVuICYmXG4gICAgICAgIFtdXG4gICAgICAgICAgLmNvbmNhdChjaGlsZHJlbilcbiAgICAgICAgICAubWFwKGNoaWxkID0+XG4gICAgICAgICAgICBjbG9uZUVsZW1lbnQoY2hpbGQsIHt0aW1lOiBwb2ludGVyQWN0aXZlICYmIHBvaW50ZXJTdGF0ZS52YWx1ZX0pXG4gICAgICAgICAgKX1cbiAgICA8L2Rpdj5cbiAgKVxufVxuXG5leHBvcnQgZGVmYXVsdCBTZWVrYmFyXG4iXX0= */"],
4276
+ css: [seekbarStyle, style, process.env.NODE_ENV === "production" ? "" : ";label:Seekbar;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIlNlZWtiYXIuanMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBMkpNIiwiZmlsZSI6IlNlZWtiYXIuanMiLCJzb3VyY2VzQ29udGVudCI6WyIvKiBAanN4SW1wb3J0U291cmNlIEBlbW90aW9uL3JlYWN0ICovXG5pbXBvcnQge3VzZVJlZHVjZXIsIHVzZVJlZiwgY2xvbmVFbGVtZW50fSBmcm9tICdyZWFjdCdcbmltcG9ydCB1c2VEaW1lbnNpb25zIGZyb20gJ3JlYWN0LWNvb2wtZGltZW5zaW9ucydcbmltcG9ydCBTaW1wbGVTbGlkZXIgZnJvbSAncGxheWVyVWkvU2ltcGxlU2xpZGVyJ1xuaW1wb3J0IHtGb3JtYXR0ZWRUaW1lfSBmcm9tICdjb250ZXh0L0kxOG4nXG5pbXBvcnQge2dldFRpbWVsaW5lU2VnbWVudHN9IGZyb20gJ3ByZW1pdW0vdGltZWxpbmUnXG5cbmNvbnN0IHNlZWtiYXJTdHlsZSA9IHtcbiAgcG9zaXRpb246ICdyZWxhdGl2ZScsXG4gIGRpc3BsYXk6ICdmbGV4JyxcbiAgYWxpZ25JdGVtczogJ2NlbnRlcicsXG4gIG1pbldpZHRoOiAnMCcsXG4gIGhlaWdodDogJzI0cHgnLFxuICBmb250U2l6ZTogJzc1JScsXG4gIGxldHRlclNwYWNpbmc6ICcxcHgnLFxuICBjb2xvcjogJyNmZmYnLFxufVxuXG5jb25zdCBzbGlkZXJTdHlsZSA9IHtcbiAgbWluV2lkdGg6ICcwJyxcbiAgZmxleDogMSxcbiAgJ0BtZWRpYSAoaG92ZXI6IGhvdmVyKSwgc2NyZWVuIGFuZCAoLW1zLWhpZ2gtY29udHJhc3Q6IGFjdGl2ZSksICgtbXMtaGlnaC1jb250cmFzdDogbm9uZSknOlxuICAgIHtcbiAgICAgICc+IGRpdjpsYXN0LW9mLXR5cGUnOiB7XG4gICAgICAgIG9wYWNpdHk6IDAsXG4gICAgICAgIHRyYW5zaXRpb246ICdvcGFjaXR5IDAuMnMgZWFzZS1vdXQnLFxuICAgICAgfSxcbiAgICB9LFxuICAnPiBkaXY6bGFzdC1vZi10eXBlJzoge1xuICAgIHdpZHRoOiAnMS4yZW0nLFxuICAgIGhlaWdodDogJzEuMmVtJyxcbiAgICBiYWNrZ3JvdW5kOiAndmFyKC0tdGhlbWUtY29sb3IsIHJlZCknLFxuICB9LFxuICAnJjpob3ZlciA+IGRpdjpsYXN0LW9mLXR5cGUnOiB7XG4gICAgb3BhY2l0eTogMSxcbiAgfSxcbn1cblxuY29uc3QgcmVkdWNlUG9pbnRlciA9IChzdGF0ZSwge3R5cGUsIHZhbHVlLCB4fSkgPT4ge1xuICBzd2l0Y2ggKHR5cGUpIHtcbiAgICBjYXNlICdtb3ZlJzpcbiAgICAgIHJldHVybiB7Li4uc3RhdGUsIGhvdmVyOiB0cnVlLCB2YWx1ZSwgeH1cbiAgICBjYXNlICdjaGFuZ2UnOlxuICAgICAgcmV0dXJuIHsuLi5zdGF0ZSwgZm9jdXNlZDogdHJ1ZSwgdmFsdWV9XG4gICAgY2FzZSAncmVsZWFzZSc6XG4gICAgICByZXR1cm4gey4uLnN0YXRlLCBmb2N1c2VkOiBmYWxzZSwgdmFsdWV9XG4gICAgY2FzZSAnbGVhdmUnOlxuICAgICAgcmV0dXJuIHsuLi5zdGF0ZSwgaG92ZXI6IGZhbHNlfVxuICAgIGRlZmF1bHQ6XG4gICAgICByZXR1cm4gc3RhdGVcbiAgfVxufVxuXG5jb25zdCBTZWVrYmFyUmFpbCA9ICgpID0+ICcnXG5cbmNvbnN0IHNlZ21lbnRlZFRyYWNrU3R5bGUgPSB7XG4gIGJveFNpemluZzogJ2JvcmRlci1ib3gnLFxuICBwb3NpdGlvbjogJ3JlbGF0aXZlJyxcbiAgbWFyZ2luOiAnMCAycHgnLFxuICBoZWlnaHQ6ICcyNHB4JyxcbiAgYm9yZGVyVG9wOiAnY2FsYygxMnB4IC0gMC4zZW0pIHNvbGlkIHRyYW5zcGFyZW50JyxcbiAgYm9yZGVyQm90dG9tOiAnY2FsYygxMnB4IC0gMC4zZW0pIHNvbGlkIHRyYW5zcGFyZW50JyxcbiAgYmFja2dyb3VuZDogJ3RyYW5zcGFyZW50JyxcbiAgJyY6aG92ZXInOiB7XG4gICAgdHJhbnNmb3JtOiAnc2NhbGUoMSwgMS41KScsXG4gIH0sXG4gICc+IGRpdic6IHtcbiAgICBoZWlnaHQ6ICcwLjZlbScsXG4gICAgYmFja2dyb3VuZENvbG9yOiAncmdiYSgyNTUsIDI1NSwgMjU1LCAwLjIpJyxcbiAgICAnJjpub3QoOmZpcnN0LW9mLXR5cGUpJzoge1xuICAgICAgcG9zaXRpb246ICdhYnNvbHV0ZScsXG4gICAgICB0b3A6ICcwJyxcbiAgICB9LFxuICB9LFxuICAnPiBkaXY6bnRoLW9mLXR5cGUoMiknOiB7XG4gICAgYmFja2dyb3VuZENvbG9yOiAncmdiYSgyNTUsIDI1NSwgMjU1LCAwLjMpJyxcbiAgfSxcbiAgJz4gZGl2Om50aC1vZi10eXBlKDMpJzoge1xuICAgIGJhY2tncm91bmRDb2xvcjogJ3ZhcigtLXRoZW1lLWNvbG9yLCByZWQpJyxcbiAgfSxcbn1cblxuY29uc3QgU2Vla2JhclRyYWNrID0gKHtzZWdtZW50cywgc3R5bGV9KSA9PlxuICBzZWdtZW50cy5tYXAoKHtsZW5ndGgsIGN1cnJlbnQsIGJ1ZmZlcmVkfSkgPT4gKFxuICAgIDxkaXZcbiAgICAgIGNzcz17W3N0eWxlLCBzZWdtZW50ZWRUcmFja1N0eWxlXX1cbiAgICAgIHN0eWxlPXt7XG4gICAgICAgIGZsZXg6IGAwICR7bGVuZ3RoICogMTAwfSVgLFxuICAgICAgICB3aWR0aDogYGNhbGMoJHtsZW5ndGggKiAxMDB9JSAtIDRweClgLFxuICAgICAgfX1cbiAgICA+XG4gICAgICA8ZGl2IHN0eWxlPXt7d2lkdGg6ICcxMDAlJ319IC8+XG4gICAgICA8ZGl2IHN0eWxlPXt7d2lkdGg6IGAke2J1ZmZlcmVkICogMTAwfSVgfX0gLz5cbiAgICAgIDxkaXYgc3R5bGU9e3t3aWR0aDogYCR7Y3VycmVudCAqIDEwMH0lYH19IC8+XG4gICAgPC9kaXY+XG4gICkpXG5cbi8vIFRPRE8gdXNlIGNsYXNzTmFtZSBpbnN0ZWFkIG9mIGNsYXNzZXMgP1xuY29uc3QgU2Vla2JhciA9ICh7XG4gIHN0eWxlLFxuICBjbGFzc2VzLFxuICBzdGFydFRpbWUgPSAwLFxuICBjdXJyZW50VGltZSxcbiAgYnVmZmVyVGltZSxcbiAgZHVyYXRpb24sXG4gIGNoYXB0ZXJzID0gW10sXG4gIG1hcmtzLFxuICB0aW1lRGlzcGxheSA9IGZhbHNlLCAvLyBUT0RPIG1vcmUgc2NhbGFibGUgd2F5P1xuICBvbkNoYW5nZSxcbiAgb25DaGFuZ2VDb21taXR0ZWQsXG4gIGNoaWxkcmVuLFxuICBvblBvaW50ZXJNb3ZlLFxuICBvblBvaW50ZXJMZWF2ZSxcbiAgLi4ucmVzdFxufSkgPT4ge1xuICBjb25zdCBbcG9pbnRlclN0YXRlLCBkaXNwYXRjaFBvaW50ZXJdID0gdXNlUmVkdWNlcihyZWR1Y2VQb2ludGVyLCB7fSlcbiAgY29uc3QgcG9pbnRlckFjdGl2ZSA9IHBvaW50ZXJTdGF0ZS5ob3ZlciB8fCBwb2ludGVyU3RhdGUuZm9jdXNlZFxuICAvLyB0byByZWZsZWN0IGJvdW5kYXJ5IHdoZW4gY29udGFpbmVyIHJlc2l6ZWRcbiAgY29uc3Qge29ic2VydmV9ID0gdXNlRGltZW5zaW9ucygpXG4gIGNvbnN0IHJlZiA9IHVzZVJlZigpXG4gIGNvbnN0IHJlY3QgPSByZWYuY3VycmVudD8uZ2V0Qm91bmRpbmdDbGllbnRSZWN0KClcbiAgY29uc3QgaGFuZGxlcnMgPSBvbkNoYW5nZUNvbW1pdHRlZCAmJiB7XG4gICAgb25Qb2ludGVyTW92ZTogKGV2ZW50LCB7dmFsdWUsIHh9KSA9PiB7XG4gICAgICBvblBvaW50ZXJNb3ZlPy4oZXZlbnQsIHt2YWx1ZSwgeH0pXG4gICAgICBkaXNwYXRjaFBvaW50ZXIoe3R5cGU6ICdtb3ZlJywgdmFsdWUsIHh9KVxuICAgIH0sXG4gICAgb25Qb2ludGVyTGVhdmU6IGV2ZW50ID0+IHtcbiAgICAgIG9uUG9pbnRlckxlYXZlPy4oZXZlbnQpXG4gICAgICBkaXNwYXRjaFBvaW50ZXIoe3R5cGU6ICdsZWF2ZSd9KVxuICAgIH0sXG4gICAgb25DaGFuZ2U6IChfLCB7dmFsdWV9KSA9PiB7XG4gICAgICBvbkNoYW5nZT8uKHZhbHVlKVxuICAgICAgZGlzcGF0Y2hQb2ludGVyKHt0eXBlOiAnY2hhbmdlJywgdmFsdWV9KVxuICAgIH0sXG4gICAgb25DaGFuZ2VDb21taXR0ZWQ6IChfLCB7dmFsdWV9KSA9PiB7XG4gICAgICBkaXNwYXRjaFBvaW50ZXIoe3R5cGU6ICdyZWxlYXNlJywgdmFsdWV9KVxuICAgICAgb25DaGFuZ2VDb21taXR0ZWQ/Lih2YWx1ZSlcbiAgICB9LFxuICB9XG4gIGNvbnN0IGVuZFRpbWUgPSBzdGFydFRpbWUgKyBkdXJhdGlvblxuICBjb25zdCBzZWdtZW50cyA9IGdldFRpbWVsaW5lU2VnbWVudHMoY2hhcHRlcnMsIHtcbiAgICBzdGFydFRpbWUsXG4gICAgY3VycmVudDogcG9pbnRlclN0YXRlLmZvY3VzZWQgPyBwb2ludGVyU3RhdGUudmFsdWUgOiBjdXJyZW50VGltZSxcbiAgICBidWZmZXJlZDogYnVmZmVyVGltZSxcbiAgICBkdXJhdGlvbixcbiAgfSlcblxuICByZXR1cm4gIShkdXJhdGlvbiA+IDApID8gKFxuICAgIDxkaXYgLz5cbiAgKSA6IChcbiAgICA8ZGl2XG4gICAgICByZWY9e2VsZW1lbnQgPT4ge1xuICAgICAgICBvYnNlcnZlKGVsZW1lbnQpXG4gICAgICAgIHJlZi5jdXJyZW50ID0gZWxlbWVudFxuICAgICAgfX1cbiAgICAgIGNzcz17W3NlZWtiYXJTdHlsZSwgc3R5bGVdfVxuICAgICAgc3R5bGU9e1xuICAgICAgICByZWN0ICYmIHtcbiAgICAgICAgICAnLS1zZWVrYmFyLWxlZnQnOiBgJHtyZWN0LmxlZnR9cHhgLFxuICAgICAgICAgICctLXNlZWtiYXItcmlnaHQnOiBgJHtyZWN0LnJpZ2h0fXB4YCxcbiAgICAgICAgICAnLS1wb2ludGVyLXgnOiBgJHtwb2ludGVyU3RhdGUueH1weGAsXG4gICAgICAgIH1cbiAgICAgIH1cbiAgICA+XG4gICAgICB7IXRpbWVEaXNwbGF5ID8gKFxuICAgICAgICAnJ1xuICAgICAgKSA6IHBvaW50ZXJBY3RpdmUgPyAoXG4gICAgICAgIDxGb3JtYXR0ZWRUaW1lIHRpbWU9e3BvaW50ZXJTdGF0ZS52YWx1ZX0gLz5cbiAgICAgICkgOiAoXG4gICAgICAgIDxGb3JtYXR0ZWRUaW1lIHRpbWU9e01hdGgubWluKE1hdGgubWF4KDAsIGN1cnJlbnRUaW1lKSwgZHVyYXRpb24pfSAvPlxuICAgICAgKX1cbiAgICAgIDxTaW1wbGVTbGlkZXJcbiAgICAgICAgY3NzPXtbc2xpZGVyU3R5bGUsIHRpbWVEaXNwbGF5ICYmIHttYXJnaW46ICcwIDFlbSd9XX1cbiAgICAgICAgY2xhc3Nlcz17Y2xhc3Nlc31cbiAgICAgICAgZGlzYWJsZWQ9eyFvbkNoYW5nZUNvbW1pdHRlZH1cbiAgICAgICAgLy8gZGlzcGxheSBmaWxsZWQgd2hlbiBzZWVrIGhhbmRsZXIgaXMgbm90IHByb3ZpZGVkLCBmcm9tIFBsYXlCb3kgYmVoYXZpb3JcbiAgICAgICAgdmFsdWU9e29uQ2hhbmdlQ29tbWl0dGVkID8gY3VycmVudFRpbWUgOiBlbmRUaW1lfVxuICAgICAgICBtaW49e3N0YXJ0VGltZX1cbiAgICAgICAgbWF4PXtlbmRUaW1lfVxuICAgICAgICBtYXJrcz17bWFya3N9XG4gICAgICAgIHsuLi5oYW5kbGVyc31cbiAgICAgICAgey4uLnJlc3R9XG4gICAgICAgIHNsb3RzPXt7UmFpbDogU2Vla2JhclJhaWwsIFRyYWNrOiBTZWVrYmFyVHJhY2t9fVxuICAgICAgICBzbG90UHJvcHM9e3t0cmFjazoge3NlZ21lbnRzfX19XG4gICAgICAvPlxuICAgICAge3RpbWVEaXNwbGF5ICYmIDxGb3JtYXR0ZWRUaW1lIHRpbWU9e2R1cmF0aW9ufSAvPn1cbiAgICAgIHtjaGlsZHJlbiAmJlxuICAgICAgICBbXVxuICAgICAgICAgIC5jb25jYXQoY2hpbGRyZW4pXG4gICAgICAgICAgLm1hcChjaGlsZCA9PlxuICAgICAgICAgICAgY2xvbmVFbGVtZW50KGNoaWxkLCB7dGltZTogcG9pbnRlckFjdGl2ZSAmJiBwb2ludGVyU3RhdGUudmFsdWV9KVxuICAgICAgICAgICl9XG4gICAgPC9kaXY+XG4gIClcbn1cblxuZXhwb3J0IGRlZmF1bHQgU2Vla2JhclxuIl19 */"],
4140
4277
  style: rect && {
4141
4278
  '--seekbar-left': `${rect.left}px`,
4142
4279
  '--seekbar-right': `${rect.right}px`,
@@ -4149,7 +4286,7 @@ const Seekbar = ({
4149
4286
  }), jsx$1(SimpleSlider, {
4150
4287
  css: [sliderStyle, timeDisplay && {
4151
4288
  margin: '0 1em'
4152
- }, process.env.NODE_ENV === "production" ? "" : ";label:Seekbar;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIlNlZWtiYXIuanMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBMktRIiwiZmlsZSI6IlNlZWtiYXIuanMiLCJzb3VyY2VzQ29udGVudCI6WyIvKiBAanN4SW1wb3J0U291cmNlIEBlbW90aW9uL3JlYWN0ICovXG5pbXBvcnQge3VzZVJlZHVjZXIsIHVzZVJlZiwgY2xvbmVFbGVtZW50fSBmcm9tICdyZWFjdCdcbmltcG9ydCB1c2VEaW1lbnNpb25zIGZyb20gJ3JlYWN0LWNvb2wtZGltZW5zaW9ucydcbmltcG9ydCBTaW1wbGVTbGlkZXIgZnJvbSAncGxheWVyVWkvU2ltcGxlU2xpZGVyJ1xuaW1wb3J0IHtGb3JtYXR0ZWRUaW1lfSBmcm9tICdjb250ZXh0L0kxOG4nXG5pbXBvcnQge2dldFRpbWVsaW5lU2VnbWVudHN9IGZyb20gJ3ByZW1pdW0vdGltZWxpbmUnXG5cbmNvbnN0IHNlZWtiYXJTdHlsZSA9IHtcbiAgcG9zaXRpb246ICdyZWxhdGl2ZScsXG4gIGRpc3BsYXk6ICdmbGV4JyxcbiAgYWxpZ25JdGVtczogJ2NlbnRlcicsXG4gIG1pbldpZHRoOiAnMCcsXG4gIGhlaWdodDogJzI0cHgnLFxuICBmb250U2l6ZTogJzc1JScsXG4gIGxldHRlclNwYWNpbmc6ICcxcHgnLFxuICBjb2xvcjogJyNmZmYnLFxufVxuXG5jb25zdCBzbGlkZXJTdHlsZSA9IHtcbiAgbWluV2lkdGg6ICcwJyxcbiAgZmxleDogMSxcbiAgJ0BtZWRpYSAoaG92ZXI6IGhvdmVyKSwgc2NyZWVuIGFuZCAoLW1zLWhpZ2gtY29udHJhc3Q6IGFjdGl2ZSksICgtbXMtaGlnaC1jb250cmFzdDogbm9uZSknOlxuICAgIHtcbiAgICAgICc+IGRpdjpsYXN0LW9mLXR5cGUnOiB7XG4gICAgICAgIG9wYWNpdHk6IDAsXG4gICAgICAgIHRyYW5zaXRpb246ICdvcGFjaXR5IDAuMnMgZWFzZS1vdXQnLFxuICAgICAgfSxcbiAgICB9LFxuICAnPiBkaXY6bGFzdC1vZi10eXBlJzoge1xuICAgIHdpZHRoOiAnMS4yZW0nLFxuICAgIGhlaWdodDogJzEuMmVtJyxcbiAgICBiYWNrZ3JvdW5kOiAndmFyKC0tdGhlbWUtY29sb3IsIHJlZCknLFxuICB9LFxuICAnJjpob3ZlciA+IGRpdjpsYXN0LW9mLXR5cGUnOiB7XG4gICAgb3BhY2l0eTogMSxcbiAgfSxcbn1cblxuY29uc3QgcmVkdWNlUG9pbnRlciA9IChzdGF0ZSwge3R5cGUsIHZhbHVlLCB4fSkgPT4ge1xuICBzd2l0Y2ggKHR5cGUpIHtcbiAgICBjYXNlICdtb3ZlJzpcbiAgICAgIHJldHVybiB7Li4uc3RhdGUsIGhvdmVyOiB0cnVlLCB2YWx1ZSwgeH1cbiAgICBjYXNlICdjaGFuZ2UnOlxuICAgICAgcmV0dXJuIHsuLi5zdGF0ZSwgZm9jdXNlZDogdHJ1ZSwgdmFsdWV9XG4gICAgY2FzZSAncmVsZWFzZSc6XG4gICAgICByZXR1cm4gey4uLnN0YXRlLCBmb2N1c2VkOiBmYWxzZSwgdmFsdWV9XG4gICAgY2FzZSAnbGVhdmUnOlxuICAgICAgcmV0dXJuIHsuLi5zdGF0ZSwgaG92ZXI6IGZhbHNlfVxuICAgIGRlZmF1bHQ6XG4gICAgICByZXR1cm4gc3RhdGVcbiAgfVxufVxuXG5jb25zdCBTZWVrYmFyUmFpbCA9ICgpID0+ICcnXG5cbmNvbnN0IHNlZ21lbnRlZFRyYWNrU3R5bGUgPSB7XG4gIHBvc2l0aW9uOiAncmVsYXRpdmUnLFxuICBtYXJnaW46ICcwIDJweCcsXG4gIGhlaWdodDogJzI0cHgnLFxuICBib3JkZXJUb3A6ICdjYWxjKDEycHggLSAwLjNlbSkgc29saWQgdHJhbnNwYXJlbnQnLFxuICBib3JkZXJCb3R0b206ICdjYWxjKDEycHggLSAwLjNlbSkgc29saWQgdHJhbnNwYXJlbnQnLFxuICBiYWNrZ3JvdW5kOiAndHJhbnNwYXJlbnQnLFxuICAnJjpob3Zlcic6IHtcbiAgICB0cmFuc2Zvcm06ICdzY2FsZSgxLCAxLjUpJyxcbiAgfSxcbiAgJz4gZGl2Jzoge1xuICAgIGhlaWdodDogJzAuNmVtJyxcbiAgICBiYWNrZ3JvdW5kQ29sb3I6ICdyZ2JhKDI1NSwgMjU1LCAyNTUsIDAuMiknLFxuICAgICcmOm5vdCg6Zmlyc3Qtb2YtdHlwZSknOiB7XG4gICAgICBwb3NpdGlvbjogJ2Fic29sdXRlJyxcbiAgICAgIHRvcDogJzAnLFxuICAgIH0sXG4gIH0sXG4gICc+IGRpdjpudGgtb2YtdHlwZSgyKSc6IHtcbiAgICBiYWNrZ3JvdW5kQ29sb3I6ICdyZ2JhKDI1NSwgMjU1LCAyNTUsIDAuMyknLFxuICB9LFxuICAnPiBkaXY6bnRoLW9mLXR5cGUoMyknOiB7XG4gICAgYmFja2dyb3VuZENvbG9yOiAndmFyKC0tdGhlbWUtY29sb3IsIHJlZCknLFxuICB9LFxufVxuXG5jb25zdCBTZWVrYmFyVHJhY2sgPSAoe3NlZ21lbnRzLCBzdHlsZX0pID0+XG4gIHNlZ21lbnRzLm1hcCgoe2xlbmd0aCwgY3VycmVudCwgYnVmZmVyZWR9KSA9PiAoXG4gICAgPGRpdlxuICAgICAgY3NzPXtbc3R5bGUsIHNlZ21lbnRlZFRyYWNrU3R5bGVdfVxuICAgICAgc3R5bGU9e3tcbiAgICAgICAgZmxleDogYDAgJHtsZW5ndGggKiAxMDB9JWAsXG4gICAgICAgIHdpZHRoOiBgY2FsYygke2xlbmd0aCAqIDEwMH0lIC0gNHB4KWAsXG4gICAgICB9fVxuICAgID5cbiAgICAgIDxkaXYgc3R5bGU9e3t3aWR0aDogJzEwMCUnfX0gLz5cbiAgICAgIDxkaXYgc3R5bGU9e3t3aWR0aDogYCR7YnVmZmVyZWQgKiAxMDB9JWB9fSAvPlxuICAgICAgPGRpdiBzdHlsZT17e3dpZHRoOiBgJHtjdXJyZW50ICogMTAwfSVgfX0gLz5cbiAgICA8L2Rpdj5cbiAgKSlcblxuLy8gVE9ETyB1c2UgY2xhc3NOYW1lIGluc3RlYWQgb2YgY2xhc3NlcyA/XG5jb25zdCBTZWVrYmFyID0gKHtcbiAgc3R5bGUsXG4gIGNsYXNzZXMsXG4gIHN0YXJ0VGltZSA9IDAsXG4gIGN1cnJlbnRUaW1lLFxuICBidWZmZXJUaW1lLFxuICBkdXJhdGlvbixcbiAgY2hhcHRlcnMgPSBbXSxcbiAgbWFya3MsXG4gIHRpbWVEaXNwbGF5ID0gZmFsc2UsIC8vIFRPRE8gbW9yZSBzY2FsYWJsZSB3YXk/XG4gIG9uQ2hhbmdlLFxuICBvbkNoYW5nZUNvbW1pdHRlZCxcbiAgY2hpbGRyZW4sXG4gIG9uUG9pbnRlck1vdmUsXG4gIG9uUG9pbnRlckxlYXZlLFxuICAuLi5yZXN0XG59KSA9PiB7XG4gIGNvbnN0IFtwb2ludGVyU3RhdGUsIGRpc3BhdGNoUG9pbnRlcl0gPSB1c2VSZWR1Y2VyKHJlZHVjZVBvaW50ZXIsIHt9KVxuICBjb25zdCBwb2ludGVyQWN0aXZlID0gcG9pbnRlclN0YXRlLmhvdmVyIHx8IHBvaW50ZXJTdGF0ZS5mb2N1c2VkXG4gIC8vIHRvIHJlZmxlY3QgYm91bmRhcnkgd2hlbiBjb250YWluZXIgcmVzaXplZFxuICBjb25zdCB7b2JzZXJ2ZX0gPSB1c2VEaW1lbnNpb25zKClcbiAgY29uc3QgcmVmID0gdXNlUmVmKClcbiAgY29uc3QgcmVjdCA9IHJlZi5jdXJyZW50Py5nZXRCb3VuZGluZ0NsaWVudFJlY3QoKVxuICBjb25zdCBoYW5kbGVycyA9IG9uQ2hhbmdlQ29tbWl0dGVkICYmIHtcbiAgICBvblBvaW50ZXJNb3ZlOiAoZXZlbnQsIHt2YWx1ZSwgeH0pID0+IHtcbiAgICAgIG9uUG9pbnRlck1vdmU/LihldmVudCwge3ZhbHVlLCB4fSlcbiAgICAgIGRpc3BhdGNoUG9pbnRlcih7dHlwZTogJ21vdmUnLCB2YWx1ZSwgeH0pXG4gICAgfSxcbiAgICBvblBvaW50ZXJMZWF2ZTogZXZlbnQgPT4ge1xuICAgICAgb25Qb2ludGVyTGVhdmU/LihldmVudClcbiAgICAgIGRpc3BhdGNoUG9pbnRlcih7dHlwZTogJ2xlYXZlJ30pXG4gICAgfSxcbiAgICBvbkNoYW5nZTogKF8sIHt2YWx1ZX0pID0+IHtcbiAgICAgIG9uQ2hhbmdlPy4odmFsdWUpXG4gICAgICBkaXNwYXRjaFBvaW50ZXIoe3R5cGU6ICdjaGFuZ2UnLCB2YWx1ZX0pXG4gICAgfSxcbiAgICBvbkNoYW5nZUNvbW1pdHRlZDogKF8sIHt2YWx1ZX0pID0+IHtcbiAgICAgIGRpc3BhdGNoUG9pbnRlcih7dHlwZTogJ3JlbGVhc2UnLCB2YWx1ZX0pXG4gICAgICBvbkNoYW5nZUNvbW1pdHRlZD8uKHZhbHVlKVxuICAgIH0sXG4gIH1cbiAgY29uc3QgZW5kVGltZSA9IHN0YXJ0VGltZSArIGR1cmF0aW9uXG4gIGNvbnN0IHNlZ21lbnRzID0gZ2V0VGltZWxpbmVTZWdtZW50cyhjaGFwdGVycywge1xuICAgIHN0YXJ0VGltZSxcbiAgICBjdXJyZW50OiBwb2ludGVyU3RhdGUuZm9jdXNlZCA/IHBvaW50ZXJTdGF0ZS52YWx1ZSA6IGN1cnJlbnRUaW1lLFxuICAgIGJ1ZmZlcmVkOiBidWZmZXJUaW1lLFxuICAgIGR1cmF0aW9uLFxuICB9KVxuXG4gIHJldHVybiAhKGR1cmF0aW9uID4gMCkgPyAoXG4gICAgPGRpdiAvPlxuICApIDogKFxuICAgIDxkaXZcbiAgICAgIHJlZj17ZWxlbWVudCA9PiB7XG4gICAgICAgIG9ic2VydmUoZWxlbWVudClcbiAgICAgICAgcmVmLmN1cnJlbnQgPSBlbGVtZW50XG4gICAgICB9fVxuICAgICAgY3NzPXtbc2Vla2JhclN0eWxlLCBzdHlsZV19XG4gICAgICBzdHlsZT17XG4gICAgICAgIHJlY3QgJiYge1xuICAgICAgICAgICctLXNlZWtiYXItbGVmdCc6IGAke3JlY3QubGVmdH1weGAsXG4gICAgICAgICAgJy0tc2Vla2Jhci1yaWdodCc6IGAke3JlY3QucmlnaHR9cHhgLFxuICAgICAgICAgICctLXBvaW50ZXIteCc6IGAke3BvaW50ZXJTdGF0ZS54fXB4YCxcbiAgICAgICAgfVxuICAgICAgfVxuICAgID5cbiAgICAgIHshdGltZURpc3BsYXkgPyAoXG4gICAgICAgICcnXG4gICAgICApIDogcG9pbnRlckFjdGl2ZSA/IChcbiAgICAgICAgPEZvcm1hdHRlZFRpbWUgdGltZT17cG9pbnRlclN0YXRlLnZhbHVlfSAvPlxuICAgICAgKSA6IChcbiAgICAgICAgPEZvcm1hdHRlZFRpbWUgdGltZT17TWF0aC5taW4oTWF0aC5tYXgoMCwgY3VycmVudFRpbWUpLCBkdXJhdGlvbil9IC8+XG4gICAgICApfVxuICAgICAgPFNpbXBsZVNsaWRlclxuICAgICAgICBjc3M9e1tzbGlkZXJTdHlsZSwgdGltZURpc3BsYXkgJiYge21hcmdpbjogJzAgMWVtJ31dfVxuICAgICAgICBjbGFzc2VzPXtjbGFzc2VzfVxuICAgICAgICBkaXNhYmxlZD17IW9uQ2hhbmdlQ29tbWl0dGVkfVxuICAgICAgICAvLyBkaXNwbGF5IGZpbGxlZCB3aGVuIHNlZWsgaGFuZGxlciBpcyBub3QgcHJvdmlkZWQsIGZyb20gUGxheUJveSBiZWhhdmlvclxuICAgICAgICB2YWx1ZT17b25DaGFuZ2VDb21taXR0ZWQgPyBjdXJyZW50VGltZSA6IGVuZFRpbWV9XG4gICAgICAgIG1pbj17c3RhcnRUaW1lfVxuICAgICAgICBtYXg9e2VuZFRpbWV9XG4gICAgICAgIG1hcmtzPXttYXJrc31cbiAgICAgICAgey4uLmhhbmRsZXJzfVxuICAgICAgICB7Li4ucmVzdH1cbiAgICAgICAgc2xvdHM9e3tSYWlsOiBTZWVrYmFyUmFpbCwgVHJhY2s6IFNlZWtiYXJUcmFja319XG4gICAgICAgIHNsb3RQcm9wcz17e3RyYWNrOiB7c2VnbWVudHN9fX1cbiAgICAgIC8+XG4gICAgICB7dGltZURpc3BsYXkgJiYgPEZvcm1hdHRlZFRpbWUgdGltZT17ZHVyYXRpb259IC8+fVxuICAgICAge2NoaWxkcmVuICYmXG4gICAgICAgIFtdXG4gICAgICAgICAgLmNvbmNhdChjaGlsZHJlbilcbiAgICAgICAgICAubWFwKGNoaWxkID0+XG4gICAgICAgICAgICBjbG9uZUVsZW1lbnQoY2hpbGQsIHt0aW1lOiBwb2ludGVyQWN0aXZlICYmIHBvaW50ZXJTdGF0ZS52YWx1ZX0pXG4gICAgICAgICAgKX1cbiAgICA8L2Rpdj5cbiAgKVxufVxuXG5leHBvcnQgZGVmYXVsdCBTZWVrYmFyXG4iXX0= */"],
4289
+ }, process.env.NODE_ENV === "production" ? "" : ";label:Seekbar;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIlNlZWtiYXIuanMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBNEtRIiwiZmlsZSI6IlNlZWtiYXIuanMiLCJzb3VyY2VzQ29udGVudCI6WyIvKiBAanN4SW1wb3J0U291cmNlIEBlbW90aW9uL3JlYWN0ICovXG5pbXBvcnQge3VzZVJlZHVjZXIsIHVzZVJlZiwgY2xvbmVFbGVtZW50fSBmcm9tICdyZWFjdCdcbmltcG9ydCB1c2VEaW1lbnNpb25zIGZyb20gJ3JlYWN0LWNvb2wtZGltZW5zaW9ucydcbmltcG9ydCBTaW1wbGVTbGlkZXIgZnJvbSAncGxheWVyVWkvU2ltcGxlU2xpZGVyJ1xuaW1wb3J0IHtGb3JtYXR0ZWRUaW1lfSBmcm9tICdjb250ZXh0L0kxOG4nXG5pbXBvcnQge2dldFRpbWVsaW5lU2VnbWVudHN9IGZyb20gJ3ByZW1pdW0vdGltZWxpbmUnXG5cbmNvbnN0IHNlZWtiYXJTdHlsZSA9IHtcbiAgcG9zaXRpb246ICdyZWxhdGl2ZScsXG4gIGRpc3BsYXk6ICdmbGV4JyxcbiAgYWxpZ25JdGVtczogJ2NlbnRlcicsXG4gIG1pbldpZHRoOiAnMCcsXG4gIGhlaWdodDogJzI0cHgnLFxuICBmb250U2l6ZTogJzc1JScsXG4gIGxldHRlclNwYWNpbmc6ICcxcHgnLFxuICBjb2xvcjogJyNmZmYnLFxufVxuXG5jb25zdCBzbGlkZXJTdHlsZSA9IHtcbiAgbWluV2lkdGg6ICcwJyxcbiAgZmxleDogMSxcbiAgJ0BtZWRpYSAoaG92ZXI6IGhvdmVyKSwgc2NyZWVuIGFuZCAoLW1zLWhpZ2gtY29udHJhc3Q6IGFjdGl2ZSksICgtbXMtaGlnaC1jb250cmFzdDogbm9uZSknOlxuICAgIHtcbiAgICAgICc+IGRpdjpsYXN0LW9mLXR5cGUnOiB7XG4gICAgICAgIG9wYWNpdHk6IDAsXG4gICAgICAgIHRyYW5zaXRpb246ICdvcGFjaXR5IDAuMnMgZWFzZS1vdXQnLFxuICAgICAgfSxcbiAgICB9LFxuICAnPiBkaXY6bGFzdC1vZi10eXBlJzoge1xuICAgIHdpZHRoOiAnMS4yZW0nLFxuICAgIGhlaWdodDogJzEuMmVtJyxcbiAgICBiYWNrZ3JvdW5kOiAndmFyKC0tdGhlbWUtY29sb3IsIHJlZCknLFxuICB9LFxuICAnJjpob3ZlciA+IGRpdjpsYXN0LW9mLXR5cGUnOiB7XG4gICAgb3BhY2l0eTogMSxcbiAgfSxcbn1cblxuY29uc3QgcmVkdWNlUG9pbnRlciA9IChzdGF0ZSwge3R5cGUsIHZhbHVlLCB4fSkgPT4ge1xuICBzd2l0Y2ggKHR5cGUpIHtcbiAgICBjYXNlICdtb3ZlJzpcbiAgICAgIHJldHVybiB7Li4uc3RhdGUsIGhvdmVyOiB0cnVlLCB2YWx1ZSwgeH1cbiAgICBjYXNlICdjaGFuZ2UnOlxuICAgICAgcmV0dXJuIHsuLi5zdGF0ZSwgZm9jdXNlZDogdHJ1ZSwgdmFsdWV9XG4gICAgY2FzZSAncmVsZWFzZSc6XG4gICAgICByZXR1cm4gey4uLnN0YXRlLCBmb2N1c2VkOiBmYWxzZSwgdmFsdWV9XG4gICAgY2FzZSAnbGVhdmUnOlxuICAgICAgcmV0dXJuIHsuLi5zdGF0ZSwgaG92ZXI6IGZhbHNlfVxuICAgIGRlZmF1bHQ6XG4gICAgICByZXR1cm4gc3RhdGVcbiAgfVxufVxuXG5jb25zdCBTZWVrYmFyUmFpbCA9ICgpID0+ICcnXG5cbmNvbnN0IHNlZ21lbnRlZFRyYWNrU3R5bGUgPSB7XG4gIGJveFNpemluZzogJ2JvcmRlci1ib3gnLFxuICBwb3NpdGlvbjogJ3JlbGF0aXZlJyxcbiAgbWFyZ2luOiAnMCAycHgnLFxuICBoZWlnaHQ6ICcyNHB4JyxcbiAgYm9yZGVyVG9wOiAnY2FsYygxMnB4IC0gMC4zZW0pIHNvbGlkIHRyYW5zcGFyZW50JyxcbiAgYm9yZGVyQm90dG9tOiAnY2FsYygxMnB4IC0gMC4zZW0pIHNvbGlkIHRyYW5zcGFyZW50JyxcbiAgYmFja2dyb3VuZDogJ3RyYW5zcGFyZW50JyxcbiAgJyY6aG92ZXInOiB7XG4gICAgdHJhbnNmb3JtOiAnc2NhbGUoMSwgMS41KScsXG4gIH0sXG4gICc+IGRpdic6IHtcbiAgICBoZWlnaHQ6ICcwLjZlbScsXG4gICAgYmFja2dyb3VuZENvbG9yOiAncmdiYSgyNTUsIDI1NSwgMjU1LCAwLjIpJyxcbiAgICAnJjpub3QoOmZpcnN0LW9mLXR5cGUpJzoge1xuICAgICAgcG9zaXRpb246ICdhYnNvbHV0ZScsXG4gICAgICB0b3A6ICcwJyxcbiAgICB9LFxuICB9LFxuICAnPiBkaXY6bnRoLW9mLXR5cGUoMiknOiB7XG4gICAgYmFja2dyb3VuZENvbG9yOiAncmdiYSgyNTUsIDI1NSwgMjU1LCAwLjMpJyxcbiAgfSxcbiAgJz4gZGl2Om50aC1vZi10eXBlKDMpJzoge1xuICAgIGJhY2tncm91bmRDb2xvcjogJ3ZhcigtLXRoZW1lLWNvbG9yLCByZWQpJyxcbiAgfSxcbn1cblxuY29uc3QgU2Vla2JhclRyYWNrID0gKHtzZWdtZW50cywgc3R5bGV9KSA9PlxuICBzZWdtZW50cy5tYXAoKHtsZW5ndGgsIGN1cnJlbnQsIGJ1ZmZlcmVkfSkgPT4gKFxuICAgIDxkaXZcbiAgICAgIGNzcz17W3N0eWxlLCBzZWdtZW50ZWRUcmFja1N0eWxlXX1cbiAgICAgIHN0eWxlPXt7XG4gICAgICAgIGZsZXg6IGAwICR7bGVuZ3RoICogMTAwfSVgLFxuICAgICAgICB3aWR0aDogYGNhbGMoJHtsZW5ndGggKiAxMDB9JSAtIDRweClgLFxuICAgICAgfX1cbiAgICA+XG4gICAgICA8ZGl2IHN0eWxlPXt7d2lkdGg6ICcxMDAlJ319IC8+XG4gICAgICA8ZGl2IHN0eWxlPXt7d2lkdGg6IGAke2J1ZmZlcmVkICogMTAwfSVgfX0gLz5cbiAgICAgIDxkaXYgc3R5bGU9e3t3aWR0aDogYCR7Y3VycmVudCAqIDEwMH0lYH19IC8+XG4gICAgPC9kaXY+XG4gICkpXG5cbi8vIFRPRE8gdXNlIGNsYXNzTmFtZSBpbnN0ZWFkIG9mIGNsYXNzZXMgP1xuY29uc3QgU2Vla2JhciA9ICh7XG4gIHN0eWxlLFxuICBjbGFzc2VzLFxuICBzdGFydFRpbWUgPSAwLFxuICBjdXJyZW50VGltZSxcbiAgYnVmZmVyVGltZSxcbiAgZHVyYXRpb24sXG4gIGNoYXB0ZXJzID0gW10sXG4gIG1hcmtzLFxuICB0aW1lRGlzcGxheSA9IGZhbHNlLCAvLyBUT0RPIG1vcmUgc2NhbGFibGUgd2F5P1xuICBvbkNoYW5nZSxcbiAgb25DaGFuZ2VDb21taXR0ZWQsXG4gIGNoaWxkcmVuLFxuICBvblBvaW50ZXJNb3ZlLFxuICBvblBvaW50ZXJMZWF2ZSxcbiAgLi4ucmVzdFxufSkgPT4ge1xuICBjb25zdCBbcG9pbnRlclN0YXRlLCBkaXNwYXRjaFBvaW50ZXJdID0gdXNlUmVkdWNlcihyZWR1Y2VQb2ludGVyLCB7fSlcbiAgY29uc3QgcG9pbnRlckFjdGl2ZSA9IHBvaW50ZXJTdGF0ZS5ob3ZlciB8fCBwb2ludGVyU3RhdGUuZm9jdXNlZFxuICAvLyB0byByZWZsZWN0IGJvdW5kYXJ5IHdoZW4gY29udGFpbmVyIHJlc2l6ZWRcbiAgY29uc3Qge29ic2VydmV9ID0gdXNlRGltZW5zaW9ucygpXG4gIGNvbnN0IHJlZiA9IHVzZVJlZigpXG4gIGNvbnN0IHJlY3QgPSByZWYuY3VycmVudD8uZ2V0Qm91bmRpbmdDbGllbnRSZWN0KClcbiAgY29uc3QgaGFuZGxlcnMgPSBvbkNoYW5nZUNvbW1pdHRlZCAmJiB7XG4gICAgb25Qb2ludGVyTW92ZTogKGV2ZW50LCB7dmFsdWUsIHh9KSA9PiB7XG4gICAgICBvblBvaW50ZXJNb3ZlPy4oZXZlbnQsIHt2YWx1ZSwgeH0pXG4gICAgICBkaXNwYXRjaFBvaW50ZXIoe3R5cGU6ICdtb3ZlJywgdmFsdWUsIHh9KVxuICAgIH0sXG4gICAgb25Qb2ludGVyTGVhdmU6IGV2ZW50ID0+IHtcbiAgICAgIG9uUG9pbnRlckxlYXZlPy4oZXZlbnQpXG4gICAgICBkaXNwYXRjaFBvaW50ZXIoe3R5cGU6ICdsZWF2ZSd9KVxuICAgIH0sXG4gICAgb25DaGFuZ2U6IChfLCB7dmFsdWV9KSA9PiB7XG4gICAgICBvbkNoYW5nZT8uKHZhbHVlKVxuICAgICAgZGlzcGF0Y2hQb2ludGVyKHt0eXBlOiAnY2hhbmdlJywgdmFsdWV9KVxuICAgIH0sXG4gICAgb25DaGFuZ2VDb21taXR0ZWQ6IChfLCB7dmFsdWV9KSA9PiB7XG4gICAgICBkaXNwYXRjaFBvaW50ZXIoe3R5cGU6ICdyZWxlYXNlJywgdmFsdWV9KVxuICAgICAgb25DaGFuZ2VDb21taXR0ZWQ/Lih2YWx1ZSlcbiAgICB9LFxuICB9XG4gIGNvbnN0IGVuZFRpbWUgPSBzdGFydFRpbWUgKyBkdXJhdGlvblxuICBjb25zdCBzZWdtZW50cyA9IGdldFRpbWVsaW5lU2VnbWVudHMoY2hhcHRlcnMsIHtcbiAgICBzdGFydFRpbWUsXG4gICAgY3VycmVudDogcG9pbnRlclN0YXRlLmZvY3VzZWQgPyBwb2ludGVyU3RhdGUudmFsdWUgOiBjdXJyZW50VGltZSxcbiAgICBidWZmZXJlZDogYnVmZmVyVGltZSxcbiAgICBkdXJhdGlvbixcbiAgfSlcblxuICByZXR1cm4gIShkdXJhdGlvbiA+IDApID8gKFxuICAgIDxkaXYgLz5cbiAgKSA6IChcbiAgICA8ZGl2XG4gICAgICByZWY9e2VsZW1lbnQgPT4ge1xuICAgICAgICBvYnNlcnZlKGVsZW1lbnQpXG4gICAgICAgIHJlZi5jdXJyZW50ID0gZWxlbWVudFxuICAgICAgfX1cbiAgICAgIGNzcz17W3NlZWtiYXJTdHlsZSwgc3R5bGVdfVxuICAgICAgc3R5bGU9e1xuICAgICAgICByZWN0ICYmIHtcbiAgICAgICAgICAnLS1zZWVrYmFyLWxlZnQnOiBgJHtyZWN0LmxlZnR9cHhgLFxuICAgICAgICAgICctLXNlZWtiYXItcmlnaHQnOiBgJHtyZWN0LnJpZ2h0fXB4YCxcbiAgICAgICAgICAnLS1wb2ludGVyLXgnOiBgJHtwb2ludGVyU3RhdGUueH1weGAsXG4gICAgICAgIH1cbiAgICAgIH1cbiAgICA+XG4gICAgICB7IXRpbWVEaXNwbGF5ID8gKFxuICAgICAgICAnJ1xuICAgICAgKSA6IHBvaW50ZXJBY3RpdmUgPyAoXG4gICAgICAgIDxGb3JtYXR0ZWRUaW1lIHRpbWU9e3BvaW50ZXJTdGF0ZS52YWx1ZX0gLz5cbiAgICAgICkgOiAoXG4gICAgICAgIDxGb3JtYXR0ZWRUaW1lIHRpbWU9e01hdGgubWluKE1hdGgubWF4KDAsIGN1cnJlbnRUaW1lKSwgZHVyYXRpb24pfSAvPlxuICAgICAgKX1cbiAgICAgIDxTaW1wbGVTbGlkZXJcbiAgICAgICAgY3NzPXtbc2xpZGVyU3R5bGUsIHRpbWVEaXNwbGF5ICYmIHttYXJnaW46ICcwIDFlbSd9XX1cbiAgICAgICAgY2xhc3Nlcz17Y2xhc3Nlc31cbiAgICAgICAgZGlzYWJsZWQ9eyFvbkNoYW5nZUNvbW1pdHRlZH1cbiAgICAgICAgLy8gZGlzcGxheSBmaWxsZWQgd2hlbiBzZWVrIGhhbmRsZXIgaXMgbm90IHByb3ZpZGVkLCBmcm9tIFBsYXlCb3kgYmVoYXZpb3JcbiAgICAgICAgdmFsdWU9e29uQ2hhbmdlQ29tbWl0dGVkID8gY3VycmVudFRpbWUgOiBlbmRUaW1lfVxuICAgICAgICBtaW49e3N0YXJ0VGltZX1cbiAgICAgICAgbWF4PXtlbmRUaW1lfVxuICAgICAgICBtYXJrcz17bWFya3N9XG4gICAgICAgIHsuLi5oYW5kbGVyc31cbiAgICAgICAgey4uLnJlc3R9XG4gICAgICAgIHNsb3RzPXt7UmFpbDogU2Vla2JhclJhaWwsIFRyYWNrOiBTZWVrYmFyVHJhY2t9fVxuICAgICAgICBzbG90UHJvcHM9e3t0cmFjazoge3NlZ21lbnRzfX19XG4gICAgICAvPlxuICAgICAge3RpbWVEaXNwbGF5ICYmIDxGb3JtYXR0ZWRUaW1lIHRpbWU9e2R1cmF0aW9ufSAvPn1cbiAgICAgIHtjaGlsZHJlbiAmJlxuICAgICAgICBbXVxuICAgICAgICAgIC5jb25jYXQoY2hpbGRyZW4pXG4gICAgICAgICAgLm1hcChjaGlsZCA9PlxuICAgICAgICAgICAgY2xvbmVFbGVtZW50KGNoaWxkLCB7dGltZTogcG9pbnRlckFjdGl2ZSAmJiBwb2ludGVyU3RhdGUudmFsdWV9KVxuICAgICAgICAgICl9XG4gICAgPC9kaXY+XG4gIClcbn1cblxuZXhwb3J0IGRlZmF1bHQgU2Vla2JhclxuIl19 */"],
4153
4290
  classes: classes,
4154
4291
  disabled: !onChangeCommitted // display filled when seek handler is not provided, from PlayBoy behavior
4155
4292
  ,
@@ -4516,17 +4653,13 @@ const MenuItemText = ({
4516
4653
  })
4517
4654
  });
4518
4655
 
4519
- MenuItemText.propTypes = {
4520
- text: PropTypes.string
4521
- };
4522
-
4523
4656
  const LoopSwitch = ({
4524
4657
  style,
4525
4658
  checked,
4526
4659
  onChange
4527
4660
  }) => jsxs("li", {
4528
4661
  role: "menuitem",
4529
- 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":"AAmLI","file":"Settings.js","sourcesContent":["/* @jsxImportSource @emotion/react */\n/* eslint-disable jsx-a11y/no-noninteractive-element-interactions */\nimport {useState, useEffect} from 'react'\nimport PropTypes from 'prop-types'\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\nMenuItemText.propTypes = {\n  text: PropTypes.string,\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,\n  type: uiType,\n  closeBy,\n  buttonPosition,\n  slots = {\n    root: uiType === 'desktop' ? DesktopContainer : SwipeableDrawer,\n  },\n  onChange,\n  onOpen,\n  onClose,\n  controller,\n}) => {\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 (controller !== 'sender' && 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            event.stopPropagation()\n            onOpen(event)\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\nSettings.propTypes = {\n  open: PropTypes.bool,\n  values: PropTypes.object,\n  sections: PropTypes.array,\n  type: PropTypes.string,\n  onChange: PropTypes.func,\n  onClose: PropTypes.func,\n}\n\nexport default Settings\n"]} */"],
4662
+ 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"]} */"],
4530
4663
  onClick: () => {
4531
4664
  onChange({
4532
4665
  name: 'loop',
@@ -4550,14 +4683,14 @@ const CloseButton = props => jsx$1("button", {
4550
4683
  ...props
4551
4684
  });
4552
4685
 
4553
- const DesktopContainer = ({
4686
+ const DesktopContainer$1 = ({
4554
4687
  open,
4555
4688
  style,
4556
4689
  children,
4557
4690
  onClose,
4558
4691
  ...rest
4559
4692
  }) => jsx$1("div", {
4560
- 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":"AAwMO","file":"Settings.js","sourcesContent":["/* @jsxImportSource @emotion/react */\n/* eslint-disable jsx-a11y/no-noninteractive-element-interactions */\nimport {useState, useEffect} from 'react'\nimport PropTypes from 'prop-types'\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\nMenuItemText.propTypes = {\n  text: PropTypes.string,\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,\n  type: uiType,\n  closeBy,\n  buttonPosition,\n  slots = {\n    root: uiType === 'desktop' ? DesktopContainer : SwipeableDrawer,\n  },\n  onChange,\n  onOpen,\n  onClose,\n  controller,\n}) => {\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 (controller !== 'sender' && 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            event.stopPropagation()\n            onOpen(event)\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\nSettings.propTypes = {\n  open: PropTypes.bool,\n  values: PropTypes.object,\n  sections: PropTypes.array,\n  type: PropTypes.string,\n  onChange: PropTypes.func,\n  onClose: PropTypes.func,\n}\n\nexport default Settings\n"]} */"],
4693
+ 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"]} */"],
4561
4694
  ...rest,
4562
4695
  children: children
4563
4696
  });
@@ -4571,18 +4704,25 @@ const matchValue = (a, b) => {
4571
4704
  const Settings = ({
4572
4705
  open,
4573
4706
  values,
4574
- sections,
4707
+ sections: originalSections,
4575
4708
  type: uiType,
4576
4709
  closeBy,
4577
4710
  buttonPosition,
4711
+ enabledSettingSections = {
4712
+ audio: true,
4713
+ subtitles: true,
4714
+ quality: true,
4715
+ speed: true,
4716
+ loop: true
4717
+ },
4578
4718
  slots = {
4579
- root: uiType === 'desktop' ? DesktopContainer : SwipeableDrawer$1
4719
+ root: uiType === 'desktop' ? DesktopContainer$1 : SwipeableDrawer$1
4580
4720
  },
4581
4721
  onChange,
4582
4722
  onOpen,
4583
- onClose,
4584
- controller
4723
+ onClose
4585
4724
  }) => {
4725
+ const sections = originalSections.filter(section => enabledSettingSections[section.name]);
4586
4726
  const Container = slots.root;
4587
4727
  const ButtonWrap = buttonPosition === 'top-right' ? TitleBarExtension : FunctionBarExtension;
4588
4728
  const commonStyle = uiType === 'desktop' ? desktopStyle$1 : mobileStyle$1;
@@ -4593,7 +4733,7 @@ const Settings = ({
4593
4733
  }
4594
4734
  }, [open]);
4595
4735
  const ref = useOnclickOutside(() => {
4596
- if (controller !== 'sender' && open && uiType === 'desktop') {
4736
+ if (open && uiType === 'desktop') {
4597
4737
  onClose();
4598
4738
  }
4599
4739
  }, {
@@ -4637,11 +4777,11 @@ const Settings = ({
4637
4777
  position: "right",
4638
4778
  children: jsx$1(Button, {
4639
4779
  startIcon: "setting",
4780
+ style: {
4781
+ order: 2
4782
+ },
4640
4783
  title: "KKS.SETTING",
4641
- onClick: event => {
4642
- event.stopPropagation();
4643
- onOpen(event);
4644
- }
4784
+ onClick: event => setTimeout(() => open ? onClose(event) : onOpen(event), 1)
4645
4785
  })
4646
4786
  }), jsx$1(Container, {
4647
4787
  style: commonStyle.overlay,
@@ -4679,7 +4819,7 @@ const Settings = ({
4679
4819
  }) : jsxs("li", {
4680
4820
  role: link ? 'menuitem' : 'menuitemradio',
4681
4821
  "aria-checked": checked,
4682
- 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":"AAkTgB","file":"Settings.js","sourcesContent":["/* @jsxImportSource @emotion/react */\n/* eslint-disable jsx-a11y/no-noninteractive-element-interactions */\nimport {useState, useEffect} from 'react'\nimport PropTypes from 'prop-types'\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\nMenuItemText.propTypes = {\n  text: PropTypes.string,\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,\n  type: uiType,\n  closeBy,\n  buttonPosition,\n  slots = {\n    root: uiType === 'desktop' ? DesktopContainer : SwipeableDrawer,\n  },\n  onChange,\n  onOpen,\n  onClose,\n  controller,\n}) => {\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 (controller !== 'sender' && 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            event.stopPropagation()\n            onOpen(event)\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\nSettings.propTypes = {\n  open: PropTypes.bool,\n  values: PropTypes.object,\n  sections: PropTypes.array,\n  type: PropTypes.string,\n  onChange: PropTypes.func,\n  onClose: PropTypes.func,\n}\n\nexport default Settings\n"]} */"],
4822
+ 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"]} */"],
4683
4823
  onClick: () => link ? navigate(link) : onChange({
4684
4824
  name: currentSection.name,
4685
4825
  value: data
@@ -4696,16 +4836,286 @@ const Settings = ({
4696
4836
  })]
4697
4837
  });
4698
4838
  };
4699
-
4700
- Settings.propTypes = {
4701
- open: PropTypes.bool,
4702
- values: PropTypes.object,
4703
- sections: PropTypes.array,
4704
- type: PropTypes.string,
4705
- onChange: PropTypes.func,
4706
- onClose: PropTypes.func
4839
+
4840
+ /* eslint-disable jsx-a11y/no-static-element-interactions */
4841
+ const containerStyle$3 = {
4842
+ position: 'absolute',
4843
+ zIndex: '2',
4844
+ top: '0',
4845
+ width: '100%',
4846
+ height: '100%',
4847
+ transform: 'translateY(-150vh)',
4848
+ opacity: '0',
4849
+ transition: 'opacity 0.5s ease, transform 0s 0.5s linear'
4850
+ };
4851
+ const openStyle$1 = {
4852
+ transform: 'translateY(0)',
4853
+ opacity: '1',
4854
+ transition: 'opacity 0.5s ease'
4855
+ };
4856
+
4857
+ const DefaultContainer = ({
4858
+ open,
4859
+ containerRef,
4860
+ children
4861
+ }) => jsx$1("div", {
4862
+ 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,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIk92ZXJsYXlQYW5lbC5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUE0QkkiLCJmaWxlIjoiT3ZlcmxheVBhbmVsLmpzIiwic291cmNlc0NvbnRlbnQiOlsiLyogZXNsaW50LWRpc2FibGUganN4LWExMXkvbm8tc3RhdGljLWVsZW1lbnQtaW50ZXJhY3Rpb25zICovXG4vKiBAanN4SW1wb3J0U291cmNlIEBlbW90aW9uL3JlYWN0ICovXG5pbXBvcnQge3VzZUVmZmVjdCwgdXNlUmVmLCB1c2VTdGF0ZX0gZnJvbSAncmVhY3QnXG5pbXBvcnQgdXNlT25jbGlja091dHNpZGUgZnJvbSAncmVhY3QtY29vbC1vbmNsaWNrb3V0c2lkZSdcbmltcG9ydCB7b259IGZyb20gJ3V0aWwvZXZlbnRzJ1xuaW1wb3J0IHtpc0Rlc2t0b3B9IGZyb20gJ3V0aWwvZW52aXJvbm1lbnQnXG5pbXBvcnQge0J1dHRvbn0gZnJvbSAnLi9idXR0b25zJ1xuaW1wb3J0IHtGdW5jdGlvbkJhckV4dGVuc2lvbiwgVGl0bGVCYXJFeHRlbnNpb259IGZyb20gJy4vdWlFeHRlbnNpb25zJ1xuXG5jb25zdCBjb250YWluZXJTdHlsZSA9IHtcbiAgcG9zaXRpb246ICdhYnNvbHV0ZScsXG4gIHpJbmRleDogJzInLFxuICB0b3A6ICcwJyxcbiAgd2lkdGg6ICcxMDAlJyxcbiAgaGVpZ2h0OiAnMTAwJScsXG4gIHRyYW5zZm9ybTogJ3RyYW5zbGF0ZVkoLTE1MHZoKScsXG4gIG9wYWNpdHk6ICcwJyxcbiAgdHJhbnNpdGlvbjogJ29wYWNpdHkgMC41cyBlYXNlLCB0cmFuc2Zvcm0gMHMgMC41cyBsaW5lYXInLFxufVxuXG5jb25zdCBvcGVuU3R5bGUgPSB7XG4gIHRyYW5zZm9ybTogJ3RyYW5zbGF0ZVkoMCknLFxuICBvcGFjaXR5OiAnMScsXG4gIHRyYW5zaXRpb246ICdvcGFjaXR5IDAuNXMgZWFzZScsXG59XG5cbmNvbnN0IERlZmF1bHRDb250YWluZXIgPSAoe29wZW4sIGNvbnRhaW5lclJlZiwgY2hpbGRyZW59KSA9PiAoXG4gIDxkaXZcbiAgICBjc3M9e1tjb250YWluZXJTdHlsZSwgb3BlbiAmJiBvcGVuU3R5bGVdfVxuICAgIHJlZj17Y29udGFpbmVyUmVmfVxuICAgIG9uQ2xpY2s9e2V2ZW50ID0+IGV2ZW50LnN0b3BQcm9wYWdhdGlvbigpfVxuICA+XG4gICAge2NoaWxkcmVufVxuICA8L2Rpdj5cbilcblxuY29uc3QgZGVza3RvcENvbnRhaW5lclN0eWxlID0ge1xuICBwb3NpdGlvbjogJ2Fic29sdXRlJyxcbiAgekluZGV4OiAnMScsXG4gIGJvdHRvbTogJ2NhbGMoNWVtICsgdmFyKC0tYm90dG9tLXNwYWNpbmcsIDByZW0pKScsXG4gIHJpZ2h0OiAnMXJlbScsXG4gIGJhY2tncm91bmQ6ICd0cmFuc3BhcmVudCcsXG4gIG91dGxpbmU6ICdub25lJyxcbiAgb3BhY2l0eTogJzAnLFxuICB0cmFuc2Zvcm06ICd0cmFuc2xhdGVZKC0xMDB2aCknLFxuICB0cmFuc2l0aW9uOiAnb3BhY2l0eSAwLjJzIGVhc2UsIHRyYW5zZm9ybSAwcyBsaW5lYXIgMC4ycycsXG59XG5cbmNvbnN0IERlc2t0b3BPcGVuU3R5bGUgPSB7XG4gIG9wYWNpdHk6ICcxJyxcbiAgdHJhbnNmb3JtOiAndHJhbnNsYXRlWSgwKScsXG4gIHRyYW5zaXRpb246ICdvcGFjaXR5IDAuMnMgZWFzZSwgdHJhbnNmb3JtIDBzJyxcbn1cblxuY29uc3QgRGVza3RvcENvbnRhaW5lciA9ICh7b3BlbiwgY29udGFpbmVyUmVmLCBjaGlsZHJlbiwgb25DbG9zZSwgLi4ucmVzdH0pID0+IChcbiAgPGRpdlxuICAgIGNzcz17W2Rlc2t0b3BDb250YWluZXJTdHlsZSwgb3BlbiAmJiBEZXNrdG9wT3BlblN0eWxlXX1cbiAgICByZWY9e2NvbnRhaW5lclJlZn1cbiAgICBvbkNsaWNrPXtldmVudCA9PiBldmVudC5zdG9wUHJvcGFnYXRpb24oKX1cbiAgICB7Li4ucmVzdH1cbiAgPlxuICAgIHtjaGlsZHJlbn1cbiAgPC9kaXY+XG4pXG5cbmNvbnN0IE92ZXJsYXlQYW5lbCA9ICh7XG4gIGJ1dHRvbkljb24sXG4gIHVpVHlwZSA9IGlzRGVza3RvcCgpID8gJ2Rlc2t0b3AnIDogJ21vYmlsZScsXG4gIGNoaWxkcmVuLFxuICBzbG90cyA9IHtcbiAgICByb290OiB1aVR5cGUgPT09ICdkZXNrdG9wJyA/IERlc2t0b3BDb250YWluZXIgOiBEZWZhdWx0Q29udGFpbmVyLFxuICB9LFxuICAuLi5yZXN0XG59KSA9PiB7XG4gIGNvbnN0IGNvbXBvbmVudHMgPSB7XG4gICAgcm9vdDogc2xvdHMucm9vdCxcbiAgfVxuICBjb25zdCBCdXR0b25XcmFwID1cbiAgICB1aVR5cGUgPT09ICdkZXNrdG9wJyA/IEZ1bmN0aW9uQmFyRXh0ZW5zaW9uIDogVGl0bGVCYXJFeHRlbnNpb25cbiAgY29uc3QgW29wZW4sIHNldE9wZW5dID0gdXNlU3RhdGUoZmFsc2UpXG4gIGNvbnN0IG9uQnV0dG9uQ2xpY2sgPSAoKSA9PiBzZXRUaW1lb3V0KCgpID0+IHNldE9wZW4oIW9wZW4pLCAxKVxuICBjb25zdCBvbkNsb3NlID0gKCkgPT4gc2V0T3BlbihmYWxzZSlcbiAgY29uc3QgY29udGFpbmVyUmVmID0gdXNlUmVmKClcbiAgY29uc3QgcmVmID0gdXNlT25jbGlja091dHNpZGUoXG4gICAgKCkgPT4ge1xuICAgICAgaWYgKG9wZW4gJiYgdWlUeXBlID09PSAnZGVza3RvcCcpIHtcbiAgICAgICAgb25DbG9zZSgpXG4gICAgICB9XG4gICAgfSxcbiAgICB7ZXZlbnRUeXBlczogWydjbGljayddfVxuICApXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgcmVmKGNvbnRhaW5lclJlZi5jdXJyZW50KVxuICAgIHJldHVybiBvbihjb250YWluZXJSZWYuY3VycmVudCwgJ2Nsb3NlLW1lbnUnLCBvbkNsb3NlKVxuICB9LCBbXSlcblxuICByZXR1cm4gKFxuICAgIDw+XG4gICAgICA8QnV0dG9uV3JhcCBwb3NpdGlvbj1cInJpZ2h0XCI+XG4gICAgICAgIDxCdXR0b24gc3RhcnRJY29uPXtidXR0b25JY29ufSBvbkNsaWNrPXtvbkJ1dHRvbkNsaWNrfSAvPlxuICAgICAgPC9CdXR0b25XcmFwPlxuICAgICAgPGNvbXBvbmVudHMucm9vdFxuICAgICAgICBjb250YWluZXJSZWY9e2NvbnRhaW5lclJlZn1cbiAgICAgICAgb3Blbj17b3Blbn1cbiAgICAgICAgey4uLnJlc3R9XG4gICAgICA+XG4gICAgICAgIHtjaGlsZHJlbn1cbiAgICAgIDwvY29tcG9uZW50cy5yb290PlxuICAgIDwvPlxuICApXG59XG5cbmV4cG9ydCBkZWZhdWx0IE92ZXJsYXlQYW5lbFxuIl19 */"],
4863
+ ref: containerRef,
4864
+ onClick: event => event.stopPropagation(),
4865
+ children: children
4866
+ });
4867
+
4868
+ const desktopContainerStyle$1 = {
4869
+ position: 'absolute',
4870
+ zIndex: '1',
4871
+ bottom: 'calc(5em + var(--bottom-spacing, 0rem))',
4872
+ right: '1rem',
4873
+ background: 'transparent',
4874
+ outline: 'none',
4875
+ opacity: '0',
4876
+ transform: 'translateY(-100vh)',
4877
+ transition: 'opacity 0.2s ease, transform 0s linear 0.2s'
4878
+ };
4879
+ const DesktopOpenStyle = {
4880
+ opacity: '1',
4881
+ transform: 'translateY(0)',
4882
+ transition: 'opacity 0.2s ease, transform 0s'
4883
+ };
4884
+
4885
+ const DesktopContainer = ({
4886
+ open,
4887
+ containerRef,
4888
+ children,
4889
+ onClose,
4890
+ ...rest
4891
+ }) => jsx$1("div", {
4892
+ 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,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIk92ZXJsYXlQYW5lbC5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUF3REkiLCJmaWxlIjoiT3ZlcmxheVBhbmVsLmpzIiwic291cmNlc0NvbnRlbnQiOlsiLyogZXNsaW50LWRpc2FibGUganN4LWExMXkvbm8tc3RhdGljLWVsZW1lbnQtaW50ZXJhY3Rpb25zICovXG4vKiBAanN4SW1wb3J0U291cmNlIEBlbW90aW9uL3JlYWN0ICovXG5pbXBvcnQge3VzZUVmZmVjdCwgdXNlUmVmLCB1c2VTdGF0ZX0gZnJvbSAncmVhY3QnXG5pbXBvcnQgdXNlT25jbGlja091dHNpZGUgZnJvbSAncmVhY3QtY29vbC1vbmNsaWNrb3V0c2lkZSdcbmltcG9ydCB7b259IGZyb20gJ3V0aWwvZXZlbnRzJ1xuaW1wb3J0IHtpc0Rlc2t0b3B9IGZyb20gJ3V0aWwvZW52aXJvbm1lbnQnXG5pbXBvcnQge0J1dHRvbn0gZnJvbSAnLi9idXR0b25zJ1xuaW1wb3J0IHtGdW5jdGlvbkJhckV4dGVuc2lvbiwgVGl0bGVCYXJFeHRlbnNpb259IGZyb20gJy4vdWlFeHRlbnNpb25zJ1xuXG5jb25zdCBjb250YWluZXJTdHlsZSA9IHtcbiAgcG9zaXRpb246ICdhYnNvbHV0ZScsXG4gIHpJbmRleDogJzInLFxuICB0b3A6ICcwJyxcbiAgd2lkdGg6ICcxMDAlJyxcbiAgaGVpZ2h0OiAnMTAwJScsXG4gIHRyYW5zZm9ybTogJ3RyYW5zbGF0ZVkoLTE1MHZoKScsXG4gIG9wYWNpdHk6ICcwJyxcbiAgdHJhbnNpdGlvbjogJ29wYWNpdHkgMC41cyBlYXNlLCB0cmFuc2Zvcm0gMHMgMC41cyBsaW5lYXInLFxufVxuXG5jb25zdCBvcGVuU3R5bGUgPSB7XG4gIHRyYW5zZm9ybTogJ3RyYW5zbGF0ZVkoMCknLFxuICBvcGFjaXR5OiAnMScsXG4gIHRyYW5zaXRpb246ICdvcGFjaXR5IDAuNXMgZWFzZScsXG59XG5cbmNvbnN0IERlZmF1bHRDb250YWluZXIgPSAoe29wZW4sIGNvbnRhaW5lclJlZiwgY2hpbGRyZW59KSA9PiAoXG4gIDxkaXZcbiAgICBjc3M9e1tjb250YWluZXJTdHlsZSwgb3BlbiAmJiBvcGVuU3R5bGVdfVxuICAgIHJlZj17Y29udGFpbmVyUmVmfVxuICAgIG9uQ2xpY2s9e2V2ZW50ID0+IGV2ZW50LnN0b3BQcm9wYWdhdGlvbigpfVxuICA+XG4gICAge2NoaWxkcmVufVxuICA8L2Rpdj5cbilcblxuY29uc3QgZGVza3RvcENvbnRhaW5lclN0eWxlID0ge1xuICBwb3NpdGlvbjogJ2Fic29sdXRlJyxcbiAgekluZGV4OiAnMScsXG4gIGJvdHRvbTogJ2NhbGMoNWVtICsgdmFyKC0tYm90dG9tLXNwYWNpbmcsIDByZW0pKScsXG4gIHJpZ2h0OiAnMXJlbScsXG4gIGJhY2tncm91bmQ6ICd0cmFuc3BhcmVudCcsXG4gIG91dGxpbmU6ICdub25lJyxcbiAgb3BhY2l0eTogJzAnLFxuICB0cmFuc2Zvcm06ICd0cmFuc2xhdGVZKC0xMDB2aCknLFxuICB0cmFuc2l0aW9uOiAnb3BhY2l0eSAwLjJzIGVhc2UsIHRyYW5zZm9ybSAwcyBsaW5lYXIgMC4ycycsXG59XG5cbmNvbnN0IERlc2t0b3BPcGVuU3R5bGUgPSB7XG4gIG9wYWNpdHk6ICcxJyxcbiAgdHJhbnNmb3JtOiAndHJhbnNsYXRlWSgwKScsXG4gIHRyYW5zaXRpb246ICdvcGFjaXR5IDAuMnMgZWFzZSwgdHJhbnNmb3JtIDBzJyxcbn1cblxuY29uc3QgRGVza3RvcENvbnRhaW5lciA9ICh7b3BlbiwgY29udGFpbmVyUmVmLCBjaGlsZHJlbiwgb25DbG9zZSwgLi4ucmVzdH0pID0+IChcbiAgPGRpdlxuICAgIGNzcz17W2Rlc2t0b3BDb250YWluZXJTdHlsZSwgb3BlbiAmJiBEZXNrdG9wT3BlblN0eWxlXX1cbiAgICByZWY9e2NvbnRhaW5lclJlZn1cbiAgICBvbkNsaWNrPXtldmVudCA9PiBldmVudC5zdG9wUHJvcGFnYXRpb24oKX1cbiAgICB7Li4ucmVzdH1cbiAgPlxuICAgIHtjaGlsZHJlbn1cbiAgPC9kaXY+XG4pXG5cbmNvbnN0IE92ZXJsYXlQYW5lbCA9ICh7XG4gIGJ1dHRvbkljb24sXG4gIHVpVHlwZSA9IGlzRGVza3RvcCgpID8gJ2Rlc2t0b3AnIDogJ21vYmlsZScsXG4gIGNoaWxkcmVuLFxuICBzbG90cyA9IHtcbiAgICByb290OiB1aVR5cGUgPT09ICdkZXNrdG9wJyA/IERlc2t0b3BDb250YWluZXIgOiBEZWZhdWx0Q29udGFpbmVyLFxuICB9LFxuICAuLi5yZXN0XG59KSA9PiB7XG4gIGNvbnN0IGNvbXBvbmVudHMgPSB7XG4gICAgcm9vdDogc2xvdHMucm9vdCxcbiAgfVxuICBjb25zdCBCdXR0b25XcmFwID1cbiAgICB1aVR5cGUgPT09ICdkZXNrdG9wJyA/IEZ1bmN0aW9uQmFyRXh0ZW5zaW9uIDogVGl0bGVCYXJFeHRlbnNpb25cbiAgY29uc3QgW29wZW4sIHNldE9wZW5dID0gdXNlU3RhdGUoZmFsc2UpXG4gIGNvbnN0IG9uQnV0dG9uQ2xpY2sgPSAoKSA9PiBzZXRUaW1lb3V0KCgpID0+IHNldE9wZW4oIW9wZW4pLCAxKVxuICBjb25zdCBvbkNsb3NlID0gKCkgPT4gc2V0T3BlbihmYWxzZSlcbiAgY29uc3QgY29udGFpbmVyUmVmID0gdXNlUmVmKClcbiAgY29uc3QgcmVmID0gdXNlT25jbGlja091dHNpZGUoXG4gICAgKCkgPT4ge1xuICAgICAgaWYgKG9wZW4gJiYgdWlUeXBlID09PSAnZGVza3RvcCcpIHtcbiAgICAgICAgb25DbG9zZSgpXG4gICAgICB9XG4gICAgfSxcbiAgICB7ZXZlbnRUeXBlczogWydjbGljayddfVxuICApXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgcmVmKGNvbnRhaW5lclJlZi5jdXJyZW50KVxuICAgIHJldHVybiBvbihjb250YWluZXJSZWYuY3VycmVudCwgJ2Nsb3NlLW1lbnUnLCBvbkNsb3NlKVxuICB9LCBbXSlcblxuICByZXR1cm4gKFxuICAgIDw+XG4gICAgICA8QnV0dG9uV3JhcCBwb3NpdGlvbj1cInJpZ2h0XCI+XG4gICAgICAgIDxCdXR0b24gc3RhcnRJY29uPXtidXR0b25JY29ufSBvbkNsaWNrPXtvbkJ1dHRvbkNsaWNrfSAvPlxuICAgICAgPC9CdXR0b25XcmFwPlxuICAgICAgPGNvbXBvbmVudHMucm9vdFxuICAgICAgICBjb250YWluZXJSZWY9e2NvbnRhaW5lclJlZn1cbiAgICAgICAgb3Blbj17b3Blbn1cbiAgICAgICAgey4uLnJlc3R9XG4gICAgICA+XG4gICAgICAgIHtjaGlsZHJlbn1cbiAgICAgIDwvY29tcG9uZW50cy5yb290PlxuICAgIDwvPlxuICApXG59XG5cbmV4cG9ydCBkZWZhdWx0IE92ZXJsYXlQYW5lbFxuIl19 */"],
4893
+ ref: containerRef,
4894
+ onClick: event => event.stopPropagation(),
4895
+ ...rest,
4896
+ children: children
4897
+ });
4898
+
4899
+ const OverlayPanel = ({
4900
+ buttonIcon,
4901
+ uiType = isDesktop() ? 'desktop' : 'mobile',
4902
+ children,
4903
+ slots = {
4904
+ root: uiType === 'desktop' ? DesktopContainer : DefaultContainer
4905
+ },
4906
+ ...rest
4907
+ }) => {
4908
+ const components = {
4909
+ root: slots.root
4910
+ };
4911
+ const ButtonWrap = uiType === 'desktop' ? FunctionBarExtension : TitleBarExtension;
4912
+ const [open, setOpen] = useState(false);
4913
+
4914
+ const onButtonClick = () => setTimeout(() => setOpen(!open), 1);
4915
+
4916
+ const onClose = () => setOpen(false);
4917
+
4918
+ const containerRef = useRef();
4919
+ const ref = useOnclickOutside(() => {
4920
+ if (open && uiType === 'desktop') {
4921
+ onClose();
4922
+ }
4923
+ }, {
4924
+ eventTypes: ['click']
4925
+ });
4926
+ useEffect(() => {
4927
+ ref(containerRef.current);
4928
+ return on(containerRef.current, 'close-menu', onClose);
4929
+ }, []);
4930
+ return jsxs(Fragment, {
4931
+ children: [jsx$1(ButtonWrap, {
4932
+ position: "right",
4933
+ children: jsx$1(Button, {
4934
+ startIcon: buttonIcon,
4935
+ onClick: onButtonClick
4936
+ })
4937
+ }), jsx$1(components.root, {
4938
+ containerRef: containerRef,
4939
+ open: open,
4940
+ ...rest,
4941
+ children: children
4942
+ })]
4943
+ });
4944
+ };
4945
+
4946
+ /* @jsxImportSource @emotion/react */
4947
+ const menuItemStyle = {
4948
+ padding: 'var(--menu-item-padding, 0.5em 1em) ',
4949
+ display: 'flex',
4950
+ alignItems: 'center',
4951
+ minWidth: 0,
4952
+ color: 'rgba(255, 255, 255, 0.6)',
4953
+ background: 'var(--menu-item-background, rgba(255, 255, 255, 0.08))',
4954
+ '> div': {
4955
+ overflow: 'hidden',
4956
+ whiteSpace: 'nowrap',
4957
+ textOverflow: 'ellipsis'
4958
+ }
4959
+ };
4960
+ const menuItemSelectedStyle = {
4961
+ color: '#FFF',
4962
+ fontWeight: '600',
4963
+ '::after': {
4964
+ content: '" "',
4965
+ display: 'inline-block',
4966
+ flex: '0 0 1em',
4967
+ height: '1em',
4968
+ marginLeft: '0.5em',
4969
+ backgroundColor: 'var(--setting-check-icon-color, #fff)',
4970
+ maskSize: 'contain',
4971
+ maskImage: `url(${iconUrls.check})`
4972
+ }
4973
+ };
4974
+
4975
+ const MenuItem = ({
4976
+ label,
4977
+ selected,
4978
+ onClick
4979
+ }) => jsx$1("div", {
4980
+ 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 */"],
4981
+ onClick: onClick,
4982
+ children: jsx$1("div", {
4983
+ children: jsx$1(FormattedMessage, {
4984
+ id: `KKS.SETTING.${label}`,
4985
+ defaultMessage: label
4986
+ })
4987
+ })
4988
+ });
4989
+
4990
+ var MenuItem$1 = MenuItem;
4991
+
4992
+ /* @jsxImportSource @emotion/react */
4993
+ const landscapeStyle = {
4994
+ display: 'flex',
4995
+ flexWrap: 'wrap',
4996
+ justifyContent: 'space-between',
4997
+ '> div': {
4998
+ boxSizing: 'border-box',
4999
+ flex: '0 0 50%',
5000
+ minWidth: 0
5001
+ },
5002
+ '> div:not(:first-of-type)': {
5003
+ marginTop: 0,
5004
+ flex: '0 0 calc(50% - 1px)'
5005
+ },
5006
+ hr: {
5007
+ height: 'auto',
5008
+ borderLeft: '1px solid transparent'
5009
+ }
5010
+ };
5011
+ const containerStyle$2 = {
5012
+ fontSize: '16px',
5013
+ background: '#262626',
5014
+ height: '100%',
5015
+ h3: {
5016
+ margin: 0,
5017
+ 'margin-block-start': 0,
5018
+ 'margin-block-end': 0,
5019
+ padding: '0.6em 1em',
5020
+ display: 'flex',
5021
+ alignItems: 'center',
5022
+ borderRadius: '4px 4px 0 0',
5023
+ fontSize: '17px',
5024
+ button: {
5025
+ fontSize: '0.6em',
5026
+ marginRight: '0.5em'
5027
+ }
5028
+ }
5029
+ };
5030
+ const twoColumnStyle = {
5031
+ hr: {
5032
+ border: 'none'
5033
+ },
5034
+ '> div ': {
5035
+ '> div:not(:first-of-type)': {
5036
+ marginTop: '0.66em'
5037
+ },
5038
+ '@media (orientation: landscape)': landscapeStyle
5039
+ },
5040
+ '> div > div > div:first-of-type': {
5041
+ padding: '0.6em 1.1em',
5042
+ fontSize: '14px',
5043
+ color: '#aaaeb5',
5044
+ background: 'var(--menu-item-background, #262626)',
5045
+ overflow: 'hidden',
5046
+ textOverflow: 'ellipsis'
5047
+ }
5048
+ };
5049
+ const desktopContainerStyle = {
5050
+ paddingBottom: '1.5em',
5051
+ width: '34em',
5052
+ minHeight: '30.5%',
5053
+ maxHeight: '63.5%',
5054
+ borderRadius: '4px',
5055
+ background: 'rgba(0, 0, 0, 0.60)',
5056
+ '--menu-item-background': 'transparent',
5057
+ '--menu-item-padding': '0.66em 1.5em',
5058
+ h3: {
5059
+ padding: '0.66em 1.5em',
5060
+ fontSize: '16px',
5061
+ color: 'rgba(255, 255, 255, 0.60)',
5062
+ background: 'rgba(0, 0, 0, 0.80)'
5063
+ }
5064
+ };
5065
+ const desktop2ColumnStyle = {
5066
+ '> div': landscapeStyle,
5067
+ '> div hr': {
5068
+ fontSize: '150%',
5069
+ borderLeft: '1px solid rgba(255, 255, 255, 0.20)'
5070
+ },
5071
+ '> div > div': {
5072
+ padding: '0 1em'
5073
+ },
5074
+ '> div > div > div:first-of-type': {
5075
+ padding: '0.66em 0.5em',
5076
+ color: '#fff',
5077
+ fontSize: '16px',
5078
+ fontWeight: '600'
5079
+ }
4707
5080
  };
4708
5081
 
5082
+ const closeMenu = event => event.currentTarget.dispatchEvent(new CustomEvent('close-menu', {
5083
+ bubbles: true
5084
+ }));
5085
+
5086
+ const LanguageMenu = ({
5087
+ uiType = 'mobile',
5088
+ title = '音声 ・ 字幕',
5089
+ sections = ['音声', '字幕'],
5090
+ sectionOptions = [],
5091
+ onChange
5092
+ }) => jsxs("div", {
5093
+ 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,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIkxhbmd1YWdlTWVudS5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUE4R0kiLCJmaWxlIjoiTGFuZ3VhZ2VNZW51LmpzIiwic291cmNlc0NvbnRlbnQiOlsiLyogQGpzeEltcG9ydFNvdXJjZSBAZW1vdGlvbi9yZWFjdCAqL1xuaW1wb3J0IHtCdXR0b259IGZyb20gJy4vYnV0dG9ucydcbmltcG9ydCBNZW51SXRlbSBmcm9tICcuL01lbnVJdGVtJ1xuXG5jb25zdCBsYW5kc2NhcGVTdHlsZSA9IHtcbiAgZGlzcGxheTogJ2ZsZXgnLFxuICBmbGV4V3JhcDogJ3dyYXAnLFxuICBqdXN0aWZ5Q29udGVudDogJ3NwYWNlLWJldHdlZW4nLFxuICAnPiBkaXYnOiB7XG4gICAgYm94U2l6aW5nOiAnYm9yZGVyLWJveCcsXG4gICAgZmxleDogJzAgMCA1MCUnLFxuICAgIG1pbldpZHRoOiAwLFxuICB9LFxuICAnPiBkaXY6bm90KDpmaXJzdC1vZi10eXBlKSc6IHtcbiAgICBtYXJnaW5Ub3A6IDAsXG4gICAgZmxleDogJzAgMCBjYWxjKDUwJSAtIDFweCknLFxuICB9LFxuICBocjoge1xuICAgIGhlaWdodDogJ2F1dG8nLFxuICAgIGJvcmRlckxlZnQ6ICcxcHggc29saWQgdHJhbnNwYXJlbnQnLFxuICB9LFxufVxuXG5jb25zdCBjb250YWluZXJTdHlsZSA9IHtcbiAgZm9udFNpemU6ICcxNnB4JyxcbiAgYmFja2dyb3VuZDogJyMyNjI2MjYnLFxuICBoZWlnaHQ6ICcxMDAlJyxcbiAgaDM6IHtcbiAgICBtYXJnaW46IDAsXG4gICAgJ21hcmdpbi1ibG9jay1zdGFydCc6IDAsXG4gICAgJ21hcmdpbi1ibG9jay1lbmQnOiAwLFxuICAgIHBhZGRpbmc6ICcwLjZlbSAxZW0nLFxuICAgIGRpc3BsYXk6ICdmbGV4JyxcbiAgICBhbGlnbkl0ZW1zOiAnY2VudGVyJyxcbiAgICBib3JkZXJSYWRpdXM6ICc0cHggNHB4IDAgMCcsXG4gICAgZm9udFNpemU6ICcxN3B4JyxcbiAgICBidXR0b246IHtcbiAgICAgIGZvbnRTaXplOiAnMC42ZW0nLFxuICAgICAgbWFyZ2luUmlnaHQ6ICcwLjVlbScsXG4gICAgfSxcbiAgfSxcbn1cblxuY29uc3QgdHdvQ29sdW1uU3R5bGUgPSB7XG4gIGhyOiB7XG4gICAgYm9yZGVyOiAnbm9uZScsXG4gIH0sXG4gICc+IGRpdiAnOiB7XG4gICAgJz4gZGl2Om5vdCg6Zmlyc3Qtb2YtdHlwZSknOiB7XG4gICAgICBtYXJnaW5Ub3A6ICcwLjY2ZW0nLFxuICAgIH0sXG4gICAgJ0BtZWRpYSAob3JpZW50YXRpb246IGxhbmRzY2FwZSknOiBsYW5kc2NhcGVTdHlsZSxcbiAgfSxcbiAgJz4gZGl2ID4gZGl2ID4gZGl2OmZpcnN0LW9mLXR5cGUnOiB7XG4gICAgcGFkZGluZzogJzAuNmVtIDEuMWVtJyxcbiAgICBmb250U2l6ZTogJzE0cHgnLFxuICAgIGNvbG9yOiAnI2FhYWViNScsXG4gICAgYmFja2dyb3VuZDogJ3ZhcigtLW1lbnUtaXRlbS1iYWNrZ3JvdW5kLCAjMjYyNjI2KScsXG4gICAgb3ZlcmZsb3c6ICdoaWRkZW4nLFxuICAgIHRleHRPdmVyZmxvdzogJ2VsbGlwc2lzJyxcbiAgfSxcbn1cblxuY29uc3QgZGVza3RvcENvbnRhaW5lclN0eWxlID0ge1xuICBwYWRkaW5nQm90dG9tOiAnMS41ZW0nLFxuICB3aWR0aDogJzM0ZW0nLFxuICBtaW5IZWlnaHQ6ICczMC41JScsXG4gIG1heEhlaWdodDogJzYzLjUlJyxcbiAgYm9yZGVyUmFkaXVzOiAnNHB4JyxcbiAgYmFja2dyb3VuZDogJ3JnYmEoMCwgMCwgMCwgMC42MCknLFxuICAnLS1tZW51LWl0ZW0tYmFja2dyb3VuZCc6ICd0cmFuc3BhcmVudCcsXG4gICctLW1lbnUtaXRlbS1wYWRkaW5nJzogJzAuNjZlbSAxLjVlbScsXG4gIGgzOiB7XG4gICAgcGFkZGluZzogJzAuNjZlbSAxLjVlbScsXG4gICAgZm9udFNpemU6ICcxNnB4JyxcbiAgICBjb2xvcjogJ3JnYmEoMjU1LCAyNTUsIDI1NSwgMC42MCknLFxuICAgIGJhY2tncm91bmQ6ICdyZ2JhKDAsIDAsIDAsIDAuODApJyxcbiAgfSxcbn1cblxuY29uc3QgZGVza3RvcDJDb2x1bW5TdHlsZSA9IHtcbiAgJz4gZGl2JzogbGFuZHNjYXBlU3R5bGUsXG4gICc+IGRpdiBocic6IHtcbiAgICBmb250U2l6ZTogJzE1MCUnLFxuICAgIGJvcmRlckxlZnQ6ICcxcHggc29saWQgIHJnYmEoMjU1LCAyNTUsIDI1NSwgMC4yMCknLFxuICB9LFxuICAnPiBkaXYgPiBkaXYnOiB7XG4gICAgcGFkZGluZzogJzAgMWVtJyxcbiAgfSxcbiAgJz4gZGl2ID4gZGl2ID4gZGl2OmZpcnN0LW9mLXR5cGUnOiB7XG4gICAgcGFkZGluZzogJzAuNjZlbSAwLjVlbScsXG4gICAgY29sb3I6ICcjZmZmJyxcbiAgICBmb250U2l6ZTogJzE2cHgnLFxuICAgIGZvbnRXZWlnaHQ6ICc2MDAnLFxuICB9LFxufVxuXG5jb25zdCBjbG9zZU1lbnUgPSBldmVudCA9PlxuICBldmVudC5jdXJyZW50VGFyZ2V0LmRpc3BhdGNoRXZlbnQoXG4gICAgbmV3IEN1c3RvbUV2ZW50KCdjbG9zZS1tZW51Jywge2J1YmJsZXM6IHRydWV9KVxuICApXG5cbmNvbnN0IExhbmd1YWdlTWVudSA9ICh7XG4gIHVpVHlwZSA9ICdtb2JpbGUnLFxuICB0aXRsZSA9ICfpn7Plo7Ag44O7IOWtl+W5lScsXG4gIHNlY3Rpb25zID0gWyfpn7Plo7AnLCAn5a2X5bmVJ10sXG4gIHNlY3Rpb25PcHRpb25zID0gW10sXG4gIG9uQ2hhbmdlLFxufSkgPT4gKFxuICA8ZGl2XG4gICAgY3NzPXtbXG4gICAgICBjb250YWluZXJTdHlsZSxcbiAgICAgIHR3b0NvbHVtblN0eWxlLFxuICAgICAgdWlUeXBlID09PSAnZGVza3RvcCcgJiYgW2Rlc2t0b3BDb250YWluZXJTdHlsZSwgZGVza3RvcDJDb2x1bW5TdHlsZV0sXG4gICAgXX1cbiAgPlxuICAgIDxoMz5cbiAgICAgIHt1aVR5cGUgIT09ICdkZXNrdG9wJyAmJiA8QnV0dG9uIHN0YXJ0SWNvbj1cImNsb3NlXCIgb25DbGljaz17Y2xvc2VNZW51fSAvPn1cbiAgICAgIHt0aXRsZX1cbiAgICA8L2gzPlxuICAgIDxkaXY+XG4gICAgICA8ZGl2PlxuICAgICAgICA8ZGl2PntzZWN0aW9uc1swXX08L2Rpdj5cbiAgICAgICAge3NlY3Rpb25PcHRpb25zWzBdLm1hcChvcHRpb24gPT4gKFxuICAgICAgICAgIDxNZW51SXRlbSB7Li4ub3B0aW9ufSBvbkNsaWNrPXtldmVudCA9PiBvbkNoYW5nZShldmVudCwgb3B0aW9uKX0gLz5cbiAgICAgICAgKSl9XG4gICAgICA8L2Rpdj5cbiAgICAgIDxociAvPlxuICAgICAgPGRpdj5cbiAgICAgICAgPGRpdj57c2VjdGlvbnNbMV19PC9kaXY+XG4gICAgICAgIHtzZWN0aW9uT3B0aW9uc1sxXS5tYXAob3B0aW9uID0+IChcbiAgICAgICAgICA8TWVudUl0ZW0gey4uLm9wdGlvbn0gb25DbGljaz17ZXZlbnQgPT4gb25DaGFuZ2UoZXZlbnQsIG9wdGlvbil9IC8+XG4gICAgICAgICkpfVxuICAgICAgPC9kaXY+XG4gICAgPC9kaXY+XG4gIDwvZGl2PlxuKVxuXG5MYW5ndWFnZU1lbnUuc3R5bGVzID0gY29udGFpbmVyU3R5bGVcbkxhbmd1YWdlTWVudS5kZXNrdG9wU3R5bGVzID0gZGVza3RvcENvbnRhaW5lclN0eWxlXG5cbmV4cG9ydCBkZWZhdWx0IExhbmd1YWdlTWVudVxuIl19 */"],
5094
+ children: [jsxs("h3", {
5095
+ children: [uiType !== 'desktop' && jsx$1(Button, {
5096
+ startIcon: "close",
5097
+ onClick: closeMenu
5098
+ }), title]
5099
+ }), jsxs("div", {
5100
+ children: [jsxs("div", {
5101
+ children: [jsx$1("div", {
5102
+ children: sections[0]
5103
+ }), sectionOptions[0].map(option => jsx$1(MenuItem$1, { ...option,
5104
+ onClick: event => onChange(event, option)
5105
+ }))]
5106
+ }), jsx$1("hr", {}), jsxs("div", {
5107
+ children: [jsx$1("div", {
5108
+ children: sections[1]
5109
+ }), sectionOptions[1].map(option => jsx$1(MenuItem$1, { ...option,
5110
+ onClick: event => onChange(event, option)
5111
+ }))]
5112
+ })]
5113
+ })]
5114
+ });
5115
+
5116
+ LanguageMenu.styles = containerStyle$2;
5117
+ LanguageMenu.desktopStyles = desktopContainerStyle;
5118
+
4709
5119
  function _EMOTION_STRINGIFIED_CSS_ERROR__$1() { return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop)."; }
4710
5120
  const panelStyle = {
4711
5121
  position: 'absolute',
@@ -7250,11 +7660,6 @@ const loadShaka = async (videoElement, config = {}, options = {}) => {
7250
7660
  },
7251
7661
  getPlaybackSpeed: () => videoElement.playbackRate,
7252
7662
  getVideoElement: () => videoElement,
7253
- setQuality: restrictions => {
7254
- if (!restrictions) return; // FIXME: Setting restrictions to {} cannot enable abr.
7255
-
7256
- player.configure('abr.restrictions', restrictions);
7257
- },
7258
7663
  getVideoQuality,
7259
7664
  getAvailableVideoQualities,
7260
7665
  isAlive: () => player.getLoadMode() !== shaka.Player.LoadMode.DESTROYED,
@@ -7273,7 +7678,7 @@ const loadPlayer = async (videoElement, {
7273
7678
  // TODO unsubscribe to release the video element
7274
7679
  // when resuming from background, video may play without no media events
7275
7680
  // on some iOS devices, pause again to workaround
7276
- if (!isIOS()) {
7681
+ if (isIOS()) {
7277
7682
  on(document, 'visibilitychange', () => setTimeout(() => videoElement.pause(), 50));
7278
7683
  }
7279
7684
 
@@ -7303,12 +7708,23 @@ const loadPlayer = async (videoElement, {
7303
7708
 
7304
7709
  reloadOnLiveStall(videoElement, {
7305
7710
  reload: () => player.reload()
7306
- }); // TODO load built-in modules here
7711
+ });
7712
+ player.preferredSettings = {}; // TODO load built-in modules here
7307
7713
 
7308
7714
  player.modules = {};
7309
7715
  return player;
7310
7716
  };
7311
7717
 
7718
+ /* eslint-disable no-param-reassign */
7719
+
7720
+ const selectVideoResolution = (_media, {
7721
+ player
7722
+ }, restrictions) => {
7723
+ player.preferredSettings.videoResolution = restrictions;
7724
+ player.configure('abr.restrictions', restrictions);
7725
+ };
7726
+ // for unit test
7727
+
7312
7728
  const baseVideoStyle = {
7313
7729
  objectFit: 'contain',
7314
7730
  width: '100%',
@@ -7333,10 +7749,10 @@ const Video = ({
7333
7749
  playbackState: targetState,
7334
7750
  currentTime: targetTime,
7335
7751
  playbackRate,
7752
+ videoResolution,
7753
+ audioTrack = {},
7336
7754
  textTracks = emptyArray,
7337
- quality,
7338
7755
  textTrack,
7339
- audio = {},
7340
7756
  liveResume,
7341
7757
  iOSFullscreenDetectThreshold,
7342
7758
  // TODO organize props
@@ -7353,7 +7769,7 @@ const Video = ({
7353
7769
  onPlaylogFired,
7354
7770
  ...videoAttributes
7355
7771
  }) => {
7356
- var _videoElement$current;
7772
+ var _videoElement$current, _videoElement$current2;
7357
7773
 
7358
7774
  const handlers = useRef();
7359
7775
  handlers.current = {
@@ -7448,17 +7864,15 @@ const Video = ({
7448
7864
  }, playbackRate);
7449
7865
  }, [playbackRate, playbackState === 'playing']);
7450
7866
  useEffect(() => {
7451
- if (player !== null && player !== void 0 && player.isAlive()) {
7452
- setQuality(videoElement.current, {
7867
+ if (player) {
7868
+ selectVideoResolution(videoElement.current, {
7453
7869
  player
7454
- }, quality);
7870
+ }, videoResolution);
7455
7871
  }
7456
- }, [quality, player]);
7457
- useEffect(() => {
7458
- setAudioTrack(videoElement.current, {
7459
- player
7460
- }, audio);
7461
- }, [(audio === null || audio === void 0 ? void 0 : audio.id) || `${audio.language || 'unk'}:${audio.label || ''}`, player]);
7872
+ }, [videoResolution, player]);
7873
+ useEffect(() => setAudioTrack(videoElement.current, {
7874
+ player
7875
+ }, audioTrack), [((_videoElement$current = videoElement.current) === null || _videoElement$current === void 0 ? void 0 : _videoElement$current.readyState) >= 2 && audioTrack.label, audioTrack.language]);
7462
7876
  useEffect(() => {
7463
7877
  if ((textTracks === null || textTracks === void 0 ? void 0 : textTracks.length) > 0 || textTrack) {
7464
7878
  return syncTextTrack(videoElement.current, {
@@ -7467,12 +7881,12 @@ const Video = ({
7467
7881
  textTracks
7468
7882
  } : textTrack);
7469
7883
  }
7470
- }, [((_videoElement$current = videoElement.current) === null || _videoElement$current === void 0 ? void 0 : _videoElement$current.readyState) >= 2 && textTracks, textTrack]);
7884
+ }, [((_videoElement$current2 = videoElement.current) === null || _videoElement$current2 === void 0 ? void 0 : _videoElement$current2.readyState) >= 2 && textTracks, textTrack]);
7471
7885
  return jsx$1("div", {
7472
7886
  ref: videoContainer,
7473
7887
  css: /*#__PURE__*/css({
7474
7888
  video: baseVideoStyle
7475
- }, process.env.NODE_ENV === "production" ? "" : ";label:Video;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIlZpZGVvLmpzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQWtLOEIiLCJmaWxlIjoiVmlkZW8uanMiLCJzb3VyY2VzQ29udGVudCI6WyIvKiBlc2xpbnQtZGlzYWJsZSBuby1wYXJhbS1yZWFzc2lnbiAqL1xuLyogQGpzeEltcG9ydFNvdXJjZSBAZW1vdGlvbi9yZWFjdCAqL1xuaW1wb3J0IHt1c2VFZmZlY3QsIHVzZUxheW91dEVmZmVjdCwgdXNlUmVmLCB1c2VTdGF0ZX0gZnJvbSAncmVhY3QnXG5cbmltcG9ydCBtdWx0aVJlZiBmcm9tICd1dGlsL211bHRpUmVmJ1xuaW1wb3J0IGxvYWRQbGF5ZXIgZnJvbSAncGxheWVyQ29yZS9sb2FkUGxheWVyJ1xuaW1wb3J0IHtcbiAgZ2V0TWVkaWFUaW1lLFxuICBzdWJzY3JpYmVQbGF5YmFja1N0YXRlLFxuICBsb2FkLFxuICBzeW5jUGxheWJhY2tTdGF0ZSxcbiAgc2VlayxcbiAgc2V0UGxheWJhY2tSYXRlLFxuICBzZXRRdWFsaXR5LFxuICBzZXRBdWRpb1RyYWNrLFxuICBzeW5jVGV4dFRyYWNrLFxufSBmcm9tICdwbGF5ZXJDb3JlL21lZGlhQmluZGluZ3MnXG5cbmNvbnN0IGJhc2VWaWRlb1N0eWxlID0ge1xuICBvYmplY3RGaXQ6ICdjb250YWluJyxcbiAgd2lkdGg6ICcxMDAlJyxcbiAgbWF4SGVpZ2h0OiAnMTAwJScsXG4gICcmOmZ1bGxzY3JlZW4nOiB7XG4gICAgcG9zaXRpb246ICdhYnNvbHV0ZScsXG4gICAgdG9wOiAwLFxuICAgIGxlZnQ6IDAsXG4gICAgYm90dG9tOiAwLFxuICAgIHJpZ2h0OiAwLFxuICB9LFxuICAnfiAuc2hha2EtdGV4dC1jb250YWluZXInOiB7XG4gICAgcG9zaXRpb246ICdhYnNvbHV0ZScsXG4gICAgaGVpZ2h0OiAnMTAwJScsXG4gICAgd2lkdGg6ICcxMDAlJyxcbiAgfSxcbn1cblxuY29uc3QgZW1wdHlBcnJheSA9IFtdXG5cbmNvbnN0IFZpZGVvID0gKHtcbiAgc291cmNlLFxuICBwbGF5YmFja1N0YXRlOiB0YXJnZXRTdGF0ZSxcbiAgY3VycmVudFRpbWU6IHRhcmdldFRpbWUsXG4gIHBsYXliYWNrUmF0ZSxcbiAgdGV4dFRyYWNrcyA9IGVtcHR5QXJyYXksXG4gIHF1YWxpdHksXG4gIHRleHRUcmFjayxcbiAgYXVkaW8gPSB7fSxcbiAgbGl2ZVJlc3VtZSxcbiAgaU9TRnVsbHNjcmVlbkRldGVjdFRocmVzaG9sZCwgLy8gVE9ETyBvcmdhbml6ZSBwcm9wc1xuICBwbHVnaW5zID0gW10sXG4gIHNoYWthLFxuICBiaXRtb3ZpbixcbiAgbGljZW5zZSxcbiAgdmFsaWRhdGlvbkhvc3QsXG4gIHZpZGVvUmVmLFxuICBwbGF5ZXJSZWYsXG4gIG9uUGxheWVyTG9hZGVkLFxuICBvblBsYXliYWNrU3RhdGVDaGFuZ2UsXG4gIG9uQmxvY2tlZEF1dG9wbGF5LFxuICBvblBsYXlsb2dGaXJlZCxcbiAgLi4udmlkZW9BdHRyaWJ1dGVzXG59KSA9PiB7XG4gIGNvbnN0IGhhbmRsZXJzID0gdXNlUmVmKClcbiAgaGFuZGxlcnMuY3VycmVudCA9IHtvblBsYXliYWNrU3RhdGVDaGFuZ2UsIG9uQmxvY2tlZEF1dG9wbGF5fVxuICBjb25zdCB2aWRlb0NvbnRhaW5lciA9IHVzZVJlZigpXG4gIGNvbnN0IHZpZGVvRWxlbWVudCA9IHVzZVJlZigpXG4gIGNvbnN0IFtwbGF5YmFja1N0YXRlLCBzZXRQbGF5YmFja1N0YXRlXSA9IHVzZVN0YXRlKCcnKVxuICBjb25zdCBbcGxheWVyLCBzZXRQbGF5ZXJdID0gdXNlU3RhdGUoKVxuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgY29uc3QgbG9hZFRhc2sgPSBsb2FkUGxheWVyKHZpZGVvRWxlbWVudC5jdXJyZW50LCB7XG4gICAgICBjb250YWluZXI6IHZpZGVvQ29udGFpbmVyLmN1cnJlbnQsXG4gICAgICBzb3VyY2UsXG4gICAgICBzaGFrYSxcbiAgICAgIGJpdG1vdmluLFxuICAgIH0pLnRoZW4oYmFzZVBsYXllciA9PiB7XG4gICAgICBzZXRQbGF5ZXIoYmFzZVBsYXllcilcbiAgICAgIG9uUGxheWVyTG9hZGVkPy4oYmFzZVBsYXllcilcbiAgICAgIGlmIChwbGF5ZXJSZWYpIHtcbiAgICAgICAgcGxheWVyUmVmLmN1cnJldCA9IGJhc2VQbGF5ZXJcbiAgICAgIH1cbiAgICAgIHJldHVybiBiYXNlUGxheWVyXG4gICAgfSlcbiAgICBjb25zdCBvZmZQbGF5YmFja1N0YXRlID0gc3Vic2NyaWJlUGxheWJhY2tTdGF0ZShcbiAgICAgIHZpZGVvRWxlbWVudC5jdXJyZW50LFxuICAgICAgKGV2ZW50LCBzdGF0ZSkgPT4ge1xuICAgICAgICBoYW5kbGVycy5jdXJyZW50Lm9uUGxheWJhY2tTdGF0ZUNoYW5nZT8uKGV2ZW50LCBzdGF0ZSlcbiAgICAgICAgLy8gZXh0ZXJuYWwgbG9naWMgbWF5IHdhbnQgdG8gY2hhbmdlIHRhcmdldFN0YXRlLCBob2xkIHBsYXliYWNrU3RhdGUgdXBkYXRlXG4gICAgICAgIC8vIHRvIHByZXZlbnQgdW53YW50ZWQgc3luY1BsYXliYWNrU3RhdGVcbiAgICAgICAgcmVxdWVzdEFuaW1hdGlvbkZyYW1lKCgpID0+IHNldFBsYXliYWNrU3RhdGUoc3RhdGUpKVxuICAgICAgfSxcbiAgICAgIHtpT1NGdWxsc2NyZWVuRGV0ZWN0VGhyZXNob2xkfVxuICAgIClcbiAgICByZXR1cm4gKCkgPT4ge1xuICAgICAgb2ZmUGxheWJhY2tTdGF0ZSgpXG4gICAgICBsb2FkVGFzay50aGVuKGN1cnJlbnRQbGF5ZXIgPT4gY3VycmVudFBsYXllcj8uZGVzdHJveSgpKVxuICAgIH1cbiAgfSwgW10pXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKFxuICAgICAgc291cmNlICYmXG4gICAgICAoc291cmNlLmxlbmd0aCA+IDAgfHwgc291cmNlLnNyYyB8fCBzb3VyY2UuaGxzIHx8IHNvdXJjZS5kYXNoKSAmJlxuICAgICAgcGxheWVyPy5pc0FsaXZlKClcbiAgICApIHtcbiAgICAgIFByb21pc2UucmVzb2x2ZShmYWxzZSlcbiAgICAgICAgLnRoZW4oKCkgPT5cbiAgICAgICAgICBsb2FkKFxuICAgICAgICAgICAgdmlkZW9FbGVtZW50LmN1cnJlbnQsXG4gICAgICAgICAgICB7cGxheWVyLCBwbHVnaW5zLCBzdGFydFRpbWU6IHRhcmdldFRpbWV9LFxuICAgICAgICAgICAgc291cmNlXG4gICAgICAgICAgKVxuICAgICAgICApXG4gICAgICAgIC5jYXRjaChlcnJvciA9PiBjb25zb2xlLndhcm4oZXJyb3IpKVxuICAgIH1cbiAgICBpZiAoKCFzb3VyY2UgfHwgc291cmNlLmxlbmd0aCA9PT0gMCkgJiYgcGxheWVyKSB7XG4gICAgICBwbGF5ZXIudW5sb2FkKClcbiAgICAgIHBsYXllci5sYXN0U3JjID0gJydcbiAgICB9XG4gIH0sIFtwbGF5ZXIsIHNvdXJjZV0pXG5cbiAgLy8gdXNlRWZmZWN0IGlzIHRvbyBsYXRlIHRvIHVubG9jayBwbGF5IG9uIFNhZmFyaVxuICAvLyBUT0RPIGNoZWNrIGlmIHRoaXMgd29yayBhZnRlciB1cGdyYWRpbmcgUmVhY3QgMThcbiAgdXNlTGF5b3V0RWZmZWN0KCgpID0+IHtcbiAgICBpZiAocGxheWVyPy5pc0FsaXZlKCkpIHtcbiAgICAgIHN5bmNQbGF5YmFja1N0YXRlKFxuICAgICAgICB2aWRlb0VsZW1lbnQuY3VycmVudCxcbiAgICAgICAge3BsYXllciwgcGx1Z2lucywgbGl2ZVJlc3VtZX0sXG4gICAgICAgIHRhcmdldFN0YXRlXG4gICAgICApPy5jYXRjaChlcnJvciA9PiB7XG4gICAgICAgIGhhbmRsZXJzLmN1cnJlbnQub25CbG9ja2VkQXV0b3BsYXk/LihlcnJvcilcbiAgICAgIH0pXG4gICAgfVxuICB9LCBbcGxheWVyICYmIHRhcmdldFN0YXRlXSlcbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBjb25zdCB7Y3VycmVudFRpbWV9ID0gZ2V0TWVkaWFUaW1lKHZpZGVvRWxlbWVudC5jdXJyZW50LCB7cGxheWVyLCBwbHVnaW5zfSlcbiAgICBpZiAocGxheWVyPy5pc0FsaXZlKCkgJiYgTWF0aC5hYnMoY3VycmVudFRpbWUgLSB0YXJnZXRUaW1lKSA+IDAuNSkge1xuICAgICAgLy8gc2Vla2luZyB1bmF2YWlsYWJsZSBjYXNlcyBhcmUgaGFuZGxlZCBieSBzZWVrIGZ1bmN0aW9uXG4gICAgICBzZWVrKHZpZGVvRWxlbWVudC5jdXJyZW50LCB7cGxheWVyLCBwbHVnaW5zfSwgdGFyZ2V0VGltZSlcbiAgICB9XG4gIH0sIFtwbGF5ZXIgJiYgdGFyZ2V0VGltZV0pXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgc2V0UGxheWJhY2tSYXRlKHZpZGVvRWxlbWVudC5jdXJyZW50LCB7cGxheWVyfSwgcGxheWJhY2tSYXRlKVxuICB9LCBbcGxheWJhY2tSYXRlLCBwbGF5YmFja1N0YXRlID09PSAncGxheWluZyddKVxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGlmIChwbGF5ZXI/LmlzQWxpdmUoKSkge1xuICAgICAgc2V0UXVhbGl0eSh2aWRlb0VsZW1lbnQuY3VycmVudCwge3BsYXllcn0sIHF1YWxpdHkpXG4gICAgfVxuICB9LCBbcXVhbGl0eSwgcGxheWVyXSlcbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBzZXRBdWRpb1RyYWNrKHZpZGVvRWxlbWVudC5jdXJyZW50LCB7cGxheWVyfSwgYXVkaW8pXG4gIH0sIFthdWRpbz8uaWQgfHwgYCR7YXVkaW8ubGFuZ3VhZ2UgfHwgJ3Vuayd9OiR7YXVkaW8ubGFiZWwgfHwgJyd9YCwgcGxheWVyXSlcbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBpZiAodGV4dFRyYWNrcz8ubGVuZ3RoID4gMCB8fCB0ZXh0VHJhY2spIHtcbiAgICAgIHJldHVybiBzeW5jVGV4dFRyYWNrKFxuICAgICAgICB2aWRlb0VsZW1lbnQuY3VycmVudCxcbiAgICAgICAge3BsYXllcn0sXG4gICAgICAgIHRleHRUcmFja3M/Lmxlbmd0aCA+IDAgPyB7dGV4dFRyYWNrc30gOiB0ZXh0VHJhY2tcbiAgICAgIClcbiAgICB9XG4gIH0sIFt2aWRlb0VsZW1lbnQuY3VycmVudD8ucmVhZHlTdGF0ZSA+PSAyICYmIHRleHRUcmFja3MsIHRleHRUcmFja10pXG5cbiAgcmV0dXJuIChcbiAgICA8ZGl2IHJlZj17dmlkZW9Db250YWluZXJ9IGNzcz17e3ZpZGVvOiBiYXNlVmlkZW9TdHlsZX19PlxuICAgICAgPHZpZGVvIC8vIGVzbGludC1kaXNhYmxlLWxpbmUganN4LWExMXkvbWVkaWEtaGFzLWNhcHRpb25cbiAgICAgICAgcmVmPXttdWx0aVJlZih2aWRlb1JlZiwgdmlkZW9FbGVtZW50KX1cbiAgICAgICAgcGxheXNJbmxpbmVcbiAgICAgICAgey4uLnZpZGVvQXR0cmlidXRlc31cbiAgICAgIC8+XG4gICAgPC9kaXY+XG4gIClcbn1cblxuZXhwb3J0IGRlZmF1bHQgVmlkZW9cbiJdfQ== */"),
7889
+ }, 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 */"),
7476
7890
  children: jsx$1("video", {
7477
7891
  // eslint-disable-line jsx-a11y/media-has-caption
7478
7892
  ref: multiRef(videoRef, videoElement),
@@ -7484,231 +7898,6 @@ const Video = ({
7484
7898
 
7485
7899
  var Video$1 = Video;
7486
7900
 
7487
- const getSpeedItems = items => items.map(value => ({
7488
- label: `${value}x`,
7489
- value
7490
- }));
7491
-
7492
- const getQualityOptions = ({
7493
- sections,
7494
- values: {
7495
- quality
7496
- }
7497
- }) => {
7498
- var _sections$find, _sections$find$items$;
7499
-
7500
- return (_sections$find = sections.find(item => item.name === 'quality')) === null || _sections$find === void 0 ? void 0 : (_sections$find$items$ = _sections$find.items.find(item => item.value === quality)) === null || _sections$find$items$ === void 0 ? void 0 : _sections$find$items$.options;
7501
- };
7502
-
7503
- const autoQualityOption = {
7504
- label: `Auto`,
7505
- options: {
7506
- maxHeight: Infinity,
7507
- minHeight: 0
7508
- },
7509
- value: 'auto'
7510
- };
7511
-
7512
- const getQualityItemsFromManifest = player => {
7513
- var _player$getAvailableV;
7514
-
7515
- return [autoQualityOption].concat(((player === null || player === void 0 ? void 0 : (_player$getAvailableV = player.getAvailableVideoQualities) === null || _player$getAvailableV === void 0 ? void 0 : _player$getAvailableV.call(player)) || []).filter(q => q.height > 0).sort((a, b) => b.height - a.height).map(q => ({
7516
- label: `${q.height}p`,
7517
- // Set the min/max height to the same value to fix the quality.
7518
- options: {
7519
- maxHeight: q.height,
7520
- minHeight: q.height
7521
- },
7522
- value: q.height
7523
- })));
7524
- };
7525
-
7526
- const getQualitySettings = (options, player) => {
7527
- // With native HLS, manifest rewrite is required to enable quality setting
7528
- // TODO let this covered by test, maybe refactor?
7529
- const items = isSafari() && !options.rewriteManifest ? [] : options.items || getQualityItemsFromManifest(player);
7530
- return items.length > 0 && items[0] && {
7531
- name: 'quality',
7532
- title: 'KKS.QUALITY',
7533
- items,
7534
- getDefault: (preferred = options.default || items[0].value) => {
7535
- const maxHeight = preferred || items[0].value;
7536
- return (nearest(items.filter(item => (item.height || item.value) <= maxHeight), item => (item.height || item.value) - maxHeight) || items[0]).value;
7537
- }
7538
- };
7539
- };
7540
-
7541
- const getSelectedAudioName = player => {
7542
- const track = getCurrentAudioTrack({}, {
7543
- player
7544
- });
7545
- /*
7546
- Sometimes, HLS manifest doesn't describe the default audio track.
7547
- Get current audio track information is undefined even though the player still has audio streaming.
7548
- For this case, we select first audio track.
7549
- More detail please refer to OTP-3450.
7550
- */
7551
-
7552
- const audioList = getAudioTracks({}, {
7553
- player
7554
- });
7555
- const currentAudio = track !== undefined ? track : audioList === null || audioList === void 0 ? void 0 : audioList[0];
7556
- return currentAudio && { ...currentAudio,
7557
- id: [currentAudio.language, currentAudio.label].join(':')
7558
- };
7559
- };
7560
-
7561
- const getDefault = (section, {
7562
- preferred,
7563
- player
7564
- }) => {
7565
- if (typeof section.getDefault === 'function') {
7566
- return section.getDefault(preferred);
7567
- }
7568
-
7569
- if (section.name === 'audio') {
7570
- return preferred !== null && preferred !== void 0 ? preferred : getSelectedAudioName(player);
7571
- }
7572
-
7573
- if (section.name === 'subtitles') {
7574
- var _ref;
7575
-
7576
- return ((_ref = section.items.find(track => {
7577
- var _track$value, _track$value2;
7578
-
7579
- return ((_track$value = track.value) === null || _track$value === void 0 ? void 0 : _track$value.label) === (preferred === null || preferred === void 0 ? void 0 : preferred.label) && ((_track$value2 = track.value) === null || _track$value2 === void 0 ? void 0 : _track$value2.language) === (preferred === null || preferred === void 0 ? void 0 : preferred.language);
7580
- }) || section.items.find(track => {
7581
- var _track$value3;
7582
-
7583
- return ((_track$value3 = track.value) === null || _track$value3 === void 0 ? void 0 : _track$value3.language) === (preferred === null || preferred === void 0 ? void 0 : preferred.language);
7584
- })) === null || _ref === void 0 ? void 0 : _ref.value) || {
7585
- language: 'off',
7586
- id: 'off'
7587
- };
7588
- }
7589
-
7590
- if (section.name === 'speed') {
7591
- return preferred !== null && preferred !== void 0 ? preferred : 1;
7592
- }
7593
- };
7594
-
7595
- const shouldProvideSettingSection = ({
7596
- name,
7597
- items
7598
- }, {
7599
- player,
7600
- loop
7601
- }) => {
7602
- // TODO keep special section
7603
- // name === 'audio' || name === 'subtitles'
7604
- if (name === 'loop') {
7605
- return !(player !== null && player !== void 0 && player.isLive()) && loop !== 'disabled';
7606
- }
7607
-
7608
- if (name === 'speed') {
7609
- return !(player !== null && player !== void 0 && player.isLive());
7610
- }
7611
-
7612
- return (items === null || items === void 0 ? void 0 : items.length) > 1;
7613
- };
7614
-
7615
- const getSettingsData = ({
7616
- media,
7617
- player,
7618
- source = [],
7619
- quality = {},
7620
- speedItems = [0.25, 0.5, 0.75, 1, 1.25, 1.5, 1.75, 2],
7621
- loop,
7622
- preferred = {},
7623
- otherSections = []
7624
- }) => {
7625
- // TODO extract base player specific things
7626
- const audioItems = getAudioTracks({}, {
7627
- player
7628
- }).filter(track => track.language && track.language !== 'und').map(track => ({
7629
- label: track.label || track.language,
7630
- value: {
7631
- language: track.language,
7632
- label: track.label,
7633
- id: [track.language, track.label].join(':')
7634
- }
7635
- })) || [];
7636
- const selectedSource = getSource(source, {
7637
- preferManifestType: 'platform'
7638
- }) || {};
7639
- const currentSpeedItems = getSpeedItems(speedItems);
7640
- const sections = [].concat(otherSections).concat([quality && getQualitySettings({ ...quality,
7641
- items: selectedSource.qualityOptions
7642
- }, player), {
7643
- name: 'subtitles',
7644
- title: 'KKS.SUBTITLES',
7645
- items: [...getTextTracks({}, {
7646
- player
7647
- }).map(track => ({
7648
- label: track.label || track.language,
7649
- value: {
7650
- language: track.language,
7651
- label: track.label,
7652
- id: [track.language, track.label].join(':')
7653
- }
7654
- })), {
7655
- label: 'OFF',
7656
- value: {
7657
- language: 'off',
7658
- id: 'off'
7659
- }
7660
- }]
7661
- }, {
7662
- name: 'audio',
7663
- title: 'KKS.AUDIO',
7664
- items: audioItems
7665
- }, {
7666
- name: 'speed',
7667
- title: 'KKS.SETTING.SPEED',
7668
- items: currentSpeedItems
7669
- }, {
7670
- type: 'switch',
7671
- name: 'loop',
7672
- title: 'KKS.SETTING.LOOP'
7673
- }]).filter(section => shouldProvideSettingSection(section, {
7674
- player,
7675
- loop,
7676
- quality
7677
- }));
7678
- const values = sections.reduce((result, section) => {
7679
- // TODO take fallback option if preferred is not available
7680
- // eslint-disable-next-line no-param-reassign
7681
- result[section.name] = getDefault(section, {
7682
- preferred: preferred[section.name],
7683
- player,
7684
- media
7685
- }) || preferred[section.name];
7686
- return result;
7687
- }, // TODO forced subtitles?
7688
- {
7689
- audio: getSelectedAudioName(player),
7690
- subtitles: 'off'
7691
- });
7692
-
7693
- if (!values.speed && (media === null || media === void 0 ? void 0 : media.playbackRate) > 0) {
7694
- values.speed = media.playbackRate;
7695
- }
7696
-
7697
- if (typeof preferred.loop === 'boolean') {
7698
- values.loop = preferred.loop;
7699
- } else {
7700
- values.loop = loop === 'disabled' ? false : Boolean(loop);
7701
- dispatchCustomEvent(media, 'loopChange', {
7702
- loop: values.loop
7703
- });
7704
- }
7705
-
7706
- return {
7707
- sections,
7708
- values
7709
- };
7710
- };
7711
-
7712
7901
  const parseVTT = data => {
7713
7902
  const lines = data.split(/\n\n/g).slice(1); // may replace with async parser to prevent blocking render
7714
7903
 
@@ -7825,11 +8014,11 @@ const SeekPreview = ({
7825
8014
 
7826
8015
  if (thumbnailsUrl) {
7827
8016
  // TODO replace with fetch
7828
- fetch(thumbnailsUrl).then(response => response.text()).then(data => {
8017
+ fetch(thumbnailsUrl).then(response => response.text()).then(data => parseThumbnails(data, thumbnailsUrl)).then(parsed => {
7829
8018
  if (!cancel) {
7830
- setThumbnails(parseThumbnails(data, thumbnailsUrl));
8019
+ setThumbnails(parsed);
7831
8020
  }
7832
- });
8021
+ }).catch(error => console.debug('Thumbnail', error));
7833
8022
  }
7834
8023
 
7835
8024
  return () => {
@@ -7886,7 +8075,40 @@ const defaultSettings = {};
7886
8075
  const shouldPause = ({
7887
8076
  userFocus,
7888
8077
  uiType
7889
- }) => uiType === 'mobile' && /settings|chapter-list/.test(userFocus) || /blocking|seekbar|share-inner/.test(userFocus);
8078
+ }) => uiType === 'mobile' && /settings|chapter-list/.test(userFocus) || /blocking|seekbar|share-inner/.test(userFocus); // TODO extract to somewhere else
8079
+
8080
+
8081
+ const menuClasses = {
8082
+ 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":"AA2EW","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 {\n  getLanguageOptions,\n  getQualityOptions,\n  getSettingsData,\n} 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    options.audioTracks.length > 0 &&\n    options.textTracks.length > 0 && (\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}\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    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    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) => 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"]} */"),
8083
+ 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":"AA4EW","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 {\n  getLanguageOptions,\n  getQualityOptions,\n  getSettingsData,\n} 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    options.audioTracks.length > 0 &&\n    options.textTracks.length > 0 && (\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}\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    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    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) => 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"]} */")
8084
+ };
8085
+
8086
+ const LanguageSettings = ({
8087
+ uiType,
8088
+ audioTracks = [],
8089
+ textTracks = [],
8090
+ getPlayer,
8091
+ onChange,
8092
+ slots = {
8093
+ LanguageMenu
8094
+ },
8095
+ ...rest
8096
+ }) => {
8097
+ const options = getLanguageOptions(getPlayer(), {
8098
+ audioTracks,
8099
+ textTracks
8100
+ });
8101
+ return options.audioTracks.length > 0 && options.textTracks.length > 0 && /*#__PURE__*/jsx(OverlayPanel, {
8102
+ buttonIcon: "subtitle",
8103
+ children: /*#__PURE__*/jsx(slots.LanguageMenu, {
8104
+ uiType: uiType,
8105
+ classes: menuClasses,
8106
+ sectionOptions: [options.audioTracks, options.textTracks],
8107
+ onChange: onChange,
8108
+ ...rest
8109
+ })
8110
+ });
8111
+ };
7890
8112
 
7891
8113
  const PremiumPlayer = ({
7892
8114
  source,
@@ -7901,6 +8123,8 @@ const PremiumPlayer = ({
7901
8123
  playbackRate: appPlaybackRate,
7902
8124
  loop,
7903
8125
  volume: appVolume,
8126
+ audioTrack: appAudioTrack,
8127
+ textTrack: appTextTrack,
7904
8128
  thumbnailsUrl,
7905
8129
  controls = {
7906
8130
  autohide: 3000
@@ -7932,6 +8156,8 @@ const PremiumPlayer = ({
7932
8156
  onPlaylogFired,
7933
8157
  ...videoProps
7934
8158
  }) => {
8159
+ var _settings$preferred, _settings$preferred2;
8160
+
7935
8161
  const components = {
7936
8162
  VolumeControl,
7937
8163
  CastUi,
@@ -8217,7 +8443,7 @@ const PremiumPlayer = ({
8217
8443
  '--playing': playbackState === 'playing' ? '1' : '0'
8218
8444
  };
8219
8445
 
8220
- const onRewind = (!isLive || playbackTime.currentTime < 0) && !isEnd && (() => setTargetTime(playbackTime.currentTime - 10));
8446
+ const onRewind = (!isLive || playbackTime.currentTime < 0) && canSeek && (() => setTargetTime(playbackTime.currentTime - 10));
8221
8447
 
8222
8448
  const onForward = (!isLive || playbackTime.currentTime < 0) && canSeek && (() => setTargetTime(playbackTime.currentTime + 10));
8223
8449
 
@@ -8351,6 +8577,8 @@ const PremiumPlayer = ({
8351
8577
  } // TODO playlist, audio / subtitle setting
8352
8578
 
8353
8579
  }, [source, castOptions.contentId, castOptions.customData]);
8580
+ const selectedAudioTrack = ((_settings$preferred = settings.preferred) === null || _settings$preferred === void 0 ? void 0 : _settings$preferred.audio) || appAudioTrack;
8581
+ const selectedTextTrack = ((_settings$preferred2 = settings.preferred) === null || _settings$preferred2 === void 0 ? void 0 : _settings$preferred2.subtitles) || appTextTrack;
8354
8582
  return /*#__PURE__*/jsx(IntlProvider, { ...intl,
8355
8583
  children: /*#__PURE__*/jsxs$1(DefaultLayout, {
8356
8584
  style: { ...style,
@@ -8370,9 +8598,9 @@ const PremiumPlayer = ({
8370
8598
  muted: appVolume <= 0
8371
8599
  }),
8372
8600
  playbackRate: settings.values.speed,
8373
- quality: qualityOptions,
8374
- audio: settings.values.audio,
8375
- textTrack: settings.values.subtitles,
8601
+ videoResolution: qualityOptions,
8602
+ audioTrack: selectedAudioTrack,
8603
+ textTrack: selectedTextTrack,
8376
8604
  loop: settings.values.loop,
8377
8605
  plugins: plugins,
8378
8606
  modulesConfig: modulesConfig,
@@ -8432,6 +8660,11 @@ const PremiumPlayer = ({
8432
8660
  onChange: uiHandlers.onChangeSettings,
8433
8661
  ...(slotProps === null || slotProps === void 0 ? void 0 : slotProps.settings),
8434
8662
  onClose: uiHandlers.onCloseSettings
8663
+ }), /*#__PURE__*/jsx(LanguageSettings, {
8664
+ uiType: uiType,
8665
+ getPlayer: () => playerRef.current,
8666
+ onChange: (_event, item) => changeSettings(item.type, item.value),
8667
+ ...slotProps.languageSettings
8435
8668
  }), /*#__PURE__*/jsx(Backdrop, {
8436
8669
  open: !playbackState || playbackState === 'loading',
8437
8670
  children: /*#__PURE__*/jsx(LoadingSpinner, {})
@@ -9028,9 +9261,9 @@ class Player {
9028
9261
  player: this.player
9029
9262
  }) || []);
9030
9263
 
9031
- _defineProperty(this, "getCurrentAudioTrack", () => getCurrentAudioTrack({}, {
9264
+ _defineProperty(this, "getCurrentAudioTrack", () => getAudioTracks({}, {
9032
9265
  player: this.player
9033
- }));
9266
+ }).find(track => track.selected));
9034
9267
 
9035
9268
  _defineProperty(this, "setAudioTrack", track => {
9036
9269
  // TODO also accept Track, name: setTextTrack