2021-06-24 04:34:34 +03:00
|
|
|
/*
|
|
|
|
Copyright 2021 The Matrix.org Foundation C.I.C.
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
import { Playback, PlaybackState } from "../../../voice/Playback";
|
2021-06-25 05:41:00 +03:00
|
|
|
import React, { ChangeEvent, CSSProperties, ReactNode } from "react";
|
2021-06-24 04:34:34 +03:00
|
|
|
import { replaceableComponent } from "../../../utils/replaceableComponent";
|
|
|
|
import { MarkedExecution } from "../../../utils/MarkedExecution";
|
|
|
|
import { percentageOf } from "../../../utils/numbers";
|
|
|
|
|
|
|
|
interface IProps {
|
|
|
|
// Playback instance to render. Cannot change during component lifecycle: create
|
|
|
|
// an all-new component instead.
|
|
|
|
playback: Playback;
|
|
|
|
|
|
|
|
// Tab index for the underlying component. Useful if the seek bar is in a managed state.
|
|
|
|
// Defaults to zero.
|
|
|
|
tabIndex?: number;
|
|
|
|
|
|
|
|
playbackPhase: PlaybackState;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface IState {
|
|
|
|
percentage: number;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface ISeekCSS extends CSSProperties {
|
|
|
|
'--fillTo': number;
|
|
|
|
}
|
|
|
|
|
2021-06-25 05:18:50 +03:00
|
|
|
const ARROW_SKIP_SECONDS = 5; // arbitrary
|
|
|
|
|
2021-06-24 04:34:34 +03:00
|
|
|
@replaceableComponent("views.audio_messages.SeekBar")
|
|
|
|
export default class SeekBar extends React.PureComponent<IProps, IState> {
|
|
|
|
// We use an animation frame request to avoid overly spamming prop updates, even if we aren't
|
|
|
|
// really using anything demanding on the CSS front.
|
|
|
|
|
|
|
|
private animationFrameFn = new MarkedExecution(
|
|
|
|
() => this.doUpdate(),
|
|
|
|
() => requestAnimationFrame(() => this.animationFrameFn.trigger()));
|
|
|
|
|
|
|
|
public static defaultProps = {
|
|
|
|
tabIndex: 0,
|
|
|
|
};
|
|
|
|
|
|
|
|
constructor(props: IProps) {
|
|
|
|
super(props);
|
|
|
|
|
|
|
|
this.state = {
|
|
|
|
percentage: 0,
|
|
|
|
};
|
|
|
|
|
|
|
|
// We don't need to de-register: the class handles this for us internally
|
|
|
|
this.props.playback.clockInfo.liveData.onUpdate(() => this.animationFrameFn.mark());
|
|
|
|
}
|
|
|
|
|
|
|
|
private doUpdate() {
|
|
|
|
this.setState({
|
|
|
|
percentage: percentageOf(
|
|
|
|
this.props.playback.clockInfo.timeSeconds,
|
|
|
|
0,
|
|
|
|
this.props.playback.clockInfo.durationSeconds),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
public left() {
|
2021-06-25 05:18:50 +03:00
|
|
|
// noinspection JSIgnoredPromiseFromCall
|
|
|
|
this.props.playback.skipTo(this.props.playback.clockInfo.timeSeconds - ARROW_SKIP_SECONDS);
|
2021-06-24 04:34:34 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
public right() {
|
2021-06-25 05:18:50 +03:00
|
|
|
// noinspection JSIgnoredPromiseFromCall
|
|
|
|
this.props.playback.skipTo(this.props.playback.clockInfo.timeSeconds + ARROW_SKIP_SECONDS);
|
2021-06-24 04:34:34 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
private onChange = (ev: ChangeEvent<HTMLInputElement>) => {
|
2021-06-25 05:18:50 +03:00
|
|
|
// Thankfully, onChange is only called when the user changes the value, not when we
|
|
|
|
// change the value on the component. We can use this as a reliable "skip to X" function.
|
|
|
|
//
|
|
|
|
// noinspection JSIgnoredPromiseFromCall
|
|
|
|
this.props.playback.skipTo(Number(ev.target.value) * this.props.playback.clockInfo.durationSeconds);
|
2021-06-24 04:34:34 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
public render(): ReactNode {
|
|
|
|
// We use a range input to avoid having to re-invent accessibility handling on
|
|
|
|
// a custom set of divs.
|
|
|
|
return <input
|
|
|
|
type="range"
|
|
|
|
className='mx_SeekBar'
|
|
|
|
tabIndex={this.props.tabIndex}
|
|
|
|
onChange={this.onChange}
|
|
|
|
min={0}
|
|
|
|
max={1}
|
|
|
|
value={this.state.percentage}
|
|
|
|
step={0.001}
|
2021-06-30 22:29:37 +03:00
|
|
|
style={{ '--fillTo': this.state.percentage } as ISeekCSS}
|
2021-06-24 04:34:34 +03:00
|
|
|
disabled={this.props.playbackPhase === PlaybackState.Decoding}
|
|
|
|
/>;
|
|
|
|
}
|
|
|
|
}
|