@kkcompany/player 2.25.0-canary.0 → 2.25.0-canary.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +15 -7
- package/dist/core.mjs +14 -14
- package/dist/index.js +439 -431
- package/dist/index.mjs +335 -326
- package/dist/modules.mjs +11 -10
- package/dist/react.mjs +150 -141
- package/package.json +1 -1
package/dist/index.mjs
CHANGED
|
@@ -76,15 +76,15 @@ function getDevice() {
|
|
|
76
76
|
if (device.type === undefined && osName === 'Android') device.type = 'tablet';
|
|
77
77
|
return device;
|
|
78
78
|
}
|
|
79
|
+
|
|
80
|
+
const isSafari = () => /^((?!chrome|android|X11|Linux).)*(safari|iPad|iPhone|Version)/i.test(navigator.userAgent);
|
|
81
|
+
|
|
79
82
|
function needNativeHls() {
|
|
80
83
|
// Don't let Android phones play HLS, even if some of them report supported
|
|
81
84
|
// This covers Samsung & OPPO special cases
|
|
82
|
-
const isAndroid = /android|X11|Linux/i.test(navigator.userAgent);
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
// none of our supported browsers other than Safari response to this
|
|
86
|
-
|
|
87
|
-
return isAndroid || /firefox/i.test(navigator.userAgent) ? '' : isSafari ? 'maybe' : document.createElement('video').canPlayType('application/vnd.apple.mpegURL');
|
|
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
88
|
}
|
|
89
89
|
|
|
90
90
|
const isDesktop = () => !getDevice().type; // TODO solve lint error:
|
|
@@ -174,7 +174,7 @@ const getDrmOptions$1 = fallbackDrm => {
|
|
|
174
174
|
* @typedef {{hls: string, dash: string}} SourceObjectAlt backward compatiable form
|
|
175
175
|
*
|
|
176
176
|
* @param {SourceObject[]|SourceObject|SourceObjectAlt|string} sourceOptions
|
|
177
|
-
* @param {{preferManifestType?: ('dash'|'hls')}} options
|
|
177
|
+
* @param {{preferManifestType?: ('dash'|'hls'|'platform')}} options
|
|
178
178
|
* @return {{src: string, type: string, drm: Object}}
|
|
179
179
|
*/
|
|
180
180
|
|
|
@@ -217,7 +217,8 @@ const getSource = (sourceOptions, {
|
|
|
217
217
|
});
|
|
218
218
|
}
|
|
219
219
|
|
|
220
|
-
const
|
|
220
|
+
const targetType = preferManifestType !== 'platform' ? preferManifestType : isSafari() ? 'hls' : 'dash';
|
|
221
|
+
const matched = sourceOptions.find(source => matchType(source, targetType));
|
|
221
222
|
const selected = matched || sourceOptions[0];
|
|
222
223
|
|
|
223
224
|
if (!selected) {
|
|
@@ -294,7 +295,7 @@ function convertToSeconds(timeString) {
|
|
|
294
295
|
function getVersion() {
|
|
295
296
|
try {
|
|
296
297
|
// eslint-disable-next-line no-undef
|
|
297
|
-
return "2.25.0-canary.
|
|
298
|
+
return "2.25.0-canary.2";
|
|
298
299
|
} catch (e) {
|
|
299
300
|
return undefined;
|
|
300
301
|
}
|
|
@@ -595,9 +596,8 @@ const load = async (media, {
|
|
|
595
596
|
startTime,
|
|
596
597
|
plugins = []
|
|
597
598
|
}, source) => {
|
|
598
|
-
const preferManifestType = needNativeHls() ? 'hls' : 'dash';
|
|
599
599
|
const preferred = getSource(source, {
|
|
600
|
-
preferManifestType
|
|
600
|
+
preferManifestType: 'platform'
|
|
601
601
|
}); // There's no use case that changing DRM options without changing manifest URL, just skip
|
|
602
602
|
|
|
603
603
|
if (player.lastSrc === (preferred === null || preferred === void 0 ? void 0 : preferred.src)) {
|
|
@@ -617,7 +617,7 @@ const load = async (media, {
|
|
|
617
617
|
player,
|
|
618
618
|
source: currentSource,
|
|
619
619
|
startTime,
|
|
620
|
-
streamFormat:
|
|
620
|
+
streamFormat: source.type,
|
|
621
621
|
reload: async () => {
|
|
622
622
|
// Bitmovin unexpectedly restores muted state, so save to restore
|
|
623
623
|
const restoreMuted = player.isMuted && {
|
|
@@ -822,31 +822,32 @@ const setPlaybackRate = (media, {
|
|
|
822
822
|
const getTextTracks = (_, {
|
|
823
823
|
player
|
|
824
824
|
}) => {
|
|
825
|
-
var _player$
|
|
825
|
+
var _player$getTextTracks;
|
|
826
826
|
|
|
827
827
|
if (!player) return [];
|
|
828
|
+
return (((_player$getTextTracks = player.getTextTracks) === null || _player$getTextTracks === void 0 ? void 0 : _player$getTextTracks.call(player)) || []).map(track => {
|
|
829
|
+
var _track$label;
|
|
828
830
|
|
|
829
|
-
|
|
830
|
-
|
|
831
|
-
|
|
832
|
-
|
|
833
|
-
|
|
834
|
-
|
|
835
|
-
|
|
836
|
-
|
|
837
|
-
|
|
838
|
-
|
|
839
|
-
|
|
840
|
-
|
|
841
|
-
|
|
842
|
-
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)) || [];
|
|
831
|
+
const label = ((_track$label = track.label) === null || _track$label === void 0 ? void 0 : _track$label.trim()) || track.language;
|
|
832
|
+
const id = `${track.language}:${label}`;
|
|
833
|
+
return { ...track,
|
|
834
|
+
label,
|
|
835
|
+
type: 'subtitles',
|
|
836
|
+
selected: player.isTextTrackVisible() ? track.active : false,
|
|
837
|
+
value: {
|
|
838
|
+
id,
|
|
839
|
+
label,
|
|
840
|
+
language: track.language
|
|
841
|
+
}
|
|
842
|
+
};
|
|
843
|
+
}).filter(track => track.label !== 'unknown' && track.language && !/^un/i.test(track.language));
|
|
843
844
|
};
|
|
844
845
|
|
|
845
846
|
const getCurrentSubtitle = (_, {
|
|
846
847
|
player
|
|
847
848
|
}) => getTextTracks({}, {
|
|
848
849
|
player
|
|
849
|
-
}).find(track => track.
|
|
850
|
+
}).find(track => track.selected) || 'off';
|
|
850
851
|
|
|
851
852
|
const textTrackLabel = 'playcraft-text-track';
|
|
852
853
|
/** @param media HTMLMediaElement */
|
|
@@ -910,6 +911,82 @@ const syncTextTrack = (media, {
|
|
|
910
911
|
player.setTextTrackVisibility(false);
|
|
911
912
|
}
|
|
912
913
|
});
|
|
914
|
+
}; // assume unique labels as defined in spec
|
|
915
|
+
|
|
916
|
+
|
|
917
|
+
const setAudioTrack = (_, {
|
|
918
|
+
player
|
|
919
|
+
}, next) => {
|
|
920
|
+
if (!next) return;
|
|
921
|
+
|
|
922
|
+
try {
|
|
923
|
+
player.selectAudioLanguage(next.language);
|
|
924
|
+
|
|
925
|
+
if (next.label) {
|
|
926
|
+
player.selectVariantsByLabel(next.label);
|
|
927
|
+
}
|
|
928
|
+
} catch (error) {
|
|
929
|
+
console.warn('Unable to set audio', error, next);
|
|
930
|
+
}
|
|
931
|
+
};
|
|
932
|
+
|
|
933
|
+
/* eslint-disable no-param-reassign */
|
|
934
|
+
|
|
935
|
+
const getSpeedItems = items => items.map(value => ({
|
|
936
|
+
label: `${value}x`,
|
|
937
|
+
value
|
|
938
|
+
}));
|
|
939
|
+
|
|
940
|
+
const getQualityOptions = ({
|
|
941
|
+
sections,
|
|
942
|
+
values: {
|
|
943
|
+
quality
|
|
944
|
+
}
|
|
945
|
+
}) => {
|
|
946
|
+
var _sections$find, _sections$find$items$;
|
|
947
|
+
|
|
948
|
+
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;
|
|
949
|
+
};
|
|
950
|
+
|
|
951
|
+
const autoQualityOption = {
|
|
952
|
+
label: `Auto`,
|
|
953
|
+
options: {
|
|
954
|
+
maxHeight: Infinity,
|
|
955
|
+
minHeight: 0
|
|
956
|
+
},
|
|
957
|
+
value: 'auto'
|
|
958
|
+
};
|
|
959
|
+
|
|
960
|
+
const getQualityItemsFromManifest = player => {
|
|
961
|
+
var _player$getAvailableV;
|
|
962
|
+
|
|
963
|
+
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 => ({
|
|
964
|
+
label: `${q.height}p`,
|
|
965
|
+
// Set the min/max height to the same value to fix the quality.
|
|
966
|
+
options: {
|
|
967
|
+
maxHeight: q.height,
|
|
968
|
+
minHeight: q.height
|
|
969
|
+
},
|
|
970
|
+
value: q.height
|
|
971
|
+
})));
|
|
972
|
+
};
|
|
973
|
+
|
|
974
|
+
const getQualitySettings = (options, player) => {
|
|
975
|
+
// With native HLS, manifest rewrite is required to enable quality setting
|
|
976
|
+
// TODO let this covered by test, maybe refactor?
|
|
977
|
+
const items = isSafari() && !options.rewriteManifest ? [] : options.items || getQualityItemsFromManifest(player);
|
|
978
|
+
|
|
979
|
+
const getDefault = (preferred = options.default || items[0].value) => {
|
|
980
|
+
const maxHeight = preferred || items[0].value;
|
|
981
|
+
return (nearest(items.filter(item => (item.height || item.value) <= maxHeight), item => (item.height || item.value) - maxHeight) || items[0]).value;
|
|
982
|
+
};
|
|
983
|
+
|
|
984
|
+
return items.length > 0 && items[0] && {
|
|
985
|
+
name: 'quality',
|
|
986
|
+
title: 'KKS.QUALITY',
|
|
987
|
+
items,
|
|
988
|
+
getDefault
|
|
989
|
+
};
|
|
913
990
|
};
|
|
914
991
|
|
|
915
992
|
const getAudioTracks = (_, {
|
|
@@ -918,50 +995,217 @@ const getAudioTracks = (_, {
|
|
|
918
995
|
if (!player) return []; // Bitmovin support is omitted as it is not needed in near future
|
|
919
996
|
|
|
920
997
|
const all = player.getVariantTracks().reduce((result, track) => {
|
|
921
|
-
var _track$
|
|
998
|
+
var _track$label;
|
|
922
999
|
|
|
923
|
-
const label = ((_track$
|
|
1000
|
+
const label = ((_track$label = track.label) === null || _track$label === void 0 ? void 0 : _track$label.trim()) || track.language;
|
|
924
1001
|
const id = `${track.language}:${label}`;
|
|
925
|
-
result[id] = { ...track,
|
|
926
|
-
|
|
1002
|
+
result[id] = result[id] || { ...track,
|
|
1003
|
+
type: 'audio',
|
|
927
1004
|
label,
|
|
928
|
-
|
|
1005
|
+
value: {
|
|
1006
|
+
id,
|
|
1007
|
+
label,
|
|
1008
|
+
language: track.language
|
|
1009
|
+
}
|
|
929
1010
|
};
|
|
1011
|
+
|
|
1012
|
+
if (track.active) {
|
|
1013
|
+
result[id].selected = true;
|
|
1014
|
+
}
|
|
1015
|
+
|
|
930
1016
|
return result;
|
|
931
1017
|
}, {});
|
|
932
|
-
|
|
1018
|
+
const allTracks = Object.values(all);
|
|
1019
|
+
const namedTracks = allTracks.filter(track => !/^un/i.test(track.language || ''));
|
|
1020
|
+
return Object.values(namedTracks.length > 0 ? namedTracks : allTracks);
|
|
933
1021
|
};
|
|
934
1022
|
|
|
935
1023
|
const getCurrentAudioTrack = (media, {
|
|
936
1024
|
player
|
|
937
1025
|
}) => getAudioTracks(media, {
|
|
938
1026
|
player
|
|
939
|
-
}).find(track => track.
|
|
1027
|
+
}).find(track => track.selected);
|
|
940
1028
|
|
|
1029
|
+
const getSelectedAudioName = player => {
|
|
1030
|
+
const track = getCurrentAudioTrack({}, {
|
|
1031
|
+
player
|
|
1032
|
+
});
|
|
1033
|
+
/*
|
|
1034
|
+
Sometimes, HLS manifest doesn't describe the default audio track.
|
|
1035
|
+
Get current audio track information is undefined even though the player still has audio streaming.
|
|
1036
|
+
For this case, we select first audio track.
|
|
1037
|
+
More detail please refer to OTP-3450.
|
|
1038
|
+
*/
|
|
941
1039
|
|
|
942
|
-
const
|
|
1040
|
+
const audioList = getAudioTracks({}, {
|
|
1041
|
+
player
|
|
1042
|
+
});
|
|
1043
|
+
const currentAudio = track !== undefined ? track : audioList === null || audioList === void 0 ? void 0 : audioList[0];
|
|
1044
|
+
return currentAudio && { ...currentAudio,
|
|
1045
|
+
id: [currentAudio.language, currentAudio.label].join(':')
|
|
1046
|
+
};
|
|
1047
|
+
};
|
|
1048
|
+
|
|
1049
|
+
const getDefault = (section, {
|
|
1050
|
+
preferred,
|
|
943
1051
|
player
|
|
944
|
-
}
|
|
945
|
-
if (
|
|
1052
|
+
}) => {
|
|
1053
|
+
if (typeof section.getDefault === 'function') {
|
|
1054
|
+
return section.getDefault(preferred);
|
|
1055
|
+
}
|
|
946
1056
|
|
|
947
|
-
|
|
948
|
-
|
|
1057
|
+
if (section.name === 'audio') {
|
|
1058
|
+
return preferred !== null && preferred !== void 0 ? preferred : getSelectedAudioName(player);
|
|
1059
|
+
}
|
|
1060
|
+
|
|
1061
|
+
if (section.name === 'subtitles') {
|
|
1062
|
+
var _ref;
|
|
1063
|
+
|
|
1064
|
+
return ((_ref = section.items.find(track => {
|
|
1065
|
+
var _track$value, _track$value2;
|
|
1066
|
+
|
|
1067
|
+
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);
|
|
1068
|
+
}) || section.items.find(track => {
|
|
1069
|
+
var _track$value3;
|
|
1070
|
+
|
|
1071
|
+
return ((_track$value3 = track.value) === null || _track$value3 === void 0 ? void 0 : _track$value3.language) === (preferred === null || preferred === void 0 ? void 0 : preferred.language);
|
|
1072
|
+
})) === null || _ref === void 0 ? void 0 : _ref.value) || {
|
|
1073
|
+
language: 'off',
|
|
1074
|
+
id: 'off'
|
|
1075
|
+
};
|
|
1076
|
+
}
|
|
1077
|
+
|
|
1078
|
+
if (section.name === 'speed') {
|
|
1079
|
+
return preferred !== null && preferred !== void 0 ? preferred : 1;
|
|
1080
|
+
}
|
|
1081
|
+
};
|
|
1082
|
+
|
|
1083
|
+
const shouldProvideSettingSection = ({
|
|
1084
|
+
name,
|
|
1085
|
+
items
|
|
1086
|
+
}, {
|
|
1087
|
+
player,
|
|
1088
|
+
loop
|
|
1089
|
+
}) => {
|
|
1090
|
+
if (name === 'loop') {
|
|
1091
|
+
return !(player !== null && player !== void 0 && player.isLive()) && loop !== 'disabled';
|
|
1092
|
+
}
|
|
1093
|
+
|
|
1094
|
+
if (name === 'speed') {
|
|
1095
|
+
return !(player !== null && player !== void 0 && player.isLive());
|
|
1096
|
+
}
|
|
1097
|
+
|
|
1098
|
+
if (name === 'subtitles' || name === 'audio') {
|
|
1099
|
+
return true;
|
|
1100
|
+
}
|
|
1101
|
+
|
|
1102
|
+
return (items === null || items === void 0 ? void 0 : items.length) > 1;
|
|
1103
|
+
};
|
|
1104
|
+
|
|
1105
|
+
const getSettingsData = ({
|
|
1106
|
+
media,
|
|
1107
|
+
player,
|
|
1108
|
+
source = [],
|
|
1109
|
+
quality = {},
|
|
1110
|
+
speedItems = [0.25, 0.5, 0.75, 1, 1.25, 1.5, 1.75, 2],
|
|
1111
|
+
loop,
|
|
1112
|
+
preferred = {}
|
|
1113
|
+
}) => {
|
|
1114
|
+
// TODO extract base player specific things
|
|
1115
|
+
const audioItems = getAudioTracks({}, {
|
|
1116
|
+
player
|
|
1117
|
+
});
|
|
1118
|
+
const selectedSource = getSource(source, {
|
|
1119
|
+
preferManifestType: 'platform'
|
|
1120
|
+
}) || {};
|
|
1121
|
+
const currentSpeedItems = getSpeedItems(speedItems);
|
|
1122
|
+
const sections = [].concat([quality && getQualitySettings({ ...quality,
|
|
1123
|
+
items: selectedSource.qualityOptions
|
|
1124
|
+
}, player), {
|
|
1125
|
+
name: 'subtitles',
|
|
1126
|
+
title: 'KKS.SUBTITLES',
|
|
1127
|
+
items: [].concat(getTextTracks({}, {
|
|
949
1128
|
player
|
|
1129
|
+
}), {
|
|
1130
|
+
label: 'OFF',
|
|
1131
|
+
value: {
|
|
1132
|
+
language: 'off',
|
|
1133
|
+
id: 'off'
|
|
1134
|
+
}
|
|
1135
|
+
})
|
|
1136
|
+
}, {
|
|
1137
|
+
name: 'audio',
|
|
1138
|
+
title: 'KKS.AUDIO',
|
|
1139
|
+
items: audioItems
|
|
1140
|
+
}, {
|
|
1141
|
+
name: 'speed',
|
|
1142
|
+
title: 'KKS.SETTING.SPEED',
|
|
1143
|
+
items: currentSpeedItems
|
|
1144
|
+
}, {
|
|
1145
|
+
type: 'switch',
|
|
1146
|
+
name: 'loop',
|
|
1147
|
+
title: 'KKS.SETTING.LOOP'
|
|
1148
|
+
}]).filter(section => shouldProvideSettingSection(section, {
|
|
1149
|
+
player,
|
|
1150
|
+
loop,
|
|
1151
|
+
quality
|
|
1152
|
+
}));
|
|
1153
|
+
const values = sections.reduce((result, section) => {
|
|
1154
|
+
// TODO take fallback option if preferred is not available
|
|
1155
|
+
// eslint-disable-next-line no-param-reassign
|
|
1156
|
+
result[section.name] = getDefault(section, {
|
|
1157
|
+
preferred: preferred[section.name],
|
|
1158
|
+
player,
|
|
1159
|
+
media
|
|
1160
|
+
}) || preferred[section.name];
|
|
1161
|
+
return result;
|
|
1162
|
+
}, // TODO forced subtitles?
|
|
1163
|
+
{
|
|
1164
|
+
audio: getSelectedAudioName(player),
|
|
1165
|
+
subtitles: 'off'
|
|
1166
|
+
});
|
|
1167
|
+
|
|
1168
|
+
if (!values.speed && (media === null || media === void 0 ? void 0 : media.playbackRate) > 0) {
|
|
1169
|
+
values.speed = media.playbackRate;
|
|
1170
|
+
}
|
|
1171
|
+
|
|
1172
|
+
if (typeof preferred.loop === 'boolean') {
|
|
1173
|
+
values.loop = preferred.loop;
|
|
1174
|
+
} else {
|
|
1175
|
+
values.loop = loop === 'disabled' ? false : Boolean(loop);
|
|
1176
|
+
dispatchCustomEvent(media, 'loopChange', {
|
|
1177
|
+
loop: values.loop
|
|
950
1178
|
});
|
|
951
|
-
|
|
1179
|
+
}
|
|
952
1180
|
|
|
953
|
-
|
|
954
|
-
|
|
955
|
-
|
|
1181
|
+
return {
|
|
1182
|
+
sections,
|
|
1183
|
+
values
|
|
1184
|
+
};
|
|
1185
|
+
};
|
|
956
1186
|
|
|
957
|
-
|
|
1187
|
+
const matchTracks = (tracks, info) => (info === null || info === void 0 ? void 0 : info.map(track => tracks.find(t => t.language === track.language && t.label === track.label)).filter(Boolean)) || tracks;
|
|
958
1188
|
|
|
959
|
-
|
|
960
|
-
|
|
1189
|
+
const getLanguageOptions = (player, info) => {
|
|
1190
|
+
const audioTracks = player && getAudioTracks({}, {
|
|
1191
|
+
player
|
|
1192
|
+
});
|
|
1193
|
+
const textTracks = getTextTracks({}, {
|
|
1194
|
+
player
|
|
1195
|
+
});
|
|
1196
|
+
const textOptions = [].concat(textTracks, {
|
|
1197
|
+
type: 'subtitles',
|
|
1198
|
+
label: 'OFF',
|
|
1199
|
+
selected: !textTracks.some(track => track.selected),
|
|
1200
|
+
value: {
|
|
1201
|
+
language: 'off',
|
|
1202
|
+
id: 'off'
|
|
961
1203
|
}
|
|
962
|
-
}
|
|
963
|
-
|
|
964
|
-
|
|
1204
|
+
});
|
|
1205
|
+
return {
|
|
1206
|
+
audioTracks: matchTracks(audioTracks, info.audioTracks),
|
|
1207
|
+
textTracks: matchTracks(textOptions, info.textTracks)
|
|
1208
|
+
};
|
|
965
1209
|
};
|
|
966
1210
|
|
|
967
1211
|
const vendors = {
|
|
@@ -4785,8 +5029,14 @@ const menuItemStyle = {
|
|
|
4785
5029
|
padding: 'var(--menu-item-padding, 0.5em 1em) ',
|
|
4786
5030
|
display: 'flex',
|
|
4787
5031
|
alignItems: 'center',
|
|
5032
|
+
minWidth: 0,
|
|
4788
5033
|
color: 'rgba(255, 255, 255, 0.6)',
|
|
4789
|
-
background: 'var(--menu-item-background, rgba(255, 255, 255, 0.08))'
|
|
5034
|
+
background: 'var(--menu-item-background, rgba(255, 255, 255, 0.08))',
|
|
5035
|
+
'> div': {
|
|
5036
|
+
overflow: 'hidden',
|
|
5037
|
+
whiteSpace: 'nowrap',
|
|
5038
|
+
textOverflow: 'ellipsis'
|
|
5039
|
+
}
|
|
4790
5040
|
};
|
|
4791
5041
|
const menuItemSelectedStyle = {
|
|
4792
5042
|
color: '#FFF',
|
|
@@ -4794,7 +5044,7 @@ const menuItemSelectedStyle = {
|
|
|
4794
5044
|
'::after': {
|
|
4795
5045
|
content: '" "',
|
|
4796
5046
|
display: 'inline-block',
|
|
4797
|
-
|
|
5047
|
+
flex: '0 0 1em',
|
|
4798
5048
|
height: '1em',
|
|
4799
5049
|
marginLeft: '0.5em',
|
|
4800
5050
|
backgroundColor: 'var(--setting-check-icon-color, #fff)',
|
|
@@ -4808,9 +5058,11 @@ const MenuItem = ({
|
|
|
4808
5058
|
selected,
|
|
4809
5059
|
onClick
|
|
4810
5060
|
}) => jsx$1("div", {
|
|
4811
|
-
css: [menuItemStyle, selected && menuItemSelectedStyle, process.env.NODE_ENV === "production" ? "" : ";label:MenuItem;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
5061
|
+
css: [menuItemStyle, selected && menuItemSelectedStyle, process.env.NODE_ENV === "production" ? "" : ";label:MenuItem;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIk1lbnVJdGVtLmpzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQW1DSSIsImZpbGUiOiJNZW51SXRlbS5qcyIsInNvdXJjZXNDb250ZW50IjpbIi8qIGVzbGludC1kaXNhYmxlIGpzeC1hMTF5L25vLXN0YXRpYy1lbGVtZW50LWludGVyYWN0aW9ucyAqL1xuLyogQGpzeEltcG9ydFNvdXJjZSBAZW1vdGlvbi9yZWFjdCAqL1xuaW1wb3J0IGljb24gZnJvbSAnc3R5bGUvaWNvbidcblxuY29uc3QgbWVudUl0ZW1TdHlsZSA9IHtcbiAgcGFkZGluZzogJ3ZhcigtLW1lbnUtaXRlbS1wYWRkaW5nLCAwLjVlbSAxZW0pICcsXG4gIGRpc3BsYXk6ICdmbGV4JyxcbiAgYWxpZ25JdGVtczogJ2NlbnRlcicsXG4gIG1pbldpZHRoOiAwLFxuICBjb2xvcjogJ3JnYmEoMjU1LCAyNTUsIDI1NSwgMC42KScsXG4gIGJhY2tncm91bmQ6ICd2YXIoLS1tZW51LWl0ZW0tYmFja2dyb3VuZCwgcmdiYSgyNTUsIDI1NSwgMjU1LCAwLjA4KSknLCBcbiAgJz4gZGl2Jzoge1xuICAgIG92ZXJmbG93OiAnaGlkZGVuJyxcbiAgICB3aGl0ZVNwYWNlOiAnbm93cmFwJyxcbiAgICB0ZXh0T3ZlcmZsb3c6ICdlbGxpcHNpcycsXG4gIH1cbn1cblxuY29uc3QgbWVudUl0ZW1TZWxlY3RlZFN0eWxlID0ge1xuICBjb2xvcjogJyNGRkYnLFxuICBmb250V2VpZ2h0OiAnNjAwJyxcbiAgJzo6YWZ0ZXInOiB7XG4gICAgY29udGVudDogJ1wiIFwiJyxcbiAgICBkaXNwbGF5OiAnaW5saW5lLWJsb2NrJyxcbiAgICBmbGV4OiAnMCAwIDFlbScsXG4gICAgaGVpZ2h0OiAnMWVtJyxcbiAgICBtYXJnaW5MZWZ0OiAnMC41ZW0nLFxuICAgIGJhY2tncm91bmRDb2xvcjogJ3ZhcigtLXNldHRpbmctY2hlY2staWNvbi1jb2xvciwgI2ZmZiknLFxuICAgIG1hc2tTaXplOiAnY29udGFpbicsXG4gICAgbWFza0ltYWdlOiBgdXJsKCR7aWNvbi5jaGVja30pYCxcbiAgfSxcbn1cblxuY29uc3QgTWVudUl0ZW0gPSAoe2xhYmVsLCBzZWxlY3RlZCwgb25DbGlja30pID0+IChcbiAgPGRpdlxuICAgIGNzcz17W21lbnVJdGVtU3R5bGUsIHNlbGVjdGVkICYmIG1lbnVJdGVtU2VsZWN0ZWRTdHlsZV19XG4gICAgb25DbGljaz17b25DbGlja31cbiAgPlxuICAgIDxkaXY+e2xhYmVsfTwvZGl2PlxuICA8L2Rpdj5cbilcblxuZXhwb3J0IGRlZmF1bHQgTWVudUl0ZW1cbiJdfQ== */"],
|
|
4812
5062
|
onClick: onClick,
|
|
4813
|
-
children:
|
|
5063
|
+
children: jsx$1("div", {
|
|
5064
|
+
children: label
|
|
5065
|
+
})
|
|
4814
5066
|
});
|
|
4815
5067
|
|
|
4816
5068
|
var MenuItem$1 = MenuItem;
|
|
@@ -4822,7 +5074,8 @@ const landscapeStyle = {
|
|
|
4822
5074
|
justifyContent: 'space-between',
|
|
4823
5075
|
'> div': {
|
|
4824
5076
|
boxSizing: 'border-box',
|
|
4825
|
-
flex: '0 0 50%'
|
|
5077
|
+
flex: '0 0 50%',
|
|
5078
|
+
minWidth: 0
|
|
4826
5079
|
},
|
|
4827
5080
|
'> div:not(:first-of-type)': {
|
|
4828
5081
|
marginTop: 0,
|
|
@@ -4915,7 +5168,7 @@ const LanguageMenu = ({
|
|
|
4915
5168
|
sectionOptions = [],
|
|
4916
5169
|
onChange
|
|
4917
5170
|
}) => jsxs("div", {
|
|
4918
|
-
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,
|
|
5171
|
+
css: [containerStyle$2, twoColumnStyle, uiType === 'desktop' && [desktopContainerStyle, desktop2ColumnStyle], process.env.NODE_ENV === "production" ? "" : ";label:LanguageMenu;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIkxhbmd1YWdlTWVudS5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUE4R0kiLCJmaWxlIjoiTGFuZ3VhZ2VNZW51LmpzIiwic291cmNlc0NvbnRlbnQiOlsiLyogQGpzeEltcG9ydFNvdXJjZSBAZW1vdGlvbi9yZWFjdCAqL1xuaW1wb3J0IHtCdXR0b259IGZyb20gJy4vYnV0dG9ucydcbmltcG9ydCBNZW51SXRlbSBmcm9tICcuL01lbnVJdGVtJ1xuXG5jb25zdCBsYW5kc2NhcGVTdHlsZSA9IHtcbiAgZGlzcGxheTogJ2ZsZXgnLFxuICBmbGV4V3JhcDogJ3dyYXAnLFxuICBqdXN0aWZ5Q29udGVudDogJ3NwYWNlLWJldHdlZW4nLFxuICAnPiBkaXYnOiB7XG4gICAgYm94U2l6aW5nOiAnYm9yZGVyLWJveCcsXG4gICAgZmxleDogJzAgMCA1MCUnLFxuICAgIG1pbldpZHRoOiAwLFxuICB9LFxuICAnPiBkaXY6bm90KDpmaXJzdC1vZi10eXBlKSc6IHtcbiAgICBtYXJnaW5Ub3A6IDAsXG4gICAgZmxleDogJzAgMCBjYWxjKDUwJSAtIDFweCknLFxuICB9LFxuICBocjoge1xuICAgIGhlaWdodDogJ2F1dG8nLFxuICAgIGJvcmRlckxlZnQ6ICcxcHggc29saWQgdHJhbnNwYXJlbnQnLFxuICB9LFxufVxuXG5jb25zdCBjb250YWluZXJTdHlsZSA9IHtcbiAgZm9udFNpemU6ICcxNnB4JyxcbiAgYmFja2dyb3VuZDogJyMyNjI2MjYnLFxuICBoZWlnaHQ6ICcxMDAlJyxcbiAgaDM6IHtcbiAgICBtYXJnaW46IDAsXG4gICAgJ21hcmdpbi1ibG9jay1zdGFydCc6IDAsXG4gICAgJ21hcmdpbi1ibG9jay1lbmQnOiAwLFxuICAgIHBhZGRpbmc6ICcwLjZlbSAxZW0nLFxuICAgIGRpc3BsYXk6ICdmbGV4JyxcbiAgICBhbGlnbkl0ZW1zOiAnY2VudGVyJyxcbiAgICBib3JkZXJSYWRpdXM6ICc0cHggNHB4IDAgMCcsXG4gICAgZm9udFNpemU6ICcxN3B4JyxcbiAgICBidXR0b246IHtcbiAgICAgIGZvbnRTaXplOiAnMC42ZW0nLFxuICAgICAgbWFyZ2luUmlnaHQ6ICcwLjVlbScsXG4gICAgfSxcbiAgfSxcbn1cblxuY29uc3QgdHdvQ29sdW1uU3R5bGUgPSB7XG4gIGhyOiB7XG4gICAgYm9yZGVyOiAnbm9uZScsXG4gIH0sXG4gICc+IGRpdiAnOiB7XG4gICAgJz4gZGl2Om5vdCg6Zmlyc3Qtb2YtdHlwZSknOiB7XG4gICAgICBtYXJnaW5Ub3A6ICcwLjY2ZW0nLFxuICAgIH0sXG4gICAgJ0BtZWRpYSAob3JpZW50YXRpb246IGxhbmRzY2FwZSknOiBsYW5kc2NhcGVTdHlsZSxcbiAgfSxcbiAgJz4gZGl2ID4gZGl2ID4gZGl2OmZpcnN0LW9mLXR5cGUnOiB7XG4gICAgcGFkZGluZzogJzAuNmVtIDEuMWVtJyxcbiAgICBmb250U2l6ZTogJzE0cHgnLFxuICAgIGNvbG9yOiAnI2FhYWViNScsXG4gICAgYmFja2dyb3VuZDogJ3ZhcigtLW1lbnUtaXRlbS1iYWNrZ3JvdW5kLCAjMjYyNjI2KScsXG4gICAgb3ZlcmZsb3c6ICdoaWRkZW4nLFxuICAgIHRleHRPdmVyZmxvdzogJ2VsbGlwc2lzJyxcbiAgfSxcbn1cblxuY29uc3QgZGVza3RvcENvbnRhaW5lclN0eWxlID0ge1xuICBwYWRkaW5nQm90dG9tOiAnMS41ZW0nLFxuICB3aWR0aDogJzM0ZW0nLFxuICBtaW5IZWlnaHQ6ICczMC41JScsXG4gIG1heEhlaWdodDogJzYzLjUlJyxcbiAgYm9yZGVyUmFkaXVzOiAnNHB4JyxcbiAgYmFja2dyb3VuZDogJ3JnYmEoMCwgMCwgMCwgMC42MCknLFxuICAnLS1tZW51LWl0ZW0tYmFja2dyb3VuZCc6ICd0cmFuc3BhcmVudCcsXG4gICctLW1lbnUtaXRlbS1wYWRkaW5nJzogJzAuNjZlbSAxLjVlbScsXG4gIGgzOiB7XG4gICAgcGFkZGluZzogJzAuNjZlbSAxLjVlbScsXG4gICAgZm9udFNpemU6ICcxNnB4JyxcbiAgICBjb2xvcjogJ3JnYmEoMjU1LCAyNTUsIDI1NSwgMC42MCknLFxuICAgIGJhY2tncm91bmQ6ICdyZ2JhKDAsIDAsIDAsIDAuODApJyxcbiAgfSxcbn1cblxuY29uc3QgZGVza3RvcDJDb2x1bW5TdHlsZSA9IHtcbiAgJz4gZGl2JzogbGFuZHNjYXBlU3R5bGUsXG4gICc+IGRpdiBocic6IHtcbiAgICBmb250U2l6ZTogJzE1MCUnLFxuICAgIGJvcmRlckxlZnQ6ICcxcHggc29saWQgIHJnYmEoMjU1LCAyNTUsIDI1NSwgMC4yMCknLFxuICB9LFxuICAnPiBkaXYgPiBkaXYnOiB7XG4gICAgcGFkZGluZzogJzAgMWVtJyxcbiAgfSxcbiAgJz4gZGl2ID4gZGl2ID4gZGl2OmZpcnN0LW9mLXR5cGUnOiB7XG4gICAgcGFkZGluZzogJzAuNjZlbSAwLjVlbScsXG4gICAgY29sb3I6ICcjZmZmJyxcbiAgICBmb250U2l6ZTogJzE2cHgnLFxuICAgIGZvbnRXZWlnaHQ6ICc2MDAnLFxuICB9LFxufVxuXG5jb25zdCBjbG9zZU1lbnUgPSBldmVudCA9PlxuICBldmVudC5jdXJyZW50VGFyZ2V0LmRpc3BhdGNoRXZlbnQoXG4gICAgbmV3IEN1c3RvbUV2ZW50KCdjbG9zZS1tZW51Jywge2J1YmJsZXM6IHRydWV9KVxuICApXG5cbmNvbnN0IExhbmd1YWdlTWVudSA9ICh7XG4gIHVpVHlwZSA9ICdtb2JpbGUnLFxuICB0aXRsZSA9ICfpn7Plo7Ag44O7IOWtl+W5lScsXG4gIHNlY3Rpb25zID0gWyfpn7Plo7AnLCAn5a2X5bmVJ10sXG4gIHNlY3Rpb25PcHRpb25zID0gW10sXG4gIG9uQ2hhbmdlLFxufSkgPT4gKFxuICA8ZGl2XG4gICAgY3NzPXtbXG4gICAgICBjb250YWluZXJTdHlsZSxcbiAgICAgIHR3b0NvbHVtblN0eWxlLFxuICAgICAgdWlUeXBlID09PSAnZGVza3RvcCcgJiYgW2Rlc2t0b3BDb250YWluZXJTdHlsZSwgZGVza3RvcDJDb2x1bW5TdHlsZV0sXG4gICAgXX1cbiAgPlxuICAgIDxoMz5cbiAgICAgIHt1aVR5cGUgIT09ICdkZXNrdG9wJyAmJiA8QnV0dG9uIHN0YXJ0SWNvbj1cImNsb3NlXCIgb25DbGljaz17Y2xvc2VNZW51fSAvPn1cbiAgICAgIHt0aXRsZX1cbiAgICA8L2gzPlxuICAgIDxkaXY+XG4gICAgICA8ZGl2PlxuICAgICAgICA8ZGl2PntzZWN0aW9uc1swXX08L2Rpdj5cbiAgICAgICAge3NlY3Rpb25PcHRpb25zWzBdLm1hcChvcHRpb24gPT4gKFxuICAgICAgICAgIDxNZW51SXRlbSB7Li4ub3B0aW9ufSBvbkNsaWNrPXtldmVudCA9PiBvbkNoYW5nZShldmVudCwgb3B0aW9uKX0gLz5cbiAgICAgICAgKSl9XG4gICAgICA8L2Rpdj5cbiAgICAgIDxociAvPlxuICAgICAgPGRpdj5cbiAgICAgICAgPGRpdj57c2VjdGlvbnNbMV19PC9kaXY+XG4gICAgICAgIHtzZWN0aW9uT3B0aW9uc1sxXS5tYXAob3B0aW9uID0+IChcbiAgICAgICAgICA8TWVudUl0ZW0gey4uLm9wdGlvbn0gb25DbGljaz17ZXZlbnQgPT4gb25DaGFuZ2UoZXZlbnQsIG9wdGlvbil9IC8+XG4gICAgICAgICkpfVxuICAgICAgPC9kaXY+XG4gICAgPC9kaXY+XG4gIDwvZGl2PlxuKVxuXG5MYW5ndWFnZU1lbnUuc3R5bGVzID0gY29udGFpbmVyU3R5bGVcbkxhbmd1YWdlTWVudS5kZXNrdG9wU3R5bGVzID0gZGVza3RvcENvbnRhaW5lclN0eWxlXG5cbmV4cG9ydCBkZWZhdWx0IExhbmd1YWdlTWVudVxuIl19 */"],
|
|
4919
5172
|
children: [jsxs("h3", {
|
|
4920
5173
|
children: [uiType !== 'desktop' && jsx$1(Button, {
|
|
4921
5174
|
startIcon: "close",
|
|
@@ -7332,7 +7585,7 @@ const loadShaka = async (videoElement, config = {}, options = {}) => {
|
|
|
7332
7585
|
// See: https://github.com/shaka-project/shaka-player/issues/3526
|
|
7333
7586
|
safeSeekOffset: 0,
|
|
7334
7587
|
rebufferingGoal: 0,
|
|
7335
|
-
...(
|
|
7588
|
+
...(isSafari() && {
|
|
7336
7589
|
preferNativeHls: true
|
|
7337
7590
|
}),
|
|
7338
7591
|
...config.streaming
|
|
@@ -7424,7 +7677,7 @@ const loadShaka = async (videoElement, config = {}, options = {}) => {
|
|
|
7424
7677
|
responseHandlers: [rewriteDashManifest]
|
|
7425
7678
|
};
|
|
7426
7679
|
|
|
7427
|
-
if (
|
|
7680
|
+
if (isSafari()) {
|
|
7428
7681
|
setupKKFariplay(player, extensionOptions);
|
|
7429
7682
|
}
|
|
7430
7683
|
|
|
@@ -7715,232 +7968,6 @@ const Video = ({
|
|
|
7715
7968
|
|
|
7716
7969
|
var Video$1 = Video;
|
|
7717
7970
|
|
|
7718
|
-
const getSpeedItems = items => items.map(value => ({
|
|
7719
|
-
label: `${value}x`,
|
|
7720
|
-
value
|
|
7721
|
-
}));
|
|
7722
|
-
|
|
7723
|
-
const getQualityOptions = ({
|
|
7724
|
-
sections,
|
|
7725
|
-
values: {
|
|
7726
|
-
quality
|
|
7727
|
-
}
|
|
7728
|
-
}) => {
|
|
7729
|
-
var _sections$find, _sections$find$items$;
|
|
7730
|
-
|
|
7731
|
-
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;
|
|
7732
|
-
};
|
|
7733
|
-
|
|
7734
|
-
const autoQualityOption = {
|
|
7735
|
-
label: `Auto`,
|
|
7736
|
-
options: {
|
|
7737
|
-
maxHeight: Infinity,
|
|
7738
|
-
minHeight: 0
|
|
7739
|
-
},
|
|
7740
|
-
value: 'auto'
|
|
7741
|
-
};
|
|
7742
|
-
|
|
7743
|
-
const getQualityItemsFromManifest = player => {
|
|
7744
|
-
var _player$getAvailableV;
|
|
7745
|
-
|
|
7746
|
-
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 => ({
|
|
7747
|
-
label: `${q.height}p`,
|
|
7748
|
-
// Set the min/max height to the same value to fix the quality.
|
|
7749
|
-
options: {
|
|
7750
|
-
maxHeight: q.height,
|
|
7751
|
-
minHeight: q.height
|
|
7752
|
-
},
|
|
7753
|
-
value: q.height
|
|
7754
|
-
})));
|
|
7755
|
-
};
|
|
7756
|
-
|
|
7757
|
-
const getQualitySettings = (options, player) => {
|
|
7758
|
-
// With native HLS, manifest rewrite is required to enable quality setting
|
|
7759
|
-
// TODO let this covered by test, maybe refactor?
|
|
7760
|
-
const items = needNativeHls() && !options.rewriteManifest ? [] : options.items || getQualityItemsFromManifest(player);
|
|
7761
|
-
return items.length > 0 && items[0] && {
|
|
7762
|
-
name: 'quality',
|
|
7763
|
-
title: 'KKS.QUALITY',
|
|
7764
|
-
items,
|
|
7765
|
-
getDefault: (preferred = options.default || items[0].value) => {
|
|
7766
|
-
const maxHeight = preferred || items[0].value;
|
|
7767
|
-
return (nearest(items.filter(item => (item.height || item.value) <= maxHeight), item => (item.height || item.value) - maxHeight) || items[0]).value;
|
|
7768
|
-
}
|
|
7769
|
-
};
|
|
7770
|
-
};
|
|
7771
|
-
|
|
7772
|
-
const getSelectedAudioName = player => {
|
|
7773
|
-
const track = getCurrentAudioTrack({}, {
|
|
7774
|
-
player
|
|
7775
|
-
});
|
|
7776
|
-
/*
|
|
7777
|
-
Sometimes, HLS manifest doesn't describe the default audio track.
|
|
7778
|
-
Get current audio track information is undefined even though the player still has audio streaming.
|
|
7779
|
-
For this case, we select first audio track.
|
|
7780
|
-
More detail please refer to OTP-3450.
|
|
7781
|
-
*/
|
|
7782
|
-
|
|
7783
|
-
const audioList = getAudioTracks({}, {
|
|
7784
|
-
player
|
|
7785
|
-
});
|
|
7786
|
-
const currentAudio = track !== undefined ? track : audioList === null || audioList === void 0 ? void 0 : audioList[0];
|
|
7787
|
-
return currentAudio && { ...currentAudio,
|
|
7788
|
-
id: [currentAudio.language, currentAudio.label].join(':')
|
|
7789
|
-
};
|
|
7790
|
-
};
|
|
7791
|
-
|
|
7792
|
-
const getDefault = (section, {
|
|
7793
|
-
preferred,
|
|
7794
|
-
player
|
|
7795
|
-
}) => {
|
|
7796
|
-
if (typeof section.getDefault === 'function') {
|
|
7797
|
-
return section.getDefault(preferred);
|
|
7798
|
-
}
|
|
7799
|
-
|
|
7800
|
-
if (section.name === 'audio') {
|
|
7801
|
-
return preferred !== null && preferred !== void 0 ? preferred : getSelectedAudioName(player);
|
|
7802
|
-
}
|
|
7803
|
-
|
|
7804
|
-
if (section.name === 'subtitles') {
|
|
7805
|
-
var _ref;
|
|
7806
|
-
|
|
7807
|
-
return ((_ref = section.items.find(track => {
|
|
7808
|
-
var _track$value, _track$value2;
|
|
7809
|
-
|
|
7810
|
-
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);
|
|
7811
|
-
}) || section.items.find(track => {
|
|
7812
|
-
var _track$value3;
|
|
7813
|
-
|
|
7814
|
-
return ((_track$value3 = track.value) === null || _track$value3 === void 0 ? void 0 : _track$value3.language) === (preferred === null || preferred === void 0 ? void 0 : preferred.language);
|
|
7815
|
-
})) === null || _ref === void 0 ? void 0 : _ref.value) || {
|
|
7816
|
-
language: 'off',
|
|
7817
|
-
id: 'off'
|
|
7818
|
-
};
|
|
7819
|
-
}
|
|
7820
|
-
|
|
7821
|
-
if (section.name === 'speed') {
|
|
7822
|
-
return preferred !== null && preferred !== void 0 ? preferred : 1;
|
|
7823
|
-
}
|
|
7824
|
-
};
|
|
7825
|
-
|
|
7826
|
-
const shouldProvideSettingSection = ({
|
|
7827
|
-
name,
|
|
7828
|
-
items
|
|
7829
|
-
}, {
|
|
7830
|
-
player,
|
|
7831
|
-
loop
|
|
7832
|
-
}) => {
|
|
7833
|
-
if (name === 'loop') {
|
|
7834
|
-
return !(player !== null && player !== void 0 && player.isLive()) && loop !== 'disabled';
|
|
7835
|
-
}
|
|
7836
|
-
|
|
7837
|
-
if (name === 'speed') {
|
|
7838
|
-
return !(player !== null && player !== void 0 && player.isLive());
|
|
7839
|
-
}
|
|
7840
|
-
|
|
7841
|
-
if (name === 'subtitles' || name === 'audio') {
|
|
7842
|
-
return true;
|
|
7843
|
-
}
|
|
7844
|
-
|
|
7845
|
-
return (items === null || items === void 0 ? void 0 : items.length) > 1;
|
|
7846
|
-
};
|
|
7847
|
-
|
|
7848
|
-
const getSettingsData = ({
|
|
7849
|
-
media,
|
|
7850
|
-
player,
|
|
7851
|
-
source = [],
|
|
7852
|
-
quality = {},
|
|
7853
|
-
speedItems = [0.25, 0.5, 0.75, 1, 1.25, 1.5, 1.75, 2],
|
|
7854
|
-
loop,
|
|
7855
|
-
preferred = {}
|
|
7856
|
-
}) => {
|
|
7857
|
-
// TODO extract base player specific things
|
|
7858
|
-
const audioItems = getAudioTracks({}, {
|
|
7859
|
-
player
|
|
7860
|
-
}).filter(track => track.language && track.language !== 'und').map(track => ({
|
|
7861
|
-
label: track.label || track.language,
|
|
7862
|
-
value: {
|
|
7863
|
-
language: track.language,
|
|
7864
|
-
label: track.label,
|
|
7865
|
-
id: [track.language, track.label].join(':')
|
|
7866
|
-
}
|
|
7867
|
-
})) || [];
|
|
7868
|
-
const selectedSource = getSource(source, {
|
|
7869
|
-
preferManifestType: needNativeHls() ? 'hls' : 'dash'
|
|
7870
|
-
}) || {};
|
|
7871
|
-
const currentSpeedItems = getSpeedItems(speedItems);
|
|
7872
|
-
const sections = [].concat([quality && getQualitySettings({ ...quality,
|
|
7873
|
-
items: selectedSource.qualityOptions
|
|
7874
|
-
}, player), {
|
|
7875
|
-
name: 'subtitles',
|
|
7876
|
-
title: 'KKS.SUBTITLES',
|
|
7877
|
-
items: [...getTextTracks({}, {
|
|
7878
|
-
player
|
|
7879
|
-
}).map(track => ({
|
|
7880
|
-
label: track.label || track.language,
|
|
7881
|
-
value: {
|
|
7882
|
-
language: track.language,
|
|
7883
|
-
label: track.label,
|
|
7884
|
-
id: [track.language, track.label].join(':')
|
|
7885
|
-
}
|
|
7886
|
-
})), {
|
|
7887
|
-
label: 'OFF',
|
|
7888
|
-
value: {
|
|
7889
|
-
language: 'off',
|
|
7890
|
-
id: 'off'
|
|
7891
|
-
}
|
|
7892
|
-
}]
|
|
7893
|
-
}, {
|
|
7894
|
-
name: 'audio',
|
|
7895
|
-
title: 'KKS.AUDIO',
|
|
7896
|
-
items: audioItems
|
|
7897
|
-
}, {
|
|
7898
|
-
name: 'speed',
|
|
7899
|
-
title: 'KKS.SETTING.SPEED',
|
|
7900
|
-
items: currentSpeedItems
|
|
7901
|
-
}, {
|
|
7902
|
-
type: 'switch',
|
|
7903
|
-
name: 'loop',
|
|
7904
|
-
title: 'KKS.SETTING.LOOP'
|
|
7905
|
-
}]).filter(section => shouldProvideSettingSection(section, {
|
|
7906
|
-
player,
|
|
7907
|
-
loop,
|
|
7908
|
-
quality
|
|
7909
|
-
}));
|
|
7910
|
-
const values = sections.reduce((result, section) => {
|
|
7911
|
-
// TODO take fallback option if preferred is not available
|
|
7912
|
-
// eslint-disable-next-line no-param-reassign
|
|
7913
|
-
result[section.name] = getDefault(section, {
|
|
7914
|
-
preferred: preferred[section.name],
|
|
7915
|
-
player,
|
|
7916
|
-
media
|
|
7917
|
-
}) || preferred[section.name];
|
|
7918
|
-
return result;
|
|
7919
|
-
}, // TODO forced subtitles?
|
|
7920
|
-
{
|
|
7921
|
-
audio: getSelectedAudioName(player),
|
|
7922
|
-
subtitles: 'off'
|
|
7923
|
-
});
|
|
7924
|
-
|
|
7925
|
-
if (!values.speed && (media === null || media === void 0 ? void 0 : media.playbackRate) > 0) {
|
|
7926
|
-
values.speed = media.playbackRate;
|
|
7927
|
-
}
|
|
7928
|
-
|
|
7929
|
-
if (typeof preferred.loop === 'boolean') {
|
|
7930
|
-
values.loop = preferred.loop;
|
|
7931
|
-
} else {
|
|
7932
|
-
values.loop = loop === 'disabled' ? false : Boolean(loop);
|
|
7933
|
-
dispatchCustomEvent(media, 'loopChange', {
|
|
7934
|
-
loop: values.loop
|
|
7935
|
-
});
|
|
7936
|
-
}
|
|
7937
|
-
|
|
7938
|
-
return {
|
|
7939
|
-
sections,
|
|
7940
|
-
values
|
|
7941
|
-
};
|
|
7942
|
-
};
|
|
7943
|
-
|
|
7944
7971
|
const parseVTT = data => {
|
|
7945
7972
|
const lines = data.split(/\n\n/g).slice(1); // may replace with async parser to prevent blocking render
|
|
7946
7973
|
|
|
@@ -8118,58 +8145,40 @@ const defaultSettings = {};
|
|
|
8118
8145
|
const shouldPause = ({
|
|
8119
8146
|
userFocus,
|
|
8120
8147
|
uiType
|
|
8121
|
-
}) => uiType === 'mobile' && /settings|chapter-list/.test(userFocus) || /blocking|seekbar|share-inner/.test(userFocus);
|
|
8122
|
-
|
|
8123
|
-
const getLanguageOptions = settingState => {
|
|
8124
|
-
var _settingState$values$, _settingState$section, _settingState$section2, _settingState$values$2, _settingState$section3, _settingState$section4;
|
|
8125
|
-
|
|
8126
|
-
const selectedAudio = settingState === null || settingState === void 0 ? void 0 : (_settingState$values$ = settingState.values.audio) === null || _settingState$values$ === void 0 ? void 0 : _settingState$values$.id;
|
|
8127
|
-
const audioTracks = ((settingState === null || settingState === void 0 ? void 0 : (_settingState$section = settingState.sections) === null || _settingState$section === void 0 ? void 0 : (_settingState$section2 = _settingState$section.find(section => section.name === 'audio')) === null || _settingState$section2 === void 0 ? void 0 : _settingState$section2.items) || []).map(item => ({
|
|
8128
|
-
section: 'audio',
|
|
8129
|
-
...item,
|
|
8130
|
-
...(item.value.id === selectedAudio && {
|
|
8131
|
-
selected: true
|
|
8132
|
-
})
|
|
8133
|
-
}));
|
|
8134
|
-
const selectedSubtitle = settingState === null || settingState === void 0 ? void 0 : (_settingState$values$2 = settingState.values.subtitles) === null || _settingState$values$2 === void 0 ? void 0 : _settingState$values$2.id;
|
|
8135
|
-
const subtitleTracks = ((settingState === null || settingState === void 0 ? void 0 : (_settingState$section3 = settingState.sections) === null || _settingState$section3 === void 0 ? void 0 : (_settingState$section4 = _settingState$section3.find(section => section.name === 'subtitles')) === null || _settingState$section4 === void 0 ? void 0 : _settingState$section4.items) || []).map(item => ({
|
|
8136
|
-
section: 'subtitles',
|
|
8137
|
-
...item,
|
|
8138
|
-
...(item.value.id === selectedSubtitle && {
|
|
8139
|
-
selected: true
|
|
8140
|
-
})
|
|
8141
|
-
}));
|
|
8142
|
-
return {
|
|
8143
|
-
audioTracks,
|
|
8144
|
-
subtitleTracks
|
|
8145
|
-
};
|
|
8146
|
-
}; // TODO extract to somewhere else
|
|
8148
|
+
}) => uiType === 'mobile' && /settings|chapter-list/.test(userFocus) || /blocking|seekbar|share-inner/.test(userFocus); // TODO extract to somewhere else
|
|
8147
8149
|
|
|
8148
8150
|
|
|
8149
8151
|
const menuClasses = {
|
|
8150
|
-
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":"AA6FW","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 {getQualityOptions, getSettingsData} from './settings'\nimport {linkMediaVolume, syncVolume} from './volume'\nimport SeekPreview from './SeekPreview'\n\nconst sizes = {\n  'small-embed': 200,\n  embed: 400,\n  'tablet-portrait': 600,\n  'tablet-landscape': 900,\n  desktop: 1200,\n}\n\nconst useLinkState = (request, dependencies = []) => {\n  const [state, setState] = useState()\n  useEffect(() => {\n    request(setState)\n  }, dependencies)\n  return state\n}\n\nconst flipState = state => (state === 'playing' ? 'paused' : 'playing')\n\n// FIXME: too few lines to split a file, looking a better place\nconst getThumbnailsUrl = source =>\n  [].concat(source).find(item => item.type === 'thumbnail')?.src\n\nconst defaultSettings = {}\n\nconst shouldPause = ({userFocus, uiType}) =>\n  (uiType === 'mobile' && /settings|chapter-list/.test(userFocus)) ||\n  /blocking|seekbar|share-inner/.test(userFocus)\n\nconst getLanguageOptions = settingState => {\n  const selectedAudio = settingState?.values.audio?.id\n  const audioTracks = (\n    settingState?.sections?.find(section => section.name === 'audio')?.items ||\n    []\n  ).map(item => ({\n    section: 'audio',\n    ...item,\n    ...(item.value.id === selectedAudio && {selected: true}),\n  }))\n  const selectedSubtitle = settingState?.values.subtitles?.id\n  const subtitleTracks = (\n    settingState?.sections?.find(section => section.name === 'subtitles')\n      ?.items || []\n  ).map(item => ({\n    section: 'subtitles',\n    ...item,\n    ...(item.value.id === selectedSubtitle && {selected: true}),\n  }))\n  return {audioTracks, subtitleTracks}\n}\n\n// TODO extract to somewhere else\nconst menuClasses = {\n  default: css(LanguageMenu.styles),\n  desktop: css(LanguageMenu.styles, LanguageMenu.desktopStyles),\n}\n\nconst LanguageSettings = ({\n  uiType,\n  audioTracks = [],\n  subtitleTracks = [],\n  onChange,\n  slots = {LanguageMenu},\n  ...rest\n}) => (\n  <OverlayPanel buttonIcon=\"subtitle\">\n    <slots.LanguageMenu\n      uiType={uiType}\n      classes={menuClasses}\n      sectionOptions={[audioTracks, subtitleTracks]}\n      onChange={onChange}\n      {...rest}\n    />\n  </OverlayPanel>\n)\n\nconst PremiumPlayer = ({\n  source,\n  startTime,\n  quality = {},\n  title,\n  metadata,\n  channelTitle,\n  chapters,\n  playbackState: appPlaybackState,\n  currentTime: appCurrentTime,\n  playbackRate: appPlaybackRate,\n  loop,\n  volume: appVolume,\n  thumbnailsUrl,\n  controls = {autohide: 3000},\n  marks = [],\n  // TODO sectionId\n  intl,\n  settings: appSettings = defaultSettings,\n  blocking,\n  plugins = [],\n  modulesConfig,\n\n  uiType = isDesktop() ? 'desktop' : 'mobile',\n  style,\n  children,\n  uiMode = 'standalone',\n  cast: castOptions = 'compatible',\n  slots,\n  slotProps = {},\n  onError,\n  onPlaybackStateChange,\n  onBack,\n  onCast,\n  onChangeNext,\n  onChangePrevious,\n  onOpenSettings,\n  onChangeSettings,\n  onPlayerLoaded,\n  onBlockedAutoplay,\n  onPlaylogFired,\n  ...videoProps\n}) => {\n  const components = {\n    VolumeControl,\n    CastUi,\n    DisplayTime,\n    LiveButton,\n    Seekbar,\n    Settings,\n    ...slots,\n  }\n  const videoRef = useRef()\n  const containerRef = useRef()\n  const playerRef = useRef()\n  const adContainerRef = useRef()\n  const refs = useRef({})\n  // TODO move RWD related to Layout\n  const {\n    currentBreakpoint: size,\n    width,\n    observe,\n  } = useDimensions({\n    polyfill: ResizeObserver,\n    breakpoints: sizes,\n  })\n  const [isUserActive, setIsUserActive] = useState(\n    uiMode === 'standalone' ? true : videoProps.autoplay\n  )\n  const [userFocus, setUserFocus] = useState('')\n  const [targetState, setTargetState] = useState(() => ({\n    playbackState:\n      appPlaybackState || (videoProps.autoplay ? 'playing' : 'paused'),\n    currentTime: startTime,\n  }))\n  const [playbackTime, setPlaybackTime] = useState({\n    currentTime: 0,\n    bufferTime: 0,\n  })\n  const togglePlay = (overrideState, {triggeredBy} = {}) => {\n    if (targetState.playbackState !== overrideState) {\n      setTargetState(state => ({\n        ...state,\n        playbackState: overrideState || flipState(state.playbackState),\n        animation:\n          triggeredBy === 'user-action' && flipState(state.playbackState),\n      }))\n    }\n  }\n\n  useEffect(() => {\n    if (appPlaybackState) {\n      togglePlay(appPlaybackState)\n    }\n  }, [appPlaybackState])\n  useEffect(() => {\n    if (uiType === 'mobile') {\n      on(document, 'visibilitychange', () => togglePlay('paused'))\n    }\n  }, [uiType, targetState.playbackState])\n  const updatePlaybackTime = event =>\n    requestAnimationFrame(\n      () =>\n        (event?.type !== 'timeupdate' || isBuffered(videoRef.current)) &&\n        setPlaybackTime(state => ({\n          ...state,\n          ...getMediaTime(videoRef.current, {\n            player: playerRef.current,\n            plugins,\n          }),\n          ...(event?.type === 'durationchange' && {\n            currentTime: state.currentTime,\n          }),\n        }))\n    )\n  const setTargetTime = time => {\n    const trimmed = Math.min(\n      time,\n      videoRef.current?.initialDuration || Infinity\n    )\n    setTargetState(state => ({\n      ...state,\n      // seek to 0 repeatedly edge case\n      currentTime: state.currentTime !== trimmed ? trimmed : trimmed + 0.01,\n    }))\n    updatePlaybackTime()\n  }\n  const [playbackState, setPlaybackState] = useState('init')\n  const [castState, setCastState] = useState('')\n  useEffect(() => {\n    if (typeof appCurrentTime === 'number') setTargetTime(appCurrentTime || 0)\n    if (typeof appCurrentTime === 'object')\n      setTargetTime(appCurrentTime?.value || 0)\n  }, [appCurrentTime])\n  const [errorData, setErrorData] = useState()\n  const errorHandler = reactEvent => {\n    onError?.(reactEvent.nativeEvent)\n    handleError(reactEvent, {\n      media: videoRef.current,\n      displayError: data => {\n        setPlaybackState('error')\n        setErrorData(current => (current?.code ? current : data))\n      },\n    })\n  }\n  const isLive = playbackTime.streamType === 'live'\n  const [settings, setSettings] = useState(() => ({\n    sections: [],\n    values: {speed: 1},\n  }))\n  const fetchSettings = () => {\n    if (!playerRef.current) {\n      return\n    }\n    const speedItems = appSettings.speedItems || (isLive ? [] : undefined)\n    setSettings(current => {\n      const {values, sections} = getSettingsData({\n        media: videoRef.current,\n        player: playerRef.current,\n        source,\n        quality,\n        speedItems,\n        loop,\n        preferred: current.preferred,\n      })\n      return {preferred: current.preferred, values, sections}\n    })\n  }\n  const lastState = useRef(playbackState)\n  const handlePlaybackStateChange = (event, state) => {\n    if (lastState.current === 'error') {\n      return\n    }\n    onPlaybackStateChange?.(event, state)\n    // previously UI playback state was synced only when\n    // exiting iOS video only fullscreen(webkitendfullscreen)\n    if (\n      /playing|paused/.test(state) &&\n      !(uiType === 'mobile' && blocking) &&\n      !shouldPause({userFocus, uiType})\n    ) {\n      togglePlay(state)\n    }\n    if (state === 'ended') {\n      togglePlay('paused')\n    }\n    if (state === 'loading' && lastState.current !== 'init') {\n      setTimeout(() => togglePlay('playing'), 1)\n    }\n    if (state === 'playing') {\n      setIsUserActive(true)\n    }\n    if (lastState.current === 'loading') {\n      fetchSettings()\n    }\n    lastState.current = state\n    setPlaybackState(state)\n  }\n\n  const changeSettings = (name, value, {keepOpen} = {}) => {\n    // TODO consider merge into useReducer?\n    onChangeSettings?.({name, value})\n    setTargetTime(playbackTime.currentTime)\n    setSettings(current => ({\n      ...current,\n      values: {...current.values, [name]: value},\n      preferred: {...current.preferred, [name]: value},\n    }))\n    if (!keepOpen) {\n      setUserFocus('')\n    }\n  }\n  const openSettings = event => {\n    const animationFrame =\n      userFocus !== 'settings' &&\n      requestAnimationFrame(() => {\n        onOpenSettings?.(event, settings)\n        // In iOS Safari, we need to update settings data\n        fetchSettings()\n      })\n    setUserFocus(current => (current === 'settings' ? '' : 'settings'))\n    return animationFrame\n  }\n  useEffect(() => {\n    if (appPlaybackRate > 0) {\n      setSettings(current => ({\n        ...current,\n        values: {...current.values, speed: appPlaybackRate},\n      }))\n    }\n  }, [appPlaybackRate])\n\n  const qualityOptions = useMemo(\n    () => getQualityOptions(settings),\n    [settings.values.quality]\n  )\n  const viewMode = useLinkState(update =>\n    onViewModeChange(videoRef.current, update)\n  )\n  const sourceOverride = useLinkState(\n    async update => {\n      const result =\n        source && (await quality.rewriteManifest?.(source, qualityOptions))\n      update(result || source)\n    },\n    [source, qualityOptions]\n  )\n  const waiting = /emptied|loading|buffering/.test(playbackState)\n  const activePlayback =\n    playbackState === 'playing' || playbackState === 'waiting'\n  const {\n    mode: autoHideMode,\n    onClick,\n    onMouseMove,\n  } = useAutoHide({\n    pinned: !controls.autohide || waiting || !activePlayback || userFocus,\n    tapToHide: uiType === 'mobile',\n    hideTimeMs: controls.autohide,\n  })\n  const mode =\n    userFocus === 'seekbar'\n      ? 'hidden'\n      : controls.autohide\n      ? autoHideMode\n      : controls\n      ? 'shown'\n      : 'hidden'\n  const controlsDisplay =\n    userFocus === 'seekbar'\n      ? 'seekbar-only'\n      : controls === 'title-only' || playbackState === 'emptied'\n      ? 'hidden'\n      : mode\n  const shouldHidePanels =\n    (controls === 'no-panel' || controlsDisplay === 'hidden') && userFocus\n  const havePlayPanel =\n    !blocking &&\n    havePointer() &&\n    uiType === 'desktop' &&\n    !waiting &&\n    !/volume|''/.test(userFocus) &&\n    !/title-only|no-panel/.test(controls) &&\n    (controls.autohide || mode === 'shown')\n\n  useEffect(() => {\n    if (shouldHidePanels) {\n      setUserFocus('')\n    }\n  }, [shouldHidePanels])\n  const {subscribe, onChange, toggleMute} = linkMediaVolume(() => ({\n    video: videoRef.current,\n    getPlayer: () => playerRef.current,\n  }))\n  const changePrevious = event => {\n    onChangePrevious(event)\n    togglePlay('paused')\n    videoRef.current.dispatchEvent(new CustomEvent('loadstart'))\n  }\n  const changeNext = event => {\n    onChangeNext(event)\n    togglePlay('paused')\n    videoRef.current.dispatchEvent(new CustomEvent('loadstart'))\n  }\n\n  useEffect(fetchSettings, [appSettings])\n  useEffect(() => {\n    // The adContainer should be set before `load` because ImaDai.load needs it.\n    plugins.forEach(plugin => plugin.setAdContainer?.(adContainerRef.current))\n  }, [])\n\n  useEffect(\n    () =>\n      dispatchChapterEvents({\n        media: videoRef.current,\n        chapters,\n        getTime: () => videoRef.current.currentTime,\n      }),\n    [chapters]\n  )\n\n  const isEnd = playbackState === 'ended'\n  const canSeek = !isEnd && playbackTime.duration > 0\n  const derivedPlaybackState =\n    (uiType === 'mobile' && blocking) || shouldPause({userFocus, uiType})\n      ? 'paused'\n      : targetState.playbackState\n  const activeThumbnailsUrl =\n    (!userFocus || userFocus === 'seekbar') &&\n    (getThumbnailsUrl(source) || thumbnailsUrl)\n  const cssVariables = {\n    '--playing': playbackState === 'playing' ? '1' : '0',\n  }\n  const onRewind =\n    (!isLive || playbackTime.currentTime < 0) &&\n    !isEnd &&\n    (() => setTargetTime(playbackTime.currentTime - 10, 'rewind'))\n  const onForward =\n    (!isLive || playbackTime.currentTime < 0) &&\n    canSeek &&\n    (() => setTargetTime(playbackTime.currentTime + 10, 'forward'))\n  const onClickBlank =\n    havePlayPanel && (() => togglePlay('', {triggeredBy: 'user-action'}))\n\n  const uiHandlers = {\n    onPlay: () => {\n      togglePlay('', {triggeredBy: 'user-action'})\n      setIsUserActive(true)\n    },\n    onClickLive: isLive && (() => setTargetTime(0, 'seekToLive')),\n    onClickSeekbar: () => {\n      refs.current.deferedSeekFocus = setTimeout(\n        () => setUserFocus('seekbar'),\n        66\n      )\n    },\n    onSeek: time => {\n      setTargetTime(time, 'seek')\n      clearTimeout(refs.current.deferedSeekFocus)\n      setTimeout(() => setUserFocus(''), 66)\n    },\n    onChangeSettings: ({name, value, keepOpen}) =>\n      changeSettings(name, value, {keepOpen}),\n    onCloseSettings: event => {\n      setUserFocus('')\n      slotProps?.settings?.onClose?.(event)\n    },\n  }\n\n  const uiElements = {\n    liveButton: uiHandlers.onClickLive && (\n      <components.LiveButton\n        usingStartOver={playbackTime.currentTime < 0}\n        onClick={uiHandlers.onClickLive}\n      />\n    ),\n    controlButtons: {\n      playButton: (\n        <PlayButton\n          playbackState={derivedPlaybackState}\n          ended={isEnd}\n          hidden={\n            uiType === 'mobile' && (waiting || /loading/.test(playbackState))\n          }\n          variant={!isUserActive && 'firstplay'}\n          onClick={uiHandlers.onPlay}\n        />\n      ),\n      rewindButton: onRewind && (\n        <Button\n          startIcon=\"rewind10\"\n          title=\"KKS.PLAYER.REWIND\"\n          disabled={isEnd}\n          onClick={onRewind}\n        />\n      ),\n      forwardButton: onForward && (\n        <ForwardButton\n          startIcon=\"forward10\"\n          title=\"KKS.PLAYER.FORWARD\"\n          disabled={!canSeek}\n          onClick={onForward}\n        />\n      ),\n      nextEpisodeButton: (\n        <Button\n          startIcon=\"next\"\n          title=\"KKS.PLAYER.NEXT\"\n          disabled={!onChangeNext}\n          onClick={changeNext}\n        />\n      ),\n      previousEpisodeButton: (\n        <Button\n          startIcon=\"previous\"\n          title=\"KKS.PLAYER.PREVIOUS\"\n          disabled={!onChangePrevious}\n          onClick={changePrevious}\n        />\n      ),\n    },\n    seekbar: (\n      <components.Seekbar\n        style={/volume/.test(userFocus) && {opacity: '0'}}\n        startTime={playbackTime.startTime}\n        currentTime={playbackTime.currentTime}\n        bufferTime={playbackTime.bufferTime}\n        duration={playbackTime.duration}\n        chapters={chapters}\n        onChange={uiHandlers.onClickSeekbar}\n        onChangeCommitted={uiHandlers.onSeek}\n        marks={marks}\n        plugins={plugins}\n        {...slotProps.seekbar}\n      >\n        {activeThumbnailsUrl && (\n          <SeekPreview\n            thumbnailsUrl={activeThumbnailsUrl}\n            duration={playbackTime.duration}\n            chapters={chapters}\n          />\n        )}\n      </components.Seekbar>\n    ),\n    displayTime: <components.DisplayTime {...playbackTime} />,\n    fullscreenButton: (\n      <FullscreenButton\n        viewMode={viewMode}\n        onClick={() => toggleFullscreen(containerRef.current)}\n      />\n    ),\n    volumeControl: width >= sizes['small-embed'] && (\n      <components.VolumeControl\n        {...{subscribe, onChange, toggleMute}}\n        onMouseOver={() => setUserFocus('volume')}\n        onMouseOut={() => setUserFocus('')}\n        {...slotProps?.volumeControl}\n      />\n    ),\n    backItems: (\n      <PlayPanel animation={targetState.animation} onClick={onClickBlank} />\n    ),\n  }\n  const compatibilityCastProps = castOptions === 'compatible' && {\n    source,\n    values: {title, ...settings.values, ...intl, modulesConfig},\n    onLoad: onCast,\n  }\n\n  useEffect(() => {\n    if (castOptions && castOptions !== 'compatible') {\n      return linkCast({source, ...castOptions})\n    } // TODO playlist, audio / subtitle setting\n  }, [source, castOptions.contentId, castOptions.customData])\n\n  return (\n    <IntlProvider {...intl}>\n      <Layout\n        style={{...style, cssVariables}}\n        type={uiType}\n        display={mode}\n        controlsDisplay={controlsDisplay}\n        size={size}\n        video={\n          <Video\n            {...videoProps}\n            videoRef={multiRef(videoRef, videoProps.videoRef)}\n            source={\n              castState !== CastState.CONNECTED &&\n              playbackState !== 'error' &&\n              sourceOverride\n            }\n            playbackState={derivedPlaybackState}\n            currentTime={targetState.currentTime}\n            {...(appVolume >= 0 && {volume: appVolume, muted: appVolume <= 0})}\n            playbackRate={settings.values.speed}\n            videoResolution={qualityOptions}\n            audio={settings.values.audio}\n            textTrack={settings.values.subtitles}\n            loop={settings.values.loop}\n            plugins={plugins}\n            modulesConfig={modulesConfig}\n            onPlaylogFired={onPlaylogFired}\n            onError={errorHandler}\n            onPlaybackStateChange={handlePlaybackStateChange}\n            onBlockedAutoplay={error => onBlockedAutoplay?.(error)}\n            onCanPlay={event => {\n              updatePlaybackTime(event)\n              videoProps.onCanPlay?.(event)\n            }}\n            onTimeUpdate={updatePlaybackTime}\n            onDurationChange={updatePlaybackTime}\n            onPlayerLoaded={player => {\n              playerRef.current = player\n              onPlayerLoaded?.(player)\n              syncVolume(videoRef.current, videoProps.muted ? 0 : appVolume)\n            }}\n          />\n        }\n        containerRef={element => {\n          containerRef.current = element\n          observe(element)\n        }}\n        backRef={adContainerRef}\n        title={title}\n        channelTitle={channelTitle}\n        backButton={onBack && <Button startIcon=\"back\" onClick={onBack} />}\n        {...(isUserActive && uiElements)}\n        onClick={onClick}\n        onMouseMove={onMouseMove}\n      >\n        <LoadingSpinner active={waiting} />\n        {errorData && <Error error={errorData} onBack={onBack} />}\n        {children}\n        {/* TODO sync Cast last played time back */}\n        {isUserActive && (\n          <components.CastUi\n            onBack={onBack}\n            onStateChange={setCastState}\n            onLoad={onCast}\n            {...compatibilityCastProps}\n            {...slotProps.castUi}\n          />\n        )}\n        {isUserActive && ( // TODO: Design <Extenstion /> flag\n          <components.Settings\n            type={uiType}\n            sections={settings.sections}\n            // TODO hasBottomPanel bottom: 8em\n            open={userFocus === 'settings'}\n            values={settings.values}\n            onOpen={openSettings}\n            onChange={uiHandlers.onChangeSettings}\n            {...slotProps?.settings}\n            onClose={uiHandlers.onCloseSettings}\n          />\n        )}\n        <LanguageSettings\n          uiType={uiType}\n          {...getLanguageOptions(settings)}\n          onChange={(_event, item) => changeSettings(item.section, 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"]} */"),
|
|
8151
|
-
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":"AA8FW","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 {getQualityOptions, getSettingsData} from './settings'\nimport {linkMediaVolume, syncVolume} from './volume'\nimport SeekPreview from './SeekPreview'\n\nconst sizes = {\n  'small-embed': 200,\n  embed: 400,\n  'tablet-portrait': 600,\n  'tablet-landscape': 900,\n  desktop: 1200,\n}\n\nconst useLinkState = (request, dependencies = []) => {\n  const [state, setState] = useState()\n  useEffect(() => {\n    request(setState)\n  }, dependencies)\n  return state\n}\n\nconst flipState = state => (state === 'playing' ? 'paused' : 'playing')\n\n// FIXME: too few lines to split a file, looking a better place\nconst getThumbnailsUrl = source =>\n  [].concat(source).find(item => item.type === 'thumbnail')?.src\n\nconst defaultSettings = {}\n\nconst shouldPause = ({userFocus, uiType}) =>\n  (uiType === 'mobile' && /settings|chapter-list/.test(userFocus)) ||\n  /blocking|seekbar|share-inner/.test(userFocus)\n\nconst getLanguageOptions = settingState => {\n  const selectedAudio = settingState?.values.audio?.id\n  const audioTracks = (\n    settingState?.sections?.find(section => section.name === 'audio')?.items ||\n    []\n  ).map(item => ({\n    section: 'audio',\n    ...item,\n    ...(item.value.id === selectedAudio && {selected: true}),\n  }))\n  const selectedSubtitle = settingState?.values.subtitles?.id\n  const subtitleTracks = (\n    settingState?.sections?.find(section => section.name === 'subtitles')\n      ?.items || []\n  ).map(item => ({\n    section: 'subtitles',\n    ...item,\n    ...(item.value.id === selectedSubtitle && {selected: true}),\n  }))\n  return {audioTracks, subtitleTracks}\n}\n\n// TODO extract to somewhere else\nconst menuClasses = {\n  default: css(LanguageMenu.styles),\n  desktop: css(LanguageMenu.styles, LanguageMenu.desktopStyles),\n}\n\nconst LanguageSettings = ({\n  uiType,\n  audioTracks = [],\n  subtitleTracks = [],\n  onChange,\n  slots = {LanguageMenu},\n  ...rest\n}) => (\n  <OverlayPanel buttonIcon=\"subtitle\">\n    <slots.LanguageMenu\n      uiType={uiType}\n      classes={menuClasses}\n      sectionOptions={[audioTracks, subtitleTracks]}\n      onChange={onChange}\n      {...rest}\n    />\n  </OverlayPanel>\n)\n\nconst PremiumPlayer = ({\n  source,\n  startTime,\n  quality = {},\n  title,\n  metadata,\n  channelTitle,\n  chapters,\n  playbackState: appPlaybackState,\n  currentTime: appCurrentTime,\n  playbackRate: appPlaybackRate,\n  loop,\n  volume: appVolume,\n  thumbnailsUrl,\n  controls = {autohide: 3000},\n  marks = [],\n  // TODO sectionId\n  intl,\n  settings: appSettings = defaultSettings,\n  blocking,\n  plugins = [],\n  modulesConfig,\n\n  uiType = isDesktop() ? 'desktop' : 'mobile',\n  style,\n  children,\n  uiMode = 'standalone',\n  cast: castOptions = 'compatible',\n  slots,\n  slotProps = {},\n  onError,\n  onPlaybackStateChange,\n  onBack,\n  onCast,\n  onChangeNext,\n  onChangePrevious,\n  onOpenSettings,\n  onChangeSettings,\n  onPlayerLoaded,\n  onBlockedAutoplay,\n  onPlaylogFired,\n  ...videoProps\n}) => {\n  const components = {\n    VolumeControl,\n    CastUi,\n    DisplayTime,\n    LiveButton,\n    Seekbar,\n    Settings,\n    ...slots,\n  }\n  const videoRef = useRef()\n  const containerRef = useRef()\n  const playerRef = useRef()\n  const adContainerRef = useRef()\n  const refs = useRef({})\n  // TODO move RWD related to Layout\n  const {\n    currentBreakpoint: size,\n    width,\n    observe,\n  } = useDimensions({\n    polyfill: ResizeObserver,\n    breakpoints: sizes,\n  })\n  const [isUserActive, setIsUserActive] = useState(\n    uiMode === 'standalone' ? true : videoProps.autoplay\n  )\n  const [userFocus, setUserFocus] = useState('')\n  const [targetState, setTargetState] = useState(() => ({\n    playbackState:\n      appPlaybackState || (videoProps.autoplay ? 'playing' : 'paused'),\n    currentTime: startTime,\n  }))\n  const [playbackTime, setPlaybackTime] = useState({\n    currentTime: 0,\n    bufferTime: 0,\n  })\n  const togglePlay = (overrideState, {triggeredBy} = {}) => {\n    if (targetState.playbackState !== overrideState) {\n      setTargetState(state => ({\n        ...state,\n        playbackState: overrideState || flipState(state.playbackState),\n        animation:\n          triggeredBy === 'user-action' && flipState(state.playbackState),\n      }))\n    }\n  }\n\n  useEffect(() => {\n    if (appPlaybackState) {\n      togglePlay(appPlaybackState)\n    }\n  }, [appPlaybackState])\n  useEffect(() => {\n    if (uiType === 'mobile') {\n      on(document, 'visibilitychange', () => togglePlay('paused'))\n    }\n  }, [uiType, targetState.playbackState])\n  const updatePlaybackTime = event =>\n    requestAnimationFrame(\n      () =>\n        (event?.type !== 'timeupdate' || isBuffered(videoRef.current)) &&\n        setPlaybackTime(state => ({\n          ...state,\n          ...getMediaTime(videoRef.current, {\n            player: playerRef.current,\n            plugins,\n          }),\n          ...(event?.type === 'durationchange' && {\n            currentTime: state.currentTime,\n          }),\n        }))\n    )\n  const setTargetTime = time => {\n    const trimmed = Math.min(\n      time,\n      videoRef.current?.initialDuration || Infinity\n    )\n    setTargetState(state => ({\n      ...state,\n      // seek to 0 repeatedly edge case\n      currentTime: state.currentTime !== trimmed ? trimmed : trimmed + 0.01,\n    }))\n    updatePlaybackTime()\n  }\n  const [playbackState, setPlaybackState] = useState('init')\n  const [castState, setCastState] = useState('')\n  useEffect(() => {\n    if (typeof appCurrentTime === 'number') setTargetTime(appCurrentTime || 0)\n    if (typeof appCurrentTime === 'object')\n      setTargetTime(appCurrentTime?.value || 0)\n  }, [appCurrentTime])\n  const [errorData, setErrorData] = useState()\n  const errorHandler = reactEvent => {\n    onError?.(reactEvent.nativeEvent)\n    handleError(reactEvent, {\n      media: videoRef.current,\n      displayError: data => {\n        setPlaybackState('error')\n        setErrorData(current => (current?.code ? current : data))\n      },\n    })\n  }\n  const isLive = playbackTime.streamType === 'live'\n  const [settings, setSettings] = useState(() => ({\n    sections: [],\n    values: {speed: 1},\n  }))\n  const fetchSettings = () => {\n    if (!playerRef.current) {\n      return\n    }\n    const speedItems = appSettings.speedItems || (isLive ? [] : undefined)\n    setSettings(current => {\n      const {values, sections} = getSettingsData({\n        media: videoRef.current,\n        player: playerRef.current,\n        source,\n        quality,\n        speedItems,\n        loop,\n        preferred: current.preferred,\n      })\n      return {preferred: current.preferred, values, sections}\n    })\n  }\n  const lastState = useRef(playbackState)\n  const handlePlaybackStateChange = (event, state) => {\n    if (lastState.current === 'error') {\n      return\n    }\n    onPlaybackStateChange?.(event, state)\n    // previously UI playback state was synced only when\n    // exiting iOS video only fullscreen(webkitendfullscreen)\n    if (\n      /playing|paused/.test(state) &&\n      !(uiType === 'mobile' && blocking) &&\n      !shouldPause({userFocus, uiType})\n    ) {\n      togglePlay(state)\n    }\n    if (state === 'ended') {\n      togglePlay('paused')\n    }\n    if (state === 'loading' && lastState.current !== 'init') {\n      setTimeout(() => togglePlay('playing'), 1)\n    }\n    if (state === 'playing') {\n      setIsUserActive(true)\n    }\n    if (lastState.current === 'loading') {\n      fetchSettings()\n    }\n    lastState.current = state\n    setPlaybackState(state)\n  }\n\n  const changeSettings = (name, value, {keepOpen} = {}) => {\n    // TODO consider merge into useReducer?\n    onChangeSettings?.({name, value})\n    setTargetTime(playbackTime.currentTime)\n    setSettings(current => ({\n      ...current,\n      values: {...current.values, [name]: value},\n      preferred: {...current.preferred, [name]: value},\n    }))\n    if (!keepOpen) {\n      setUserFocus('')\n    }\n  }\n  const openSettings = event => {\n    const animationFrame =\n      userFocus !== 'settings' &&\n      requestAnimationFrame(() => {\n        onOpenSettings?.(event, settings)\n        // In iOS Safari, we need to update settings data\n        fetchSettings()\n      })\n    setUserFocus(current => (current === 'settings' ? '' : 'settings'))\n    return animationFrame\n  }\n  useEffect(() => {\n    if (appPlaybackRate > 0) {\n      setSettings(current => ({\n        ...current,\n        values: {...current.values, speed: appPlaybackRate},\n      }))\n    }\n  }, [appPlaybackRate])\n\n  const qualityOptions = useMemo(\n    () => getQualityOptions(settings),\n    [settings.values.quality]\n  )\n  const viewMode = useLinkState(update =>\n    onViewModeChange(videoRef.current, update)\n  )\n  const sourceOverride = useLinkState(\n    async update => {\n      const result =\n        source && (await quality.rewriteManifest?.(source, qualityOptions))\n      update(result || source)\n    },\n    [source, qualityOptions]\n  )\n  const waiting = /emptied|loading|buffering/.test(playbackState)\n  const activePlayback =\n    playbackState === 'playing' || playbackState === 'waiting'\n  const {\n    mode: autoHideMode,\n    onClick,\n    onMouseMove,\n  } = useAutoHide({\n    pinned: !controls.autohide || waiting || !activePlayback || userFocus,\n    tapToHide: uiType === 'mobile',\n    hideTimeMs: controls.autohide,\n  })\n  const mode =\n    userFocus === 'seekbar'\n      ? 'hidden'\n      : controls.autohide\n      ? autoHideMode\n      : controls\n      ? 'shown'\n      : 'hidden'\n  const controlsDisplay =\n    userFocus === 'seekbar'\n      ? 'seekbar-only'\n      : controls === 'title-only' || playbackState === 'emptied'\n      ? 'hidden'\n      : mode\n  const shouldHidePanels =\n    (controls === 'no-panel' || controlsDisplay === 'hidden') && userFocus\n  const havePlayPanel =\n    !blocking &&\n    havePointer() &&\n    uiType === 'desktop' &&\n    !waiting &&\n    !/volume|''/.test(userFocus) &&\n    !/title-only|no-panel/.test(controls) &&\n    (controls.autohide || mode === 'shown')\n\n  useEffect(() => {\n    if (shouldHidePanels) {\n      setUserFocus('')\n    }\n  }, [shouldHidePanels])\n  const {subscribe, onChange, toggleMute} = linkMediaVolume(() => ({\n    video: videoRef.current,\n    getPlayer: () => playerRef.current,\n  }))\n  const changePrevious = event => {\n    onChangePrevious(event)\n    togglePlay('paused')\n    videoRef.current.dispatchEvent(new CustomEvent('loadstart'))\n  }\n  const changeNext = event => {\n    onChangeNext(event)\n    togglePlay('paused')\n    videoRef.current.dispatchEvent(new CustomEvent('loadstart'))\n  }\n\n  useEffect(fetchSettings, [appSettings])\n  useEffect(() => {\n    // The adContainer should be set before `load` because ImaDai.load needs it.\n    plugins.forEach(plugin => plugin.setAdContainer?.(adContainerRef.current))\n  }, [])\n\n  useEffect(\n    () =>\n      dispatchChapterEvents({\n        media: videoRef.current,\n        chapters,\n        getTime: () => videoRef.current.currentTime,\n      }),\n    [chapters]\n  )\n\n  const isEnd = playbackState === 'ended'\n  const canSeek = !isEnd && playbackTime.duration > 0\n  const derivedPlaybackState =\n    (uiType === 'mobile' && blocking) || shouldPause({userFocus, uiType})\n      ? 'paused'\n      : targetState.playbackState\n  const activeThumbnailsUrl =\n    (!userFocus || userFocus === 'seekbar') &&\n    (getThumbnailsUrl(source) || thumbnailsUrl)\n  const cssVariables = {\n    '--playing': playbackState === 'playing' ? '1' : '0',\n  }\n  const onRewind =\n    (!isLive || playbackTime.currentTime < 0) &&\n    !isEnd &&\n    (() => setTargetTime(playbackTime.currentTime - 10, 'rewind'))\n  const onForward =\n    (!isLive || playbackTime.currentTime < 0) &&\n    canSeek &&\n    (() => setTargetTime(playbackTime.currentTime + 10, 'forward'))\n  const onClickBlank =\n    havePlayPanel && (() => togglePlay('', {triggeredBy: 'user-action'}))\n\n  const uiHandlers = {\n    onPlay: () => {\n      togglePlay('', {triggeredBy: 'user-action'})\n      setIsUserActive(true)\n    },\n    onClickLive: isLive && (() => setTargetTime(0, 'seekToLive')),\n    onClickSeekbar: () => {\n      refs.current.deferedSeekFocus = setTimeout(\n        () => setUserFocus('seekbar'),\n        66\n      )\n    },\n    onSeek: time => {\n      setTargetTime(time, 'seek')\n      clearTimeout(refs.current.deferedSeekFocus)\n      setTimeout(() => setUserFocus(''), 66)\n    },\n    onChangeSettings: ({name, value, keepOpen}) =>\n      changeSettings(name, value, {keepOpen}),\n    onCloseSettings: event => {\n      setUserFocus('')\n      slotProps?.settings?.onClose?.(event)\n    },\n  }\n\n  const uiElements = {\n    liveButton: uiHandlers.onClickLive && (\n      <components.LiveButton\n        usingStartOver={playbackTime.currentTime < 0}\n        onClick={uiHandlers.onClickLive}\n      />\n    ),\n    controlButtons: {\n      playButton: (\n        <PlayButton\n          playbackState={derivedPlaybackState}\n          ended={isEnd}\n          hidden={\n            uiType === 'mobile' && (waiting || /loading/.test(playbackState))\n          }\n          variant={!isUserActive && 'firstplay'}\n          onClick={uiHandlers.onPlay}\n        />\n      ),\n      rewindButton: onRewind && (\n        <Button\n          startIcon=\"rewind10\"\n          title=\"KKS.PLAYER.REWIND\"\n          disabled={isEnd}\n          onClick={onRewind}\n        />\n      ),\n      forwardButton: onForward && (\n        <ForwardButton\n          startIcon=\"forward10\"\n          title=\"KKS.PLAYER.FORWARD\"\n          disabled={!canSeek}\n          onClick={onForward}\n        />\n      ),\n      nextEpisodeButton: (\n        <Button\n          startIcon=\"next\"\n          title=\"KKS.PLAYER.NEXT\"\n          disabled={!onChangeNext}\n          onClick={changeNext}\n        />\n      ),\n      previousEpisodeButton: (\n        <Button\n          startIcon=\"previous\"\n          title=\"KKS.PLAYER.PREVIOUS\"\n          disabled={!onChangePrevious}\n          onClick={changePrevious}\n        />\n      ),\n    },\n    seekbar: (\n      <components.Seekbar\n        style={/volume/.test(userFocus) && {opacity: '0'}}\n        startTime={playbackTime.startTime}\n        currentTime={playbackTime.currentTime}\n        bufferTime={playbackTime.bufferTime}\n        duration={playbackTime.duration}\n        chapters={chapters}\n        onChange={uiHandlers.onClickSeekbar}\n        onChangeCommitted={uiHandlers.onSeek}\n        marks={marks}\n        plugins={plugins}\n        {...slotProps.seekbar}\n      >\n        {activeThumbnailsUrl && (\n          <SeekPreview\n            thumbnailsUrl={activeThumbnailsUrl}\n            duration={playbackTime.duration}\n            chapters={chapters}\n          />\n        )}\n      </components.Seekbar>\n    ),\n    displayTime: <components.DisplayTime {...playbackTime} />,\n    fullscreenButton: (\n      <FullscreenButton\n        viewMode={viewMode}\n        onClick={() => toggleFullscreen(containerRef.current)}\n      />\n    ),\n    volumeControl: width >= sizes['small-embed'] && (\n      <components.VolumeControl\n        {...{subscribe, onChange, toggleMute}}\n        onMouseOver={() => setUserFocus('volume')}\n        onMouseOut={() => setUserFocus('')}\n        {...slotProps?.volumeControl}\n      />\n    ),\n    backItems: (\n      <PlayPanel animation={targetState.animation} onClick={onClickBlank} />\n    ),\n  }\n  const compatibilityCastProps = castOptions === 'compatible' && {\n    source,\n    values: {title, ...settings.values, ...intl, modulesConfig},\n    onLoad: onCast,\n  }\n\n  useEffect(() => {\n    if (castOptions && castOptions !== 'compatible') {\n      return linkCast({source, ...castOptions})\n    } // TODO playlist, audio / subtitle setting\n  }, [source, castOptions.contentId, castOptions.customData])\n\n  return (\n    <IntlProvider {...intl}>\n      <Layout\n        style={{...style, cssVariables}}\n        type={uiType}\n        display={mode}\n        controlsDisplay={controlsDisplay}\n        size={size}\n        video={\n          <Video\n            {...videoProps}\n            videoRef={multiRef(videoRef, videoProps.videoRef)}\n            source={\n              castState !== CastState.CONNECTED &&\n              playbackState !== 'error' &&\n              sourceOverride\n            }\n            playbackState={derivedPlaybackState}\n            currentTime={targetState.currentTime}\n            {...(appVolume >= 0 && {volume: appVolume, muted: appVolume <= 0})}\n            playbackRate={settings.values.speed}\n            videoResolution={qualityOptions}\n            audio={settings.values.audio}\n            textTrack={settings.values.subtitles}\n            loop={settings.values.loop}\n            plugins={plugins}\n            modulesConfig={modulesConfig}\n            onPlaylogFired={onPlaylogFired}\n            onError={errorHandler}\n            onPlaybackStateChange={handlePlaybackStateChange}\n            onBlockedAutoplay={error => onBlockedAutoplay?.(error)}\n            onCanPlay={event => {\n              updatePlaybackTime(event)\n              videoProps.onCanPlay?.(event)\n            }}\n            onTimeUpdate={updatePlaybackTime}\n            onDurationChange={updatePlaybackTime}\n            onPlayerLoaded={player => {\n              playerRef.current = player\n              onPlayerLoaded?.(player)\n              syncVolume(videoRef.current, videoProps.muted ? 0 : appVolume)\n            }}\n          />\n        }\n        containerRef={element => {\n          containerRef.current = element\n          observe(element)\n        }}\n        backRef={adContainerRef}\n        title={title}\n        channelTitle={channelTitle}\n        backButton={onBack && <Button startIcon=\"back\" onClick={onBack} />}\n        {...(isUserActive && uiElements)}\n        onClick={onClick}\n        onMouseMove={onMouseMove}\n      >\n        <LoadingSpinner active={waiting} />\n        {errorData && <Error error={errorData} onBack={onBack} />}\n        {children}\n        {/* TODO sync Cast last played time back */}\n        {isUserActive && (\n          <components.CastUi\n            onBack={onBack}\n            onStateChange={setCastState}\n            onLoad={onCast}\n            {...compatibilityCastProps}\n            {...slotProps.castUi}\n          />\n        )}\n        {isUserActive && ( // TODO: Design <Extenstion /> flag\n          <components.Settings\n            type={uiType}\n            sections={settings.sections}\n            // TODO hasBottomPanel bottom: 8em\n            open={userFocus === 'settings'}\n            values={settings.values}\n            onOpen={openSettings}\n            onChange={uiHandlers.onChangeSettings}\n            {...slotProps?.settings}\n            onClose={uiHandlers.onCloseSettings}\n          />\n        )}\n        <LanguageSettings\n          uiType={uiType}\n          {...getLanguageOptions(settings)}\n          onChange={(_event, item) => changeSettings(item.section, 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"]} */")
|
|
8152
|
+
default: /*#__PURE__*/css$1(LanguageMenu.styles, process.env.NODE_ENV === "production" ? "" : ";label:default;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["PremiumPlayer.js"],"names":[],"mappings":"AAuEW","file":"PremiumPlayer.js","sourcesContent":["/* eslint-disable no-nested-ternary */\nimport {useEffect, useMemo, useRef, useState} from 'react'\nimport {css} from '@emotion/css'\nimport useDimensions from 'react-cool-dimensions'\nimport {ResizeObserver} from '@juggle/resize-observer'\n\nimport {handleError} from 'playerCore/errors'\nimport {on} from 'util/events'\nimport {havePointer, isDesktop} from 'util/environment'\nimport multiRef from 'util/multiRef'\nimport {onViewModeChange, toggleFullscreen} from 'util/viewModes'\nimport {useAutoHide} from 'hooks'\nimport {IntlProvider} from 'context/I18n'\nimport {getMediaTime, isBuffered} from 'playerCore/mediaBindings'\nimport Error from 'playerUi/Error'\nimport Layout from 'playerUi/DefaultLayout'\nimport {\n  Button,\n  LiveButton,\n  FullscreenButton,\n  PlayButton,\n  ForwardButton,\n} from 'playerUi/buttons'\nimport Seekbar from 'playerUi/Seekbar'\nimport DisplayTime from 'playerUi/DisplayTime'\nimport LoadingSpinner from 'playerUi/LoadingSpinner'\nimport Backdrop from 'playerUi/Backdrop'\nimport Settings from 'playerUi/Settings'\nimport OverlayPanel from 'playerUi/OverlayPanel'\nimport LanguageMenu from 'playerUi/LanguageMenu'\nimport PlayPanel from 'playerUi/PlayPane'\nimport VolumeControl from 'component/VolumeControl'\nimport CastUi from 'cast/CastUi'\nimport {linkCast} from 'cast/framework'\nimport {CastState} from 'Enum'\nimport {dispatchChapterEvents} from './timeline'\nimport Video from '../Video'\nimport {getLanguageOptions, getQualityOptions, getSettingsData} from './settings'\nimport {linkMediaVolume, syncVolume} from './volume'\nimport SeekPreview from './SeekPreview'\n\nconst sizes = {\n  'small-embed': 200,\n  embed: 400,\n  'tablet-portrait': 600,\n  'tablet-landscape': 900,\n  desktop: 1200,\n}\n\nconst useLinkState = (request, dependencies = []) => {\n  const [state, setState] = useState()\n  useEffect(() => {\n    request(setState)\n  }, dependencies)\n  return state\n}\n\nconst flipState = state => (state === 'playing' ? 'paused' : 'playing')\n\n// FIXME: too few lines to split a file, looking a better place\nconst getThumbnailsUrl = source =>\n  [].concat(source).find(item => item.type === 'thumbnail')?.src\n\nconst defaultSettings = {}\n\nconst shouldPause = ({userFocus, uiType}) =>\n  (uiType === 'mobile' && /settings|chapter-list/.test(userFocus)) ||\n  /blocking|seekbar|share-inner/.test(userFocus)\n\n// TODO extract to somewhere else\nconst menuClasses = {\n  default: css(LanguageMenu.styles),\n  desktop: css(LanguageMenu.styles, LanguageMenu.desktopStyles),\n}\n\nconst LanguageSettings = ({\n  uiType,\n  audioTracks = [],\n  subtitleTracks = [],\n  getPlayer,\n  onChange,\n  slots = {LanguageMenu},\n  ...rest\n}) => {\n  const options = getLanguageOptions(getPlayer(), {audioTracks, subtitleTracks})\n  return (\n    <OverlayPanel buttonIcon=\"subtitle\">\n      <slots.LanguageMenu\n        uiType={uiType}\n        classes={menuClasses}\n        sectionOptions={[options.audioTracks, options.textTracks]}\n        onChange={onChange}\n        {...rest}\n      />\n    </OverlayPanel>\n  )\n}\n\nconst PremiumPlayer = ({\n  source,\n  startTime,\n  quality = {},\n  title,\n  metadata,\n  channelTitle,\n  chapters,\n  playbackState: appPlaybackState,\n  currentTime: appCurrentTime,\n  playbackRate: appPlaybackRate,\n  loop,\n  volume: appVolume,\n  thumbnailsUrl,\n  controls = {autohide: 3000},\n  marks = [],\n  // TODO sectionId\n  intl,\n  settings: appSettings = defaultSettings,\n  blocking,\n  plugins = [],\n  modulesConfig,\n\n  uiType = isDesktop() ? 'desktop' : 'mobile',\n  style,\n  children,\n  uiMode = 'standalone',\n  cast: castOptions = 'compatible',\n  slots,\n  slotProps = {},\n  onError,\n  onPlaybackStateChange,\n  onBack,\n  onCast,\n  onChangeNext,\n  onChangePrevious,\n  onOpenSettings,\n  onChangeSettings,\n  onPlayerLoaded,\n  onBlockedAutoplay,\n  onPlaylogFired,\n  ...videoProps\n}) => {\n  const components = {\n    VolumeControl,\n    CastUi,\n    DisplayTime,\n    LiveButton,\n    Seekbar,\n    Settings,\n    ...slots,\n  }\n  const videoRef = useRef()\n  const containerRef = useRef()\n  const playerRef = useRef()\n  const adContainerRef = useRef()\n  const refs = useRef({})\n  // TODO move RWD related to Layout\n  const {\n    currentBreakpoint: size,\n    width,\n    observe,\n  } = useDimensions({\n    polyfill: ResizeObserver,\n    breakpoints: sizes,\n  })\n  const [isUserActive, setIsUserActive] = useState(\n    uiMode === 'standalone' ? true : videoProps.autoplay\n  )\n  const [userFocus, setUserFocus] = useState('')\n  const [targetState, setTargetState] = useState(() => ({\n    playbackState:\n      appPlaybackState || (videoProps.autoplay ? 'playing' : 'paused'),\n    currentTime: startTime,\n  }))\n  const [playbackTime, setPlaybackTime] = useState({\n    currentTime: 0,\n    bufferTime: 0,\n  })\n  const togglePlay = (overrideState, {triggeredBy} = {}) => {\n    if (targetState.playbackState !== overrideState) {\n      setTargetState(state => ({\n        ...state,\n        playbackState: overrideState || flipState(state.playbackState),\n        animation:\n          triggeredBy === 'user-action' && flipState(state.playbackState),\n      }))\n    }\n  }\n\n  useEffect(() => {\n    if (appPlaybackState) {\n      togglePlay(appPlaybackState)\n    }\n  }, [appPlaybackState])\n  useEffect(() => {\n    if (uiType === 'mobile') {\n      on(document, 'visibilitychange', () => togglePlay('paused'))\n    }\n  }, [uiType, targetState.playbackState])\n  const updatePlaybackTime = event =>\n    requestAnimationFrame(\n      () =>\n        (event?.type !== 'timeupdate' || isBuffered(videoRef.current)) &&\n        setPlaybackTime(state => ({\n          ...state,\n          ...getMediaTime(videoRef.current, {\n            player: playerRef.current,\n            plugins,\n          }),\n          ...(event?.type === 'durationchange' && {\n            currentTime: state.currentTime,\n          }),\n        }))\n    )\n  const setTargetTime = time => {\n    const trimmed = Math.min(\n      time,\n      videoRef.current?.initialDuration || Infinity\n    )\n    setTargetState(state => ({\n      ...state,\n      // seek to 0 repeatedly edge case\n      currentTime: state.currentTime !== trimmed ? trimmed : trimmed + 0.01,\n    }))\n    updatePlaybackTime()\n  }\n  const [playbackState, setPlaybackState] = useState('init')\n  const [castState, setCastState] = useState('')\n  useEffect(() => {\n    if (typeof appCurrentTime === 'number') setTargetTime(appCurrentTime || 0)\n    if (typeof appCurrentTime === 'object')\n      setTargetTime(appCurrentTime?.value || 0)\n  }, [appCurrentTime])\n  const [errorData, setErrorData] = useState()\n  const errorHandler = reactEvent => {\n    onError?.(reactEvent.nativeEvent)\n    handleError(reactEvent, {\n      media: videoRef.current,\n      displayError: data => {\n        setPlaybackState('error')\n        setErrorData(current => (current?.code ? current : data))\n      },\n    })\n  }\n  const isLive = playbackTime.streamType === 'live'\n  const [settings, setSettings] = useState(() => ({\n    sections: [],\n    values: {speed: 1},\n  }))\n  const fetchSettings = () => {\n    if (!playerRef.current) {\n      return\n    }\n    const speedItems = appSettings.speedItems || (isLive ? [] : undefined)\n    setSettings(current => {\n      const {values, sections} = getSettingsData({\n        media: videoRef.current,\n        player: playerRef.current,\n        source,\n        quality,\n        speedItems,\n        loop,\n        preferred: current.preferred,\n      })\n      return {preferred: current.preferred, values, sections}\n    })\n  }\n  const lastState = useRef(playbackState)\n  const handlePlaybackStateChange = (event, state) => {\n    if (lastState.current === 'error') {\n      return\n    }\n    onPlaybackStateChange?.(event, state)\n    // previously UI playback state was synced only when\n    // exiting iOS video only fullscreen(webkitendfullscreen)\n    if (\n      /playing|paused/.test(state) &&\n      !(uiType === 'mobile' && blocking) &&\n      !shouldPause({userFocus, uiType})\n    ) {\n      togglePlay(state)\n    }\n    if (state === 'ended') {\n      togglePlay('paused')\n    }\n    if (state === 'loading' && lastState.current !== 'init') {\n      setTimeout(() => togglePlay('playing'), 1)\n    }\n    if (state === 'playing') {\n      setIsUserActive(true)\n    }\n    if (lastState.current === 'loading') {\n      fetchSettings()\n    }\n    lastState.current = state\n    setPlaybackState(state)\n  }\n\n  const changeSettings = (name, value, {keepOpen} = {}) => {\n    // TODO consider merge into useReducer?\n    onChangeSettings?.({name, value})\n    setTargetTime(playbackTime.currentTime)\n    setSettings(current => ({\n      ...current,\n      values: {...current.values, [name]: value},\n      preferred: {...current.preferred, [name]: value},\n    }))\n    if (!keepOpen) {\n      setUserFocus('')\n    }\n  }\n  const openSettings = event => {\n    const animationFrame =\n      userFocus !== 'settings' &&\n      requestAnimationFrame(() => {\n        onOpenSettings?.(event, settings)\n        // In iOS Safari, we need to update settings data\n        fetchSettings()\n      })\n    setUserFocus(current => (current === 'settings' ? '' : 'settings'))\n    return animationFrame\n  }\n  useEffect(() => {\n    if (appPlaybackRate > 0) {\n      setSettings(current => ({\n        ...current,\n        values: {...current.values, speed: appPlaybackRate},\n      }))\n    }\n  }, [appPlaybackRate])\n\n  const qualityOptions = useMemo(\n    () => getQualityOptions(settings),\n    [settings.values.quality]\n  )\n  const viewMode = useLinkState(update =>\n    onViewModeChange(videoRef.current, update)\n  )\n  const sourceOverride = useLinkState(\n    async update => {\n      const result =\n        source && (await quality.rewriteManifest?.(source, qualityOptions))\n      update(result || source)\n    },\n    [source, qualityOptions]\n  )\n  const waiting = /emptied|loading|buffering/.test(playbackState)\n  const activePlayback =\n    playbackState === 'playing' || playbackState === 'waiting'\n  const {\n    mode: autoHideMode,\n    onClick,\n    onMouseMove,\n  } = useAutoHide({\n    pinned: !controls.autohide || waiting || !activePlayback || userFocus,\n    tapToHide: uiType === 'mobile',\n    hideTimeMs: controls.autohide,\n  })\n  const mode =\n    userFocus === 'seekbar'\n      ? 'hidden'\n      : controls.autohide\n      ? autoHideMode\n      : controls\n      ? 'shown'\n      : 'hidden'\n  const controlsDisplay =\n    userFocus === 'seekbar'\n      ? 'seekbar-only'\n      : controls === 'title-only' || playbackState === 'emptied'\n      ? 'hidden'\n      : mode\n  const shouldHidePanels =\n    (controls === 'no-panel' || controlsDisplay === 'hidden') && userFocus\n  const havePlayPanel =\n    !blocking &&\n    havePointer() &&\n    uiType === 'desktop' &&\n    !waiting &&\n    !/volume|''/.test(userFocus) &&\n    !/title-only|no-panel/.test(controls) &&\n    (controls.autohide || mode === 'shown')\n\n  useEffect(() => {\n    if (shouldHidePanels) {\n      setUserFocus('')\n    }\n  }, [shouldHidePanels])\n  const {subscribe, onChange, toggleMute} = linkMediaVolume(() => ({\n    video: videoRef.current,\n    getPlayer: () => playerRef.current,\n  }))\n  const changePrevious = event => {\n    onChangePrevious(event)\n    togglePlay('paused')\n    videoRef.current.dispatchEvent(new CustomEvent('loadstart'))\n  }\n  const changeNext = event => {\n    onChangeNext(event)\n    togglePlay('paused')\n    videoRef.current.dispatchEvent(new CustomEvent('loadstart'))\n  }\n\n  useEffect(fetchSettings, [appSettings])\n  useEffect(() => {\n    // The adContainer should be set before `load` because ImaDai.load needs it.\n    plugins.forEach(plugin => plugin.setAdContainer?.(adContainerRef.current))\n  }, [])\n\n  useEffect(\n    () =>\n      dispatchChapterEvents({\n        media: videoRef.current,\n        chapters,\n        getTime: () => videoRef.current.currentTime,\n      }),\n    [chapters]\n  )\n\n  const isEnd = playbackState === 'ended'\n  const canSeek = !isEnd && playbackTime.duration > 0\n  const derivedPlaybackState =\n    (uiType === 'mobile' && blocking) || shouldPause({userFocus, uiType})\n      ? 'paused'\n      : targetState.playbackState\n  const activeThumbnailsUrl =\n    (!userFocus || userFocus === 'seekbar') &&\n    (getThumbnailsUrl(source) || thumbnailsUrl)\n  const cssVariables = {\n    '--playing': playbackState === 'playing' ? '1' : '0',\n  }\n  const onRewind =\n    (!isLive || playbackTime.currentTime < 0) &&\n    !isEnd &&\n    (() => setTargetTime(playbackTime.currentTime - 10, 'rewind'))\n  const onForward =\n    (!isLive || playbackTime.currentTime < 0) &&\n    canSeek &&\n    (() => setTargetTime(playbackTime.currentTime + 10, 'forward'))\n  const onClickBlank =\n    havePlayPanel && (() => togglePlay('', {triggeredBy: 'user-action'}))\n\n  const uiHandlers = {\n    onPlay: () => {\n      togglePlay('', {triggeredBy: 'user-action'})\n      setIsUserActive(true)\n    },\n    onClickLive: isLive && (() => setTargetTime(0, 'seekToLive')),\n    onClickSeekbar: () => {\n      refs.current.deferedSeekFocus = setTimeout(\n        () => setUserFocus('seekbar'),\n        66\n      )\n    },\n    onSeek: time => {\n      setTargetTime(time, 'seek')\n      clearTimeout(refs.current.deferedSeekFocus)\n      setTimeout(() => setUserFocus(''), 66)\n    },\n    onChangeSettings: ({name, value, keepOpen}) =>\n      changeSettings(name, value, {keepOpen}),\n    onCloseSettings: event => {\n      setUserFocus('')\n      slotProps?.settings?.onClose?.(event)\n    },\n  }\n\n  const uiElements = {\n    liveButton: uiHandlers.onClickLive && (\n      <components.LiveButton\n        usingStartOver={playbackTime.currentTime < 0}\n        onClick={uiHandlers.onClickLive}\n      />\n    ),\n    controlButtons: {\n      playButton: (\n        <PlayButton\n          playbackState={derivedPlaybackState}\n          ended={isEnd}\n          hidden={\n            uiType === 'mobile' && (waiting || /loading/.test(playbackState))\n          }\n          variant={!isUserActive && 'firstplay'}\n          onClick={uiHandlers.onPlay}\n        />\n      ),\n      rewindButton: onRewind && (\n        <Button\n          startIcon=\"rewind10\"\n          title=\"KKS.PLAYER.REWIND\"\n          disabled={isEnd}\n          onClick={onRewind}\n        />\n      ),\n      forwardButton: onForward && (\n        <ForwardButton\n          startIcon=\"forward10\"\n          title=\"KKS.PLAYER.FORWARD\"\n          disabled={!canSeek}\n          onClick={onForward}\n        />\n      ),\n      nextEpisodeButton: (\n        <Button\n          startIcon=\"next\"\n          title=\"KKS.PLAYER.NEXT\"\n          disabled={!onChangeNext}\n          onClick={changeNext}\n        />\n      ),\n      previousEpisodeButton: (\n        <Button\n          startIcon=\"previous\"\n          title=\"KKS.PLAYER.PREVIOUS\"\n          disabled={!onChangePrevious}\n          onClick={changePrevious}\n        />\n      ),\n    },\n    seekbar: (\n      <components.Seekbar\n        style={/volume/.test(userFocus) && {opacity: '0'}}\n        startTime={playbackTime.startTime}\n        currentTime={playbackTime.currentTime}\n        bufferTime={playbackTime.bufferTime}\n        duration={playbackTime.duration}\n        chapters={chapters}\n        onChange={uiHandlers.onClickSeekbar}\n        onChangeCommitted={uiHandlers.onSeek}\n        marks={marks}\n        plugins={plugins}\n        {...slotProps.seekbar}\n      >\n        {activeThumbnailsUrl && (\n          <SeekPreview\n            thumbnailsUrl={activeThumbnailsUrl}\n            duration={playbackTime.duration}\n            chapters={chapters}\n          />\n        )}\n      </components.Seekbar>\n    ),\n    displayTime: <components.DisplayTime {...playbackTime} />,\n    fullscreenButton: (\n      <FullscreenButton\n        viewMode={viewMode}\n        onClick={() => toggleFullscreen(containerRef.current)}\n      />\n    ),\n    volumeControl: width >= sizes['small-embed'] && (\n      <components.VolumeControl\n        {...{subscribe, onChange, toggleMute}}\n        onMouseOver={() => setUserFocus('volume')}\n        onMouseOut={() => setUserFocus('')}\n        {...slotProps?.volumeControl}\n      />\n    ),\n    backItems: (\n      <PlayPanel animation={targetState.animation} onClick={onClickBlank} />\n    ),\n  }\n  const compatibilityCastProps = castOptions === 'compatible' && {\n    source,\n    values: {title, ...settings.values, ...intl, modulesConfig},\n    onLoad: onCast,\n  }\n\n  useEffect(() => {\n    if (castOptions && castOptions !== 'compatible') {\n      return linkCast({source, ...castOptions})\n    } // TODO playlist, audio / subtitle setting\n  }, [source, castOptions.contentId, castOptions.customData])\n\n  return (\n    <IntlProvider {...intl}>\n      <Layout\n        style={{...style, cssVariables}}\n        type={uiType}\n        display={mode}\n        controlsDisplay={controlsDisplay}\n        size={size}\n        video={\n          <Video\n            {...videoProps}\n            videoRef={multiRef(videoRef, videoProps.videoRef)}\n            source={\n              castState !== CastState.CONNECTED &&\n              playbackState !== 'error' &&\n              sourceOverride\n            }\n            playbackState={derivedPlaybackState}\n            currentTime={targetState.currentTime}\n            {...(appVolume >= 0 && {volume: appVolume, muted: appVolume <= 0})}\n            playbackRate={settings.values.speed}\n            videoResolution={qualityOptions}\n            audio={settings.values.audio}\n            textTrack={settings.values.subtitles}\n            loop={settings.values.loop}\n            plugins={plugins}\n            modulesConfig={modulesConfig}\n            onPlaylogFired={onPlaylogFired}\n            onError={errorHandler}\n            onPlaybackStateChange={handlePlaybackStateChange}\n            onBlockedAutoplay={error => onBlockedAutoplay?.(error)}\n            onCanPlay={event => {\n              updatePlaybackTime(event)\n              videoProps.onCanPlay?.(event)\n            }}\n            onTimeUpdate={updatePlaybackTime}\n            onDurationChange={updatePlaybackTime}\n            onPlayerLoaded={player => {\n              playerRef.current = player\n              onPlayerLoaded?.(player)\n              syncVolume(videoRef.current, videoProps.muted ? 0 : appVolume)\n            }}\n          />\n        }\n        containerRef={element => {\n          containerRef.current = element\n          observe(element)\n        }}\n        backRef={adContainerRef}\n        title={title}\n        channelTitle={channelTitle}\n        backButton={onBack && <Button startIcon=\"back\" onClick={onBack} />}\n        {...(isUserActive && uiElements)}\n        onClick={onClick}\n        onMouseMove={onMouseMove}\n      >\n        <LoadingSpinner active={waiting} />\n        {errorData && <Error error={errorData} onBack={onBack} />}\n        {children}\n        {/* TODO sync Cast last played time back */}\n        {isUserActive && (\n          <components.CastUi\n            onBack={onBack}\n            onStateChange={setCastState}\n            onLoad={onCast}\n            {...compatibilityCastProps}\n            {...slotProps.castUi}\n          />\n        )}\n        {isUserActive && ( // TODO: Design <Extenstion /> flag\n          <components.Settings\n            type={uiType}\n            sections={settings.sections}\n            // TODO hasBottomPanel bottom: 8em\n            open={userFocus === 'settings'}\n            values={settings.values}\n            onOpen={openSettings}\n            onChange={uiHandlers.onChangeSettings}\n            {...slotProps?.settings}\n            onClose={uiHandlers.onCloseSettings}\n          />\n        )}\n        <LanguageSettings\n          uiType={uiType}\n          getPlayer={() => playerRef.current}\n          onChange={(_event, item) => changeSettings(item.type, item.value)}\n          {...slotProps.languageSettings}\n        />\n        <Backdrop open={!playbackState || playbackState === 'loading'}>\n          <LoadingSpinner />\n        </Backdrop>\n      </Layout>\n    </IntlProvider>\n  )\n}\n\nexport default PremiumPlayer\n"]} */"),
|
|
8153
|
+
desktop: /*#__PURE__*/css$1(LanguageMenu.styles, LanguageMenu.desktopStyles, process.env.NODE_ENV === "production" ? "" : ";label:desktop;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["PremiumPlayer.js"],"names":[],"mappings":"AAwEW","file":"PremiumPlayer.js","sourcesContent":["/* eslint-disable no-nested-ternary */\nimport {useEffect, useMemo, useRef, useState} from 'react'\nimport {css} from '@emotion/css'\nimport useDimensions from 'react-cool-dimensions'\nimport {ResizeObserver} from '@juggle/resize-observer'\n\nimport {handleError} from 'playerCore/errors'\nimport {on} from 'util/events'\nimport {havePointer, isDesktop} from 'util/environment'\nimport multiRef from 'util/multiRef'\nimport {onViewModeChange, toggleFullscreen} from 'util/viewModes'\nimport {useAutoHide} from 'hooks'\nimport {IntlProvider} from 'context/I18n'\nimport {getMediaTime, isBuffered} from 'playerCore/mediaBindings'\nimport Error from 'playerUi/Error'\nimport Layout from 'playerUi/DefaultLayout'\nimport {\n  Button,\n  LiveButton,\n  FullscreenButton,\n  PlayButton,\n  ForwardButton,\n} from 'playerUi/buttons'\nimport Seekbar from 'playerUi/Seekbar'\nimport DisplayTime from 'playerUi/DisplayTime'\nimport LoadingSpinner from 'playerUi/LoadingSpinner'\nimport Backdrop from 'playerUi/Backdrop'\nimport Settings from 'playerUi/Settings'\nimport OverlayPanel from 'playerUi/OverlayPanel'\nimport LanguageMenu from 'playerUi/LanguageMenu'\nimport PlayPanel from 'playerUi/PlayPane'\nimport VolumeControl from 'component/VolumeControl'\nimport CastUi from 'cast/CastUi'\nimport {linkCast} from 'cast/framework'\nimport {CastState} from 'Enum'\nimport {dispatchChapterEvents} from './timeline'\nimport Video from '../Video'\nimport {getLanguageOptions, getQualityOptions, getSettingsData} from './settings'\nimport {linkMediaVolume, syncVolume} from './volume'\nimport SeekPreview from './SeekPreview'\n\nconst sizes = {\n  'small-embed': 200,\n  embed: 400,\n  'tablet-portrait': 600,\n  'tablet-landscape': 900,\n  desktop: 1200,\n}\n\nconst useLinkState = (request, dependencies = []) => {\n  const [state, setState] = useState()\n  useEffect(() => {\n    request(setState)\n  }, dependencies)\n  return state\n}\n\nconst flipState = state => (state === 'playing' ? 'paused' : 'playing')\n\n// FIXME: too few lines to split a file, looking a better place\nconst getThumbnailsUrl = source =>\n  [].concat(source).find(item => item.type === 'thumbnail')?.src\n\nconst defaultSettings = {}\n\nconst shouldPause = ({userFocus, uiType}) =>\n  (uiType === 'mobile' && /settings|chapter-list/.test(userFocus)) ||\n  /blocking|seekbar|share-inner/.test(userFocus)\n\n// TODO extract to somewhere else\nconst menuClasses = {\n  default: css(LanguageMenu.styles),\n  desktop: css(LanguageMenu.styles, LanguageMenu.desktopStyles),\n}\n\nconst LanguageSettings = ({\n  uiType,\n  audioTracks = [],\n  subtitleTracks = [],\n  getPlayer,\n  onChange,\n  slots = {LanguageMenu},\n  ...rest\n}) => {\n  const options = getLanguageOptions(getPlayer(), {audioTracks, subtitleTracks})\n  return (\n    <OverlayPanel buttonIcon=\"subtitle\">\n      <slots.LanguageMenu\n        uiType={uiType}\n        classes={menuClasses}\n        sectionOptions={[options.audioTracks, options.textTracks]}\n        onChange={onChange}\n        {...rest}\n      />\n    </OverlayPanel>\n  )\n}\n\nconst PremiumPlayer = ({\n  source,\n  startTime,\n  quality = {},\n  title,\n  metadata,\n  channelTitle,\n  chapters,\n  playbackState: appPlaybackState,\n  currentTime: appCurrentTime,\n  playbackRate: appPlaybackRate,\n  loop,\n  volume: appVolume,\n  thumbnailsUrl,\n  controls = {autohide: 3000},\n  marks = [],\n  // TODO sectionId\n  intl,\n  settings: appSettings = defaultSettings,\n  blocking,\n  plugins = [],\n  modulesConfig,\n\n  uiType = isDesktop() ? 'desktop' : 'mobile',\n  style,\n  children,\n  uiMode = 'standalone',\n  cast: castOptions = 'compatible',\n  slots,\n  slotProps = {},\n  onError,\n  onPlaybackStateChange,\n  onBack,\n  onCast,\n  onChangeNext,\n  onChangePrevious,\n  onOpenSettings,\n  onChangeSettings,\n  onPlayerLoaded,\n  onBlockedAutoplay,\n  onPlaylogFired,\n  ...videoProps\n}) => {\n  const components = {\n    VolumeControl,\n    CastUi,\n    DisplayTime,\n    LiveButton,\n    Seekbar,\n    Settings,\n    ...slots,\n  }\n  const videoRef = useRef()\n  const containerRef = useRef()\n  const playerRef = useRef()\n  const adContainerRef = useRef()\n  const refs = useRef({})\n  // TODO move RWD related to Layout\n  const {\n    currentBreakpoint: size,\n    width,\n    observe,\n  } = useDimensions({\n    polyfill: ResizeObserver,\n    breakpoints: sizes,\n  })\n  const [isUserActive, setIsUserActive] = useState(\n    uiMode === 'standalone' ? true : videoProps.autoplay\n  )\n  const [userFocus, setUserFocus] = useState('')\n  const [targetState, setTargetState] = useState(() => ({\n    playbackState:\n      appPlaybackState || (videoProps.autoplay ? 'playing' : 'paused'),\n    currentTime: startTime,\n  }))\n  const [playbackTime, setPlaybackTime] = useState({\n    currentTime: 0,\n    bufferTime: 0,\n  })\n  const togglePlay = (overrideState, {triggeredBy} = {}) => {\n    if (targetState.playbackState !== overrideState) {\n      setTargetState(state => ({\n        ...state,\n        playbackState: overrideState || flipState(state.playbackState),\n        animation:\n          triggeredBy === 'user-action' && flipState(state.playbackState),\n      }))\n    }\n  }\n\n  useEffect(() => {\n    if (appPlaybackState) {\n      togglePlay(appPlaybackState)\n    }\n  }, [appPlaybackState])\n  useEffect(() => {\n    if (uiType === 'mobile') {\n      on(document, 'visibilitychange', () => togglePlay('paused'))\n    }\n  }, [uiType, targetState.playbackState])\n  const updatePlaybackTime = event =>\n    requestAnimationFrame(\n      () =>\n        (event?.type !== 'timeupdate' || isBuffered(videoRef.current)) &&\n        setPlaybackTime(state => ({\n          ...state,\n          ...getMediaTime(videoRef.current, {\n            player: playerRef.current,\n            plugins,\n          }),\n          ...(event?.type === 'durationchange' && {\n            currentTime: state.currentTime,\n          }),\n        }))\n    )\n  const setTargetTime = time => {\n    const trimmed = Math.min(\n      time,\n      videoRef.current?.initialDuration || Infinity\n    )\n    setTargetState(state => ({\n      ...state,\n      // seek to 0 repeatedly edge case\n      currentTime: state.currentTime !== trimmed ? trimmed : trimmed + 0.01,\n    }))\n    updatePlaybackTime()\n  }\n  const [playbackState, setPlaybackState] = useState('init')\n  const [castState, setCastState] = useState('')\n  useEffect(() => {\n    if (typeof appCurrentTime === 'number') setTargetTime(appCurrentTime || 0)\n    if (typeof appCurrentTime === 'object')\n      setTargetTime(appCurrentTime?.value || 0)\n  }, [appCurrentTime])\n  const [errorData, setErrorData] = useState()\n  const errorHandler = reactEvent => {\n    onError?.(reactEvent.nativeEvent)\n    handleError(reactEvent, {\n      media: videoRef.current,\n      displayError: data => {\n        setPlaybackState('error')\n        setErrorData(current => (current?.code ? current : data))\n      },\n    })\n  }\n  const isLive = playbackTime.streamType === 'live'\n  const [settings, setSettings] = useState(() => ({\n    sections: [],\n    values: {speed: 1},\n  }))\n  const fetchSettings = () => {\n    if (!playerRef.current) {\n      return\n    }\n    const speedItems = appSettings.speedItems || (isLive ? [] : undefined)\n    setSettings(current => {\n      const {values, sections} = getSettingsData({\n        media: videoRef.current,\n        player: playerRef.current,\n        source,\n        quality,\n        speedItems,\n        loop,\n        preferred: current.preferred,\n      })\n      return {preferred: current.preferred, values, sections}\n    })\n  }\n  const lastState = useRef(playbackState)\n  const handlePlaybackStateChange = (event, state) => {\n    if (lastState.current === 'error') {\n      return\n    }\n    onPlaybackStateChange?.(event, state)\n    // previously UI playback state was synced only when\n    // exiting iOS video only fullscreen(webkitendfullscreen)\n    if (\n      /playing|paused/.test(state) &&\n      !(uiType === 'mobile' && blocking) &&\n      !shouldPause({userFocus, uiType})\n    ) {\n      togglePlay(state)\n    }\n    if (state === 'ended') {\n      togglePlay('paused')\n    }\n    if (state === 'loading' && lastState.current !== 'init') {\n      setTimeout(() => togglePlay('playing'), 1)\n    }\n    if (state === 'playing') {\n      setIsUserActive(true)\n    }\n    if (lastState.current === 'loading') {\n      fetchSettings()\n    }\n    lastState.current = state\n    setPlaybackState(state)\n  }\n\n  const changeSettings = (name, value, {keepOpen} = {}) => {\n    // TODO consider merge into useReducer?\n    onChangeSettings?.({name, value})\n    setTargetTime(playbackTime.currentTime)\n    setSettings(current => ({\n      ...current,\n      values: {...current.values, [name]: value},\n      preferred: {...current.preferred, [name]: value},\n    }))\n    if (!keepOpen) {\n      setUserFocus('')\n    }\n  }\n  const openSettings = event => {\n    const animationFrame =\n      userFocus !== 'settings' &&\n      requestAnimationFrame(() => {\n        onOpenSettings?.(event, settings)\n        // In iOS Safari, we need to update settings data\n        fetchSettings()\n      })\n    setUserFocus(current => (current === 'settings' ? '' : 'settings'))\n    return animationFrame\n  }\n  useEffect(() => {\n    if (appPlaybackRate > 0) {\n      setSettings(current => ({\n        ...current,\n        values: {...current.values, speed: appPlaybackRate},\n      }))\n    }\n  }, [appPlaybackRate])\n\n  const qualityOptions = useMemo(\n    () => getQualityOptions(settings),\n    [settings.values.quality]\n  )\n  const viewMode = useLinkState(update =>\n    onViewModeChange(videoRef.current, update)\n  )\n  const sourceOverride = useLinkState(\n    async update => {\n      const result =\n        source && (await quality.rewriteManifest?.(source, qualityOptions))\n      update(result || source)\n    },\n    [source, qualityOptions]\n  )\n  const waiting = /emptied|loading|buffering/.test(playbackState)\n  const activePlayback =\n    playbackState === 'playing' || playbackState === 'waiting'\n  const {\n    mode: autoHideMode,\n    onClick,\n    onMouseMove,\n  } = useAutoHide({\n    pinned: !controls.autohide || waiting || !activePlayback || userFocus,\n    tapToHide: uiType === 'mobile',\n    hideTimeMs: controls.autohide,\n  })\n  const mode =\n    userFocus === 'seekbar'\n      ? 'hidden'\n      : controls.autohide\n      ? autoHideMode\n      : controls\n      ? 'shown'\n      : 'hidden'\n  const controlsDisplay =\n    userFocus === 'seekbar'\n      ? 'seekbar-only'\n      : controls === 'title-only' || playbackState === 'emptied'\n      ? 'hidden'\n      : mode\n  const shouldHidePanels =\n    (controls === 'no-panel' || controlsDisplay === 'hidden') && userFocus\n  const havePlayPanel =\n    !blocking &&\n    havePointer() &&\n    uiType === 'desktop' &&\n    !waiting &&\n    !/volume|''/.test(userFocus) &&\n    !/title-only|no-panel/.test(controls) &&\n    (controls.autohide || mode === 'shown')\n\n  useEffect(() => {\n    if (shouldHidePanels) {\n      setUserFocus('')\n    }\n  }, [shouldHidePanels])\n  const {subscribe, onChange, toggleMute} = linkMediaVolume(() => ({\n    video: videoRef.current,\n    getPlayer: () => playerRef.current,\n  }))\n  const changePrevious = event => {\n    onChangePrevious(event)\n    togglePlay('paused')\n    videoRef.current.dispatchEvent(new CustomEvent('loadstart'))\n  }\n  const changeNext = event => {\n    onChangeNext(event)\n    togglePlay('paused')\n    videoRef.current.dispatchEvent(new CustomEvent('loadstart'))\n  }\n\n  useEffect(fetchSettings, [appSettings])\n  useEffect(() => {\n    // The adContainer should be set before `load` because ImaDai.load needs it.\n    plugins.forEach(plugin => plugin.setAdContainer?.(adContainerRef.current))\n  }, [])\n\n  useEffect(\n    () =>\n      dispatchChapterEvents({\n        media: videoRef.current,\n        chapters,\n        getTime: () => videoRef.current.currentTime,\n      }),\n    [chapters]\n  )\n\n  const isEnd = playbackState === 'ended'\n  const canSeek = !isEnd && playbackTime.duration > 0\n  const derivedPlaybackState =\n    (uiType === 'mobile' && blocking) || shouldPause({userFocus, uiType})\n      ? 'paused'\n      : targetState.playbackState\n  const activeThumbnailsUrl =\n    (!userFocus || userFocus === 'seekbar') &&\n    (getThumbnailsUrl(source) || thumbnailsUrl)\n  const cssVariables = {\n    '--playing': playbackState === 'playing' ? '1' : '0',\n  }\n  const onRewind =\n    (!isLive || playbackTime.currentTime < 0) &&\n    !isEnd &&\n    (() => setTargetTime(playbackTime.currentTime - 10, 'rewind'))\n  const onForward =\n    (!isLive || playbackTime.currentTime < 0) &&\n    canSeek &&\n    (() => setTargetTime(playbackTime.currentTime + 10, 'forward'))\n  const onClickBlank =\n    havePlayPanel && (() => togglePlay('', {triggeredBy: 'user-action'}))\n\n  const uiHandlers = {\n    onPlay: () => {\n      togglePlay('', {triggeredBy: 'user-action'})\n      setIsUserActive(true)\n    },\n    onClickLive: isLive && (() => setTargetTime(0, 'seekToLive')),\n    onClickSeekbar: () => {\n      refs.current.deferedSeekFocus = setTimeout(\n        () => setUserFocus('seekbar'),\n        66\n      )\n    },\n    onSeek: time => {\n      setTargetTime(time, 'seek')\n      clearTimeout(refs.current.deferedSeekFocus)\n      setTimeout(() => setUserFocus(''), 66)\n    },\n    onChangeSettings: ({name, value, keepOpen}) =>\n      changeSettings(name, value, {keepOpen}),\n    onCloseSettings: event => {\n      setUserFocus('')\n      slotProps?.settings?.onClose?.(event)\n    },\n  }\n\n  const uiElements = {\n    liveButton: uiHandlers.onClickLive && (\n      <components.LiveButton\n        usingStartOver={playbackTime.currentTime < 0}\n        onClick={uiHandlers.onClickLive}\n      />\n    ),\n    controlButtons: {\n      playButton: (\n        <PlayButton\n          playbackState={derivedPlaybackState}\n          ended={isEnd}\n          hidden={\n            uiType === 'mobile' && (waiting || /loading/.test(playbackState))\n          }\n          variant={!isUserActive && 'firstplay'}\n          onClick={uiHandlers.onPlay}\n        />\n      ),\n      rewindButton: onRewind && (\n        <Button\n          startIcon=\"rewind10\"\n          title=\"KKS.PLAYER.REWIND\"\n          disabled={isEnd}\n          onClick={onRewind}\n        />\n      ),\n      forwardButton: onForward && (\n        <ForwardButton\n          startIcon=\"forward10\"\n          title=\"KKS.PLAYER.FORWARD\"\n          disabled={!canSeek}\n          onClick={onForward}\n        />\n      ),\n      nextEpisodeButton: (\n        <Button\n          startIcon=\"next\"\n          title=\"KKS.PLAYER.NEXT\"\n          disabled={!onChangeNext}\n          onClick={changeNext}\n        />\n      ),\n      previousEpisodeButton: (\n        <Button\n          startIcon=\"previous\"\n          title=\"KKS.PLAYER.PREVIOUS\"\n          disabled={!onChangePrevious}\n          onClick={changePrevious}\n        />\n      ),\n    },\n    seekbar: (\n      <components.Seekbar\n        style={/volume/.test(userFocus) && {opacity: '0'}}\n        startTime={playbackTime.startTime}\n        currentTime={playbackTime.currentTime}\n        bufferTime={playbackTime.bufferTime}\n        duration={playbackTime.duration}\n        chapters={chapters}\n        onChange={uiHandlers.onClickSeekbar}\n        onChangeCommitted={uiHandlers.onSeek}\n        marks={marks}\n        plugins={plugins}\n        {...slotProps.seekbar}\n      >\n        {activeThumbnailsUrl && (\n          <SeekPreview\n            thumbnailsUrl={activeThumbnailsUrl}\n            duration={playbackTime.duration}\n            chapters={chapters}\n          />\n        )}\n      </components.Seekbar>\n    ),\n    displayTime: <components.DisplayTime {...playbackTime} />,\n    fullscreenButton: (\n      <FullscreenButton\n        viewMode={viewMode}\n        onClick={() => toggleFullscreen(containerRef.current)}\n      />\n    ),\n    volumeControl: width >= sizes['small-embed'] && (\n      <components.VolumeControl\n        {...{subscribe, onChange, toggleMute}}\n        onMouseOver={() => setUserFocus('volume')}\n        onMouseOut={() => setUserFocus('')}\n        {...slotProps?.volumeControl}\n      />\n    ),\n    backItems: (\n      <PlayPanel animation={targetState.animation} onClick={onClickBlank} />\n    ),\n  }\n  const compatibilityCastProps = castOptions === 'compatible' && {\n    source,\n    values: {title, ...settings.values, ...intl, modulesConfig},\n    onLoad: onCast,\n  }\n\n  useEffect(() => {\n    if (castOptions && castOptions !== 'compatible') {\n      return linkCast({source, ...castOptions})\n    } // TODO playlist, audio / subtitle setting\n  }, [source, castOptions.contentId, castOptions.customData])\n\n  return (\n    <IntlProvider {...intl}>\n      <Layout\n        style={{...style, cssVariables}}\n        type={uiType}\n        display={mode}\n        controlsDisplay={controlsDisplay}\n        size={size}\n        video={\n          <Video\n            {...videoProps}\n            videoRef={multiRef(videoRef, videoProps.videoRef)}\n            source={\n              castState !== CastState.CONNECTED &&\n              playbackState !== 'error' &&\n              sourceOverride\n            }\n            playbackState={derivedPlaybackState}\n            currentTime={targetState.currentTime}\n            {...(appVolume >= 0 && {volume: appVolume, muted: appVolume <= 0})}\n            playbackRate={settings.values.speed}\n            videoResolution={qualityOptions}\n            audio={settings.values.audio}\n            textTrack={settings.values.subtitles}\n            loop={settings.values.loop}\n            plugins={plugins}\n            modulesConfig={modulesConfig}\n            onPlaylogFired={onPlaylogFired}\n            onError={errorHandler}\n            onPlaybackStateChange={handlePlaybackStateChange}\n            onBlockedAutoplay={error => onBlockedAutoplay?.(error)}\n            onCanPlay={event => {\n              updatePlaybackTime(event)\n              videoProps.onCanPlay?.(event)\n            }}\n            onTimeUpdate={updatePlaybackTime}\n            onDurationChange={updatePlaybackTime}\n            onPlayerLoaded={player => {\n              playerRef.current = player\n              onPlayerLoaded?.(player)\n              syncVolume(videoRef.current, videoProps.muted ? 0 : appVolume)\n            }}\n          />\n        }\n        containerRef={element => {\n          containerRef.current = element\n          observe(element)\n        }}\n        backRef={adContainerRef}\n        title={title}\n        channelTitle={channelTitle}\n        backButton={onBack && <Button startIcon=\"back\" onClick={onBack} />}\n        {...(isUserActive && uiElements)}\n        onClick={onClick}\n        onMouseMove={onMouseMove}\n      >\n        <LoadingSpinner active={waiting} />\n        {errorData && <Error error={errorData} onBack={onBack} />}\n        {children}\n        {/* TODO sync Cast last played time back */}\n        {isUserActive && (\n          <components.CastUi\n            onBack={onBack}\n            onStateChange={setCastState}\n            onLoad={onCast}\n            {...compatibilityCastProps}\n            {...slotProps.castUi}\n          />\n        )}\n        {isUserActive && ( // TODO: Design <Extenstion /> flag\n          <components.Settings\n            type={uiType}\n            sections={settings.sections}\n            // TODO hasBottomPanel bottom: 8em\n            open={userFocus === 'settings'}\n            values={settings.values}\n            onOpen={openSettings}\n            onChange={uiHandlers.onChangeSettings}\n            {...slotProps?.settings}\n            onClose={uiHandlers.onCloseSettings}\n          />\n        )}\n        <LanguageSettings\n          uiType={uiType}\n          getPlayer={() => playerRef.current}\n          onChange={(_event, item) => changeSettings(item.type, item.value)}\n          {...slotProps.languageSettings}\n        />\n        <Backdrop open={!playbackState || playbackState === 'loading'}>\n          <LoadingSpinner />\n        </Backdrop>\n      </Layout>\n    </IntlProvider>\n  )\n}\n\nexport default PremiumPlayer\n"]} */")
|
|
8152
8154
|
};
|
|
8153
8155
|
|
|
8154
8156
|
const LanguageSettings = ({
|
|
8155
8157
|
uiType,
|
|
8156
8158
|
audioTracks = [],
|
|
8157
8159
|
subtitleTracks = [],
|
|
8160
|
+
getPlayer,
|
|
8158
8161
|
onChange,
|
|
8159
8162
|
slots = {
|
|
8160
8163
|
LanguageMenu
|
|
8161
8164
|
},
|
|
8162
8165
|
...rest
|
|
8163
|
-
}) =>
|
|
8164
|
-
|
|
8165
|
-
|
|
8166
|
-
|
|
8167
|
-
|
|
8168
|
-
|
|
8169
|
-
|
|
8170
|
-
|
|
8171
|
-
|
|
8172
|
-
|
|
8166
|
+
}) => {
|
|
8167
|
+
const options = getLanguageOptions(getPlayer(), {
|
|
8168
|
+
audioTracks,
|
|
8169
|
+
subtitleTracks
|
|
8170
|
+
});
|
|
8171
|
+
return /*#__PURE__*/jsx(OverlayPanel, {
|
|
8172
|
+
buttonIcon: "subtitle",
|
|
8173
|
+
children: /*#__PURE__*/jsx(slots.LanguageMenu, {
|
|
8174
|
+
uiType: uiType,
|
|
8175
|
+
classes: menuClasses,
|
|
8176
|
+
sectionOptions: [options.audioTracks, options.textTracks],
|
|
8177
|
+
onChange: onChange,
|
|
8178
|
+
...rest
|
|
8179
|
+
})
|
|
8180
|
+
});
|
|
8181
|
+
};
|
|
8173
8182
|
|
|
8174
8183
|
const PremiumPlayer = ({
|
|
8175
8184
|
source,
|
|
@@ -8716,8 +8725,8 @@ const PremiumPlayer = ({
|
|
|
8716
8725
|
onClose: uiHandlers.onCloseSettings
|
|
8717
8726
|
}), /*#__PURE__*/jsx(LanguageSettings, {
|
|
8718
8727
|
uiType: uiType,
|
|
8719
|
-
|
|
8720
|
-
onChange: (_event, item) => changeSettings(item.
|
|
8728
|
+
getPlayer: () => playerRef.current,
|
|
8729
|
+
onChange: (_event, item) => changeSettings(item.type, item.value),
|
|
8721
8730
|
...slotProps.languageSettings
|
|
8722
8731
|
}), /*#__PURE__*/jsx(Backdrop, {
|
|
8723
8732
|
open: !playbackState || playbackState === 'loading',
|
|
@@ -9315,9 +9324,9 @@ class Player {
|
|
|
9315
9324
|
player: this.player
|
|
9316
9325
|
}) || []);
|
|
9317
9326
|
|
|
9318
|
-
_defineProperty(this, "getCurrentAudioTrack", () =>
|
|
9327
|
+
_defineProperty(this, "getCurrentAudioTrack", () => getAudioTracks({}, {
|
|
9319
9328
|
player: this.player
|
|
9320
|
-
}));
|
|
9329
|
+
}).find(track => track.selected));
|
|
9321
9330
|
|
|
9322
9331
|
_defineProperty(this, "setAudioTrack", track => {
|
|
9323
9332
|
// TODO also accept Track, name: setTextTrack
|