@kkcompany/player 2.25.0-canary.21 → 2.25.0-canary.22
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 +12 -0
- package/dist/core.mjs +1 -1
- package/dist/index.js +55 -41
- package/dist/index.mjs +54 -40
- package/dist/modules.mjs +2 -2
- package/dist/react.mjs +55 -41
- package/package.json +1 -1
package/dist/index.mjs
CHANGED
|
@@ -762,30 +762,29 @@ const syncTextTrack = (media, {
|
|
|
762
762
|
|
|
763
763
|
Array.from(cues).forEach(cue => customTextTrack.addCue(new VTTCue(cue.startTime, cue.endTime, cue.text)));
|
|
764
764
|
customTextTrack.mode = 'showing';
|
|
765
|
-
|
|
766
|
-
|
|
767
|
-
} else {
|
|
768
|
-
if (customTextTrack) {
|
|
769
|
-
customTextTrack.mode = 'hidden';
|
|
770
|
-
}
|
|
765
|
+
return;
|
|
766
|
+
}
|
|
771
767
|
|
|
772
|
-
|
|
773
|
-
|
|
774
|
-
|
|
768
|
+
if (customTextTrack) {
|
|
769
|
+
customTextTrack.mode = 'hidden';
|
|
770
|
+
}
|
|
775
771
|
|
|
776
|
-
|
|
777
|
-
|
|
778
|
-
|
|
779
|
-
|
|
780
|
-
|
|
781
|
-
|
|
772
|
+
const matchedTrack = getTextTracks(media, {
|
|
773
|
+
player
|
|
774
|
+
}).find(t => (!language || t.language === language) && (!label || t.label === label));
|
|
775
|
+
|
|
776
|
+
if (matchedTrack) {
|
|
777
|
+
player === null || player === void 0 ? void 0 : player.selectTextTrack(matchedTrack);
|
|
778
|
+
return;
|
|
782
779
|
}
|
|
783
780
|
|
|
784
|
-
|
|
785
|
-
|
|
786
|
-
|
|
787
|
-
}
|
|
788
|
-
}
|
|
781
|
+
player.setTextTrackVisibility(false);
|
|
782
|
+
|
|
783
|
+
if (!/off|none/i.test(label + language)) {
|
|
784
|
+
console.warn(`turn off text track: ${label} ${language} is not found`);
|
|
785
|
+
}
|
|
786
|
+
|
|
787
|
+
return ((_media$textTracks = media.textTracks) === null || _media$textTracks === void 0 ? void 0 : _media$textTracks.addEventListener) && once(media.textTracks, 'change', () => player.setTextTrackVisibility(false));
|
|
789
788
|
}; // assume unique labels as defined in spec
|
|
790
789
|
|
|
791
790
|
|
|
@@ -812,7 +811,7 @@ function convertToSeconds(timeString) {
|
|
|
812
811
|
function getVersion() {
|
|
813
812
|
try {
|
|
814
813
|
// eslint-disable-next-line no-undef
|
|
815
|
-
return "2.25.0-canary.
|
|
814
|
+
return "2.25.0-canary.22";
|
|
816
815
|
} catch (e) {
|
|
817
816
|
return undefined;
|
|
818
817
|
}
|
|
@@ -1137,11 +1136,9 @@ const getLanguageOptions = (player, sorted) => {
|
|
|
1137
1136
|
});
|
|
1138
1137
|
const textOptions = [].concat(textTracks, {
|
|
1139
1138
|
type: 'subtitles',
|
|
1140
|
-
label: 'none',
|
|
1141
1139
|
language: 'none',
|
|
1142
1140
|
selected: !textTracks.some(track => track.selected),
|
|
1143
1141
|
value: {
|
|
1144
|
-
label: 'none',
|
|
1145
1142
|
language: 'none',
|
|
1146
1143
|
id: 'off'
|
|
1147
1144
|
}
|
|
@@ -3864,7 +3861,7 @@ const SliderRail = ({
|
|
|
3864
3861
|
css: [style, orientation !== 'horizontal' && {
|
|
3865
3862
|
borderRadius: '0.2em',
|
|
3866
3863
|
alignSelf: 'normal'
|
|
3867
|
-
}, process.env.NODE_ENV === "production" ? "" : ";label:SliderRail;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIlNpbXBsZVNsaWRlci5qc3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBcUhJIiwiZmlsZSI6IlNpbXBsZVNsaWRlci5qc3giLCJzb3VyY2VzQ29udGVudCI6WyIvKiBlc2xpbnQtZGlzYWJsZSBuby1wYXJhbS1yZWFzc2lnbiAqL1xuLyogZXNsaW50LWRpc2FibGUganN4LWExMXkvbm8tc3RhdGljLWVsZW1lbnQtaW50ZXJhY3Rpb25zICovXG4vKiBlc2xpbnQtZGlzYWJsZSBqc3gtYTExeS9jbGljay1ldmVudHMtaGF2ZS1rZXktZXZlbnRzICovXG4vKiBAanN4SW1wb3J0U291cmNlIEBlbW90aW9uL3JlYWN0ICovXG5pbXBvcnQge3VzZVN0YXRlLCB1c2VSZWZ9IGZyb20gJ3JlYWN0J1xuXG5pbXBvcnQge2dldFBvaW50ZXJEYXRhfSBmcm9tICd1dGlsL2dlc3R1cmVzJ1xuXG5jb25zdCBjb250YWluZXJTdHlsZSA9IHtcbiAgcG9zaXRpb246ICdyZWxhdGl2ZScsXG4gIHdpZHRoOiAnMTAwJScsXG4gIGhlaWdodDogJzEwMCUnLFxuICBkaXNwbGF5OiAnZmxleCcsXG4gIGFsaWduSXRlbXM6ICdjZW50ZXInLFxuICBjdXJzb3I6ICdwb2ludGVyJyxcbiAgdXNlclNlbGVjdDogJ25vbmUnLFxuICB0b3VjaEFjdGlvbjogJ25vbmUnLFxufVxuXG5jb25zdCBkaXNhYmxlZFN0eWxlID0ge1xuICBwb2ludGVyRXZlbnRzOiAnbm9uZScsXG59XG5cbmNvbnN0IHJhaWxTdHlsZSA9IHtcbiAgZmxleDogJzEwMCUnLFxuICBiYWNrZ3JvdW5kOiAncmdiYSgyNTUsIDI1NSwgMjU1LCAwLjIpJyxcbn1cblxuY29uc3QgdHJhY2tTdHlsZSA9IHtcbiAgcG9zaXRpb246ICdhYnNvbHV0ZScsXG4gIHdpZHRoOiAnNHB4JyxcbiAgaGVpZ2h0OiAnMTAwJScsXG4gIGJhY2tncm91bmRDb2xvcjogJ3ZhcigtLXNlbmRlci1zZWVrYmFyLWJhY2tncm91bmQsICNmZmYpJyxcbn1cblxuY29uc3QgbWFya1N0eWxlID0ge1xuICBwb3NpdGlvbjogJ2Fic29sdXRlJyxcbiAgaGVpZ2h0OiByYWlsU3R5bGUuaGVpZ2h0LFxuICB3aWR0aDogJzRweCcsXG4gIHRyYW5zZm9ybTogJ3RyYW5zbGF0ZVgoLTUwJSknLFxuICBiYWNrZ3JvdW5kQ29sb3I6ICcjZmY5ODM1Jyxcbn1cblxuY29uc3QgdGh1bWJTdHlsZSA9IHtcbiAgcG9zaXRpb246ICdhYnNvbHV0ZScsXG4gIGhlaWdodDogJzAuNjZlbScsXG4gIHdpZHRoOiAnMC42NmVtJyxcbiAgYm9yZGVyUmFkaXVzOiAnMTAwJScsXG4gIGJhY2tncm91bmRDb2xvcjogJyNmZmYnLFxuICBib3hTaGFkb3c6ICcwIDJweCAycHggMCByZ2JhKDAsIDAsIDAsIDAuNSknLFxuICB0cmFuc2Zvcm06ICd0cmFuc2xhdGVZKGNhbGModmFyKC0tc2xpZGVyLXRodW1iLXkpIC0gMC4zNWVtKSknLFxufVxuXG5jb25zdCBnZXRQb2ludGVyVmFsdWUgPSAoe29yaWVudGF0aW9uLCB4LCB5LCBsZWZ0LCBib3R0b20sIGhlaWdodCwgd2lkdGh9KSA9PlxuICBNYXRoLm1heChcbiAgICAwLFxuICAgIG9yaWVudGF0aW9uID09PSAndmVydGljYWwnXG4gICAgICA/IE1hdGgubWluKChib3R0b20gLSB5KSAvIGhlaWdodCwgMSlcbiAgICAgIDogTWF0aC5taW4oKHggLSBsZWZ0KSAvIHdpZHRoLCAxKSxcbiAgKVxuXG5jb25zdCBkZWJvdW5jZWRQb2ludGVySGFuZGxlcnMgPSAoe3N0YXRlLCBvbk1vdmUsIG9uTGVhdmV9KSA9PiB7XG4gIGNvbnN0IGVtaXQgPSAoKSA9PiB7XG4gICAgaWYgKCFzdGF0ZS5zY2hlZHVsZWQpIHtcbiAgICAgIHJldHVyblxuICAgIH1cbiAgICBpZiAoc3RhdGUudHlwZSA9PT0gJ2xlYXZlJykge1xuICAgICAgb25MZWF2ZT8uKHN0YXRlLmV2ZW50LCBzdGF0ZSlcbiAgICB9IGVsc2Uge1xuICAgICAgb25Nb3ZlKHN0YXRlLmV2ZW50LCBzdGF0ZSlcbiAgICB9XG4gICAgc3RhdGUuc2NoZWR1bGVkID0gZmFsc2VcbiAgfVxuICBjb25zdCBzY2hlZHVsZSA9ICgpID0+IHtcbiAgICBpZiAoc3RhdGUuc2NoZWR1bGVkKSB7XG4gICAgICByZXR1cm5cbiAgICB9XG4gICAgc3RhdGUuc2NoZWR1bGVkID0gdHJ1ZVxuICAgIHJlcXVlc3RBbmltYXRpb25GcmFtZShlbWl0KVxuICB9XG5cbiAgcmV0dXJuIHtcbiAgICBvblBvaW50ZXJNb3ZlOiBldmVudCA9PiB7XG4gICAgICBjb25zdCB0eXBlID1cbiAgICAgICAgZXZlbnQuYnV0dG9ucyA+IDAgfHwgZXZlbnQudG91Y2hlcz8ubGVuZ3RoID4gMCA/ICdjaGFuZ2UnIDogJ21vdmUnXG4gICAgICBPYmplY3QuYXNzaWduKHN0YXRlLCB7ZXZlbnQsIHR5cGUsIC4uLmdldFBvaW50ZXJEYXRhKGV2ZW50KX0pXG4gICAgICBzY2hlZHVsZSgpXG4gICAgfSxcbiAgICBvblBvaW50ZXJMZWF2ZTogZXZlbnQgPT4ge1xuICAgICAgY29uc3QgdHlwZSA9ICdsZWF2ZSdcbiAgICAgIE9iamVjdC5hc3NpZ24oc3RhdGUsIHtldmVudCwgdHlwZX0pXG4gICAgICBzY2hlZHVsZSgpXG4gICAgfSxcbiAgICBlbWl0LFxuICB9XG59XG5cbmNvbnN0IGV2ZW50SGFuZGxlcnMgPSAoe1xuICBvblBvaW50ZXJEb3duLFxuICBvblBvaW50ZXJNb3ZlLFxuICBvblBvaW50ZXJMZWF2ZSxcbiAgb25Qb2ludGVyVXAsXG59KSA9PiAoe1xuICBvblBvaW50ZXJEb3duLFxuICBvblBvaW50ZXJNb3ZlLFxuICBvblBvaW50ZXJMZWF2ZSxcbiAgb25Qb2ludGVyVXAsXG4gIG9uVG91Y2hTdGFydDogb25Qb2ludGVyRG93bixcbiAgb25Ub3VjaE1vdmU6IG9uUG9pbnRlck1vdmUsXG4gIG9uVG91Y2hFbmQ6IGV2ZW50ID0+IHtcbiAgICBvblBvaW50ZXJMZWF2ZShldmVudClcbiAgICBvblBvaW50ZXJVcChldmVudClcbiAgfSxcbn0pXG5cbmNvbnN0IFNsaWRlclJhaWwgPSAoe29yaWVudGF0aW9uLCBzdHlsZSwgLi4ucmVzdH0pID0+IChcbiAgPGRpdlxuICAgIGNzcz17W3N0eWxlLCBvcmllbnRhdGlvbiAhPT0gJ2hvcml6b250YWwnICYmIHtib3JkZXJSYWRpdXM6ICcwLjJlbScsIGFsaWduU2VsZjogJ25vcm1hbCd9XX1cbiAgICB7Li4ucmVzdH1cbiAgLz5cbilcblxuY29uc3QgU2xpZGVyVHJhY2sgPSAoe3ZhbHVlLCBvcmllbnRhdGlvbiwgc3R5bGUsIC4uLnJlc3R9KSA9PiAoXG4gIDxkaXZcbiAgICBjc3M9e1tcbiAgICAgIHN0eWxlLFxuICAgICAgb3JpZW50YXRpb24gPT09ICd2ZXJ0aWNhbCdcbiAgICAgICAgPyB7Ym90dG9tOiAnMCcsIGhlaWdodDogYCR7dmFsdWUgKiAxMDB9JWB9XG4gICAgICAgIDoge3dpZHRoOiBgJHt2YWx1ZSAqIDEwMH0lYH0sXG4gICAgXX1cbiAgICB7Li4ucmVzdH1cbiAgLz5cbilcblxuY29uc3QgZGVmYXVsdFNsb3RzID0ge1xuICBSYWlsOiBTbGlkZXJSYWlsLFxuICBUcmFjazogU2xpZGVyVHJhY2ssXG59XG5cbi8vIFRPRE8gYWxpZ24gd2l0aCBtYXRlcmlhbCB1aSBtb3JlLCBtb3ZlIHNwZWNpYWwgaGFuZGxpbmcgb2YgcG9pbnRlciBldmVudHNcbmNvbnN0IFNpbXBsZVNsaWRlciA9ICh7XG4gIG1pbiA9IDAsXG4gIG1heCA9IDEwMCxcbiAgdmFsdWUsXG4gIHNlY29uZGFyeVRyYWNrVmFsdWUsIC8vIFRPRE8gYSBiZXR0ZXIgbmFtZVxuICBtYXJrcyA9IFtdLFxuICBjbGFzc05hbWUgPSAnJyxcbiAgY2xhc3NlcyA9IHt9LFxuICBkaXNhYmxlZCxcbiAgb25Qb2ludGVyTW92ZSxcbiAgb25Qb2ludGVyTGVhdmUsXG4gIG9uUG9pbnRlckRvd24sXG4gIG9uQ2hhbmdlLFxuICBvbkNoYW5nZUNvbW1pdHRlZCxcbiAgb3JpZW50YXRpb24gPSAnaG9yaXpvbnRhbCcsXG4gIHNsb3RzID0gZGVmYXVsdFNsb3RzLFxuICBzbG90UHJvcHMgPSB7fSxcbn0pID0+IHtcbiAgY29uc3QgcG9pbnRlclN0YXRlID0gdXNlUmVmKHt9KVxuICBjb25zdCBbZm9jdXNWYWx1ZSwgc2V0Rm9jdXNWYWx1ZV0gPSB1c2VTdGF0ZSgtSW5maW5pdHkpXG4gIGNvbnN0IHRodW1iUG9zaXRpb24gPVxuICAgICgoZm9jdXNWYWx1ZSA+PSBtaW4gPyBmb2N1c1ZhbHVlIDogdmFsdWUpIC0gbWluKSAvIChtYXggLSBtaW4pXG4gIGNvbnN0IHN1YlRyYWNrUG9zaXRpb24gPSAoc2Vjb25kYXJ5VHJhY2tWYWx1ZSAtIG1pbikgLyAobWF4IC0gbWluKVxuICBjb25zdCBwb2ludGVySGFuZGxlcnMgPSBkZWJvdW5jZWRQb2ludGVySGFuZGxlcnMoe1xuICAgIHN0YXRlOiBwb2ludGVyU3RhdGUuY3VycmVudCxcbiAgICBvbk1vdmU6IChldmVudCwge3R5cGUsIHgsIHksIHdpZHRoLCBsZWZ0LCBoZWlnaHQsIGJvdHRvbX0pID0+IHtcbiAgICAgIGNvbnN0IHBvaW50ZXJEYXRhID0ge29yaWVudGF0aW9uLCB4LCB5LCBsZWZ0LCBib3R0b20sIHdpZHRoLCBoZWlnaHR9XG4gICAgICBjb25zdCBwb2ludGVyVmFsdWUgPSBtaW4gKyAobWF4IC0gbWluKSAqIGdldFBvaW50ZXJWYWx1ZShwb2ludGVyRGF0YSlcblxuICAgICAgb25Qb2ludGVyTW92ZT8uKGV2ZW50LCB7dmFsdWU6IHBvaW50ZXJWYWx1ZSwgeCwgeX0pXG4gICAgICBpZiAodHlwZSA9PT0gJ2NoYW5nZScpIHtcbiAgICAgICAgc2V0Rm9jdXNWYWx1ZShwb2ludGVyVmFsdWUpXG4gICAgICAgIG9uQ2hhbmdlPy4oZXZlbnQsIHt2YWx1ZTogcG9pbnRlclZhbHVlLCB4LCB5fSlcbiAgICAgIH1cbiAgICB9LFxuICAgIG9uTGVhdmU6ICgpID0+
|
|
3864
|
+
}, process.env.NODE_ENV === "production" ? "" : ";label:SliderRail;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SimpleSlider.jsx"],"names":[],"mappings":"AAqHI","file":"SimpleSlider.jsx","sourcesContent":["/* eslint-disable no-param-reassign */\n/* eslint-disable jsx-a11y/no-static-element-interactions */\n/* eslint-disable jsx-a11y/click-events-have-key-events */\n/* @jsxImportSource @emotion/react */\nimport {useState, useRef} from 'react'\n\nimport {getPointerData} from 'util/gestures'\n\nconst containerStyle = {\n  position: 'relative',\n  width: '100%',\n  height: '100%',\n  display: 'flex',\n  alignItems: 'center',\n  cursor: 'pointer',\n  userSelect: 'none',\n  touchAction: 'none',\n}\n\nconst disabledStyle = {\n  pointerEvents: 'none',\n}\n\nconst railStyle = {\n  flex: '100%',\n  background: 'rgba(255, 255, 255, 0.2)',\n}\n\nconst trackStyle = {\n  position: 'absolute',\n  width: '4px',\n  height: '100%',\n  backgroundColor: 'var(--sender-seekbar-background, #fff)',\n}\n\nconst markStyle = {\n  position: 'absolute',\n  height: railStyle.height,\n  width: '4px',\n  transform: 'translateX(-50%)',\n  backgroundColor: '#ff9835',\n}\n\nconst thumbStyle = {\n  position: 'absolute',\n  height: '0.66em',\n  width: '0.66em',\n  borderRadius: '100%',\n  backgroundColor: '#fff',\n  boxShadow: '0 2px 2px 0 rgba(0, 0, 0, 0.5)',\n  transform: 'translateY(calc(var(--slider-thumb-y) - 0.35em))',\n}\n\nconst getPointerValue = ({orientation, x, y, left, bottom, height, width}) =>\n  Math.max(\n    0,\n    orientation === 'vertical'\n      ? Math.min((bottom - y) / height, 1)\n      : Math.min((x - left) / width, 1),\n  )\n\nconst debouncedPointerHandlers = ({state, onMove, onLeave}) => {\n  const emit = () => {\n    if (!state.scheduled) {\n      return\n    }\n    if (state.type === 'leave') {\n      onLeave?.(state.event, state)\n    } else {\n      onMove(state.event, state)\n    }\n    state.scheduled = false\n  }\n  const schedule = () => {\n    if (state.scheduled) {\n      return\n    }\n    state.scheduled = true\n    requestAnimationFrame(emit)\n  }\n\n  return {\n    onPointerMove: event => {\n      const type =\n        event.buttons > 0 || event.touches?.length > 0 ? 'change' : 'move'\n      Object.assign(state, {event, type, ...getPointerData(event)})\n      schedule()\n    },\n    onPointerLeave: event => {\n      const type = 'leave'\n      Object.assign(state, {event, type})\n      schedule()\n    },\n    emit,\n  }\n}\n\nconst eventHandlers = ({\n  onPointerDown,\n  onPointerMove,\n  onPointerLeave,\n  onPointerUp,\n}) => ({\n  onPointerDown,\n  onPointerMove,\n  onPointerLeave,\n  onPointerUp,\n  onTouchStart: onPointerDown,\n  onTouchMove: onPointerMove,\n  onTouchEnd: event => {\n    onPointerLeave(event)\n    onPointerUp(event)\n  },\n})\n\nconst SliderRail = ({orientation, style, ...rest}) => (\n  <div\n    css={[style, orientation !== 'horizontal' && {borderRadius: '0.2em', alignSelf: 'normal'}]}\n    {...rest}\n  />\n)\n\nconst SliderTrack = ({value, orientation, style, ...rest}) => (\n  <div\n    css={[\n      style,\n      orientation === 'vertical'\n        ? {bottom: '0', height: `${value * 100}%`}\n        : {width: `${value * 100}%`},\n    ]}\n    {...rest}\n  />\n)\n\nconst defaultSlots = {\n  Rail: SliderRail,\n  Track: SliderTrack,\n}\n\n// TODO align with material ui more, move special handling of pointer events\nconst SimpleSlider = ({\n  min = 0,\n  max = 100,\n  value,\n  secondaryTrackValue, // TODO a better name\n  marks = [],\n  className = '',\n  classes = {},\n  disabled,\n  onPointerMove,\n  onPointerLeave,\n  onPointerDown,\n  onChange,\n  onChangeCommitted,\n  orientation = 'horizontal',\n  slots = defaultSlots,\n  slotProps = {},\n}) => {\n  const pointerState = useRef({})\n  const [focusValue, setFocusValue] = useState(-Infinity)\n  const thumbPosition =\n    ((focusValue >= min ? focusValue : value) - min) / (max - min)\n  const subTrackPosition = (secondaryTrackValue - min) / (max - min)\n  const pointerHandlers = debouncedPointerHandlers({\n    state: pointerState.current,\n    onMove: (event, {type, x, y, width, left, height, bottom}) => {\n      const pointerData = {orientation, x, y, left, bottom, width, height}\n      const pointerValue = min + (max - min) * getPointerValue(pointerData)\n\n      onPointerMove?.(event, {value: pointerValue, x, y})\n      if (type === 'change') {\n        setFocusValue(pointerValue)\n        onChange?.(event, {value: pointerValue, x, y})\n      }\n    },\n    onLeave: () => onPointerLeave?.(),\n  })\n  const handlePointerUp = event => {\n    if (event.pointerId) {\n      event.currentTarget.releasePointerCapture(event.pointerId)\n    }\n    const {x, y, width, left, height, bottom} = getPointerData(event)\n    const pointerValue =\n      min +\n      (max - min) *\n        getPointerValue({orientation, x, y, left, bottom, width, height})\n\n    pointerHandlers.emit()\n    onChangeCommitted?.(event, {value: pointerValue})\n    setFocusValue()\n  }\n\n  const thumbPositionStyle = {\n    bottom: `calc(var(--slider-thumb-y) - 0.35em)`,\n    left: `calc(var(--slider-thumb-x) - 0.25em)`,\n  }\n\n  return (\n    <div\n      className={className}\n      css={[containerStyle, disabled && disabledStyle]}\n      style={{\n        ...(orientation === 'vertical' && {\n          flexDirection: 'column',\n          '--slider-thumb-y': `${thumbPosition * 100}%`,\n        }),\n        ...(orientation === 'horizontal' && {\n          '--slider-thumb-x': `${thumbPosition * 100}%`,\n        }),\n      }}\n      {...eventHandlers({\n        onPointerDown: event => {\n          if (event.type === 'pointerdown') {\n            event.currentTarget.setPointerCapture(event.pointerId)\n            onPointerDown?.()\n          }\n          pointerHandlers.onPointerMove(event)\n        },\n        ...pointerHandlers,\n        onPointerUp: handlePointerUp,\n      })}\n    >\n      <slots.Rail style={railStyle} className={classes.rail} {...slotProps.rail} />\n      {secondaryTrackValue && (\n        <slots.Track\n          style={{...trackStyle, backgroundColor: 'rgba(255, 255, 255, 0.3)'}}\n          orientation={orientation}\n          value={subTrackPosition}\n          {...slotProps.track}\n        />\n      )}\n      <slots.Track\n        className={classes.track}\n        style={trackStyle}\n        orientation={orientation}\n        value={thumbPosition}\n        {...slotProps.track}\n      />\n      {marks.map(position => (\n        <div\n          key={position}\n          css={markStyle}\n          className={classes.marked}\n          style={{left: `${(position / max) * 100}%`}}\n        />\n      ))}\n      {onChange && !disabled ? (\n        <div\n          css={thumbStyle}\n          className={classes.thumb}\n          style={thumbPositionStyle}\n        />\n      ) : (\n        <div />\n      )}\n    </div>\n  )\n}\n\nexport default SimpleSlider\n"]} */"],
|
|
3868
3865
|
...rest
|
|
3869
3866
|
});
|
|
3870
3867
|
|
|
@@ -3879,7 +3876,7 @@ const SliderTrack = ({
|
|
|
3879
3876
|
height: `${value * 100}%`
|
|
3880
3877
|
} : {
|
|
3881
3878
|
width: `${value * 100}%`
|
|
3882
|
-
}, process.env.NODE_ENV === "production" ? "" : ";label:SliderTrack;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIlNpbXBsZVNsaWRlci5qc3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBNEhJIiwiZmlsZSI6IlNpbXBsZVNsaWRlci5qc3giLCJzb3VyY2VzQ29udGVudCI6WyIvKiBlc2xpbnQtZGlzYWJsZSBuby1wYXJhbS1yZWFzc2lnbiAqL1xuLyogZXNsaW50LWRpc2FibGUganN4LWExMXkvbm8tc3RhdGljLWVsZW1lbnQtaW50ZXJhY3Rpb25zICovXG4vKiBlc2xpbnQtZGlzYWJsZSBqc3gtYTExeS9jbGljay1ldmVudHMtaGF2ZS1rZXktZXZlbnRzICovXG4vKiBAanN4SW1wb3J0U291cmNlIEBlbW90aW9uL3JlYWN0ICovXG5pbXBvcnQge3VzZVN0YXRlLCB1c2VSZWZ9IGZyb20gJ3JlYWN0J1xuXG5pbXBvcnQge2dldFBvaW50ZXJEYXRhfSBmcm9tICd1dGlsL2dlc3R1cmVzJ1xuXG5jb25zdCBjb250YWluZXJTdHlsZSA9IHtcbiAgcG9zaXRpb246ICdyZWxhdGl2ZScsXG4gIHdpZHRoOiAnMTAwJScsXG4gIGhlaWdodDogJzEwMCUnLFxuICBkaXNwbGF5OiAnZmxleCcsXG4gIGFsaWduSXRlbXM6ICdjZW50ZXInLFxuICBjdXJzb3I6ICdwb2ludGVyJyxcbiAgdXNlclNlbGVjdDogJ25vbmUnLFxuICB0b3VjaEFjdGlvbjogJ25vbmUnLFxufVxuXG5jb25zdCBkaXNhYmxlZFN0eWxlID0ge1xuICBwb2ludGVyRXZlbnRzOiAnbm9uZScsXG59XG5cbmNvbnN0IHJhaWxTdHlsZSA9IHtcbiAgZmxleDogJzEwMCUnLFxuICBiYWNrZ3JvdW5kOiAncmdiYSgyNTUsIDI1NSwgMjU1LCAwLjIpJyxcbn1cblxuY29uc3QgdHJhY2tTdHlsZSA9IHtcbiAgcG9zaXRpb246ICdhYnNvbHV0ZScsXG4gIHdpZHRoOiAnNHB4JyxcbiAgaGVpZ2h0OiAnMTAwJScsXG4gIGJhY2tncm91bmRDb2xvcjogJ3ZhcigtLXNlbmRlci1zZWVrYmFyLWJhY2tncm91bmQsICNmZmYpJyxcbn1cblxuY29uc3QgbWFya1N0eWxlID0ge1xuICBwb3NpdGlvbjogJ2Fic29sdXRlJyxcbiAgaGVpZ2h0OiByYWlsU3R5bGUuaGVpZ2h0LFxuICB3aWR0aDogJzRweCcsXG4gIHRyYW5zZm9ybTogJ3RyYW5zbGF0ZVgoLTUwJSknLFxuICBiYWNrZ3JvdW5kQ29sb3I6ICcjZmY5ODM1Jyxcbn1cblxuY29uc3QgdGh1bWJTdHlsZSA9IHtcbiAgcG9zaXRpb246ICdhYnNvbHV0ZScsXG4gIGhlaWdodDogJzAuNjZlbScsXG4gIHdpZHRoOiAnMC42NmVtJyxcbiAgYm9yZGVyUmFkaXVzOiAnMTAwJScsXG4gIGJhY2tncm91bmRDb2xvcjogJyNmZmYnLFxuICBib3hTaGFkb3c6ICcwIDJweCAycHggMCByZ2JhKDAsIDAsIDAsIDAuNSknLFxuICB0cmFuc2Zvcm06ICd0cmFuc2xhdGVZKGNhbGModmFyKC0tc2xpZGVyLXRodW1iLXkpIC0gMC4zNWVtKSknLFxufVxuXG5jb25zdCBnZXRQb2ludGVyVmFsdWUgPSAoe29yaWVudGF0aW9uLCB4LCB5LCBsZWZ0LCBib3R0b20sIGhlaWdodCwgd2lkdGh9KSA9PlxuICBNYXRoLm1heChcbiAgICAwLFxuICAgIG9yaWVudGF0aW9uID09PSAndmVydGljYWwnXG4gICAgICA/IE1hdGgubWluKChib3R0b20gLSB5KSAvIGhlaWdodCwgMSlcbiAgICAgIDogTWF0aC5taW4oKHggLSBsZWZ0KSAvIHdpZHRoLCAxKSxcbiAgKVxuXG5jb25zdCBkZWJvdW5jZWRQb2ludGVySGFuZGxlcnMgPSAoe3N0YXRlLCBvbk1vdmUsIG9uTGVhdmV9KSA9PiB7XG4gIGNvbnN0IGVtaXQgPSAoKSA9PiB7XG4gICAgaWYgKCFzdGF0ZS5zY2hlZHVsZWQpIHtcbiAgICAgIHJldHVyblxuICAgIH1cbiAgICBpZiAoc3RhdGUudHlwZSA9PT0gJ2xlYXZlJykge1xuICAgICAgb25MZWF2ZT8uKHN0YXRlLmV2ZW50LCBzdGF0ZSlcbiAgICB9IGVsc2Uge1xuICAgICAgb25Nb3ZlKHN0YXRlLmV2ZW50LCBzdGF0ZSlcbiAgICB9XG4gICAgc3RhdGUuc2NoZWR1bGVkID0gZmFsc2VcbiAgfVxuICBjb25zdCBzY2hlZHVsZSA9ICgpID0+IHtcbiAgICBpZiAoc3RhdGUuc2NoZWR1bGVkKSB7XG4gICAgICByZXR1cm5cbiAgICB9XG4gICAgc3RhdGUuc2NoZWR1bGVkID0gdHJ1ZVxuICAgIHJlcXVlc3RBbmltYXRpb25GcmFtZShlbWl0KVxuICB9XG5cbiAgcmV0dXJuIHtcbiAgICBvblBvaW50ZXJNb3ZlOiBldmVudCA9PiB7XG4gICAgICBjb25zdCB0eXBlID1cbiAgICAgICAgZXZlbnQuYnV0dG9ucyA+IDAgfHwgZXZlbnQudG91Y2hlcz8ubGVuZ3RoID4gMCA/ICdjaGFuZ2UnIDogJ21vdmUnXG4gICAgICBPYmplY3QuYXNzaWduKHN0YXRlLCB7ZXZlbnQsIHR5cGUsIC4uLmdldFBvaW50ZXJEYXRhKGV2ZW50KX0pXG4gICAgICBzY2hlZHVsZSgpXG4gICAgfSxcbiAgICBvblBvaW50ZXJMZWF2ZTogZXZlbnQgPT4ge1xuICAgICAgY29uc3QgdHlwZSA9ICdsZWF2ZSdcbiAgICAgIE9iamVjdC5hc3NpZ24oc3RhdGUsIHtldmVudCwgdHlwZX0pXG4gICAgICBzY2hlZHVsZSgpXG4gICAgfSxcbiAgICBlbWl0LFxuICB9XG59XG5cbmNvbnN0IGV2ZW50SGFuZGxlcnMgPSAoe1xuICBvblBvaW50ZXJEb3duLFxuICBvblBvaW50ZXJNb3ZlLFxuICBvblBvaW50ZXJMZWF2ZSxcbiAgb25Qb2ludGVyVXAsXG59KSA9PiAoe1xuICBvblBvaW50ZXJEb3duLFxuICBvblBvaW50ZXJNb3ZlLFxuICBvblBvaW50ZXJMZWF2ZSxcbiAgb25Qb2ludGVyVXAsXG4gIG9uVG91Y2hTdGFydDogb25Qb2ludGVyRG93bixcbiAgb25Ub3VjaE1vdmU6IG9uUG9pbnRlck1vdmUsXG4gIG9uVG91Y2hFbmQ6IGV2ZW50ID0+IHtcbiAgICBvblBvaW50ZXJMZWF2ZShldmVudClcbiAgICBvblBvaW50ZXJVcChldmVudClcbiAgfSxcbn0pXG5cbmNvbnN0IFNsaWRlclJhaWwgPSAoe29yaWVudGF0aW9uLCBzdHlsZSwgLi4ucmVzdH0pID0+IChcbiAgPGRpdlxuICAgIGNzcz17W3N0eWxlLCBvcmllbnRhdGlvbiAhPT0gJ2hvcml6b250YWwnICYmIHtib3JkZXJSYWRpdXM6ICcwLjJlbScsIGFsaWduU2VsZjogJ25vcm1hbCd9XX1cbiAgICB7Li4ucmVzdH1cbiAgLz5cbilcblxuY29uc3QgU2xpZGVyVHJhY2sgPSAoe3ZhbHVlLCBvcmllbnRhdGlvbiwgc3R5bGUsIC4uLnJlc3R9KSA9PiAoXG4gIDxkaXZcbiAgICBjc3M9e1tcbiAgICAgIHN0eWxlLFxuICAgICAgb3JpZW50YXRpb24gPT09ICd2ZXJ0aWNhbCdcbiAgICAgICAgPyB7Ym90dG9tOiAnMCcsIGhlaWdodDogYCR7dmFsdWUgKiAxMDB9JWB9XG4gICAgICAgIDoge3dpZHRoOiBgJHt2YWx1ZSAqIDEwMH0lYH0sXG4gICAgXX1cbiAgICB7Li4ucmVzdH1cbiAgLz5cbilcblxuY29uc3QgZGVmYXVsdFNsb3RzID0ge1xuICBSYWlsOiBTbGlkZXJSYWlsLFxuICBUcmFjazogU2xpZGVyVHJhY2ssXG59XG5cbi8vIFRPRE8gYWxpZ24gd2l0aCBtYXRlcmlhbCB1aSBtb3JlLCBtb3ZlIHNwZWNpYWwgaGFuZGxpbmcgb2YgcG9pbnRlciBldmVudHNcbmNvbnN0IFNpbXBsZVNsaWRlciA9ICh7XG4gIG1pbiA9IDAsXG4gIG1heCA9IDEwMCxcbiAgdmFsdWUsXG4gIHNlY29uZGFyeVRyYWNrVmFsdWUsIC8vIFRPRE8gYSBiZXR0ZXIgbmFtZVxuICBtYXJrcyA9IFtdLFxuICBjbGFzc05hbWUgPSAnJyxcbiAgY2xhc3NlcyA9IHt9LFxuICBkaXNhYmxlZCxcbiAgb25Qb2ludGVyTW92ZSxcbiAgb25Qb2ludGVyTGVhdmUsXG4gIG9uUG9pbnRlckRvd24sXG4gIG9uQ2hhbmdlLFxuICBvbkNoYW5nZUNvbW1pdHRlZCxcbiAgb3JpZW50YXRpb24gPSAnaG9yaXpvbnRhbCcsXG4gIHNsb3RzID0gZGVmYXVsdFNsb3RzLFxuICBzbG90UHJvcHMgPSB7fSxcbn0pID0+IHtcbiAgY29uc3QgcG9pbnRlclN0YXRlID0gdXNlUmVmKHt9KVxuICBjb25zdCBbZm9jdXNWYWx1ZSwgc2V0Rm9jdXNWYWx1ZV0gPSB1c2VTdGF0ZSgtSW5maW5pdHkpXG4gIGNvbnN0IHRodW1iUG9zaXRpb24gPVxuICAgICgoZm9jdXNWYWx1ZSA+PSBtaW4gPyBmb2N1c1ZhbHVlIDogdmFsdWUpIC0gbWluKSAvIChtYXggLSBtaW4pXG4gIGNvbnN0IHN1YlRyYWNrUG9zaXRpb24gPSAoc2Vjb25kYXJ5VHJhY2tWYWx1ZSAtIG1pbikgLyAobWF4IC0gbWluKVxuICBjb25zdCBwb2ludGVySGFuZGxlcnMgPSBkZWJvdW5jZWRQb2ludGVySGFuZGxlcnMoe1xuICAgIHN0YXRlOiBwb2ludGVyU3RhdGUuY3VycmVudCxcbiAgICBvbk1vdmU6IChldmVudCwge3R5cGUsIHgsIHksIHdpZHRoLCBsZWZ0LCBoZWlnaHQsIGJvdHRvbX0pID0+IHtcbiAgICAgIGNvbnN0IHBvaW50ZXJEYXRhID0ge29yaWVudGF0aW9uLCB4LCB5LCBsZWZ0LCBib3R0b20sIHdpZHRoLCBoZWlnaHR9XG4gICAgICBjb25zdCBwb2ludGVyVmFsdWUgPSBtaW4gKyAobWF4IC0gbWluKSAqIGdldFBvaW50ZXJWYWx1ZShwb2ludGVyRGF0YSlcblxuICAgICAgb25Qb2ludGVyTW92ZT8uKGV2ZW50LCB7dmFsdWU6IHBvaW50ZXJWYWx1ZSwgeCwgeX0pXG4gICAgICBpZiAodHlwZSA9PT0gJ2NoYW5nZScpIHtcbiAgICAgICAgc2V0Rm9jdXNWYWx1ZShwb2ludGVyVmFsdWUpXG4gICAgICAgIG9uQ2hhbmdlPy4oZXZlbnQsIHt2YWx1ZTogcG9pbnRlclZhbHVlLCB4LCB5fSlcbiAgICAgIH1cbiAgICB9LFxuICAgIG9uTGVhdmU6ICgpID0+
|
|
3879
|
+
}, process.env.NODE_ENV === "production" ? "" : ";label:SliderTrack;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SimpleSlider.jsx"],"names":[],"mappings":"AA4HI","file":"SimpleSlider.jsx","sourcesContent":["/* eslint-disable no-param-reassign */\n/* eslint-disable jsx-a11y/no-static-element-interactions */\n/* eslint-disable jsx-a11y/click-events-have-key-events */\n/* @jsxImportSource @emotion/react */\nimport {useState, useRef} from 'react'\n\nimport {getPointerData} from 'util/gestures'\n\nconst containerStyle = {\n  position: 'relative',\n  width: '100%',\n  height: '100%',\n  display: 'flex',\n  alignItems: 'center',\n  cursor: 'pointer',\n  userSelect: 'none',\n  touchAction: 'none',\n}\n\nconst disabledStyle = {\n  pointerEvents: 'none',\n}\n\nconst railStyle = {\n  flex: '100%',\n  background: 'rgba(255, 255, 255, 0.2)',\n}\n\nconst trackStyle = {\n  position: 'absolute',\n  width: '4px',\n  height: '100%',\n  backgroundColor: 'var(--sender-seekbar-background, #fff)',\n}\n\nconst markStyle = {\n  position: 'absolute',\n  height: railStyle.height,\n  width: '4px',\n  transform: 'translateX(-50%)',\n  backgroundColor: '#ff9835',\n}\n\nconst thumbStyle = {\n  position: 'absolute',\n  height: '0.66em',\n  width: '0.66em',\n  borderRadius: '100%',\n  backgroundColor: '#fff',\n  boxShadow: '0 2px 2px 0 rgba(0, 0, 0, 0.5)',\n  transform: 'translateY(calc(var(--slider-thumb-y) - 0.35em))',\n}\n\nconst getPointerValue = ({orientation, x, y, left, bottom, height, width}) =>\n  Math.max(\n    0,\n    orientation === 'vertical'\n      ? Math.min((bottom - y) / height, 1)\n      : Math.min((x - left) / width, 1),\n  )\n\nconst debouncedPointerHandlers = ({state, onMove, onLeave}) => {\n  const emit = () => {\n    if (!state.scheduled) {\n      return\n    }\n    if (state.type === 'leave') {\n      onLeave?.(state.event, state)\n    } else {\n      onMove(state.event, state)\n    }\n    state.scheduled = false\n  }\n  const schedule = () => {\n    if (state.scheduled) {\n      return\n    }\n    state.scheduled = true\n    requestAnimationFrame(emit)\n  }\n\n  return {\n    onPointerMove: event => {\n      const type =\n        event.buttons > 0 || event.touches?.length > 0 ? 'change' : 'move'\n      Object.assign(state, {event, type, ...getPointerData(event)})\n      schedule()\n    },\n    onPointerLeave: event => {\n      const type = 'leave'\n      Object.assign(state, {event, type})\n      schedule()\n    },\n    emit,\n  }\n}\n\nconst eventHandlers = ({\n  onPointerDown,\n  onPointerMove,\n  onPointerLeave,\n  onPointerUp,\n}) => ({\n  onPointerDown,\n  onPointerMove,\n  onPointerLeave,\n  onPointerUp,\n  onTouchStart: onPointerDown,\n  onTouchMove: onPointerMove,\n  onTouchEnd: event => {\n    onPointerLeave(event)\n    onPointerUp(event)\n  },\n})\n\nconst SliderRail = ({orientation, style, ...rest}) => (\n  <div\n    css={[style, orientation !== 'horizontal' && {borderRadius: '0.2em', alignSelf: 'normal'}]}\n    {...rest}\n  />\n)\n\nconst SliderTrack = ({value, orientation, style, ...rest}) => (\n  <div\n    css={[\n      style,\n      orientation === 'vertical'\n        ? {bottom: '0', height: `${value * 100}%`}\n        : {width: `${value * 100}%`},\n    ]}\n    {...rest}\n  />\n)\n\nconst defaultSlots = {\n  Rail: SliderRail,\n  Track: SliderTrack,\n}\n\n// TODO align with material ui more, move special handling of pointer events\nconst SimpleSlider = ({\n  min = 0,\n  max = 100,\n  value,\n  secondaryTrackValue, // TODO a better name\n  marks = [],\n  className = '',\n  classes = {},\n  disabled,\n  onPointerMove,\n  onPointerLeave,\n  onPointerDown,\n  onChange,\n  onChangeCommitted,\n  orientation = 'horizontal',\n  slots = defaultSlots,\n  slotProps = {},\n}) => {\n  const pointerState = useRef({})\n  const [focusValue, setFocusValue] = useState(-Infinity)\n  const thumbPosition =\n    ((focusValue >= min ? focusValue : value) - min) / (max - min)\n  const subTrackPosition = (secondaryTrackValue - min) / (max - min)\n  const pointerHandlers = debouncedPointerHandlers({\n    state: pointerState.current,\n    onMove: (event, {type, x, y, width, left, height, bottom}) => {\n      const pointerData = {orientation, x, y, left, bottom, width, height}\n      const pointerValue = min + (max - min) * getPointerValue(pointerData)\n\n      onPointerMove?.(event, {value: pointerValue, x, y})\n      if (type === 'change') {\n        setFocusValue(pointerValue)\n        onChange?.(event, {value: pointerValue, x, y})\n      }\n    },\n    onLeave: () => onPointerLeave?.(),\n  })\n  const handlePointerUp = event => {\n    if (event.pointerId) {\n      event.currentTarget.releasePointerCapture(event.pointerId)\n    }\n    const {x, y, width, left, height, bottom} = getPointerData(event)\n    const pointerValue =\n      min +\n      (max - min) *\n        getPointerValue({orientation, x, y, left, bottom, width, height})\n\n    pointerHandlers.emit()\n    onChangeCommitted?.(event, {value: pointerValue})\n    setFocusValue()\n  }\n\n  const thumbPositionStyle = {\n    bottom: `calc(var(--slider-thumb-y) - 0.35em)`,\n    left: `calc(var(--slider-thumb-x) - 0.25em)`,\n  }\n\n  return (\n    <div\n      className={className}\n      css={[containerStyle, disabled && disabledStyle]}\n      style={{\n        ...(orientation === 'vertical' && {\n          flexDirection: 'column',\n          '--slider-thumb-y': `${thumbPosition * 100}%`,\n        }),\n        ...(orientation === 'horizontal' && {\n          '--slider-thumb-x': `${thumbPosition * 100}%`,\n        }),\n      }}\n      {...eventHandlers({\n        onPointerDown: event => {\n          if (event.type === 'pointerdown') {\n            event.currentTarget.setPointerCapture(event.pointerId)\n            onPointerDown?.()\n          }\n          pointerHandlers.onPointerMove(event)\n        },\n        ...pointerHandlers,\n        onPointerUp: handlePointerUp,\n      })}\n    >\n      <slots.Rail style={railStyle} className={classes.rail} {...slotProps.rail} />\n      {secondaryTrackValue && (\n        <slots.Track\n          style={{...trackStyle, backgroundColor: 'rgba(255, 255, 255, 0.3)'}}\n          orientation={orientation}\n          value={subTrackPosition}\n          {...slotProps.track}\n        />\n      )}\n      <slots.Track\n        className={classes.track}\n        style={trackStyle}\n        orientation={orientation}\n        value={thumbPosition}\n        {...slotProps.track}\n      />\n      {marks.map(position => (\n        <div\n          key={position}\n          css={markStyle}\n          className={classes.marked}\n          style={{left: `${(position / max) * 100}%`}}\n        />\n      ))}\n      {onChange && !disabled ? (\n        <div\n          css={thumbStyle}\n          className={classes.thumb}\n          style={thumbPositionStyle}\n        />\n      ) : (\n        <div />\n      )}\n    </div>\n  )\n}\n\nexport default SimpleSlider\n"]} */"],
|
|
3883
3880
|
...rest
|
|
3884
3881
|
});
|
|
3885
3882
|
|
|
@@ -3985,7 +3982,7 @@ const SimpleSlider = ({
|
|
|
3985
3982
|
};
|
|
3986
3983
|
return jsxs("div", {
|
|
3987
3984
|
className: className,
|
|
3988
|
-
css: [containerStyle$4, disabled && disabledStyle, process.env.NODE_ENV === "production" ? "" : ";label:SimpleSlider;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIlNpbXBsZVNsaWRlci5qc3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBd01NIiwiZmlsZSI6IlNpbXBsZVNsaWRlci5qc3giLCJzb3VyY2VzQ29udGVudCI6WyIvKiBlc2xpbnQtZGlzYWJsZSBuby1wYXJhbS1yZWFzc2lnbiAqL1xuLyogZXNsaW50LWRpc2FibGUganN4LWExMXkvbm8tc3RhdGljLWVsZW1lbnQtaW50ZXJhY3Rpb25zICovXG4vKiBlc2xpbnQtZGlzYWJsZSBqc3gtYTExeS9jbGljay1ldmVudHMtaGF2ZS1rZXktZXZlbnRzICovXG4vKiBAanN4SW1wb3J0U291cmNlIEBlbW90aW9uL3JlYWN0ICovXG5pbXBvcnQge3VzZVN0YXRlLCB1c2VSZWZ9IGZyb20gJ3JlYWN0J1xuXG5pbXBvcnQge2dldFBvaW50ZXJEYXRhfSBmcm9tICd1dGlsL2dlc3R1cmVzJ1xuXG5jb25zdCBjb250YWluZXJTdHlsZSA9IHtcbiAgcG9zaXRpb246ICdyZWxhdGl2ZScsXG4gIHdpZHRoOiAnMTAwJScsXG4gIGhlaWdodDogJzEwMCUnLFxuICBkaXNwbGF5OiAnZmxleCcsXG4gIGFsaWduSXRlbXM6ICdjZW50ZXInLFxuICBjdXJzb3I6ICdwb2ludGVyJyxcbiAgdXNlclNlbGVjdDogJ25vbmUnLFxuICB0b3VjaEFjdGlvbjogJ25vbmUnLFxufVxuXG5jb25zdCBkaXNhYmxlZFN0eWxlID0ge1xuICBwb2ludGVyRXZlbnRzOiAnbm9uZScsXG59XG5cbmNvbnN0IHJhaWxTdHlsZSA9IHtcbiAgZmxleDogJzEwMCUnLFxuICBiYWNrZ3JvdW5kOiAncmdiYSgyNTUsIDI1NSwgMjU1LCAwLjIpJyxcbn1cblxuY29uc3QgdHJhY2tTdHlsZSA9IHtcbiAgcG9zaXRpb246ICdhYnNvbHV0ZScsXG4gIHdpZHRoOiAnNHB4JyxcbiAgaGVpZ2h0OiAnMTAwJScsXG4gIGJhY2tncm91bmRDb2xvcjogJ3ZhcigtLXNlbmRlci1zZWVrYmFyLWJhY2tncm91bmQsICNmZmYpJyxcbn1cblxuY29uc3QgbWFya1N0eWxlID0ge1xuICBwb3NpdGlvbjogJ2Fic29sdXRlJyxcbiAgaGVpZ2h0OiByYWlsU3R5bGUuaGVpZ2h0LFxuICB3aWR0aDogJzRweCcsXG4gIHRyYW5zZm9ybTogJ3RyYW5zbGF0ZVgoLTUwJSknLFxuICBiYWNrZ3JvdW5kQ29sb3I6ICcjZmY5ODM1Jyxcbn1cblxuY29uc3QgdGh1bWJTdHlsZSA9IHtcbiAgcG9zaXRpb246ICdhYnNvbHV0ZScsXG4gIGhlaWdodDogJzAuNjZlbScsXG4gIHdpZHRoOiAnMC42NmVtJyxcbiAgYm9yZGVyUmFkaXVzOiAnMTAwJScsXG4gIGJhY2tncm91bmRDb2xvcjogJyNmZmYnLFxuICBib3hTaGFkb3c6ICcwIDJweCAycHggMCByZ2JhKDAsIDAsIDAsIDAuNSknLFxuICB0cmFuc2Zvcm06ICd0cmFuc2xhdGVZKGNhbGModmFyKC0tc2xpZGVyLXRodW1iLXkpIC0gMC4zNWVtKSknLFxufVxuXG5jb25zdCBnZXRQb2ludGVyVmFsdWUgPSAoe29yaWVudGF0aW9uLCB4LCB5LCBsZWZ0LCBib3R0b20sIGhlaWdodCwgd2lkdGh9KSA9PlxuICBNYXRoLm1heChcbiAgICAwLFxuICAgIG9yaWVudGF0aW9uID09PSAndmVydGljYWwnXG4gICAgICA/IE1hdGgubWluKChib3R0b20gLSB5KSAvIGhlaWdodCwgMSlcbiAgICAgIDogTWF0aC5taW4oKHggLSBsZWZ0KSAvIHdpZHRoLCAxKSxcbiAgKVxuXG5jb25zdCBkZWJvdW5jZWRQb2ludGVySGFuZGxlcnMgPSAoe3N0YXRlLCBvbk1vdmUsIG9uTGVhdmV9KSA9PiB7XG4gIGNvbnN0IGVtaXQgPSAoKSA9PiB7XG4gICAgaWYgKCFzdGF0ZS5zY2hlZHVsZWQpIHtcbiAgICAgIHJldHVyblxuICAgIH1cbiAgICBpZiAoc3RhdGUudHlwZSA9PT0gJ2xlYXZlJykge1xuICAgICAgb25MZWF2ZT8uKHN0YXRlLmV2ZW50LCBzdGF0ZSlcbiAgICB9IGVsc2Uge1xuICAgICAgb25Nb3ZlKHN0YXRlLmV2ZW50LCBzdGF0ZSlcbiAgICB9XG4gICAgc3RhdGUuc2NoZWR1bGVkID0gZmFsc2VcbiAgfVxuICBjb25zdCBzY2hlZHVsZSA9ICgpID0+IHtcbiAgICBpZiAoc3RhdGUuc2NoZWR1bGVkKSB7XG4gICAgICByZXR1cm5cbiAgICB9XG4gICAgc3RhdGUuc2NoZWR1bGVkID0gdHJ1ZVxuICAgIHJlcXVlc3RBbmltYXRpb25GcmFtZShlbWl0KVxuICB9XG5cbiAgcmV0dXJuIHtcbiAgICBvblBvaW50ZXJNb3ZlOiBldmVudCA9PiB7XG4gICAgICBjb25zdCB0eXBlID1cbiAgICAgICAgZXZlbnQuYnV0dG9ucyA+IDAgfHwgZXZlbnQudG91Y2hlcz8ubGVuZ3RoID4gMCA/ICdjaGFuZ2UnIDogJ21vdmUnXG4gICAgICBPYmplY3QuYXNzaWduKHN0YXRlLCB7ZXZlbnQsIHR5cGUsIC4uLmdldFBvaW50ZXJEYXRhKGV2ZW50KX0pXG4gICAgICBzY2hlZHVsZSgpXG4gICAgfSxcbiAgICBvblBvaW50ZXJMZWF2ZTogZXZlbnQgPT4ge1xuICAgICAgY29uc3QgdHlwZSA9ICdsZWF2ZSdcbiAgICAgIE9iamVjdC5hc3NpZ24oc3RhdGUsIHtldmVudCwgdHlwZX0pXG4gICAgICBzY2hlZHVsZSgpXG4gICAgfSxcbiAgICBlbWl0LFxuICB9XG59XG5cbmNvbnN0IGV2ZW50SGFuZGxlcnMgPSAoe1xuICBvblBvaW50ZXJEb3duLFxuICBvblBvaW50ZXJNb3ZlLFxuICBvblBvaW50ZXJMZWF2ZSxcbiAgb25Qb2ludGVyVXAsXG59KSA9PiAoe1xuICBvblBvaW50ZXJEb3duLFxuICBvblBvaW50ZXJNb3ZlLFxuICBvblBvaW50ZXJMZWF2ZSxcbiAgb25Qb2ludGVyVXAsXG4gIG9uVG91Y2hTdGFydDogb25Qb2ludGVyRG93bixcbiAgb25Ub3VjaE1vdmU6IG9uUG9pbnRlck1vdmUsXG4gIG9uVG91Y2hFbmQ6IGV2ZW50ID0+IHtcbiAgICBvblBvaW50ZXJMZWF2ZShldmVudClcbiAgICBvblBvaW50ZXJVcChldmVudClcbiAgfSxcbn0pXG5cbmNvbnN0IFNsaWRlclJhaWwgPSAoe29yaWVudGF0aW9uLCBzdHlsZSwgLi4ucmVzdH0pID0+IChcbiAgPGRpdlxuICAgIGNzcz17W3N0eWxlLCBvcmllbnRhdGlvbiAhPT0gJ2hvcml6b250YWwnICYmIHtib3JkZXJSYWRpdXM6ICcwLjJlbScsIGFsaWduU2VsZjogJ25vcm1hbCd9XX1cbiAgICB7Li4ucmVzdH1cbiAgLz5cbilcblxuY29uc3QgU2xpZGVyVHJhY2sgPSAoe3ZhbHVlLCBvcmllbnRhdGlvbiwgc3R5bGUsIC4uLnJlc3R9KSA9PiAoXG4gIDxkaXZcbiAgICBjc3M9e1tcbiAgICAgIHN0eWxlLFxuICAgICAgb3JpZW50YXRpb24gPT09ICd2ZXJ0aWNhbCdcbiAgICAgICAgPyB7Ym90dG9tOiAnMCcsIGhlaWdodDogYCR7dmFsdWUgKiAxMDB9JWB9XG4gICAgICAgIDoge3dpZHRoOiBgJHt2YWx1ZSAqIDEwMH0lYH0sXG4gICAgXX1cbiAgICB7Li4ucmVzdH1cbiAgLz5cbilcblxuY29uc3QgZGVmYXVsdFNsb3RzID0ge1xuICBSYWlsOiBTbGlkZXJSYWlsLFxuICBUcmFjazogU2xpZGVyVHJhY2ssXG59XG5cbi8vIFRPRE8gYWxpZ24gd2l0aCBtYXRlcmlhbCB1aSBtb3JlLCBtb3ZlIHNwZWNpYWwgaGFuZGxpbmcgb2YgcG9pbnRlciBldmVudHNcbmNvbnN0IFNpbXBsZVNsaWRlciA9ICh7XG4gIG1pbiA9IDAsXG4gIG1heCA9IDEwMCxcbiAgdmFsdWUsXG4gIHNlY29uZGFyeVRyYWNrVmFsdWUsIC8vIFRPRE8gYSBiZXR0ZXIgbmFtZVxuICBtYXJrcyA9IFtdLFxuICBjbGFzc05hbWUgPSAnJyxcbiAgY2xhc3NlcyA9IHt9LFxuICBkaXNhYmxlZCxcbiAgb25Qb2ludGVyTW92ZSxcbiAgb25Qb2ludGVyTGVhdmUsXG4gIG9uUG9pbnRlckRvd24sXG4gIG9uQ2hhbmdlLFxuICBvbkNoYW5nZUNvbW1pdHRlZCxcbiAgb3JpZW50YXRpb24gPSAnaG9yaXpvbnRhbCcsXG4gIHNsb3RzID0gZGVmYXVsdFNsb3RzLFxuICBzbG90UHJvcHMgPSB7fSxcbn0pID0+IHtcbiAgY29uc3QgcG9pbnRlclN0YXRlID0gdXNlUmVmKHt9KVxuICBjb25zdCBbZm9jdXNWYWx1ZSwgc2V0Rm9jdXNWYWx1ZV0gPSB1c2VTdGF0ZSgtSW5maW5pdHkpXG4gIGNvbnN0IHRodW1iUG9zaXRpb24gPVxuICAgICgoZm9jdXNWYWx1ZSA+PSBtaW4gPyBmb2N1c1ZhbHVlIDogdmFsdWUpIC0gbWluKSAvIChtYXggLSBtaW4pXG4gIGNvbnN0IHN1YlRyYWNrUG9zaXRpb24gPSAoc2Vjb25kYXJ5VHJhY2tWYWx1ZSAtIG1pbikgLyAobWF4IC0gbWluKVxuICBjb25zdCBwb2ludGVySGFuZGxlcnMgPSBkZWJvdW5jZWRQb2ludGVySGFuZGxlcnMoe1xuICAgIHN0YXRlOiBwb2ludGVyU3RhdGUuY3VycmVudCxcbiAgICBvbk1vdmU6IChldmVudCwge3R5cGUsIHgsIHksIHdpZHRoLCBsZWZ0LCBoZWlnaHQsIGJvdHRvbX0pID0+IHtcbiAgICAgIGNvbnN0IHBvaW50ZXJEYXRhID0ge29yaWVudGF0aW9uLCB4LCB5LCBsZWZ0LCBib3R0b20sIHdpZHRoLCBoZWlnaHR9XG4gICAgICBjb25zdCBwb2ludGVyVmFsdWUgPSBtaW4gKyAobWF4IC0gbWluKSAqIGdldFBvaW50ZXJWYWx1ZShwb2ludGVyRGF0YSlcblxuICAgICAgb25Qb2ludGVyTW92ZT8uKGV2ZW50LCB7dmFsdWU6IHBvaW50ZXJWYWx1ZSwgeCwgeX0pXG4gICAgICBpZiAodHlwZSA9PT0gJ2NoYW5nZScpIHtcbiAgICAgICAgc2V0Rm9jdXNWYWx1ZShwb2ludGVyVmFsdWUpXG4gICAgICAgIG9uQ2hhbmdlPy4oZXZlbnQsIHt2YWx1ZTogcG9pbnRlclZhbHVlLCB4LCB5fSlcbiAgICAgIH1cbiAgICB9LFxuICAgIG9uTGVhdmU6ICgpID0+
|
|
3985
|
+
css: [containerStyle$4, disabled && disabledStyle, process.env.NODE_ENV === "production" ? "" : ";label:SimpleSlider;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SimpleSlider.jsx"],"names":[],"mappings":"AAwMM","file":"SimpleSlider.jsx","sourcesContent":["/* eslint-disable no-param-reassign */\n/* eslint-disable jsx-a11y/no-static-element-interactions */\n/* eslint-disable jsx-a11y/click-events-have-key-events */\n/* @jsxImportSource @emotion/react */\nimport {useState, useRef} from 'react'\n\nimport {getPointerData} from 'util/gestures'\n\nconst containerStyle = {\n  position: 'relative',\n  width: '100%',\n  height: '100%',\n  display: 'flex',\n  alignItems: 'center',\n  cursor: 'pointer',\n  userSelect: 'none',\n  touchAction: 'none',\n}\n\nconst disabledStyle = {\n  pointerEvents: 'none',\n}\n\nconst railStyle = {\n  flex: '100%',\n  background: 'rgba(255, 255, 255, 0.2)',\n}\n\nconst trackStyle = {\n  position: 'absolute',\n  width: '4px',\n  height: '100%',\n  backgroundColor: 'var(--sender-seekbar-background, #fff)',\n}\n\nconst markStyle = {\n  position: 'absolute',\n  height: railStyle.height,\n  width: '4px',\n  transform: 'translateX(-50%)',\n  backgroundColor: '#ff9835',\n}\n\nconst thumbStyle = {\n  position: 'absolute',\n  height: '0.66em',\n  width: '0.66em',\n  borderRadius: '100%',\n  backgroundColor: '#fff',\n  boxShadow: '0 2px 2px 0 rgba(0, 0, 0, 0.5)',\n  transform: 'translateY(calc(var(--slider-thumb-y) - 0.35em))',\n}\n\nconst getPointerValue = ({orientation, x, y, left, bottom, height, width}) =>\n  Math.max(\n    0,\n    orientation === 'vertical'\n      ? Math.min((bottom - y) / height, 1)\n      : Math.min((x - left) / width, 1),\n  )\n\nconst debouncedPointerHandlers = ({state, onMove, onLeave}) => {\n  const emit = () => {\n    if (!state.scheduled) {\n      return\n    }\n    if (state.type === 'leave') {\n      onLeave?.(state.event, state)\n    } else {\n      onMove(state.event, state)\n    }\n    state.scheduled = false\n  }\n  const schedule = () => {\n    if (state.scheduled) {\n      return\n    }\n    state.scheduled = true\n    requestAnimationFrame(emit)\n  }\n\n  return {\n    onPointerMove: event => {\n      const type =\n        event.buttons > 0 || event.touches?.length > 0 ? 'change' : 'move'\n      Object.assign(state, {event, type, ...getPointerData(event)})\n      schedule()\n    },\n    onPointerLeave: event => {\n      const type = 'leave'\n      Object.assign(state, {event, type})\n      schedule()\n    },\n    emit,\n  }\n}\n\nconst eventHandlers = ({\n  onPointerDown,\n  onPointerMove,\n  onPointerLeave,\n  onPointerUp,\n}) => ({\n  onPointerDown,\n  onPointerMove,\n  onPointerLeave,\n  onPointerUp,\n  onTouchStart: onPointerDown,\n  onTouchMove: onPointerMove,\n  onTouchEnd: event => {\n    onPointerLeave(event)\n    onPointerUp(event)\n  },\n})\n\nconst SliderRail = ({orientation, style, ...rest}) => (\n  <div\n    css={[style, orientation !== 'horizontal' && {borderRadius: '0.2em', alignSelf: 'normal'}]}\n    {...rest}\n  />\n)\n\nconst SliderTrack = ({value, orientation, style, ...rest}) => (\n  <div\n    css={[\n      style,\n      orientation === 'vertical'\n        ? {bottom: '0', height: `${value * 100}%`}\n        : {width: `${value * 100}%`},\n    ]}\n    {...rest}\n  />\n)\n\nconst defaultSlots = {\n  Rail: SliderRail,\n  Track: SliderTrack,\n}\n\n// TODO align with material ui more, move special handling of pointer events\nconst SimpleSlider = ({\n  min = 0,\n  max = 100,\n  value,\n  secondaryTrackValue, // TODO a better name\n  marks = [],\n  className = '',\n  classes = {},\n  disabled,\n  onPointerMove,\n  onPointerLeave,\n  onPointerDown,\n  onChange,\n  onChangeCommitted,\n  orientation = 'horizontal',\n  slots = defaultSlots,\n  slotProps = {},\n}) => {\n  const pointerState = useRef({})\n  const [focusValue, setFocusValue] = useState(-Infinity)\n  const thumbPosition =\n    ((focusValue >= min ? focusValue : value) - min) / (max - min)\n  const subTrackPosition = (secondaryTrackValue - min) / (max - min)\n  const pointerHandlers = debouncedPointerHandlers({\n    state: pointerState.current,\n    onMove: (event, {type, x, y, width, left, height, bottom}) => {\n      const pointerData = {orientation, x, y, left, bottom, width, height}\n      const pointerValue = min + (max - min) * getPointerValue(pointerData)\n\n      onPointerMove?.(event, {value: pointerValue, x, y})\n      if (type === 'change') {\n        setFocusValue(pointerValue)\n        onChange?.(event, {value: pointerValue, x, y})\n      }\n    },\n    onLeave: () => onPointerLeave?.(),\n  })\n  const handlePointerUp = event => {\n    if (event.pointerId) {\n      event.currentTarget.releasePointerCapture(event.pointerId)\n    }\n    const {x, y, width, left, height, bottom} = getPointerData(event)\n    const pointerValue =\n      min +\n      (max - min) *\n        getPointerValue({orientation, x, y, left, bottom, width, height})\n\n    pointerHandlers.emit()\n    onChangeCommitted?.(event, {value: pointerValue})\n    setFocusValue()\n  }\n\n  const thumbPositionStyle = {\n    bottom: `calc(var(--slider-thumb-y) - 0.35em)`,\n    left: `calc(var(--slider-thumb-x) - 0.25em)`,\n  }\n\n  return (\n    <div\n      className={className}\n      css={[containerStyle, disabled && disabledStyle]}\n      style={{\n        ...(orientation === 'vertical' && {\n          flexDirection: 'column',\n          '--slider-thumb-y': `${thumbPosition * 100}%`,\n        }),\n        ...(orientation === 'horizontal' && {\n          '--slider-thumb-x': `${thumbPosition * 100}%`,\n        }),\n      }}\n      {...eventHandlers({\n        onPointerDown: event => {\n          if (event.type === 'pointerdown') {\n            event.currentTarget.setPointerCapture(event.pointerId)\n            onPointerDown?.()\n          }\n          pointerHandlers.onPointerMove(event)\n        },\n        ...pointerHandlers,\n        onPointerUp: handlePointerUp,\n      })}\n    >\n      <slots.Rail style={railStyle} className={classes.rail} {...slotProps.rail} />\n      {secondaryTrackValue && (\n        <slots.Track\n          style={{...trackStyle, backgroundColor: 'rgba(255, 255, 255, 0.3)'}}\n          orientation={orientation}\n          value={subTrackPosition}\n          {...slotProps.track}\n        />\n      )}\n      <slots.Track\n        className={classes.track}\n        style={trackStyle}\n        orientation={orientation}\n        value={thumbPosition}\n        {...slotProps.track}\n      />\n      {marks.map(position => (\n        <div\n          key={position}\n          css={markStyle}\n          className={classes.marked}\n          style={{left: `${(position / max) * 100}%`}}\n        />\n      ))}\n      {onChange && !disabled ? (\n        <div\n          css={thumbStyle}\n          className={classes.thumb}\n          style={thumbPositionStyle}\n        />\n      ) : (\n        <div />\n      )}\n    </div>\n  )\n}\n\nexport default SimpleSlider\n"]} */"],
|
|
3989
3986
|
style: { ...(orientation === 'vertical' && {
|
|
3990
3987
|
flexDirection: 'column',
|
|
3991
3988
|
'--slider-thumb-y': `${thumbPosition * 100}%`
|
|
@@ -3994,7 +3991,6 @@ const SimpleSlider = ({
|
|
|
3994
3991
|
'--slider-thumb-x': `${thumbPosition * 100}%`
|
|
3995
3992
|
})
|
|
3996
3993
|
},
|
|
3997
|
-
onClick: event => event.stopPropagation(),
|
|
3998
3994
|
...eventHandlers({
|
|
3999
3995
|
onPointerDown: event => {
|
|
4000
3996
|
if (event.type === 'pointerdown') {
|
|
@@ -4875,12 +4871,12 @@ const containerStyle$3 = {
|
|
|
4875
4871
|
height: '100%',
|
|
4876
4872
|
transform: 'translateY(-150vh)',
|
|
4877
4873
|
opacity: '0',
|
|
4878
|
-
transition: 'opacity 0.1s ease, transform 0s 0.1s
|
|
4874
|
+
transition: 'opacity 0.1s ease, transform 0s linear 0.1s'
|
|
4879
4875
|
};
|
|
4880
4876
|
const openStyle$1 = {
|
|
4881
4877
|
transform: 'translateY(0)',
|
|
4882
4878
|
opacity: '1',
|
|
4883
|
-
transition: 'opacity 0.2s ease'
|
|
4879
|
+
transition: 'opacity 0.2s ease, transform 0s'
|
|
4884
4880
|
};
|
|
4885
4881
|
|
|
4886
4882
|
const DefaultContainer = ({
|
|
@@ -4888,7 +4884,7 @@ const DefaultContainer = ({
|
|
|
4888
4884
|
containerRef,
|
|
4889
4885
|
children
|
|
4890
4886
|
}) => jsx$1("div", {
|
|
4891
|
-
css: [containerStyle$3, open && openStyle$1, process.env.NODE_ENV === "production" ? "" : ";label:DefaultContainer;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
4887
|
+
css: [containerStyle$3, open && openStyle$1, process.env.NODE_ENV === "production" ? "" : ";label:DefaultContainer;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIk92ZXJsYXlQYW5lbC5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUE0QkkiLCJmaWxlIjoiT3ZlcmxheVBhbmVsLmpzIiwic291cmNlc0NvbnRlbnQiOlsiLyogZXNsaW50LWRpc2FibGUganN4LWExMXkvbm8tc3RhdGljLWVsZW1lbnQtaW50ZXJhY3Rpb25zICovXG4vKiBAanN4SW1wb3J0U291cmNlIEBlbW90aW9uL3JlYWN0ICovXG5pbXBvcnQge3VzZUVmZmVjdCwgdXNlUmVmLCB1c2VTdGF0ZX0gZnJvbSAncmVhY3QnXG5pbXBvcnQgdXNlT25jbGlja091dHNpZGUgZnJvbSAncmVhY3QtY29vbC1vbmNsaWNrb3V0c2lkZSdcbmltcG9ydCB7b259IGZyb20gJ3V0aWwvZXZlbnRzJ1xuaW1wb3J0IHtpc0Rlc2t0b3B9IGZyb20gJ3V0aWwvZW52aXJvbm1lbnQnXG5pbXBvcnQge0J1dHRvbn0gZnJvbSAnLi9idXR0b25zJ1xuaW1wb3J0IHtGdW5jdGlvbkJhckV4dGVuc2lvbiwgVGl0bGVCYXJFeHRlbnNpb259IGZyb20gJy4vdWlFeHRlbnNpb25zJ1xuXG5jb25zdCBjb250YWluZXJTdHlsZSA9IHtcbiAgcG9zaXRpb246ICdhYnNvbHV0ZScsXG4gIHpJbmRleDogJzInLFxuICB0b3A6ICcwJyxcbiAgd2lkdGg6ICcxMDAlJyxcbiAgaGVpZ2h0OiAnMTAwJScsXG4gIHRyYW5zZm9ybTogJ3RyYW5zbGF0ZVkoLTE1MHZoKScsXG4gIG9wYWNpdHk6ICcwJyxcbiAgdHJhbnNpdGlvbjogJ29wYWNpdHkgMC4xcyBlYXNlLCB0cmFuc2Zvcm0gMHMgbGluZWFyIDAuMXMnLFxufVxuXG5jb25zdCBvcGVuU3R5bGUgPSB7XG4gIHRyYW5zZm9ybTogJ3RyYW5zbGF0ZVkoMCknLFxuICBvcGFjaXR5OiAnMScsXG4gIHRyYW5zaXRpb246ICdvcGFjaXR5IDAuMnMgZWFzZSwgdHJhbnNmb3JtIDBzJyxcbn1cblxuY29uc3QgRGVmYXVsdENvbnRhaW5lciA9ICh7b3BlbiwgY29udGFpbmVyUmVmLCBjaGlsZHJlbn0pID0+IChcbiAgPGRpdlxuICAgIGNzcz17W2NvbnRhaW5lclN0eWxlLCBvcGVuICYmIG9wZW5TdHlsZV19XG4gICAgcmVmPXtjb250YWluZXJSZWZ9XG4gICAgb25DbGljaz17ZXZlbnQgPT4gZXZlbnQuc3RvcFByb3BhZ2F0aW9uKCl9XG4gID5cbiAgICB7Y2hpbGRyZW59XG4gIDwvZGl2PlxuKVxuXG5jb25zdCBkZXNrdG9wQ29udGFpbmVyU3R5bGUgPSB7XG4gIHBvc2l0aW9uOiAnYWJzb2x1dGUnLFxuICB6SW5kZXg6ICcyJyxcbiAgYm90dG9tOiAnY2FsYygzZW0gKyB2YXIoLS1ib3R0b20tc3BhY2luZywgMHJlbSkpJyxcbiAgcmlnaHQ6ICdjYWxjKHZhcigtLXNwYWNpbmcsIDAuNzVlbSkgKyAwLjVlbSArIDAuNXJlbSknLFxuICBiYWNrZ3JvdW5kOiAndHJhbnNwYXJlbnQnLFxuICBvdXRsaW5lOiAnbm9uZScsXG4gIG9wYWNpdHk6ICcwJyxcbiAgdHJhbnNmb3JtOiAndHJhbnNsYXRlWSgtMTAwdmgpJyxcbiAgdHJhbnNpdGlvbjogJ29wYWNpdHkgMC4xcyBlYXNlLCB0cmFuc2Zvcm0gMHMgbGluZWFyIDAuMXMnLFxufVxuXG5jb25zdCBEZXNrdG9wT3BlblN0eWxlID0ge1xuICBvcGFjaXR5OiAnMScsXG4gIHRyYW5zZm9ybTogJ3RyYW5zbGF0ZVkoMCknLFxuICB0cmFuc2l0aW9uOiAnb3BhY2l0eSAwLjJzIGVhc2UsIHRyYW5zZm9ybSAwcycsXG59XG5cbmNvbnN0IERlc2t0b3BDb250YWluZXIgPSAoe29wZW4sIGNvbnRhaW5lclJlZiwgY2hpbGRyZW4sIG9uQ2xvc2UsIC4uLnJlc3R9KSA9PiAoXG4gIDxkaXZcbiAgICBjc3M9e1tkZXNrdG9wQ29udGFpbmVyU3R5bGUsIG9wZW4gJiYgRGVza3RvcE9wZW5TdHlsZV19XG4gICAgcmVmPXtjb250YWluZXJSZWZ9XG4gICAgb25DbGljaz17ZXZlbnQgPT4gZXZlbnQuc3RvcFByb3BhZ2F0aW9uKCl9XG4gICAgey4uLnJlc3R9XG4gID5cbiAgICB7Y2hpbGRyZW59XG4gIDwvZGl2PlxuKVxuXG5jb25zdCBPdmVybGF5UGFuZWwgPSAoe1xuICBidXR0b25JY29uLFxuICB0aXRsZSxcbiAgdWlUeXBlID0gaXNEZXNrdG9wKCkgPyAnZGVza3RvcCcgOiAnbW9iaWxlJyxcbiAgY2hpbGRyZW4sXG4gIHNsb3RzID0ge1xuICAgIHJvb3Q6IHVpVHlwZSA9PT0gJ2Rlc2t0b3AnID8gRGVza3RvcENvbnRhaW5lciA6IERlZmF1bHRDb250YWluZXIsXG4gIH0sXG4gIC4uLnJlc3Rcbn0pID0+IHtcbiAgY29uc3QgY29tcG9uZW50cyA9IHtcbiAgICByb290OiBzbG90cy5yb290LFxuICB9XG4gIGNvbnN0IEJ1dHRvbldyYXAgPVxuICAgIHVpVHlwZSA9PT0gJ2Rlc2t0b3AnID8gRnVuY3Rpb25CYXJFeHRlbnNpb24gOiBUaXRsZUJhckV4dGVuc2lvblxuICBjb25zdCBbb3Blbiwgc2V0T3Blbl0gPSB1c2VTdGF0ZShmYWxzZSlcbiAgY29uc3Qgb25PcGVuID0gZXZlbnQgPT4ge1xuICAgIHNldFRpbWVvdXQoKCkgPT4gc2V0T3Blbih0cnVlKSwgMSlcbiAgICBjb25zdCBjb250YWluZXIgPSBldmVudC5jdXJyZW50VGFyZ2V0XG4gICAgY29uc3QgY2xvc2VNZW51ID0gKCkgPT4gc2V0T3BlbihmYWxzZSlcbiAgICBjb250YWluZXIuZGlzcGF0Y2hFdmVudChcbiAgICAgIG5ldyBDdXN0b21FdmVudCgnZm9jdXMtbWVudScsIHtcbiAgICAgICAgZGV0YWlsOiB7c3RhdHVzOiAnb3BlbicsIGNsb3NlTWVudX0sXG4gICAgICAgIGJ1YmJsZXM6IHRydWUsXG4gICAgICB9KVxuICAgIClcbiAgfVxuICBjb25zdCBjb250YWluZXJSZWYgPSB1c2VSZWYoKVxuICBjb25zdCBvbkNsb3NlID0gKCkgPT4ge1xuICAgIHNldE9wZW4oZmFsc2UpXG4gICAgY29udGFpbmVyUmVmLmN1cnJlbnQuZGlzcGF0Y2hFdmVudChcbiAgICAgIG5ldyBDdXN0b21FdmVudCgnZm9jdXMtbWVudScsIHtkZXRhaWw6IHtzdGF0dXM6ICdjbG9zZWQnfSwgYnViYmxlczogdHJ1ZX0pXG4gICAgKVxuICB9XG4gIGNvbnN0IHJlZiA9IHVzZU9uY2xpY2tPdXRzaWRlKFxuICAgICgpID0+IHtcbiAgICAgIGlmIChvcGVuICYmIHVpVHlwZSA9PT0gJ2Rlc2t0b3AnKSB7XG4gICAgICAgIG9uQ2xvc2UoKVxuICAgICAgfVxuICAgIH0sXG4gICAge2V2ZW50VHlwZXM6IFsnY2xpY2snXX1cbiAgKVxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIHJlZihjb250YWluZXJSZWYuY3VycmVudClcbiAgICByZXR1cm4gb24oY29udGFpbmVyUmVmLmN1cnJlbnQsICdjbG9zZS1tZW51Jywgb25DbG9zZSlcbiAgfSwgW10pXG5cbiAgcmV0dXJuIChcbiAgICA8PlxuICAgICAgPEJ1dHRvbldyYXAgcG9zaXRpb249XCJyaWdodFwiPlxuICAgICAgICA8QnV0dG9uXG4gICAgICAgICAgdGl0bGU9e3RpdGxlfVxuICAgICAgICAgIHN0YXJ0SWNvbj17YnV0dG9uSWNvbn1cbiAgICAgICAgICBvbkNsaWNrPXtvcGVuID8gdW5kZWZpbmVkIDogb25PcGVufVxuICAgICAgICAvPlxuICAgICAgPC9CdXR0b25XcmFwPlxuICAgICAgPGNvbXBvbmVudHMucm9vdCBjb250YWluZXJSZWY9e2NvbnRhaW5lclJlZn0gb3Blbj17b3Blbn0gey4uLnJlc3R9PlxuICAgICAgICB7Y2hpbGRyZW59XG4gICAgICA8L2NvbXBvbmVudHMucm9vdD5cbiAgICA8Lz5cbiAgKVxufVxuXG5leHBvcnQgZGVmYXVsdCBPdmVybGF5UGFuZWxcbiJdfQ== */"],
|
|
4892
4888
|
ref: containerRef,
|
|
4893
4889
|
onClick: event => event.stopPropagation(),
|
|
4894
4890
|
children: children
|
|
@@ -4903,7 +4899,7 @@ const desktopContainerStyle$1 = {
|
|
|
4903
4899
|
outline: 'none',
|
|
4904
4900
|
opacity: '0',
|
|
4905
4901
|
transform: 'translateY(-100vh)',
|
|
4906
|
-
transition: 'opacity 0.
|
|
4902
|
+
transition: 'opacity 0.1s ease, transform 0s linear 0.1s'
|
|
4907
4903
|
};
|
|
4908
4904
|
const DesktopOpenStyle = {
|
|
4909
4905
|
opacity: '1',
|
|
@@ -4918,7 +4914,7 @@ const DesktopContainer = ({
|
|
|
4918
4914
|
onClose,
|
|
4919
4915
|
...rest
|
|
4920
4916
|
}) => jsx$1("div", {
|
|
4921
|
-
css: [desktopContainerStyle$1, open && DesktopOpenStyle, process.env.NODE_ENV === "production" ? "" : ";label:DesktopContainer;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
4917
|
+
css: [desktopContainerStyle$1, open && DesktopOpenStyle, process.env.NODE_ENV === "production" ? "" : ";label:DesktopContainer;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIk92ZXJsYXlQYW5lbC5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUF3REkiLCJmaWxlIjoiT3ZlcmxheVBhbmVsLmpzIiwic291cmNlc0NvbnRlbnQiOlsiLyogZXNsaW50LWRpc2FibGUganN4LWExMXkvbm8tc3RhdGljLWVsZW1lbnQtaW50ZXJhY3Rpb25zICovXG4vKiBAanN4SW1wb3J0U291cmNlIEBlbW90aW9uL3JlYWN0ICovXG5pbXBvcnQge3VzZUVmZmVjdCwgdXNlUmVmLCB1c2VTdGF0ZX0gZnJvbSAncmVhY3QnXG5pbXBvcnQgdXNlT25jbGlja091dHNpZGUgZnJvbSAncmVhY3QtY29vbC1vbmNsaWNrb3V0c2lkZSdcbmltcG9ydCB7b259IGZyb20gJ3V0aWwvZXZlbnRzJ1xuaW1wb3J0IHtpc0Rlc2t0b3B9IGZyb20gJ3V0aWwvZW52aXJvbm1lbnQnXG5pbXBvcnQge0J1dHRvbn0gZnJvbSAnLi9idXR0b25zJ1xuaW1wb3J0IHtGdW5jdGlvbkJhckV4dGVuc2lvbiwgVGl0bGVCYXJFeHRlbnNpb259IGZyb20gJy4vdWlFeHRlbnNpb25zJ1xuXG5jb25zdCBjb250YWluZXJTdHlsZSA9IHtcbiAgcG9zaXRpb246ICdhYnNvbHV0ZScsXG4gIHpJbmRleDogJzInLFxuICB0b3A6ICcwJyxcbiAgd2lkdGg6ICcxMDAlJyxcbiAgaGVpZ2h0OiAnMTAwJScsXG4gIHRyYW5zZm9ybTogJ3RyYW5zbGF0ZVkoLTE1MHZoKScsXG4gIG9wYWNpdHk6ICcwJyxcbiAgdHJhbnNpdGlvbjogJ29wYWNpdHkgMC4xcyBlYXNlLCB0cmFuc2Zvcm0gMHMgbGluZWFyIDAuMXMnLFxufVxuXG5jb25zdCBvcGVuU3R5bGUgPSB7XG4gIHRyYW5zZm9ybTogJ3RyYW5zbGF0ZVkoMCknLFxuICBvcGFjaXR5OiAnMScsXG4gIHRyYW5zaXRpb246ICdvcGFjaXR5IDAuMnMgZWFzZSwgdHJhbnNmb3JtIDBzJyxcbn1cblxuY29uc3QgRGVmYXVsdENvbnRhaW5lciA9ICh7b3BlbiwgY29udGFpbmVyUmVmLCBjaGlsZHJlbn0pID0+IChcbiAgPGRpdlxuICAgIGNzcz17W2NvbnRhaW5lclN0eWxlLCBvcGVuICYmIG9wZW5TdHlsZV19XG4gICAgcmVmPXtjb250YWluZXJSZWZ9XG4gICAgb25DbGljaz17ZXZlbnQgPT4gZXZlbnQuc3RvcFByb3BhZ2F0aW9uKCl9XG4gID5cbiAgICB7Y2hpbGRyZW59XG4gIDwvZGl2PlxuKVxuXG5jb25zdCBkZXNrdG9wQ29udGFpbmVyU3R5bGUgPSB7XG4gIHBvc2l0aW9uOiAnYWJzb2x1dGUnLFxuICB6SW5kZXg6ICcyJyxcbiAgYm90dG9tOiAnY2FsYygzZW0gKyB2YXIoLS1ib3R0b20tc3BhY2luZywgMHJlbSkpJyxcbiAgcmlnaHQ6ICdjYWxjKHZhcigtLXNwYWNpbmcsIDAuNzVlbSkgKyAwLjVlbSArIDAuNXJlbSknLFxuICBiYWNrZ3JvdW5kOiAndHJhbnNwYXJlbnQnLFxuICBvdXRsaW5lOiAnbm9uZScsXG4gIG9wYWNpdHk6ICcwJyxcbiAgdHJhbnNmb3JtOiAndHJhbnNsYXRlWSgtMTAwdmgpJyxcbiAgdHJhbnNpdGlvbjogJ29wYWNpdHkgMC4xcyBlYXNlLCB0cmFuc2Zvcm0gMHMgbGluZWFyIDAuMXMnLFxufVxuXG5jb25zdCBEZXNrdG9wT3BlblN0eWxlID0ge1xuICBvcGFjaXR5OiAnMScsXG4gIHRyYW5zZm9ybTogJ3RyYW5zbGF0ZVkoMCknLFxuICB0cmFuc2l0aW9uOiAnb3BhY2l0eSAwLjJzIGVhc2UsIHRyYW5zZm9ybSAwcycsXG59XG5cbmNvbnN0IERlc2t0b3BDb250YWluZXIgPSAoe29wZW4sIGNvbnRhaW5lclJlZiwgY2hpbGRyZW4sIG9uQ2xvc2UsIC4uLnJlc3R9KSA9PiAoXG4gIDxkaXZcbiAgICBjc3M9e1tkZXNrdG9wQ29udGFpbmVyU3R5bGUsIG9wZW4gJiYgRGVza3RvcE9wZW5TdHlsZV19XG4gICAgcmVmPXtjb250YWluZXJSZWZ9XG4gICAgb25DbGljaz17ZXZlbnQgPT4gZXZlbnQuc3RvcFByb3BhZ2F0aW9uKCl9XG4gICAgey4uLnJlc3R9XG4gID5cbiAgICB7Y2hpbGRyZW59XG4gIDwvZGl2PlxuKVxuXG5jb25zdCBPdmVybGF5UGFuZWwgPSAoe1xuICBidXR0b25JY29uLFxuICB0aXRsZSxcbiAgdWlUeXBlID0gaXNEZXNrdG9wKCkgPyAnZGVza3RvcCcgOiAnbW9iaWxlJyxcbiAgY2hpbGRyZW4sXG4gIHNsb3RzID0ge1xuICAgIHJvb3Q6IHVpVHlwZSA9PT0gJ2Rlc2t0b3AnID8gRGVza3RvcENvbnRhaW5lciA6IERlZmF1bHRDb250YWluZXIsXG4gIH0sXG4gIC4uLnJlc3Rcbn0pID0+IHtcbiAgY29uc3QgY29tcG9uZW50cyA9IHtcbiAgICByb290OiBzbG90cy5yb290LFxuICB9XG4gIGNvbnN0IEJ1dHRvbldyYXAgPVxuICAgIHVpVHlwZSA9PT0gJ2Rlc2t0b3AnID8gRnVuY3Rpb25CYXJFeHRlbnNpb24gOiBUaXRsZUJhckV4dGVuc2lvblxuICBjb25zdCBbb3Blbiwgc2V0T3Blbl0gPSB1c2VTdGF0ZShmYWxzZSlcbiAgY29uc3Qgb25PcGVuID0gZXZlbnQgPT4ge1xuICAgIHNldFRpbWVvdXQoKCkgPT4gc2V0T3Blbih0cnVlKSwgMSlcbiAgICBjb25zdCBjb250YWluZXIgPSBldmVudC5jdXJyZW50VGFyZ2V0XG4gICAgY29uc3QgY2xvc2VNZW51ID0gKCkgPT4gc2V0T3BlbihmYWxzZSlcbiAgICBjb250YWluZXIuZGlzcGF0Y2hFdmVudChcbiAgICAgIG5ldyBDdXN0b21FdmVudCgnZm9jdXMtbWVudScsIHtcbiAgICAgICAgZGV0YWlsOiB7c3RhdHVzOiAnb3BlbicsIGNsb3NlTWVudX0sXG4gICAgICAgIGJ1YmJsZXM6IHRydWUsXG4gICAgICB9KVxuICAgIClcbiAgfVxuICBjb25zdCBjb250YWluZXJSZWYgPSB1c2VSZWYoKVxuICBjb25zdCBvbkNsb3NlID0gKCkgPT4ge1xuICAgIHNldE9wZW4oZmFsc2UpXG4gICAgY29udGFpbmVyUmVmLmN1cnJlbnQuZGlzcGF0Y2hFdmVudChcbiAgICAgIG5ldyBDdXN0b21FdmVudCgnZm9jdXMtbWVudScsIHtkZXRhaWw6IHtzdGF0dXM6ICdjbG9zZWQnfSwgYnViYmxlczogdHJ1ZX0pXG4gICAgKVxuICB9XG4gIGNvbnN0IHJlZiA9IHVzZU9uY2xpY2tPdXRzaWRlKFxuICAgICgpID0+IHtcbiAgICAgIGlmIChvcGVuICYmIHVpVHlwZSA9PT0gJ2Rlc2t0b3AnKSB7XG4gICAgICAgIG9uQ2xvc2UoKVxuICAgICAgfVxuICAgIH0sXG4gICAge2V2ZW50VHlwZXM6IFsnY2xpY2snXX1cbiAgKVxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIHJlZihjb250YWluZXJSZWYuY3VycmVudClcbiAgICByZXR1cm4gb24oY29udGFpbmVyUmVmLmN1cnJlbnQsICdjbG9zZS1tZW51Jywgb25DbG9zZSlcbiAgfSwgW10pXG5cbiAgcmV0dXJuIChcbiAgICA8PlxuICAgICAgPEJ1dHRvbldyYXAgcG9zaXRpb249XCJyaWdodFwiPlxuICAgICAgICA8QnV0dG9uXG4gICAgICAgICAgdGl0bGU9e3RpdGxlfVxuICAgICAgICAgIHN0YXJ0SWNvbj17YnV0dG9uSWNvbn1cbiAgICAgICAgICBvbkNsaWNrPXtvcGVuID8gdW5kZWZpbmVkIDogb25PcGVufVxuICAgICAgICAvPlxuICAgICAgPC9CdXR0b25XcmFwPlxuICAgICAgPGNvbXBvbmVudHMucm9vdCBjb250YWluZXJSZWY9e2NvbnRhaW5lclJlZn0gb3Blbj17b3Blbn0gey4uLnJlc3R9PlxuICAgICAgICB7Y2hpbGRyZW59XG4gICAgICA8L2NvbXBvbmVudHMucm9vdD5cbiAgICA8Lz5cbiAgKVxufVxuXG5leHBvcnQgZGVmYXVsdCBPdmVybGF5UGFuZWxcbiJdfQ== */"],
|
|
4922
4918
|
ref: containerRef,
|
|
4923
4919
|
onClick: event => event.stopPropagation(),
|
|
4924
4920
|
...rest,
|
|
@@ -4943,9 +4939,14 @@ const OverlayPanel = ({
|
|
|
4943
4939
|
|
|
4944
4940
|
const onOpen = event => {
|
|
4945
4941
|
setTimeout(() => setOpen(true), 1);
|
|
4946
|
-
event.currentTarget
|
|
4942
|
+
const container = event.currentTarget;
|
|
4943
|
+
|
|
4944
|
+
const closeMenu = () => setOpen(false);
|
|
4945
|
+
|
|
4946
|
+
container.dispatchEvent(new CustomEvent('focus-menu', {
|
|
4947
4947
|
detail: {
|
|
4948
|
-
status: 'open'
|
|
4948
|
+
status: 'open',
|
|
4949
|
+
closeMenu
|
|
4949
4950
|
},
|
|
4950
4951
|
bubbles: true
|
|
4951
4952
|
}));
|
|
@@ -4999,6 +5000,7 @@ const menuItemStyle = {
|
|
|
4999
5000
|
minWidth: 0,
|
|
5000
5001
|
color: 'rgba(255, 255, 255, 0.6)',
|
|
5001
5002
|
background: 'var(--menu-item-background, rgba(255, 255, 255, 0.08))',
|
|
5003
|
+
cursor: 'pointer',
|
|
5002
5004
|
'> div': {
|
|
5003
5005
|
overflow: 'hidden',
|
|
5004
5006
|
whiteSpace: 'nowrap',
|
|
@@ -5025,7 +5027,7 @@ const MenuItem = ({
|
|
|
5025
5027
|
selected,
|
|
5026
5028
|
onClick
|
|
5027
5029
|
}) => jsx$1("div", {
|
|
5028
|
-
css: [menuItemStyle, selected && menuItemSelectedStyle, process.env.NODE_ENV === "production" ? "" : ";label:MenuItem;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
5030
|
+
css: [menuItemStyle, selected && menuItemSelectedStyle, process.env.NODE_ENV === "production" ? "" : ";label:MenuItem;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIk1lbnVJdGVtLmpzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQXFDSSIsImZpbGUiOiJNZW51SXRlbS5qcyIsInNvdXJjZXNDb250ZW50IjpbIi8qIEBqc3hJbXBvcnRTb3VyY2UgQGVtb3Rpb24vcmVhY3QgKi9cbi8qIGVzbGludC1kaXNhYmxlIGpzeC1hMTF5L25vLXN0YXRpYy1lbGVtZW50LWludGVyYWN0aW9ucyAqL1xuaW1wb3J0IGljb24gZnJvbSAnc3R5bGUvaWNvbidcbmltcG9ydCB7Rm9ybWF0dGVkTWVzc2FnZX0gZnJvbSAnY29udGV4dC9JMThuJ1xuXG5jb25zdCBtZW51SXRlbVN0eWxlID0ge1xuICBwYWRkaW5nOiAndmFyKC0tbWVudS1pdGVtLXBhZGRpbmcsIDAuNWVtIDFlbSkgJyxcbiAgZGlzcGxheTogJ2ZsZXgnLFxuICBhbGlnbkl0ZW1zOiAnY2VudGVyJyxcbiAgbWluV2lkdGg6IDAsXG4gIGNvbG9yOiAncmdiYSgyNTUsIDI1NSwgMjU1LCAwLjYpJyxcbiAgYmFja2dyb3VuZDogJ3ZhcigtLW1lbnUtaXRlbS1iYWNrZ3JvdW5kLCByZ2JhKDI1NSwgMjU1LCAyNTUsIDAuMDgpKScsXG4gIGN1cnNvcjogJ3BvaW50ZXInLFxuICAnPiBkaXYnOiB7XG4gICAgb3ZlcmZsb3c6ICdoaWRkZW4nLFxuICAgIHdoaXRlU3BhY2U6ICdub3dyYXAnLFxuICAgIHRleHRPdmVyZmxvdzogJ2VsbGlwc2lzJyxcbiAgfSxcbn1cblxuY29uc3QgbWVudUl0ZW1TZWxlY3RlZFN0eWxlID0ge1xuICBjb2xvcjogJyNGRkYnLFxuICBmb250V2VpZ2h0OiAnNjAwJyxcbiAgJzo6YWZ0ZXInOiB7XG4gICAgY29udGVudDogJ1wiIFwiJyxcbiAgICBkaXNwbGF5OiAnaW5saW5lLWJsb2NrJyxcbiAgICBmbGV4OiAnMCAwIDFlbScsXG4gICAgaGVpZ2h0OiAnMWVtJyxcbiAgICBtYXJnaW5MZWZ0OiAnMC41ZW0nLFxuICAgIGJhY2tncm91bmRDb2xvcjogJ3ZhcigtLXNldHRpbmctY2hlY2staWNvbi1jb2xvciwgI2ZmZiknLFxuICAgIG1hc2tTaXplOiAnY29udGFpbicsXG4gICAgbWFza0ltYWdlOiBgdXJsKCR7aWNvbi5jaGVja30pYCxcbiAgfSxcbn1cblxuY29uc3QgTWVudUl0ZW0gPSAoe2xhYmVsLCBzZWxlY3RlZCwgb25DbGlja30pID0+IChcbiAgPGRpdlxuICAgIGNzcz17W21lbnVJdGVtU3R5bGUsIHNlbGVjdGVkICYmIG1lbnVJdGVtU2VsZWN0ZWRTdHlsZV19XG4gICAgb25DbGljaz17b25DbGlja31cbiAgPlxuICAgIDxkaXY+XG4gICAgICA8Rm9ybWF0dGVkTWVzc2FnZSBpZD17YEtLUy5TRVRUSU5HLiR7bGFiZWx9YH0gZGVmYXVsdE1lc3NhZ2U9e2xhYmVsfSAvPlxuICAgIDwvZGl2PlxuICA8L2Rpdj5cbilcblxuZXhwb3J0IGRlZmF1bHQgTWVudUl0ZW1cbiJdfQ== */"],
|
|
5029
5031
|
onClick: onClick,
|
|
5030
5032
|
children: jsx$1("div", {
|
|
5031
5033
|
children: jsx$1(FormattedMessage, {
|
|
@@ -8796,8 +8798,8 @@ const shouldPause = ({
|
|
|
8796
8798
|
|
|
8797
8799
|
|
|
8798
8800
|
const menuClasses = {
|
|
8799
|
-
default: /*#__PURE__*/css$1(LanguageMenu.styles, process.env.NODE_ENV === "production" ? "" : ";label:default;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["PremiumPlayer.js"],"names":[],"mappings":"AAuEW","file":"PremiumPlayer.js","sourcesContent":["/* eslint-disable no-nested-ternary */\nimport {useEffect, useMemo, useRef, useState} from 'react'\nimport {css} from '@emotion/css'\nimport useDimensions from 'react-cool-dimensions'\nimport {ResizeObserver} from '@juggle/resize-observer'\n\nimport {handleError} from 'playerCore/errors'\nimport {on} from 'util/events'\nimport {isDesktop} from 'util/environment'\nimport multiRef from 'util/multiRef'\nimport {onViewModeChange, toggleFullscreen} from 'util/viewModes'\nimport {useAutoHide} from 'hooks'\nimport {IntlProvider} from 'context/I18n'\nimport {getMediaTime, isBuffered} from 'playerCore/mediaBindings'\nimport Error from 'playerUi/Error'\nimport Layout from 'playerUi/DefaultLayout'\nimport {\n  Button,\n  LiveButton,\n  FullscreenButton,\n  PlayButton,\n  ForwardButton,\n} from 'playerUi/buttons'\nimport Seekbar from 'playerUi/Seekbar'\nimport DisplayTime from 'playerUi/DisplayTime'\nimport LoadingSpinner from 'playerUi/LoadingSpinner'\nimport Backdrop from 'playerUi/Backdrop'\nimport Settings from 'playerUi/Settings'\nimport OverlayPanel from 'playerUi/OverlayPanel'\nimport LanguageMenu from 'playerUi/LanguageMenu'\nimport PlayPanel from 'playerUi/PlayPane'\nimport VolumeControl from 'component/VolumeControl'\nimport CastUi from 'cast/CastUi'\nimport {linkCast} from 'cast/framework'\nimport {CastState} from 'Enum'\nimport {dispatchChapterEvents} from './timeline'\nimport Video from '../Video'\nimport {getLanguageOptions, getQualityOptions, getSettingsData} from './settings'\nimport {linkMediaVolume, syncVolume} from './volume'\nimport SeekPreview from './SeekPreview'\n\nconst sizes = {\n  'small-embed': 200,\n  embed: 400,\n  'tablet-portrait': 600,\n  'tablet-landscape': 900,\n  desktop: 1200,\n}\n\nconst useLinkState = (request, dependencies = []) => {\n  const [state, setState] = useState()\n  useEffect(() => {\n    request(setState)\n  }, dependencies)\n  return state\n}\n\nconst flipState = state => (state === 'playing' ? 'paused' : 'playing')\n\n// FIXME: too few lines to split a file, looking a better place\nconst getThumbnailsUrl = source =>\n  [].concat(source).find(item => item.type === 'thumbnail')?.src\n\nconst defaultSettings = {}\n\nconst shouldPause = ({userFocus, uiType}) =>\n  (uiType === 'mobile' && /settings|chapter-list/.test(userFocus)) ||\n  /blocking|seekbar|share-inner/.test(userFocus)\n\n// TODO extract to somewhere else\nconst menuClasses = {\n  default: css(LanguageMenu.styles),\n  desktop: css(LanguageMenu.styles, LanguageMenu.desktopStyles),\n}\n\nconst LanguageSettings = ({\n  uiType,\n  audioTracks = [],\n  textTracks = [],\n  getPlayer,\n  onChange,\n  slots = {LanguageMenu},\n  ...rest\n}) => {\n  const options = getLanguageOptions(getPlayer(), {audioTracks, textTracks})\n  return (\n    audioTracks.length > 0 &&\n    textTracks.length > 0 && (\n    <OverlayPanel title=\"字幕・音声を切り替える\" buttonIcon=\"subtitle\">\n      <slots.LanguageMenu\n        uiType={uiType}\n        classes={menuClasses}\n        sectionOptions={[options.audioTracks, options.textTracks]}\n        onChange={onChange}\n        {...rest}\n      />\n    </OverlayPanel>\n    )\n  )\n}\n\nconst PremiumPlayer = ({\n  source,\n  startTime,\n  quality = {},\n  title,\n  metadata,\n  channelTitle,\n  chapters,\n  playbackState: appPlaybackState,\n  currentTime: appCurrentTime,\n  playbackRate: appPlaybackRate,\n  loop,\n  volume: appVolume,\n  audioTrack: appAudioTrack,\n  textTrack: appTextTrack,\n  thumbnailsUrl,\n  controls = {autohide: 3000},\n  marks = [],\n  // TODO sectionId\n  intl,\n  settings: appSettings = defaultSettings,\n  blocking,\n  plugins = [],\n  modulesConfig,\n\n  uiType = isDesktop() ? 'desktop' : 'mobile',\n  style,\n  children,\n  uiMode = 'standalone',\n  cast: castOptions = 'compatible',\n  slots,\n  slotProps = {},\n  onError,\n  onPlaybackStateChange,\n  onBack,\n  onCast,\n  onChangeNext,\n  onChangePrevious,\n  onOpenSettings,\n  onChangeSettings,\n  onPlayerLoaded,\n  onBlockedAutoplay,\n  onPlaylogFired,\n  ...videoProps\n}) => {\n  const components = {\n    VolumeControl,\n    CastUi,\n    DisplayTime,\n    LiveButton,\n    Seekbar,\n    Settings,\n    ...slots,\n  }\n  const videoRef = useRef()\n  const containerRef = useRef()\n  const playerRef = useRef()\n  const adContainerRef = useRef()\n  const refs = useRef({})\n  // TODO move RWD related to Layout\n  const {\n    currentBreakpoint: size,\n    width,\n    observe,\n  } = useDimensions({\n    polyfill: ResizeObserver,\n    breakpoints: sizes,\n  })\n  const [isUserActive, setIsUserActive] = useState(\n    uiMode === 'standalone' ? true : videoProps.autoplay\n  )\n  const [userFocus, setUserFocus] = useState('')\n  const [targetState, setTargetState] = useState(() => ({\n    playbackState:\n      appPlaybackState || (videoProps.autoplay ? 'playing' : 'paused'),\n    currentTime: startTime,\n  }))\n  const [playbackTime, setPlaybackTime] = useState({\n    currentTime: 0,\n    bufferTime: 0,\n  })\n  const togglePlay = (overrideState, {triggeredBy} = {}) => {\n    if (targetState.playbackState !== overrideState) {\n      setTargetState(state => ({\n        ...state,\n        playbackState: overrideState || flipState(state.playbackState),\n        animation:\n          triggeredBy === 'user-action' && flipState(state.playbackState),\n      }))\n    }\n  }\n\n  useEffect(() => {\n    if (appPlaybackState) {\n      togglePlay(appPlaybackState)\n    }\n  }, [appPlaybackState])\n  useEffect(() => {\n    if (uiType === 'mobile') {\n      on(document, 'visibilitychange', () => togglePlay('paused'))\n    }\n  }, [uiType, targetState.playbackState])\n  const updatePlaybackTime = event =>\n    requestAnimationFrame(\n      () =>\n        (event?.type !== 'timeupdate' || isBuffered(videoRef.current)) &&\n        setPlaybackTime(state => ({\n          ...state,\n          ...getMediaTime(videoRef.current, {\n            player: playerRef.current,\n            plugins,\n          }),\n          ...(event?.type === 'durationchange' && {\n            currentTime: state.currentTime,\n          }),\n        }))\n    )\n  const setTargetTime = time => {\n    const trimmed = Math.min(\n      time,\n      videoRef.current?.initialDuration || Infinity\n    )\n    setTargetState(state => ({\n      ...state,\n      // seek to 0 repeatedly edge case\n      currentTime: state.currentTime !== trimmed ? trimmed : trimmed + 0.01,\n    }))\n    updatePlaybackTime()\n  }\n  const [playbackState, setPlaybackState] = useState('init')\n  const [castState, setCastState] = useState('')\n  useEffect(() => {\n    if (typeof appCurrentTime === 'number') setTargetTime(appCurrentTime || 0)\n    if (typeof appCurrentTime === 'object')\n      setTargetTime(appCurrentTime?.value || 0)\n  }, [appCurrentTime])\n  const [errorData, setErrorData] = useState()\n  const errorHandler = reactEvent => {\n    onError?.(reactEvent.nativeEvent)\n    handleError(reactEvent, {\n      media: videoRef.current,\n      displayError: data => {\n        setPlaybackState('error')\n        setErrorData(current => (current?.code ? current : data))\n      },\n    })\n  }\n  const isLive = playbackTime.streamType === 'live'\n  const [settings, setSettings] = useState(() => ({\n    sections: [],\n    values: {speed: 1},\n  }))\n  const fetchSettings = () => {\n    if (!playerRef.current) {\n      return\n    }\n    const speedItems = appSettings.speedItems || (isLive ? [] : undefined)\n    setSettings(current => {\n      const {values, sections} = getSettingsData({\n        media: videoRef.current,\n        player: playerRef.current,\n        source,\n        quality,\n        speedItems,\n        loop,\n        preferred: current.preferred,\n        otherSections: appSettings.sections,\n      })\n      return {preferred: current.preferred, values, sections}\n    })\n  }\n  const lastState = useRef(playbackState)\n  const handlePlaybackStateChange = (event, state) => {\n    if (lastState.current === 'error') {\n      return\n    }\n    onPlaybackStateChange?.(event, state)\n    // previously UI playback state was synced only when\n    // exiting iOS video only fullscreen(webkitendfullscreen)\n    if (\n      /playing|paused/.test(state) &&\n      !(uiType === 'mobile' && blocking) &&\n      !shouldPause({userFocus, uiType})\n    ) {\n      togglePlay(state)\n    }\n    if (state === 'ended') {\n      togglePlay('paused')\n    }\n    if (state === 'loading' && lastState.current !== 'init') {\n      setTimeout(() => togglePlay('playing'), 1)\n    }\n    if (state === 'playing') {\n      setIsUserActive(true)\n    }\n    if (lastState.current === 'loading') {\n      fetchSettings()\n    }\n    lastState.current = state\n    setPlaybackState(state)\n  }\n\n  const changeSettings = (name, value, {keepOpen} = {}) => {\n    // TODO consider merge into useReducer?\n    onChangeSettings?.({name, value})\n    setTargetTime(playbackTime.currentTime)\n    setSettings(current => ({\n      ...current,\n      values: {...current.values, [name]: value},\n      preferred: {...current.preferred, [name]: value},\n    }))\n    if (!keepOpen) {\n      setUserFocus('')\n    }\n  }\n  const openSettings = event => {\n    const animationFrame =\n      userFocus !== 'settings' &&\n      requestAnimationFrame(() => {\n        onOpenSettings?.(event, settings)\n        // In iOS Safari, we need to update settings data\n        fetchSettings()\n      })\n    setUserFocus(current => (current === 'settings' ? '' : 'settings'))\n    return animationFrame\n  }\n  useEffect(() => {\n    if (appPlaybackRate > 0) {\n      setSettings(current => ({\n        ...current,\n        values: {...current.values, speed: appPlaybackRate},\n      }))\n    }\n  }, [appPlaybackRate])\n\n  const qualityOptions = useMemo(\n    () => getQualityOptions(settings),\n    [settings.values.quality]\n  )\n  const viewMode = useLinkState(update =>\n    onViewModeChange(videoRef.current, update)\n  )\n  const sourceOverride = useLinkState(\n    async update => {\n      const result =\n        source && (await quality.rewriteManifest?.(source, qualityOptions))\n      update(result || source)\n    },\n    [source, qualityOptions]\n  )\n  const waiting = /emptied|loading|buffering/.test(playbackState)\n  const activePlayback =\n    playbackState === 'playing' || playbackState === 'waiting'\n  const {\n    mode: autoHideMode,\n    onClick,\n    onMouseMove,\n  } = useAutoHide({\n    pinned: !controls.autohide || waiting || !activePlayback || userFocus,\n    tapToHide: uiType === 'mobile',\n    hideTimeMs: controls.autohide,\n  })\n  const mode =\n    userFocus === 'seekbar'\n      ? 'hidden'\n      : controls.autohide\n      ? autoHideMode\n      : controls\n      ? 'shown'\n      : 'hidden'\n  const controlsDisplay =\n    userFocus === 'seekbar'\n      ? 'seekbar-only'\n      : controls === 'title-only' || playbackState === 'emptied'\n      ? 'hidden'\n      : mode\n  const shouldHidePanels =\n    (controls === 'no-panel' || controlsDisplay === 'hidden') && userFocus\n  const havePlayPanel =\n    !blocking &&\n    uiType === 'desktop' &&\n    !waiting &&\n    !/volume|''/.test(userFocus) &&\n    !/title-only|no-panel/.test(controls) &&\n    (controls.autohide || mode === 'shown')\n\n  useEffect(() => {\n    if (shouldHidePanels) {\n      setUserFocus('')\n    }\n  }, [shouldHidePanels])\n  useEffect(() => \n    on(containerRef.current, 'focus-menu', event => {\n      if (event.detail?.status === 'open') {\n        setUserFocus('menu')\n      }\n      if (event.detail?.status === 'closed') {\n        setUserFocus(current => (current === 'menu' ? '' : current))\n      }\n    })\n  , [])\n  const {subscribe, onChange, toggleMute} = linkMediaVolume(() => ({\n    video: videoRef.current,\n    getPlayer: () => playerRef.current,\n  }))\n  const changePrevious = event => {\n    onChangePrevious(event)\n    togglePlay('paused')\n    videoRef.current.dispatchEvent(new CustomEvent('loadstart'))\n  }\n  const changeNext = event => {\n    onChangeNext(event)\n    togglePlay('paused')\n    videoRef.current.dispatchEvent(new CustomEvent('loadstart'))\n  }\n\n  useEffect(fetchSettings, [appSettings])\n  useEffect(() => {\n    // The adContainer should be set before `load` because ImaDai.load needs it.\n    plugins.forEach(plugin => plugin.setAdContainer?.(adContainerRef.current))\n  }, [])\n\n  useEffect(\n    () =>\n      dispatchChapterEvents({\n        media: videoRef.current,\n        chapters,\n        getTime: () => videoRef.current.currentTime,\n      }),\n    [chapters]\n  )\n\n  const isEnd = playbackState === 'ended'\n  const canSeek = !isEnd && playbackTime.duration > 0\n  const derivedPlaybackState =\n    (uiType === 'mobile' && blocking) || shouldPause({userFocus, uiType})\n      ? 'paused'\n      : targetState.playbackState\n  const activeThumbnailsUrl =\n    (!userFocus || userFocus === 'seekbar') &&\n    (getThumbnailsUrl(source) || thumbnailsUrl)\n  const cssVariables = {\n    '--playing': playbackState === 'playing' ? '1' : '0',\n  }\n  const onRewind =\n    (!isLive || playbackTime.currentTime < 0) &&\n    canSeek &&\n    (() => setTargetTime(playbackTime.currentTime - 10, 'rewind'))\n  const onForward =\n    (!isLive || playbackTime.currentTime < 0) &&\n    canSeek &&\n    (() => setTargetTime(playbackTime.currentTime + 10, 'forward'))\n  const onClickBlank =\n    havePlayPanel && (() => togglePlay('', {triggeredBy: 'user-action'}))\n\n  const uiHandlers = {\n    onPlay: () => {\n      togglePlay('', {triggeredBy: 'user-action'})\n      setIsUserActive(true)\n    },\n    onClickLive: isLive && (() => setTargetTime(0, 'seekToLive')),\n    onClickSeekbar: () => {\n      refs.current.deferedSeekFocus = setTimeout(\n        () => setUserFocus('seekbar'),\n        66\n      )\n    },\n    onSeek: time => {\n      setTargetTime(time, 'seek')\n      clearTimeout(refs.current.deferedSeekFocus)\n      setTimeout(() => setUserFocus(''), 66)\n    },\n    onChangeSettings: ({name, value, keepOpen}) =>\n      changeSettings(name, value, {keepOpen}),\n    onCloseSettings: event => {\n      setUserFocus('')\n      slotProps?.settings?.onClose?.(event)\n    },\n  }\n\n  const uiElements = {\n    liveButton: uiHandlers.onClickLive && (\n      <components.LiveButton\n        usingStartOver={playbackTime.currentTime < 0}\n        onClick={uiHandlers.onClickLive}\n      />\n    ),\n    controlButtons: {\n      playButton: (\n        <PlayButton\n          playbackState={derivedPlaybackState}\n          ended={isEnd}\n          hidden={\n            uiType === 'mobile' && (waiting || /loading/.test(playbackState))\n          }\n          variant={!isUserActive && 'firstplay'}\n          onClick={uiHandlers.onPlay}\n        />\n      ),\n      rewindButton: onRewind && (\n        <Button\n          startIcon=\"rewind10\"\n          title=\"KKS.PLAYER.REWIND\"\n          disabled={isEnd}\n          onClick={onRewind}\n        />\n      ),\n      forwardButton: onForward && (\n        <ForwardButton\n          startIcon=\"forward10\"\n          title=\"KKS.PLAYER.FORWARD\"\n          disabled={!canSeek}\n          onClick={onForward}\n        />\n      ),\n      nextEpisodeButton: (\n        <Button\n          startIcon=\"next\"\n          title=\"KKS.PLAYER.NEXT\"\n          disabled={!onChangeNext}\n          onClick={changeNext}\n        />\n      ),\n      previousEpisodeButton: (\n        <Button\n          startIcon=\"previous\"\n          title=\"KKS.PLAYER.PREVIOUS\"\n          disabled={!onChangePrevious}\n          onClick={changePrevious}\n        />\n      ),\n    },\n    seekbar: (\n      <components.Seekbar\n        style={/volume/.test(userFocus) && {opacity: '0'}}\n        startTime={playbackTime.startTime}\n        currentTime={playbackTime.currentTime}\n        bufferTime={playbackTime.bufferTime}\n        duration={playbackTime.duration}\n        chapters={chapters}\n        onChange={uiHandlers.onClickSeekbar}\n        onChangeCommitted={uiHandlers.onSeek}\n        marks={marks}\n        plugins={plugins}\n        {...slotProps.seekbar}\n      >\n        {activeThumbnailsUrl && (\n          <SeekPreview\n            thumbnailsUrl={activeThumbnailsUrl}\n            duration={playbackTime.duration}\n            chapters={chapters}\n          />\n        )}\n      </components.Seekbar>\n    ),\n    displayTime: <components.DisplayTime {...playbackTime} />,\n    fullscreenButton: (\n      <FullscreenButton\n        viewMode={viewMode}\n        onClick={() => toggleFullscreen(containerRef.current)}\n      />\n    ),\n    volumeControl: width >= sizes['small-embed'] && (\n      <components.VolumeControl\n        {...{subscribe, onChange, toggleMute}}\n        onMouseOver={() => setUserFocus('volume')}\n        onMouseOut={() => setUserFocus('')}\n        {...slotProps?.volumeControl}\n      />\n    ),\n    backItems: (\n      <PlayPanel animation={targetState.animation} onClick={onClickBlank} />\n    ),\n  }\n  const compatibilityCastProps = castOptions === 'compatible' && {\n    source,\n    values: {title, ...settings.values, ...intl, modulesConfig},\n    onLoad: onCast,\n  }\n\n  useEffect(() => {\n    if (castOptions && castOptions !== 'compatible') {\n      return linkCast({source, ...castOptions})\n    } // TODO playlist, audio / subtitle setting\n  }, [source, castOptions.contentId, castOptions.customData])\n  const selectedAudioTrack = settings.preferred?.audio || appAudioTrack\n  const selectedTextTrack = settings.preferred?.subtitles || appTextTrack\n\n  return (\n    <IntlProvider {...intl}>\n      <Layout\n        style={{...style, cssVariables}}\n        type={uiType}\n        display={mode}\n        controlsDisplay={controlsDisplay}\n        size={size}\n        video={\n          <Video\n            {...videoProps}\n            videoRef={multiRef(videoRef, videoProps.videoRef)}\n            source={\n              castState !== CastState.CONNECTED &&\n              playbackState !== 'error' &&\n              sourceOverride\n            }\n            playbackState={derivedPlaybackState}\n            currentTime={targetState.currentTime}\n            {...(appVolume >= 0 && {volume: appVolume, muted: appVolume <= 0})}\n            playbackRate={settings.values.speed}\n            videoResolution={qualityOptions}\n            audioTrack={selectedAudioTrack}\n            textTrack={selectedTextTrack}\n            loop={settings.values.loop}\n            plugins={plugins}\n            modulesConfig={modulesConfig}\n            onPlaylogFired={onPlaylogFired}\n            onError={errorHandler}\n            onPlaybackStateChange={handlePlaybackStateChange}\n            onBlockedAutoplay={error => onBlockedAutoplay?.(error)}\n            onCanPlay={event => {\n              updatePlaybackTime(event)\n              videoProps.onCanPlay?.(event)\n            }}\n            onTimeUpdate={updatePlaybackTime}\n            onDurationChange={updatePlaybackTime}\n            onPlayerLoaded={player => {\n              playerRef.current = player\n              onPlayerLoaded?.(player)\n              syncVolume(videoRef.current, videoProps.muted ? 0 : appVolume)\n            }}\n          />\n        }\n        containerRef={element => {\n          containerRef.current = element\n          observe(element)\n        }}\n        backRef={adContainerRef}\n        title={title}\n        channelTitle={channelTitle}\n        backButton={onBack && <Button startIcon=\"back\" onClick={onBack} />}\n        {...(isUserActive && uiElements)}\n        onClick={onClick}\n        onMouseMove={onMouseMove}\n      >\n        <LoadingSpinner active={waiting} />\n        {errorData && <Error error={errorData} onBack={onBack} />}\n        {children}\n        {/* TODO sync Cast last played time back */}\n        {isUserActive && (\n          <components.CastUi\n            onBack={onBack}\n            onStateChange={setCastState}\n            onLoad={onCast}\n            {...compatibilityCastProps}\n            {...slotProps.castUi}\n          />\n        )}\n        {isUserActive && ( // TODO: Design <Extenstion /> flag\n          <components.Settings\n            type={uiType}\n            sections={settings.sections}\n            // TODO hasBottomPanel bottom: 8em\n            open={userFocus === 'settings'}\n            values={settings.values}\n            onOpen={openSettings}\n            onChange={uiHandlers.onChangeSettings}\n            {...slotProps?.settings}\n            onClose={uiHandlers.onCloseSettings}\n          />\n        )}\n        <LanguageSettings\n          uiType={uiType}\n          getPlayer={() => playerRef.current}\n          onChange={(_event, item) =>\n            uiHandlers.onChangeSettings({\n              name: item.type,\n              value: item.value,\n              keepOpen: true,\n            })\n          }\n          {...slotProps.languageSettings}\n        />\n        <Backdrop open={!playbackState || playbackState === 'loading'}>\n          <LoadingSpinner />\n        </Backdrop>\n      </Layout>\n    </IntlProvider>\n  )\n}\n\nexport default PremiumPlayer\n"]} */"),
|
|
8800
|
-
desktop: /*#__PURE__*/css$1(LanguageMenu.styles, LanguageMenu.desktopStyles, process.env.NODE_ENV === "production" ? "" : ";label:desktop;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["PremiumPlayer.js"],"names":[],"mappings":"AAwEW","file":"PremiumPlayer.js","sourcesContent":["/* eslint-disable no-nested-ternary */\nimport {useEffect, useMemo, useRef, useState} from 'react'\nimport {css} from '@emotion/css'\nimport useDimensions from 'react-cool-dimensions'\nimport {ResizeObserver} from '@juggle/resize-observer'\n\nimport {handleError} from 'playerCore/errors'\nimport {on} from 'util/events'\nimport {isDesktop} from 'util/environment'\nimport multiRef from 'util/multiRef'\nimport {onViewModeChange, toggleFullscreen} from 'util/viewModes'\nimport {useAutoHide} from 'hooks'\nimport {IntlProvider} from 'context/I18n'\nimport {getMediaTime, isBuffered} from 'playerCore/mediaBindings'\nimport Error from 'playerUi/Error'\nimport Layout from 'playerUi/DefaultLayout'\nimport {\n  Button,\n  LiveButton,\n  FullscreenButton,\n  PlayButton,\n  ForwardButton,\n} from 'playerUi/buttons'\nimport Seekbar from 'playerUi/Seekbar'\nimport DisplayTime from 'playerUi/DisplayTime'\nimport LoadingSpinner from 'playerUi/LoadingSpinner'\nimport Backdrop from 'playerUi/Backdrop'\nimport Settings from 'playerUi/Settings'\nimport OverlayPanel from 'playerUi/OverlayPanel'\nimport LanguageMenu from 'playerUi/LanguageMenu'\nimport PlayPanel from 'playerUi/PlayPane'\nimport VolumeControl from 'component/VolumeControl'\nimport CastUi from 'cast/CastUi'\nimport {linkCast} from 'cast/framework'\nimport {CastState} from 'Enum'\nimport {dispatchChapterEvents} from './timeline'\nimport Video from '../Video'\nimport {getLanguageOptions, getQualityOptions, getSettingsData} from './settings'\nimport {linkMediaVolume, syncVolume} from './volume'\nimport SeekPreview from './SeekPreview'\n\nconst sizes = {\n  'small-embed': 200,\n  embed: 400,\n  'tablet-portrait': 600,\n  'tablet-landscape': 900,\n  desktop: 1200,\n}\n\nconst useLinkState = (request, dependencies = []) => {\n  const [state, setState] = useState()\n  useEffect(() => {\n    request(setState)\n  }, dependencies)\n  return state\n}\n\nconst flipState = state => (state === 'playing' ? 'paused' : 'playing')\n\n// FIXME: too few lines to split a file, looking a better place\nconst getThumbnailsUrl = source =>\n  [].concat(source).find(item => item.type === 'thumbnail')?.src\n\nconst defaultSettings = {}\n\nconst shouldPause = ({userFocus, uiType}) =>\n  (uiType === 'mobile' && /settings|chapter-list/.test(userFocus)) ||\n  /blocking|seekbar|share-inner/.test(userFocus)\n\n// TODO extract to somewhere else\nconst menuClasses = {\n  default: css(LanguageMenu.styles),\n  desktop: css(LanguageMenu.styles, LanguageMenu.desktopStyles),\n}\n\nconst LanguageSettings = ({\n  uiType,\n  audioTracks = [],\n  textTracks = [],\n  getPlayer,\n  onChange,\n  slots = {LanguageMenu},\n  ...rest\n}) => {\n  const options = getLanguageOptions(getPlayer(), {audioTracks, textTracks})\n  return (\n    audioTracks.length > 0 &&\n    textTracks.length > 0 && (\n    <OverlayPanel title=\"字幕・音声を切り替える\" buttonIcon=\"subtitle\">\n      <slots.LanguageMenu\n        uiType={uiType}\n        classes={menuClasses}\n        sectionOptions={[options.audioTracks, options.textTracks]}\n        onChange={onChange}\n        {...rest}\n      />\n    </OverlayPanel>\n    )\n  )\n}\n\nconst PremiumPlayer = ({\n  source,\n  startTime,\n  quality = {},\n  title,\n  metadata,\n  channelTitle,\n  chapters,\n  playbackState: appPlaybackState,\n  currentTime: appCurrentTime,\n  playbackRate: appPlaybackRate,\n  loop,\n  volume: appVolume,\n  audioTrack: appAudioTrack,\n  textTrack: appTextTrack,\n  thumbnailsUrl,\n  controls = {autohide: 3000},\n  marks = [],\n  // TODO sectionId\n  intl,\n  settings: appSettings = defaultSettings,\n  blocking,\n  plugins = [],\n  modulesConfig,\n\n  uiType = isDesktop() ? 'desktop' : 'mobile',\n  style,\n  children,\n  uiMode = 'standalone',\n  cast: castOptions = 'compatible',\n  slots,\n  slotProps = {},\n  onError,\n  onPlaybackStateChange,\n  onBack,\n  onCast,\n  onChangeNext,\n  onChangePrevious,\n  onOpenSettings,\n  onChangeSettings,\n  onPlayerLoaded,\n  onBlockedAutoplay,\n  onPlaylogFired,\n  ...videoProps\n}) => {\n  const components = {\n    VolumeControl,\n    CastUi,\n    DisplayTime,\n    LiveButton,\n    Seekbar,\n    Settings,\n    ...slots,\n  }\n  const videoRef = useRef()\n  const containerRef = useRef()\n  const playerRef = useRef()\n  const adContainerRef = useRef()\n  const refs = useRef({})\n  // TODO move RWD related to Layout\n  const {\n    currentBreakpoint: size,\n    width,\n    observe,\n  } = useDimensions({\n    polyfill: ResizeObserver,\n    breakpoints: sizes,\n  })\n  const [isUserActive, setIsUserActive] = useState(\n    uiMode === 'standalone' ? true : videoProps.autoplay\n  )\n  const [userFocus, setUserFocus] = useState('')\n  const [targetState, setTargetState] = useState(() => ({\n    playbackState:\n      appPlaybackState || (videoProps.autoplay ? 'playing' : 'paused'),\n    currentTime: startTime,\n  }))\n  const [playbackTime, setPlaybackTime] = useState({\n    currentTime: 0,\n    bufferTime: 0,\n  })\n  const togglePlay = (overrideState, {triggeredBy} = {}) => {\n    if (targetState.playbackState !== overrideState) {\n      setTargetState(state => ({\n        ...state,\n        playbackState: overrideState || flipState(state.playbackState),\n        animation:\n          triggeredBy === 'user-action' && flipState(state.playbackState),\n      }))\n    }\n  }\n\n  useEffect(() => {\n    if (appPlaybackState) {\n      togglePlay(appPlaybackState)\n    }\n  }, [appPlaybackState])\n  useEffect(() => {\n    if (uiType === 'mobile') {\n      on(document, 'visibilitychange', () => togglePlay('paused'))\n    }\n  }, [uiType, targetState.playbackState])\n  const updatePlaybackTime = event =>\n    requestAnimationFrame(\n      () =>\n        (event?.type !== 'timeupdate' || isBuffered(videoRef.current)) &&\n        setPlaybackTime(state => ({\n          ...state,\n          ...getMediaTime(videoRef.current, {\n            player: playerRef.current,\n            plugins,\n          }),\n          ...(event?.type === 'durationchange' && {\n            currentTime: state.currentTime,\n          }),\n        }))\n    )\n  const setTargetTime = time => {\n    const trimmed = Math.min(\n      time,\n      videoRef.current?.initialDuration || Infinity\n    )\n    setTargetState(state => ({\n      ...state,\n      // seek to 0 repeatedly edge case\n      currentTime: state.currentTime !== trimmed ? trimmed : trimmed + 0.01,\n    }))\n    updatePlaybackTime()\n  }\n  const [playbackState, setPlaybackState] = useState('init')\n  const [castState, setCastState] = useState('')\n  useEffect(() => {\n    if (typeof appCurrentTime === 'number') setTargetTime(appCurrentTime || 0)\n    if (typeof appCurrentTime === 'object')\n      setTargetTime(appCurrentTime?.value || 0)\n  }, [appCurrentTime])\n  const [errorData, setErrorData] = useState()\n  const errorHandler = reactEvent => {\n    onError?.(reactEvent.nativeEvent)\n    handleError(reactEvent, {\n      media: videoRef.current,\n      displayError: data => {\n        setPlaybackState('error')\n        setErrorData(current => (current?.code ? current : data))\n      },\n    })\n  }\n  const isLive = playbackTime.streamType === 'live'\n  const [settings, setSettings] = useState(() => ({\n    sections: [],\n    values: {speed: 1},\n  }))\n  const fetchSettings = () => {\n    if (!playerRef.current) {\n      return\n    }\n    const speedItems = appSettings.speedItems || (isLive ? [] : undefined)\n    setSettings(current => {\n      const {values, sections} = getSettingsData({\n        media: videoRef.current,\n        player: playerRef.current,\n        source,\n        quality,\n        speedItems,\n        loop,\n        preferred: current.preferred,\n        otherSections: appSettings.sections,\n      })\n      return {preferred: current.preferred, values, sections}\n    })\n  }\n  const lastState = useRef(playbackState)\n  const handlePlaybackStateChange = (event, state) => {\n    if (lastState.current === 'error') {\n      return\n    }\n    onPlaybackStateChange?.(event, state)\n    // previously UI playback state was synced only when\n    // exiting iOS video only fullscreen(webkitendfullscreen)\n    if (\n      /playing|paused/.test(state) &&\n      !(uiType === 'mobile' && blocking) &&\n      !shouldPause({userFocus, uiType})\n    ) {\n      togglePlay(state)\n    }\n    if (state === 'ended') {\n      togglePlay('paused')\n    }\n    if (state === 'loading' && lastState.current !== 'init') {\n      setTimeout(() => togglePlay('playing'), 1)\n    }\n    if (state === 'playing') {\n      setIsUserActive(true)\n    }\n    if (lastState.current === 'loading') {\n      fetchSettings()\n    }\n    lastState.current = state\n    setPlaybackState(state)\n  }\n\n  const changeSettings = (name, value, {keepOpen} = {}) => {\n    // TODO consider merge into useReducer?\n    onChangeSettings?.({name, value})\n    setTargetTime(playbackTime.currentTime)\n    setSettings(current => ({\n      ...current,\n      values: {...current.values, [name]: value},\n      preferred: {...current.preferred, [name]: value},\n    }))\n    if (!keepOpen) {\n      setUserFocus('')\n    }\n  }\n  const openSettings = event => {\n    const animationFrame =\n      userFocus !== 'settings' &&\n      requestAnimationFrame(() => {\n        onOpenSettings?.(event, settings)\n        // In iOS Safari, we need to update settings data\n        fetchSettings()\n      })\n    setUserFocus(current => (current === 'settings' ? '' : 'settings'))\n    return animationFrame\n  }\n  useEffect(() => {\n    if (appPlaybackRate > 0) {\n      setSettings(current => ({\n        ...current,\n        values: {...current.values, speed: appPlaybackRate},\n      }))\n    }\n  }, [appPlaybackRate])\n\n  const qualityOptions = useMemo(\n    () => getQualityOptions(settings),\n    [settings.values.quality]\n  )\n  const viewMode = useLinkState(update =>\n    onViewModeChange(videoRef.current, update)\n  )\n  const sourceOverride = useLinkState(\n    async update => {\n      const result =\n        source && (await quality.rewriteManifest?.(source, qualityOptions))\n      update(result || source)\n    },\n    [source, qualityOptions]\n  )\n  const waiting = /emptied|loading|buffering/.test(playbackState)\n  const activePlayback =\n    playbackState === 'playing' || playbackState === 'waiting'\n  const {\n    mode: autoHideMode,\n    onClick,\n    onMouseMove,\n  } = useAutoHide({\n    pinned: !controls.autohide || waiting || !activePlayback || userFocus,\n    tapToHide: uiType === 'mobile',\n    hideTimeMs: controls.autohide,\n  })\n  const mode =\n    userFocus === 'seekbar'\n      ? 'hidden'\n      : controls.autohide\n      ? autoHideMode\n      : controls\n      ? 'shown'\n      : 'hidden'\n  const controlsDisplay =\n    userFocus === 'seekbar'\n      ? 'seekbar-only'\n      : controls === 'title-only' || playbackState === 'emptied'\n      ? 'hidden'\n      : mode\n  const shouldHidePanels =\n    (controls === 'no-panel' || controlsDisplay === 'hidden') && userFocus\n  const havePlayPanel =\n    !blocking &&\n    uiType === 'desktop' &&\n    !waiting &&\n    !/volume|''/.test(userFocus) &&\n    !/title-only|no-panel/.test(controls) &&\n    (controls.autohide || mode === 'shown')\n\n  useEffect(() => {\n    if (shouldHidePanels) {\n      setUserFocus('')\n    }\n  }, [shouldHidePanels])\n  useEffect(() => \n    on(containerRef.current, 'focus-menu', event => {\n      if (event.detail?.status === 'open') {\n        setUserFocus('menu')\n      }\n      if (event.detail?.status === 'closed') {\n        setUserFocus(current => (current === 'menu' ? '' : current))\n      }\n    })\n  , [])\n  const {subscribe, onChange, toggleMute} = linkMediaVolume(() => ({\n    video: videoRef.current,\n    getPlayer: () => playerRef.current,\n  }))\n  const changePrevious = event => {\n    onChangePrevious(event)\n    togglePlay('paused')\n    videoRef.current.dispatchEvent(new CustomEvent('loadstart'))\n  }\n  const changeNext = event => {\n    onChangeNext(event)\n    togglePlay('paused')\n    videoRef.current.dispatchEvent(new CustomEvent('loadstart'))\n  }\n\n  useEffect(fetchSettings, [appSettings])\n  useEffect(() => {\n    // The adContainer should be set before `load` because ImaDai.load needs it.\n    plugins.forEach(plugin => plugin.setAdContainer?.(adContainerRef.current))\n  }, [])\n\n  useEffect(\n    () =>\n      dispatchChapterEvents({\n        media: videoRef.current,\n        chapters,\n        getTime: () => videoRef.current.currentTime,\n      }),\n    [chapters]\n  )\n\n  const isEnd = playbackState === 'ended'\n  const canSeek = !isEnd && playbackTime.duration > 0\n  const derivedPlaybackState =\n    (uiType === 'mobile' && blocking) || shouldPause({userFocus, uiType})\n      ? 'paused'\n      : targetState.playbackState\n  const activeThumbnailsUrl =\n    (!userFocus || userFocus === 'seekbar') &&\n    (getThumbnailsUrl(source) || thumbnailsUrl)\n  const cssVariables = {\n    '--playing': playbackState === 'playing' ? '1' : '0',\n  }\n  const onRewind =\n    (!isLive || playbackTime.currentTime < 0) &&\n    canSeek &&\n    (() => setTargetTime(playbackTime.currentTime - 10, 'rewind'))\n  const onForward =\n    (!isLive || playbackTime.currentTime < 0) &&\n    canSeek &&\n    (() => setTargetTime(playbackTime.currentTime + 10, 'forward'))\n  const onClickBlank =\n    havePlayPanel && (() => togglePlay('', {triggeredBy: 'user-action'}))\n\n  const uiHandlers = {\n    onPlay: () => {\n      togglePlay('', {triggeredBy: 'user-action'})\n      setIsUserActive(true)\n    },\n    onClickLive: isLive && (() => setTargetTime(0, 'seekToLive')),\n    onClickSeekbar: () => {\n      refs.current.deferedSeekFocus = setTimeout(\n        () => setUserFocus('seekbar'),\n        66\n      )\n    },\n    onSeek: time => {\n      setTargetTime(time, 'seek')\n      clearTimeout(refs.current.deferedSeekFocus)\n      setTimeout(() => setUserFocus(''), 66)\n    },\n    onChangeSettings: ({name, value, keepOpen}) =>\n      changeSettings(name, value, {keepOpen}),\n    onCloseSettings: event => {\n      setUserFocus('')\n      slotProps?.settings?.onClose?.(event)\n    },\n  }\n\n  const uiElements = {\n    liveButton: uiHandlers.onClickLive && (\n      <components.LiveButton\n        usingStartOver={playbackTime.currentTime < 0}\n        onClick={uiHandlers.onClickLive}\n      />\n    ),\n    controlButtons: {\n      playButton: (\n        <PlayButton\n          playbackState={derivedPlaybackState}\n          ended={isEnd}\n          hidden={\n            uiType === 'mobile' && (waiting || /loading/.test(playbackState))\n          }\n          variant={!isUserActive && 'firstplay'}\n          onClick={uiHandlers.onPlay}\n        />\n      ),\n      rewindButton: onRewind && (\n        <Button\n          startIcon=\"rewind10\"\n          title=\"KKS.PLAYER.REWIND\"\n          disabled={isEnd}\n          onClick={onRewind}\n        />\n      ),\n      forwardButton: onForward && (\n        <ForwardButton\n          startIcon=\"forward10\"\n          title=\"KKS.PLAYER.FORWARD\"\n          disabled={!canSeek}\n          onClick={onForward}\n        />\n      ),\n      nextEpisodeButton: (\n        <Button\n          startIcon=\"next\"\n          title=\"KKS.PLAYER.NEXT\"\n          disabled={!onChangeNext}\n          onClick={changeNext}\n        />\n      ),\n      previousEpisodeButton: (\n        <Button\n          startIcon=\"previous\"\n          title=\"KKS.PLAYER.PREVIOUS\"\n          disabled={!onChangePrevious}\n          onClick={changePrevious}\n        />\n      ),\n    },\n    seekbar: (\n      <components.Seekbar\n        style={/volume/.test(userFocus) && {opacity: '0'}}\n        startTime={playbackTime.startTime}\n        currentTime={playbackTime.currentTime}\n        bufferTime={playbackTime.bufferTime}\n        duration={playbackTime.duration}\n        chapters={chapters}\n        onChange={uiHandlers.onClickSeekbar}\n        onChangeCommitted={uiHandlers.onSeek}\n        marks={marks}\n        plugins={plugins}\n        {...slotProps.seekbar}\n      >\n        {activeThumbnailsUrl && (\n          <SeekPreview\n            thumbnailsUrl={activeThumbnailsUrl}\n            duration={playbackTime.duration}\n            chapters={chapters}\n          />\n        )}\n      </components.Seekbar>\n    ),\n    displayTime: <components.DisplayTime {...playbackTime} />,\n    fullscreenButton: (\n      <FullscreenButton\n        viewMode={viewMode}\n        onClick={() => toggleFullscreen(containerRef.current)}\n      />\n    ),\n    volumeControl: width >= sizes['small-embed'] && (\n      <components.VolumeControl\n        {...{subscribe, onChange, toggleMute}}\n        onMouseOver={() => setUserFocus('volume')}\n        onMouseOut={() => setUserFocus('')}\n        {...slotProps?.volumeControl}\n      />\n    ),\n    backItems: (\n      <PlayPanel animation={targetState.animation} onClick={onClickBlank} />\n    ),\n  }\n  const compatibilityCastProps = castOptions === 'compatible' && {\n    source,\n    values: {title, ...settings.values, ...intl, modulesConfig},\n    onLoad: onCast,\n  }\n\n  useEffect(() => {\n    if (castOptions && castOptions !== 'compatible') {\n      return linkCast({source, ...castOptions})\n    } // TODO playlist, audio / subtitle setting\n  }, [source, castOptions.contentId, castOptions.customData])\n  const selectedAudioTrack = settings.preferred?.audio || appAudioTrack\n  const selectedTextTrack = settings.preferred?.subtitles || appTextTrack\n\n  return (\n    <IntlProvider {...intl}>\n      <Layout\n        style={{...style, cssVariables}}\n        type={uiType}\n        display={mode}\n        controlsDisplay={controlsDisplay}\n        size={size}\n        video={\n          <Video\n            {...videoProps}\n            videoRef={multiRef(videoRef, videoProps.videoRef)}\n            source={\n              castState !== CastState.CONNECTED &&\n              playbackState !== 'error' &&\n              sourceOverride\n            }\n            playbackState={derivedPlaybackState}\n            currentTime={targetState.currentTime}\n            {...(appVolume >= 0 && {volume: appVolume, muted: appVolume <= 0})}\n            playbackRate={settings.values.speed}\n            videoResolution={qualityOptions}\n            audioTrack={selectedAudioTrack}\n            textTrack={selectedTextTrack}\n            loop={settings.values.loop}\n            plugins={plugins}\n            modulesConfig={modulesConfig}\n            onPlaylogFired={onPlaylogFired}\n            onError={errorHandler}\n            onPlaybackStateChange={handlePlaybackStateChange}\n            onBlockedAutoplay={error => onBlockedAutoplay?.(error)}\n            onCanPlay={event => {\n              updatePlaybackTime(event)\n              videoProps.onCanPlay?.(event)\n            }}\n            onTimeUpdate={updatePlaybackTime}\n            onDurationChange={updatePlaybackTime}\n            onPlayerLoaded={player => {\n              playerRef.current = player\n              onPlayerLoaded?.(player)\n              syncVolume(videoRef.current, videoProps.muted ? 0 : appVolume)\n            }}\n          />\n        }\n        containerRef={element => {\n          containerRef.current = element\n          observe(element)\n        }}\n        backRef={adContainerRef}\n        title={title}\n        channelTitle={channelTitle}\n        backButton={onBack && <Button startIcon=\"back\" onClick={onBack} />}\n        {...(isUserActive && uiElements)}\n        onClick={onClick}\n        onMouseMove={onMouseMove}\n      >\n        <LoadingSpinner active={waiting} />\n        {errorData && <Error error={errorData} onBack={onBack} />}\n        {children}\n        {/* TODO sync Cast last played time back */}\n        {isUserActive && (\n          <components.CastUi\n            onBack={onBack}\n            onStateChange={setCastState}\n            onLoad={onCast}\n            {...compatibilityCastProps}\n            {...slotProps.castUi}\n          />\n        )}\n        {isUserActive && ( // TODO: Design <Extenstion /> flag\n          <components.Settings\n            type={uiType}\n            sections={settings.sections}\n            // TODO hasBottomPanel bottom: 8em\n            open={userFocus === 'settings'}\n            values={settings.values}\n            onOpen={openSettings}\n            onChange={uiHandlers.onChangeSettings}\n            {...slotProps?.settings}\n            onClose={uiHandlers.onCloseSettings}\n          />\n        )}\n        <LanguageSettings\n          uiType={uiType}\n          getPlayer={() => playerRef.current}\n          onChange={(_event, item) =>\n            uiHandlers.onChangeSettings({\n              name: item.type,\n              value: item.value,\n              keepOpen: true,\n            })\n          }\n          {...slotProps.languageSettings}\n        />\n        <Backdrop open={!playbackState || playbackState === 'loading'}>\n          <LoadingSpinner />\n        </Backdrop>\n      </Layout>\n    </IntlProvider>\n  )\n}\n\nexport default PremiumPlayer\n"]} */")
|
|
8801
|
+
default: /*#__PURE__*/css$1(LanguageMenu.styles, process.env.NODE_ENV === "production" ? "" : ";label:default;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["PremiumPlayer.js"],"names":[],"mappings":"AAuEW","file":"PremiumPlayer.js","sourcesContent":["/* eslint-disable no-nested-ternary */\nimport {useEffect, useMemo, useRef, useState} from 'react'\nimport {css} from '@emotion/css'\nimport useDimensions from 'react-cool-dimensions'\nimport {ResizeObserver} from '@juggle/resize-observer'\n\nimport {handleError} from 'playerCore/errors'\nimport {on} from 'util/events'\nimport {isDesktop} from 'util/environment'\nimport multiRef from 'util/multiRef'\nimport {onViewModeChange, toggleFullscreen} from 'util/viewModes'\nimport {useAutoHide} from 'hooks'\nimport {IntlProvider} from 'context/I18n'\nimport {getMediaTime, isBuffered} from 'playerCore/mediaBindings'\nimport Error from 'playerUi/Error'\nimport Layout from 'playerUi/DefaultLayout'\nimport {\n  Button,\n  LiveButton,\n  FullscreenButton,\n  PlayButton,\n  ForwardButton,\n} from 'playerUi/buttons'\nimport Seekbar from 'playerUi/Seekbar'\nimport DisplayTime from 'playerUi/DisplayTime'\nimport LoadingSpinner from 'playerUi/LoadingSpinner'\nimport Backdrop from 'playerUi/Backdrop'\nimport Settings from 'playerUi/Settings'\nimport OverlayPanel from 'playerUi/OverlayPanel'\nimport LanguageMenu from 'playerUi/LanguageMenu'\nimport PlayPanel from 'playerUi/PlayPane'\nimport VolumeControl from 'component/VolumeControl'\nimport CastUi from 'cast/CastUi'\nimport {linkCast} from 'cast/framework'\nimport {CastState} from 'Enum'\nimport {dispatchChapterEvents} from './timeline'\nimport Video from '../Video'\nimport {getLanguageOptions, getQualityOptions, getSettingsData} from './settings'\nimport {linkMediaVolume, syncVolume} from './volume'\nimport SeekPreview from './SeekPreview'\n\nconst sizes = {\n  'small-embed': 200,\n  embed: 400,\n  'tablet-portrait': 600,\n  'tablet-landscape': 900,\n  desktop: 1200,\n}\n\nconst useLinkState = (request, dependencies = []) => {\n  const [state, setState] = useState()\n  useEffect(() => {\n    request(setState)\n  }, dependencies)\n  return state\n}\n\nconst flipState = state => (state === 'playing' ? 'paused' : 'playing')\n\n// FIXME: too few lines to split a file, looking a better place\nconst getThumbnailsUrl = source =>\n  [].concat(source).find(item => item.type === 'thumbnail')?.src\n\nconst defaultSettings = {}\n\nconst shouldPause = ({userFocus, uiType}) =>\n  (uiType === 'mobile' && /settings|chapter-list/.test(userFocus)) ||\n  /blocking|seekbar|share-inner/.test(userFocus)\n\n// TODO extract to somewhere else\nconst menuClasses = {\n  default: css(LanguageMenu.styles),\n  desktop: css(LanguageMenu.styles, LanguageMenu.desktopStyles),\n}\n\nconst LanguageSettings = ({\n  uiType,\n  audioTracks = [],\n  textTracks = [],\n  getPlayer,\n  onChange,\n  slots = {LanguageMenu},\n  ...rest\n}) => {\n  const options = getLanguageOptions(getPlayer(), {audioTracks, textTracks})\n  return (\n    audioTracks.length > 0 &&\n    textTracks.length > 0 && (\n      <OverlayPanel\n        uiType={uiType}\n        title=\"字幕・音声を切り替える\"\n        buttonIcon=\"subtitle\"\n      >\n      <slots.LanguageMenu\n        uiType={uiType}\n        classes={menuClasses}\n        sectionOptions={[options.audioTracks, options.textTracks]}\n        onChange={onChange}\n        {...rest}\n      />\n    </OverlayPanel>\n    )\n  )\n}\n\nconst PremiumPlayer = ({\n  source,\n  startTime,\n  quality = {},\n  title,\n  metadata,\n  channelTitle,\n  chapters,\n  playbackState: appPlaybackState,\n  currentTime: appCurrentTime,\n  playbackRate: appPlaybackRate,\n  loop,\n  volume: appVolume,\n  audioTrack: appAudioTrack,\n  textTrack: appTextTrack,\n  thumbnailsUrl,\n  controls = {autohide: 3000},\n  marks = [],\n  // TODO sectionId\n  intl,\n  settings: appSettings = defaultSettings,\n  blocking,\n  plugins = [],\n  modulesConfig,\n\n  uiType = isDesktop() ? 'desktop' : 'mobile',\n  style,\n  children,\n  uiMode = 'standalone',\n  cast: castOptions = 'compatible',\n  slots,\n  slotProps = {},\n  onError,\n  onPlaybackStateChange,\n  onBack,\n  onCast,\n  onChangeNext,\n  onChangePrevious,\n  onOpenSettings,\n  onChangeSettings,\n  onPlayerLoaded,\n  onBlockedAutoplay,\n  onPlaylogFired,\n  ...videoProps\n}) => {\n  const components = {\n    VolumeControl,\n    CastUi,\n    DisplayTime,\n    LiveButton,\n    Seekbar,\n    Settings,\n    ...slots,\n  }\n  const videoRef = useRef()\n  const containerRef = useRef()\n  const playerRef = useRef()\n  const adContainerRef = useRef()\n  const refs = useRef({})\n  // TODO move RWD related to Layout\n  const {\n    currentBreakpoint: size,\n    width,\n    observe,\n  } = useDimensions({\n    polyfill: ResizeObserver,\n    breakpoints: sizes,\n  })\n  const [isUserActive, setIsUserActive] = useState(\n    uiMode === 'standalone' ? true : videoProps.autoplay\n  )\n  const [userFocus, setUserFocus] = useState('')\n  const [targetState, setTargetState] = useState(() => ({\n    playbackState:\n      appPlaybackState || (videoProps.autoplay ? 'playing' : 'paused'),\n    currentTime: startTime,\n  }))\n  const [playbackTime, setPlaybackTime] = useState({\n    currentTime: 0,\n    bufferTime: 0,\n  })\n  const togglePlay = (overrideState, {triggeredBy} = {}) => {\n    if (targetState.playbackState !== overrideState) {\n      setTargetState(state => ({\n        ...state,\n        playbackState: overrideState || flipState(state.playbackState),\n        animation:\n          triggeredBy === 'user-action' && flipState(state.playbackState),\n      }))\n    }\n  }\n\n  useEffect(() => {\n    if (appPlaybackState) {\n      togglePlay(appPlaybackState)\n    }\n  }, [appPlaybackState])\n  useEffect(() => {\n    if (uiType === 'mobile') {\n      on(document, 'visibilitychange', () => togglePlay('paused'))\n    }\n  }, [uiType, targetState.playbackState])\n  const updatePlaybackTime = event =>\n    requestAnimationFrame(\n      () =>\n        (event?.type !== 'timeupdate' || isBuffered(videoRef.current)) &&\n        setPlaybackTime(state => ({\n          ...state,\n          ...getMediaTime(videoRef.current, {\n            player: playerRef.current,\n            plugins,\n          }),\n          ...(event?.type === 'durationchange' && {\n            currentTime: state.currentTime,\n          }),\n        }))\n    )\n  const setTargetTime = time => {\n    const trimmed = Math.min(\n      time,\n      videoRef.current?.initialDuration || Infinity\n    )\n    setTargetState(state => ({\n      ...state,\n      // seek to 0 repeatedly edge case\n      currentTime: state.currentTime !== trimmed ? trimmed : trimmed + 0.01,\n    }))\n    updatePlaybackTime()\n  }\n  const [playbackState, setPlaybackState] = useState('init')\n  const [castState, setCastState] = useState('')\n  useEffect(() => {\n    if (typeof appCurrentTime === 'number') setTargetTime(appCurrentTime || 0)\n    if (typeof appCurrentTime === 'object')\n      setTargetTime(appCurrentTime?.value || 0)\n  }, [appCurrentTime])\n  const [errorData, setErrorData] = useState()\n  const errorHandler = reactEvent => {\n    onError?.(reactEvent.nativeEvent)\n    handleError(reactEvent, {\n      media: videoRef.current,\n      displayError: data => {\n        setPlaybackState('error')\n        setErrorData(current => (current?.code ? current : data))\n      },\n    })\n  }\n  const isLive = playbackTime.streamType === 'live'\n  const [settings, setSettings] = useState(() => ({\n    sections: [],\n    values: {speed: 1},\n  }))\n  const fetchSettings = () => {\n    if (!playerRef.current) {\n      return\n    }\n    const speedItems = appSettings.speedItems || (isLive ? [] : undefined)\n    setSettings(current => {\n      const {values, sections} = getSettingsData({\n        media: videoRef.current,\n        player: playerRef.current,\n        source,\n        quality,\n        speedItems,\n        loop,\n        preferred: current.preferred,\n        otherSections: appSettings.sections,\n      })\n      return {preferred: current.preferred, values, sections}\n    })\n  }\n  const lastState = useRef(playbackState)\n  const handlePlaybackStateChange = (event, state) => {\n    if (lastState.current === 'error') {\n      return\n    }\n    onPlaybackStateChange?.(event, state)\n    // previously UI playback state was synced only when\n    // exiting iOS video only fullscreen(webkitendfullscreen)\n    if (\n      /playing|paused/.test(state) &&\n      !(uiType === 'mobile' && blocking) &&\n      !shouldPause({userFocus, uiType})\n    ) {\n      togglePlay(state)\n    }\n    if (state === 'ended') {\n      togglePlay('paused')\n    }\n    if (state === 'loading' && lastState.current !== 'init') {\n      setTimeout(() => togglePlay('playing'), 1)\n    }\n    if (state === 'playing') {\n      setIsUserActive(true)\n    }\n    if (lastState.current === 'loading') {\n      fetchSettings()\n    }\n    lastState.current = state\n    setPlaybackState(state)\n  }\n\n  const changeSettings = (name, value, {keepOpen} = {}) => {\n    // TODO consider merge into useReducer?\n    onChangeSettings?.({name, value})\n    setTargetTime(playbackTime.currentTime)\n    setSettings(current => ({\n      ...current,\n      values: {...current.values, [name]: value},\n      preferred: {...current.preferred, [name]: value},\n    }))\n    if (!keepOpen) {\n      setUserFocus('')\n    }\n  }\n  const openSettings = event => {\n    const animationFrame =\n      userFocus !== 'settings' &&\n      requestAnimationFrame(() => {\n        onOpenSettings?.(event, settings)\n        // In iOS Safari, we need to update settings data\n        fetchSettings()\n      })\n    setUserFocus(current => (current === 'settings' ? '' : 'settings'))\n    return animationFrame\n  }\n  useEffect(() => {\n    if (appPlaybackRate > 0) {\n      setSettings(current => ({\n        ...current,\n        values: {...current.values, speed: appPlaybackRate},\n      }))\n    }\n  }, [appPlaybackRate])\n\n  const qualityOptions = useMemo(\n    () => getQualityOptions(settings),\n    [settings.values.quality]\n  )\n  const viewMode = useLinkState(update =>\n    onViewModeChange(videoRef.current, update)\n  )\n  const sourceOverride = useLinkState(\n    async update => {\n      const result =\n        source && (await quality.rewriteManifest?.(source, qualityOptions))\n      update(result || source)\n    },\n    [source, qualityOptions]\n  )\n  const waiting = /emptied|loading|buffering/.test(playbackState)\n  const activePlayback =\n    playbackState === 'playing' || playbackState === 'waiting'\n  const {\n    mode: autoHideMode,\n    onClick,\n    onMouseMove,\n  } = useAutoHide({\n    pinned: !controls.autohide || waiting || !activePlayback || userFocus,\n    tapToHide: uiType === 'mobile',\n    hideTimeMs: controls.autohide,\n  })\n  const mode =\n    userFocus === 'seekbar'\n      ? 'hidden'\n      : controls.autohide\n      ? autoHideMode\n      : controls\n      ? 'shown'\n      : 'hidden'\n  const controlsDisplay =\n    userFocus === 'seekbar'\n      ? 'seekbar-only'\n      : controls === 'title-only' || playbackState === 'emptied'\n      ? 'hidden'\n      : mode\n  const shouldHidePanels =\n    (controls === 'no-panel' ||\n      controlsDisplay === 'hidden' ||\n      playbackState === 'ended') &&\n    userFocus\n  const havePlayPanel =\n    !blocking &&\n    uiType === 'desktop' &&\n    !waiting &&\n    !/volume|''/.test(userFocus) &&\n    !/title-only|no-panel/.test(controls) &&\n    (controls.autohide || mode === 'shown')\n\n  useEffect(() => {\n    if (shouldHidePanels) {\n      setUserFocus('')\n      if (refs.current.closeMenu) {\n        refs.current.closeMenu()\n        refs.current.closeMenu = undefined\n      }\n    }\n  }, [shouldHidePanels])\n  useEffect(() => \n    on(containerRef.current, 'focus-menu', event => {\n      if (event.detail?.status === 'open') {\n        setUserFocus('menu')\n        refs.current.closeMenu = () => event.detail.closeMenu?.()\n      }\n      if (event.detail?.status === 'closed') {\n        setUserFocus(current => (current === 'menu' ? '' : current))\n      }\n    })\n  , [])\n  const {subscribe, onChange, toggleMute} = linkMediaVolume(() => ({\n    video: videoRef.current,\n    getPlayer: () => playerRef.current,\n  }))\n  const changePrevious = event => {\n    onChangePrevious(event)\n    togglePlay('paused')\n    videoRef.current.dispatchEvent(new CustomEvent('loadstart'))\n  }\n  const changeNext = event => {\n    onChangeNext(event)\n    togglePlay('paused')\n    videoRef.current.dispatchEvent(new CustomEvent('loadstart'))\n  }\n\n  useEffect(fetchSettings, [appSettings])\n  useEffect(() => {\n    // The adContainer should be set before `load` because ImaDai.load needs it.\n    plugins.forEach(plugin => plugin.setAdContainer?.(adContainerRef.current))\n  }, [])\n\n  useEffect(\n    () =>\n      dispatchChapterEvents({\n        media: videoRef.current,\n        chapters,\n        getTime: () => videoRef.current.currentTime,\n      }),\n    [chapters]\n  )\n\n  const isEnd = playbackState === 'ended'\n  const canSeek = !isEnd && playbackTime.duration > 0\n  const derivedPlaybackState =\n    (uiType === 'mobile' && blocking) || shouldPause({userFocus, uiType})\n      ? 'paused'\n      : targetState.playbackState\n  const activeThumbnailsUrl =\n    (!userFocus || userFocus === 'seekbar') &&\n    (getThumbnailsUrl(source) || thumbnailsUrl)\n  const cssVariables = {\n    '--playing': playbackState === 'playing' ? '1' : '0',\n  }\n  const onRewind =\n    (!isLive || playbackTime.currentTime < 0) &&\n    canSeek &&\n    (() => setTargetTime(playbackTime.currentTime - 10, 'rewind'))\n  const onForward =\n    (!isLive || playbackTime.currentTime < 0) &&\n    canSeek &&\n    (() => setTargetTime(playbackTime.currentTime + 10, 'forward'))\n  const onClickBlank =\n    havePlayPanel && (() => togglePlay('', {triggeredBy: 'user-action'}))\n\n  const uiHandlers = {\n    onPlay: () => {\n      togglePlay('', {triggeredBy: 'user-action'})\n      setIsUserActive(true)\n    },\n    onClickLive: isLive && (() => setTargetTime(0, 'seekToLive')),\n    onClickSeekbar: () => {\n      refs.current.deferedSeekFocus = setTimeout(\n        () => setUserFocus('seekbar'),\n        66\n      )\n    },\n    onSeek: time => {\n      setTargetTime(time, 'seek')\n      clearTimeout(refs.current.deferedSeekFocus)\n      setTimeout(() => setUserFocus(''), 66)\n    },\n    onChangeSettings: ({name, value, keepOpen}) =>\n      changeSettings(name, value, {keepOpen}),\n    onCloseSettings: event => {\n      setUserFocus('')\n      slotProps?.settings?.onClose?.(event)\n    },\n  }\n\n  const uiElements = {\n    liveButton: uiHandlers.onClickLive && (\n      <components.LiveButton\n        usingStartOver={playbackTime.currentTime < 0}\n        onClick={uiHandlers.onClickLive}\n      />\n    ),\n    controlButtons: {\n      playButton: (\n        <PlayButton\n          playbackState={derivedPlaybackState}\n          ended={isEnd}\n          hidden={\n            uiType === 'mobile' && (waiting || /loading/.test(playbackState))\n          }\n          variant={!isUserActive && 'firstplay'}\n          onClick={uiHandlers.onPlay}\n        />\n      ),\n      rewindButton: onRewind && (\n        <Button\n          startIcon=\"rewind10\"\n          title=\"KKS.PLAYER.REWIND\"\n          disabled={isEnd}\n          onClick={onRewind}\n        />\n      ),\n      forwardButton: onForward && (\n        <ForwardButton\n          startIcon=\"forward10\"\n          title=\"KKS.PLAYER.FORWARD\"\n          disabled={!canSeek}\n          onClick={onForward}\n        />\n      ),\n      nextEpisodeButton: (\n        <Button\n          startIcon=\"next\"\n          title=\"KKS.PLAYER.NEXT\"\n          disabled={!onChangeNext}\n          onClick={changeNext}\n        />\n      ),\n      previousEpisodeButton: (\n        <Button\n          startIcon=\"previous\"\n          title=\"KKS.PLAYER.PREVIOUS\"\n          disabled={!onChangePrevious}\n          onClick={changePrevious}\n        />\n      ),\n    },\n    seekbar: (\n      <components.Seekbar\n        style={/volume/.test(userFocus) && {opacity: '0'}}\n        startTime={playbackTime.startTime}\n        currentTime={playbackTime.currentTime}\n        bufferTime={playbackTime.bufferTime}\n        duration={playbackTime.duration}\n        chapters={chapters}\n        onChange={uiHandlers.onClickSeekbar}\n        onChangeCommitted={uiHandlers.onSeek}\n        marks={marks}\n        plugins={plugins}\n        {...slotProps.seekbar}\n      >\n        {activeThumbnailsUrl && (\n          <SeekPreview\n            thumbnailsUrl={activeThumbnailsUrl}\n            duration={playbackTime.duration}\n            chapters={chapters}\n          />\n        )}\n      </components.Seekbar>\n    ),\n    displayTime: <components.DisplayTime {...playbackTime} />,\n    fullscreenButton: (\n      <FullscreenButton\n        viewMode={viewMode}\n        onClick={() => toggleFullscreen(containerRef.current)}\n      />\n    ),\n    volumeControl: width >= sizes['small-embed'] && (\n      <components.VolumeControl\n        {...{subscribe, onChange, toggleMute}}\n        onMouseOver={() => setUserFocus('volume')}\n        onMouseOut={() => setUserFocus('')}\n        {...slotProps?.volumeControl}\n      />\n    ),\n    backItems: (\n      <PlayPanel animation={targetState.animation} onClick={onClickBlank} />\n    ),\n  }\n  const compatibilityCastProps = castOptions === 'compatible' && {\n    source,\n    values: {title, ...settings.values, ...intl, modulesConfig},\n    onLoad: onCast,\n  }\n\n  useEffect(() => {\n    if (castOptions && castOptions !== 'compatible') {\n      return linkCast({source, ...castOptions})\n    } // TODO playlist, audio / subtitle setting\n  }, [source, castOptions.contentId, castOptions.customData])\n  const selectedAudioTrack = settings.preferred?.audio || appAudioTrack\n  const selectedTextTrack = settings.preferred?.subtitles || appTextTrack\n\n  return (\n    <IntlProvider {...intl}>\n      <Layout\n        style={{...style, cssVariables}}\n        type={uiType}\n        display={mode}\n        controlsDisplay={controlsDisplay}\n        size={size}\n        video={\n          <Video\n            {...videoProps}\n            videoRef={multiRef(videoRef, videoProps.videoRef)}\n            source={\n              castState !== CastState.CONNECTED &&\n              playbackState !== 'error' &&\n              sourceOverride\n            }\n            playbackState={derivedPlaybackState}\n            currentTime={targetState.currentTime}\n            {...(appVolume >= 0 && {volume: appVolume, muted: appVolume <= 0})}\n            playbackRate={settings.values.speed}\n            videoResolution={qualityOptions}\n            audioTrack={selectedAudioTrack}\n            textTrack={selectedTextTrack}\n            loop={settings.values.loop}\n            plugins={plugins}\n            modulesConfig={modulesConfig}\n            onPlaylogFired={onPlaylogFired}\n            onError={errorHandler}\n            onPlaybackStateChange={handlePlaybackStateChange}\n            onBlockedAutoplay={error => onBlockedAutoplay?.(error)}\n            onCanPlay={event => {\n              updatePlaybackTime(event)\n              videoProps.onCanPlay?.(event)\n            }}\n            onTimeUpdate={updatePlaybackTime}\n            onDurationChange={updatePlaybackTime}\n            onPlayerLoaded={player => {\n              playerRef.current = player\n              onPlayerLoaded?.(player)\n              syncVolume(videoRef.current, videoProps.muted ? 0 : appVolume)\n            }}\n          />\n        }\n        containerRef={element => {\n          containerRef.current = element\n          observe(element)\n        }}\n        backRef={adContainerRef}\n        title={title}\n        channelTitle={channelTitle}\n        backButton={onBack && <Button startIcon=\"back\" onClick={onBack} />}\n        {...(isUserActive && uiElements)}\n        onClick={onClick}\n        onMouseMove={onMouseMove}\n      >\n        <LoadingSpinner active={waiting} />\n        {errorData && <Error error={errorData} onBack={onBack} />}\n        {children}\n        {/* TODO sync Cast last played time back */}\n        {isUserActive && (\n          <components.CastUi\n            onBack={onBack}\n            onStateChange={setCastState}\n            onLoad={onCast}\n            {...compatibilityCastProps}\n            {...slotProps.castUi}\n          />\n        )}\n        {isUserActive && ( // TODO: Design <Extenstion /> flag\n          <components.Settings\n            type={uiType}\n            sections={settings.sections}\n            // TODO hasBottomPanel bottom: 8em\n            open={userFocus === 'settings'}\n            values={settings.values}\n            onOpen={openSettings}\n            onChange={uiHandlers.onChangeSettings}\n            {...slotProps?.settings}\n            onClose={uiHandlers.onCloseSettings}\n          />\n        )}\n        <LanguageSettings\n          uiType={uiType}\n          getPlayer={() => playerRef.current}\n          onChange={(_event, item) =>\n            uiHandlers.onChangeSettings({\n              name: item.type,\n              value: item.value,\n              keepOpen: true,\n            })\n          }\n          {...slotProps.languageSettings}\n        />\n        <Backdrop open={!playbackState || playbackState === 'loading'}>\n          <LoadingSpinner />\n        </Backdrop>\n      </Layout>\n    </IntlProvider>\n  )\n}\n\nexport default PremiumPlayer\n"]} */"),
|
|
8802
|
+
desktop: /*#__PURE__*/css$1(LanguageMenu.styles, LanguageMenu.desktopStyles, process.env.NODE_ENV === "production" ? "" : ";label:desktop;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["PremiumPlayer.js"],"names":[],"mappings":"AAwEW","file":"PremiumPlayer.js","sourcesContent":["/* eslint-disable no-nested-ternary */\nimport {useEffect, useMemo, useRef, useState} from 'react'\nimport {css} from '@emotion/css'\nimport useDimensions from 'react-cool-dimensions'\nimport {ResizeObserver} from '@juggle/resize-observer'\n\nimport {handleError} from 'playerCore/errors'\nimport {on} from 'util/events'\nimport {isDesktop} from 'util/environment'\nimport multiRef from 'util/multiRef'\nimport {onViewModeChange, toggleFullscreen} from 'util/viewModes'\nimport {useAutoHide} from 'hooks'\nimport {IntlProvider} from 'context/I18n'\nimport {getMediaTime, isBuffered} from 'playerCore/mediaBindings'\nimport Error from 'playerUi/Error'\nimport Layout from 'playerUi/DefaultLayout'\nimport {\n  Button,\n  LiveButton,\n  FullscreenButton,\n  PlayButton,\n  ForwardButton,\n} from 'playerUi/buttons'\nimport Seekbar from 'playerUi/Seekbar'\nimport DisplayTime from 'playerUi/DisplayTime'\nimport LoadingSpinner from 'playerUi/LoadingSpinner'\nimport Backdrop from 'playerUi/Backdrop'\nimport Settings from 'playerUi/Settings'\nimport OverlayPanel from 'playerUi/OverlayPanel'\nimport LanguageMenu from 'playerUi/LanguageMenu'\nimport PlayPanel from 'playerUi/PlayPane'\nimport VolumeControl from 'component/VolumeControl'\nimport CastUi from 'cast/CastUi'\nimport {linkCast} from 'cast/framework'\nimport {CastState} from 'Enum'\nimport {dispatchChapterEvents} from './timeline'\nimport Video from '../Video'\nimport {getLanguageOptions, getQualityOptions, getSettingsData} from './settings'\nimport {linkMediaVolume, syncVolume} from './volume'\nimport SeekPreview from './SeekPreview'\n\nconst sizes = {\n  'small-embed': 200,\n  embed: 400,\n  'tablet-portrait': 600,\n  'tablet-landscape': 900,\n  desktop: 1200,\n}\n\nconst useLinkState = (request, dependencies = []) => {\n  const [state, setState] = useState()\n  useEffect(() => {\n    request(setState)\n  }, dependencies)\n  return state\n}\n\nconst flipState = state => (state === 'playing' ? 'paused' : 'playing')\n\n// FIXME: too few lines to split a file, looking a better place\nconst getThumbnailsUrl = source =>\n  [].concat(source).find(item => item.type === 'thumbnail')?.src\n\nconst defaultSettings = {}\n\nconst shouldPause = ({userFocus, uiType}) =>\n  (uiType === 'mobile' && /settings|chapter-list/.test(userFocus)) ||\n  /blocking|seekbar|share-inner/.test(userFocus)\n\n// TODO extract to somewhere else\nconst menuClasses = {\n  default: css(LanguageMenu.styles),\n  desktop: css(LanguageMenu.styles, LanguageMenu.desktopStyles),\n}\n\nconst LanguageSettings = ({\n  uiType,\n  audioTracks = [],\n  textTracks = [],\n  getPlayer,\n  onChange,\n  slots = {LanguageMenu},\n  ...rest\n}) => {\n  const options = getLanguageOptions(getPlayer(), {audioTracks, textTracks})\n  return (\n    audioTracks.length > 0 &&\n    textTracks.length > 0 && (\n      <OverlayPanel\n        uiType={uiType}\n        title=\"字幕・音声を切り替える\"\n        buttonIcon=\"subtitle\"\n      >\n      <slots.LanguageMenu\n        uiType={uiType}\n        classes={menuClasses}\n        sectionOptions={[options.audioTracks, options.textTracks]}\n        onChange={onChange}\n        {...rest}\n      />\n    </OverlayPanel>\n    )\n  )\n}\n\nconst PremiumPlayer = ({\n  source,\n  startTime,\n  quality = {},\n  title,\n  metadata,\n  channelTitle,\n  chapters,\n  playbackState: appPlaybackState,\n  currentTime: appCurrentTime,\n  playbackRate: appPlaybackRate,\n  loop,\n  volume: appVolume,\n  audioTrack: appAudioTrack,\n  textTrack: appTextTrack,\n  thumbnailsUrl,\n  controls = {autohide: 3000},\n  marks = [],\n  // TODO sectionId\n  intl,\n  settings: appSettings = defaultSettings,\n  blocking,\n  plugins = [],\n  modulesConfig,\n\n  uiType = isDesktop() ? 'desktop' : 'mobile',\n  style,\n  children,\n  uiMode = 'standalone',\n  cast: castOptions = 'compatible',\n  slots,\n  slotProps = {},\n  onError,\n  onPlaybackStateChange,\n  onBack,\n  onCast,\n  onChangeNext,\n  onChangePrevious,\n  onOpenSettings,\n  onChangeSettings,\n  onPlayerLoaded,\n  onBlockedAutoplay,\n  onPlaylogFired,\n  ...videoProps\n}) => {\n  const components = {\n    VolumeControl,\n    CastUi,\n    DisplayTime,\n    LiveButton,\n    Seekbar,\n    Settings,\n    ...slots,\n  }\n  const videoRef = useRef()\n  const containerRef = useRef()\n  const playerRef = useRef()\n  const adContainerRef = useRef()\n  const refs = useRef({})\n  // TODO move RWD related to Layout\n  const {\n    currentBreakpoint: size,\n    width,\n    observe,\n  } = useDimensions({\n    polyfill: ResizeObserver,\n    breakpoints: sizes,\n  })\n  const [isUserActive, setIsUserActive] = useState(\n    uiMode === 'standalone' ? true : videoProps.autoplay\n  )\n  const [userFocus, setUserFocus] = useState('')\n  const [targetState, setTargetState] = useState(() => ({\n    playbackState:\n      appPlaybackState || (videoProps.autoplay ? 'playing' : 'paused'),\n    currentTime: startTime,\n  }))\n  const [playbackTime, setPlaybackTime] = useState({\n    currentTime: 0,\n    bufferTime: 0,\n  })\n  const togglePlay = (overrideState, {triggeredBy} = {}) => {\n    if (targetState.playbackState !== overrideState) {\n      setTargetState(state => ({\n        ...state,\n        playbackState: overrideState || flipState(state.playbackState),\n        animation:\n          triggeredBy === 'user-action' && flipState(state.playbackState),\n      }))\n    }\n  }\n\n  useEffect(() => {\n    if (appPlaybackState) {\n      togglePlay(appPlaybackState)\n    }\n  }, [appPlaybackState])\n  useEffect(() => {\n    if (uiType === 'mobile') {\n      on(document, 'visibilitychange', () => togglePlay('paused'))\n    }\n  }, [uiType, targetState.playbackState])\n  const updatePlaybackTime = event =>\n    requestAnimationFrame(\n      () =>\n        (event?.type !== 'timeupdate' || isBuffered(videoRef.current)) &&\n        setPlaybackTime(state => ({\n          ...state,\n          ...getMediaTime(videoRef.current, {\n            player: playerRef.current,\n            plugins,\n          }),\n          ...(event?.type === 'durationchange' && {\n            currentTime: state.currentTime,\n          }),\n        }))\n    )\n  const setTargetTime = time => {\n    const trimmed = Math.min(\n      time,\n      videoRef.current?.initialDuration || Infinity\n    )\n    setTargetState(state => ({\n      ...state,\n      // seek to 0 repeatedly edge case\n      currentTime: state.currentTime !== trimmed ? trimmed : trimmed + 0.01,\n    }))\n    updatePlaybackTime()\n  }\n  const [playbackState, setPlaybackState] = useState('init')\n  const [castState, setCastState] = useState('')\n  useEffect(() => {\n    if (typeof appCurrentTime === 'number') setTargetTime(appCurrentTime || 0)\n    if (typeof appCurrentTime === 'object')\n      setTargetTime(appCurrentTime?.value || 0)\n  }, [appCurrentTime])\n  const [errorData, setErrorData] = useState()\n  const errorHandler = reactEvent => {\n    onError?.(reactEvent.nativeEvent)\n    handleError(reactEvent, {\n      media: videoRef.current,\n      displayError: data => {\n        setPlaybackState('error')\n        setErrorData(current => (current?.code ? current : data))\n      },\n    })\n  }\n  const isLive = playbackTime.streamType === 'live'\n  const [settings, setSettings] = useState(() => ({\n    sections: [],\n    values: {speed: 1},\n  }))\n  const fetchSettings = () => {\n    if (!playerRef.current) {\n      return\n    }\n    const speedItems = appSettings.speedItems || (isLive ? [] : undefined)\n    setSettings(current => {\n      const {values, sections} = getSettingsData({\n        media: videoRef.current,\n        player: playerRef.current,\n        source,\n        quality,\n        speedItems,\n        loop,\n        preferred: current.preferred,\n        otherSections: appSettings.sections,\n      })\n      return {preferred: current.preferred, values, sections}\n    })\n  }\n  const lastState = useRef(playbackState)\n  const handlePlaybackStateChange = (event, state) => {\n    if (lastState.current === 'error') {\n      return\n    }\n    onPlaybackStateChange?.(event, state)\n    // previously UI playback state was synced only when\n    // exiting iOS video only fullscreen(webkitendfullscreen)\n    if (\n      /playing|paused/.test(state) &&\n      !(uiType === 'mobile' && blocking) &&\n      !shouldPause({userFocus, uiType})\n    ) {\n      togglePlay(state)\n    }\n    if (state === 'ended') {\n      togglePlay('paused')\n    }\n    if (state === 'loading' && lastState.current !== 'init') {\n      setTimeout(() => togglePlay('playing'), 1)\n    }\n    if (state === 'playing') {\n      setIsUserActive(true)\n    }\n    if (lastState.current === 'loading') {\n      fetchSettings()\n    }\n    lastState.current = state\n    setPlaybackState(state)\n  }\n\n  const changeSettings = (name, value, {keepOpen} = {}) => {\n    // TODO consider merge into useReducer?\n    onChangeSettings?.({name, value})\n    setTargetTime(playbackTime.currentTime)\n    setSettings(current => ({\n      ...current,\n      values: {...current.values, [name]: value},\n      preferred: {...current.preferred, [name]: value},\n    }))\n    if (!keepOpen) {\n      setUserFocus('')\n    }\n  }\n  const openSettings = event => {\n    const animationFrame =\n      userFocus !== 'settings' &&\n      requestAnimationFrame(() => {\n        onOpenSettings?.(event, settings)\n        // In iOS Safari, we need to update settings data\n        fetchSettings()\n      })\n    setUserFocus(current => (current === 'settings' ? '' : 'settings'))\n    return animationFrame\n  }\n  useEffect(() => {\n    if (appPlaybackRate > 0) {\n      setSettings(current => ({\n        ...current,\n        values: {...current.values, speed: appPlaybackRate},\n      }))\n    }\n  }, [appPlaybackRate])\n\n  const qualityOptions = useMemo(\n    () => getQualityOptions(settings),\n    [settings.values.quality]\n  )\n  const viewMode = useLinkState(update =>\n    onViewModeChange(videoRef.current, update)\n  )\n  const sourceOverride = useLinkState(\n    async update => {\n      const result =\n        source && (await quality.rewriteManifest?.(source, qualityOptions))\n      update(result || source)\n    },\n    [source, qualityOptions]\n  )\n  const waiting = /emptied|loading|buffering/.test(playbackState)\n  const activePlayback =\n    playbackState === 'playing' || playbackState === 'waiting'\n  const {\n    mode: autoHideMode,\n    onClick,\n    onMouseMove,\n  } = useAutoHide({\n    pinned: !controls.autohide || waiting || !activePlayback || userFocus,\n    tapToHide: uiType === 'mobile',\n    hideTimeMs: controls.autohide,\n  })\n  const mode =\n    userFocus === 'seekbar'\n      ? 'hidden'\n      : controls.autohide\n      ? autoHideMode\n      : controls\n      ? 'shown'\n      : 'hidden'\n  const controlsDisplay =\n    userFocus === 'seekbar'\n      ? 'seekbar-only'\n      : controls === 'title-only' || playbackState === 'emptied'\n      ? 'hidden'\n      : mode\n  const shouldHidePanels =\n    (controls === 'no-panel' ||\n      controlsDisplay === 'hidden' ||\n      playbackState === 'ended') &&\n    userFocus\n  const havePlayPanel =\n    !blocking &&\n    uiType === 'desktop' &&\n    !waiting &&\n    !/volume|''/.test(userFocus) &&\n    !/title-only|no-panel/.test(controls) &&\n    (controls.autohide || mode === 'shown')\n\n  useEffect(() => {\n    if (shouldHidePanels) {\n      setUserFocus('')\n      if (refs.current.closeMenu) {\n        refs.current.closeMenu()\n        refs.current.closeMenu = undefined\n      }\n    }\n  }, [shouldHidePanels])\n  useEffect(() => \n    on(containerRef.current, 'focus-menu', event => {\n      if (event.detail?.status === 'open') {\n        setUserFocus('menu')\n        refs.current.closeMenu = () => event.detail.closeMenu?.()\n      }\n      if (event.detail?.status === 'closed') {\n        setUserFocus(current => (current === 'menu' ? '' : current))\n      }\n    })\n  , [])\n  const {subscribe, onChange, toggleMute} = linkMediaVolume(() => ({\n    video: videoRef.current,\n    getPlayer: () => playerRef.current,\n  }))\n  const changePrevious = event => {\n    onChangePrevious(event)\n    togglePlay('paused')\n    videoRef.current.dispatchEvent(new CustomEvent('loadstart'))\n  }\n  const changeNext = event => {\n    onChangeNext(event)\n    togglePlay('paused')\n    videoRef.current.dispatchEvent(new CustomEvent('loadstart'))\n  }\n\n  useEffect(fetchSettings, [appSettings])\n  useEffect(() => {\n    // The adContainer should be set before `load` because ImaDai.load needs it.\n    plugins.forEach(plugin => plugin.setAdContainer?.(adContainerRef.current))\n  }, [])\n\n  useEffect(\n    () =>\n      dispatchChapterEvents({\n        media: videoRef.current,\n        chapters,\n        getTime: () => videoRef.current.currentTime,\n      }),\n    [chapters]\n  )\n\n  const isEnd = playbackState === 'ended'\n  const canSeek = !isEnd && playbackTime.duration > 0\n  const derivedPlaybackState =\n    (uiType === 'mobile' && blocking) || shouldPause({userFocus, uiType})\n      ? 'paused'\n      : targetState.playbackState\n  const activeThumbnailsUrl =\n    (!userFocus || userFocus === 'seekbar') &&\n    (getThumbnailsUrl(source) || thumbnailsUrl)\n  const cssVariables = {\n    '--playing': playbackState === 'playing' ? '1' : '0',\n  }\n  const onRewind =\n    (!isLive || playbackTime.currentTime < 0) &&\n    canSeek &&\n    (() => setTargetTime(playbackTime.currentTime - 10, 'rewind'))\n  const onForward =\n    (!isLive || playbackTime.currentTime < 0) &&\n    canSeek &&\n    (() => setTargetTime(playbackTime.currentTime + 10, 'forward'))\n  const onClickBlank =\n    havePlayPanel && (() => togglePlay('', {triggeredBy: 'user-action'}))\n\n  const uiHandlers = {\n    onPlay: () => {\n      togglePlay('', {triggeredBy: 'user-action'})\n      setIsUserActive(true)\n    },\n    onClickLive: isLive && (() => setTargetTime(0, 'seekToLive')),\n    onClickSeekbar: () => {\n      refs.current.deferedSeekFocus = setTimeout(\n        () => setUserFocus('seekbar'),\n        66\n      )\n    },\n    onSeek: time => {\n      setTargetTime(time, 'seek')\n      clearTimeout(refs.current.deferedSeekFocus)\n      setTimeout(() => setUserFocus(''), 66)\n    },\n    onChangeSettings: ({name, value, keepOpen}) =>\n      changeSettings(name, value, {keepOpen}),\n    onCloseSettings: event => {\n      setUserFocus('')\n      slotProps?.settings?.onClose?.(event)\n    },\n  }\n\n  const uiElements = {\n    liveButton: uiHandlers.onClickLive && (\n      <components.LiveButton\n        usingStartOver={playbackTime.currentTime < 0}\n        onClick={uiHandlers.onClickLive}\n      />\n    ),\n    controlButtons: {\n      playButton: (\n        <PlayButton\n          playbackState={derivedPlaybackState}\n          ended={isEnd}\n          hidden={\n            uiType === 'mobile' && (waiting || /loading/.test(playbackState))\n          }\n          variant={!isUserActive && 'firstplay'}\n          onClick={uiHandlers.onPlay}\n        />\n      ),\n      rewindButton: onRewind && (\n        <Button\n          startIcon=\"rewind10\"\n          title=\"KKS.PLAYER.REWIND\"\n          disabled={isEnd}\n          onClick={onRewind}\n        />\n      ),\n      forwardButton: onForward && (\n        <ForwardButton\n          startIcon=\"forward10\"\n          title=\"KKS.PLAYER.FORWARD\"\n          disabled={!canSeek}\n          onClick={onForward}\n        />\n      ),\n      nextEpisodeButton: (\n        <Button\n          startIcon=\"next\"\n          title=\"KKS.PLAYER.NEXT\"\n          disabled={!onChangeNext}\n          onClick={changeNext}\n        />\n      ),\n      previousEpisodeButton: (\n        <Button\n          startIcon=\"previous\"\n          title=\"KKS.PLAYER.PREVIOUS\"\n          disabled={!onChangePrevious}\n          onClick={changePrevious}\n        />\n      ),\n    },\n    seekbar: (\n      <components.Seekbar\n        style={/volume/.test(userFocus) && {opacity: '0'}}\n        startTime={playbackTime.startTime}\n        currentTime={playbackTime.currentTime}\n        bufferTime={playbackTime.bufferTime}\n        duration={playbackTime.duration}\n        chapters={chapters}\n        onChange={uiHandlers.onClickSeekbar}\n        onChangeCommitted={uiHandlers.onSeek}\n        marks={marks}\n        plugins={plugins}\n        {...slotProps.seekbar}\n      >\n        {activeThumbnailsUrl && (\n          <SeekPreview\n            thumbnailsUrl={activeThumbnailsUrl}\n            duration={playbackTime.duration}\n            chapters={chapters}\n          />\n        )}\n      </components.Seekbar>\n    ),\n    displayTime: <components.DisplayTime {...playbackTime} />,\n    fullscreenButton: (\n      <FullscreenButton\n        viewMode={viewMode}\n        onClick={() => toggleFullscreen(containerRef.current)}\n      />\n    ),\n    volumeControl: width >= sizes['small-embed'] && (\n      <components.VolumeControl\n        {...{subscribe, onChange, toggleMute}}\n        onMouseOver={() => setUserFocus('volume')}\n        onMouseOut={() => setUserFocus('')}\n        {...slotProps?.volumeControl}\n      />\n    ),\n    backItems: (\n      <PlayPanel animation={targetState.animation} onClick={onClickBlank} />\n    ),\n  }\n  const compatibilityCastProps = castOptions === 'compatible' && {\n    source,\n    values: {title, ...settings.values, ...intl, modulesConfig},\n    onLoad: onCast,\n  }\n\n  useEffect(() => {\n    if (castOptions && castOptions !== 'compatible') {\n      return linkCast({source, ...castOptions})\n    } // TODO playlist, audio / subtitle setting\n  }, [source, castOptions.contentId, castOptions.customData])\n  const selectedAudioTrack = settings.preferred?.audio || appAudioTrack\n  const selectedTextTrack = settings.preferred?.subtitles || appTextTrack\n\n  return (\n    <IntlProvider {...intl}>\n      <Layout\n        style={{...style, cssVariables}}\n        type={uiType}\n        display={mode}\n        controlsDisplay={controlsDisplay}\n        size={size}\n        video={\n          <Video\n            {...videoProps}\n            videoRef={multiRef(videoRef, videoProps.videoRef)}\n            source={\n              castState !== CastState.CONNECTED &&\n              playbackState !== 'error' &&\n              sourceOverride\n            }\n            playbackState={derivedPlaybackState}\n            currentTime={targetState.currentTime}\n            {...(appVolume >= 0 && {volume: appVolume, muted: appVolume <= 0})}\n            playbackRate={settings.values.speed}\n            videoResolution={qualityOptions}\n            audioTrack={selectedAudioTrack}\n            textTrack={selectedTextTrack}\n            loop={settings.values.loop}\n            plugins={plugins}\n            modulesConfig={modulesConfig}\n            onPlaylogFired={onPlaylogFired}\n            onError={errorHandler}\n            onPlaybackStateChange={handlePlaybackStateChange}\n            onBlockedAutoplay={error => onBlockedAutoplay?.(error)}\n            onCanPlay={event => {\n              updatePlaybackTime(event)\n              videoProps.onCanPlay?.(event)\n            }}\n            onTimeUpdate={updatePlaybackTime}\n            onDurationChange={updatePlaybackTime}\n            onPlayerLoaded={player => {\n              playerRef.current = player\n              onPlayerLoaded?.(player)\n              syncVolume(videoRef.current, videoProps.muted ? 0 : appVolume)\n            }}\n          />\n        }\n        containerRef={element => {\n          containerRef.current = element\n          observe(element)\n        }}\n        backRef={adContainerRef}\n        title={title}\n        channelTitle={channelTitle}\n        backButton={onBack && <Button startIcon=\"back\" onClick={onBack} />}\n        {...(isUserActive && uiElements)}\n        onClick={onClick}\n        onMouseMove={onMouseMove}\n      >\n        <LoadingSpinner active={waiting} />\n        {errorData && <Error error={errorData} onBack={onBack} />}\n        {children}\n        {/* TODO sync Cast last played time back */}\n        {isUserActive && (\n          <components.CastUi\n            onBack={onBack}\n            onStateChange={setCastState}\n            onLoad={onCast}\n            {...compatibilityCastProps}\n            {...slotProps.castUi}\n          />\n        )}\n        {isUserActive && ( // TODO: Design <Extenstion /> flag\n          <components.Settings\n            type={uiType}\n            sections={settings.sections}\n            // TODO hasBottomPanel bottom: 8em\n            open={userFocus === 'settings'}\n            values={settings.values}\n            onOpen={openSettings}\n            onChange={uiHandlers.onChangeSettings}\n            {...slotProps?.settings}\n            onClose={uiHandlers.onCloseSettings}\n          />\n        )}\n        <LanguageSettings\n          uiType={uiType}\n          getPlayer={() => playerRef.current}\n          onChange={(_event, item) =>\n            uiHandlers.onChangeSettings({\n              name: item.type,\n              value: item.value,\n              keepOpen: true,\n            })\n          }\n          {...slotProps.languageSettings}\n        />\n        <Backdrop open={!playbackState || playbackState === 'loading'}>\n          <LoadingSpinner />\n        </Backdrop>\n      </Layout>\n    </IntlProvider>\n  )\n}\n\nexport default PremiumPlayer\n"]} */")
|
|
8801
8803
|
};
|
|
8802
8804
|
|
|
8803
8805
|
const LanguageSettings = ({
|
|
@@ -8816,6 +8818,7 @@ const LanguageSettings = ({
|
|
|
8816
8818
|
textTracks
|
|
8817
8819
|
});
|
|
8818
8820
|
return audioTracks.length > 0 && textTracks.length > 0 && /*#__PURE__*/jsx(OverlayPanel, {
|
|
8821
|
+
uiType: uiType,
|
|
8819
8822
|
title: "\u5B57\u5E55\u30FB\u97F3\u58F0\u3092\u5207\u308A\u66FF\u3048\u308B",
|
|
8820
8823
|
buttonIcon: "subtitle",
|
|
8821
8824
|
children: /*#__PURE__*/jsx(slots.LanguageMenu, {
|
|
@@ -9108,21 +9111,32 @@ const PremiumPlayer = ({
|
|
|
9108
9111
|
});
|
|
9109
9112
|
const mode = userFocus === 'seekbar' ? 'hidden' : controls.autohide ? autoHideMode : controls ? 'shown' : 'hidden';
|
|
9110
9113
|
const controlsDisplay = userFocus === 'seekbar' ? 'seekbar-only' : controls === 'title-only' || playbackState === 'emptied' ? 'hidden' : mode;
|
|
9111
|
-
const shouldHidePanels = (controls === 'no-panel' || controlsDisplay === 'hidden') && userFocus;
|
|
9114
|
+
const shouldHidePanels = (controls === 'no-panel' || controlsDisplay === 'hidden' || playbackState === 'ended') && userFocus;
|
|
9112
9115
|
const havePlayPanel = !blocking && uiType === 'desktop' && !waiting && !/volume|''/.test(userFocus) && !/title-only|no-panel/.test(controls) && (controls.autohide || mode === 'shown');
|
|
9113
9116
|
useEffect(() => {
|
|
9114
9117
|
if (shouldHidePanels) {
|
|
9115
9118
|
setUserFocus('');
|
|
9119
|
+
|
|
9120
|
+
if (refs.current.closeMenu) {
|
|
9121
|
+
refs.current.closeMenu();
|
|
9122
|
+
refs.current.closeMenu = undefined;
|
|
9123
|
+
}
|
|
9116
9124
|
}
|
|
9117
9125
|
}, [shouldHidePanels]);
|
|
9118
9126
|
useEffect(() => on(containerRef.current, 'focus-menu', event => {
|
|
9119
|
-
var _event$detail, _event$
|
|
9127
|
+
var _event$detail, _event$detail3;
|
|
9120
9128
|
|
|
9121
9129
|
if (((_event$detail = event.detail) === null || _event$detail === void 0 ? void 0 : _event$detail.status) === 'open') {
|
|
9122
9130
|
setUserFocus('menu');
|
|
9131
|
+
|
|
9132
|
+
refs.current.closeMenu = () => {
|
|
9133
|
+
var _event$detail$closeMe, _event$detail2;
|
|
9134
|
+
|
|
9135
|
+
return (_event$detail$closeMe = (_event$detail2 = event.detail).closeMenu) === null || _event$detail$closeMe === void 0 ? void 0 : _event$detail$closeMe.call(_event$detail2);
|
|
9136
|
+
};
|
|
9123
9137
|
}
|
|
9124
9138
|
|
|
9125
|
-
if (((_event$
|
|
9139
|
+
if (((_event$detail3 = event.detail) === null || _event$detail3 === void 0 ? void 0 : _event$detail3.status) === 'closed') {
|
|
9126
9140
|
setUserFocus(current => current === 'menu' ? '' : current);
|
|
9127
9141
|
}
|
|
9128
9142
|
}), []);
|