Safari audio stream

Safari iOS fix video stream
Safari video playback with sound
Some fixes
This commit is contained in:
morethanwords 2020-08-30 13:43:57 +03:00
parent e05e9afa37
commit 9ae707aa7b
9 changed files with 156 additions and 40 deletions

View File

@ -2,9 +2,15 @@ import { MTDocument } from "../types";
import { $rootScope } from "../lib/utils"; import { $rootScope } from "../lib/utils";
import appMessagesManager from "../lib/appManagers/appMessagesManager"; import appMessagesManager from "../lib/appManagers/appMessagesManager";
import appDocsManager from "../lib/appManagers/appDocsManager"; import appDocsManager from "../lib/appManagers/appDocsManager";
import { isSafari } from "../lib/config";
import { CancellablePromise, deferredPromise } from "../lib/polyfill";
// TODO: если удалить сообщение, и при этом аудио будет играть - оно не остановится, и можно будет по нему перейти вникуда // TODO: если удалить сообщение, и при этом аудио будет играть - оно не остановится, и можно будет по нему перейти вникуда
// TODO: Safari: проверить стрим, включить его и сразу попробовать включить видео или другую песню
// TODO: Safari: попробовать замаскировать подгрузку последнего чанка
// TODO: Safari: пофиксить момент, когда заканчивается песня и пытаешься включить её заново - прогресс сразу в конце
type HTMLMediaElement = HTMLAudioElement | HTMLVideoElement; type HTMLMediaElement = HTMLAudioElement | HTMLVideoElement;
type MediaType = 'voice' | 'audio' | 'round'; type MediaType = 'voice' | 'audio' | 'round';
@ -13,6 +19,8 @@ class AppMediaPlaybackController {
private container: HTMLElement; private container: HTMLElement;
private media: {[mid: string]: HTMLMediaElement} = {}; private media: {[mid: string]: HTMLMediaElement} = {};
private playingMedia: HTMLMediaElement; private playingMedia: HTMLMediaElement;
private waitingMediaForLoad: {[mid: string]: CancellablePromise<void>} = {};
public willBePlayedMedia: HTMLMediaElement; public willBePlayedMedia: HTMLMediaElement;
@ -26,17 +34,22 @@ class AppMediaPlaybackController {
document.body.append(this.container); document.body.append(this.container);
} }
public addMedia(doc: MTDocument, mid: number): HTMLMediaElement { public addMedia(doc: MTDocument, mid: number, autoload = true): HTMLMediaElement {
if(this.media[mid]) return this.media[mid]; if(this.media[mid]) return this.media[mid];
const media = document.createElement(doc.type == 'round' ? 'video' : 'audio'); const media = document.createElement(doc.type == 'round' ? 'video' : 'audio');
//const source = document.createElement('source'); //const source = document.createElement('source');
//source.type = doc.type == 'voice' && !opusDecodeController.isPlaySupported() ? 'audio/wav' : doc.mime_type; //source.type = doc.type == 'voice' && !opusDecodeController.isPlaySupported() ? 'audio/wav' : doc.mime_type;
media.dataset.mid = '' + mid;
media.dataset.type = doc.type;
media.autoplay = false; //media.autoplay = true;
media.volume = 1; media.volume = 1;
//media.append(source); //media.append(source);
this.container.append(media);
media.addEventListener('playing', () => { media.addEventListener('playing', () => {
if(this.playingMedia != media) { if(this.playingMedia != media) {
if(this.playingMedia && !this.playingMedia.paused) { if(this.playingMedia && !this.playingMedia.paused) {
@ -68,20 +81,78 @@ class AppMediaPlaybackController {
media.addEventListener('error', onError); media.addEventListener('error', onError);
const deferred = deferredPromise<void>();
if(autoload) {
deferred.resolve();
} else {
this.waitingMediaForLoad[mid] = deferred;
}
// если что - загрузит voice или round заранее, так правильнее
const downloadPromise: Promise<any> = !doc.supportsStreaming ? appDocsManager.downloadDocNew(doc.id) : Promise.resolve(); const downloadPromise: Promise<any> = !doc.supportsStreaming ? appDocsManager.downloadDocNew(doc.id) : Promise.resolve();
Promise.all([deferred, downloadPromise]).then(() => {
//media.autoplay = true;
//console.log('will set media url:', media, doc, doc.type, doc.url);
downloadPromise.then(() => { if(doc.type == 'audio' && doc.supportsStreaming && isSafari) {
//if(doc.type != 'round') { this.handleSafariStreamable(media);
this.container.append(media); }
//}
//source.src = doc.url;
media.src = doc.url; media.src = doc.url;
}, onError); }, onError);
return this.media[mid] = media; return this.media[mid] = media;
} }
// safari подгрузит последний чанк и песня включится,
// при этом этот чанк нельзя руками отдать из SW, потому что браузер тогда теряется
private handleSafariStreamable(media: HTMLMediaElement) {
media.addEventListener('play', () => {
/* if(media.readyState == 4) { // https://developer.mozilla.org/ru/docs/Web/API/XMLHttpRequest/readyState
return;
} */
//media.volume = 0;
const currentTime = media.currentTime;
//this.setSafariBuffering(media, true);
media.addEventListener('progress', () => {
media.currentTime = media.duration - 1;
media.addEventListener('progress', () => {
media.currentTime = currentTime;
//media.volume = 1;
//this.setSafariBuffering(media, false);
if(!media.paused) {
media.play()/* .catch(() => {}) */;
}
}, {once: true});
}, {once: true});
}/* , {once: true} */);
}
public resolveWaitingForLoadMedia(mid: number) {
const promise = this.waitingMediaForLoad[mid];
if(promise) {
promise.resolve();
delete this.waitingMediaForLoad[mid];
}
}
/**
* Only for audio
*/
public isSafariBuffering(media: HTMLMediaElement) {
/// @ts-ignore
return !!media.safariBuffering;
}
private setSafariBuffering(media: HTMLMediaElement, value: boolean) {
// @ts-ignore
media.safariBuffering = value;
}
onPause = (e: Event) => { onPause = (e: Event) => {
$rootScope.$broadcast('audio_pause'); $rootScope.$broadcast('audio_pause');
}; };
@ -89,8 +160,20 @@ class AppMediaPlaybackController {
onEnded = (e: Event) => { onEnded = (e: Event) => {
this.onPause(e); this.onPause(e);
//console.log('on media end');
if(this.nextMid) { if(this.nextMid) {
this.media[this.nextMid].play(); const media = this.media[this.nextMid];
/* if(isSafari) {
media.autoplay = true;
} */
this.resolveWaitingForLoadMedia(this.nextMid);
setTimeout(() => {
media.play()//.catch(() => {});
}, 0);
} }
}; };
@ -106,7 +189,7 @@ class AppMediaPlaybackController {
if(this.playingMedia != media) { if(this.playingMedia != media) {
return; return;
} }
for(let m of value.history) { for(let m of value.history) {
if(m > mid) { if(m > mid) {
this.nextMid = m; this.nextMid = m;
@ -118,7 +201,7 @@ class AppMediaPlaybackController {
[this.prevMid, this.nextMid].filter(Boolean).forEach(mid => { [this.prevMid, this.nextMid].filter(Boolean).forEach(mid => {
const message = appMessagesManager.getMessage(mid); const message = appMessagesManager.getMessage(mid);
this.addMedia(message.media.document, mid); this.addMedia(message.media.document, mid, false);
}); });
//console.log('loadSiblingsAudio', audio, type, mid, value, this.prevMid, this.nextMid); //console.log('loadSiblingsAudio', audio, type, mid, value, this.prevMid, this.nextMid);
@ -143,10 +226,6 @@ class AppMediaPlaybackController {
public willBePlayed(media: HTMLMediaElement) { public willBePlayed(media: HTMLMediaElement) {
this.willBePlayedMedia = media; this.willBePlayedMedia = media;
} }
public mediaExists(mid: number) {
return !!this.media[mid];
}
} }
const appMediaPlaybackController = new AppMediaPlaybackController(); const appMediaPlaybackController = new AppMediaPlaybackController();

View File

@ -5,8 +5,9 @@ import ProgressivePreloader from "./preloader";
import { MediaProgressLine } from "../lib/mediaPlayer"; import { MediaProgressLine } from "../lib/mediaPlayer";
import appMediaPlaybackController from "./appMediaPlaybackController"; import appMediaPlaybackController from "./appMediaPlaybackController";
import { MTDocument } from "../types"; import { MTDocument } from "../types";
import { mediaSizes } from "../lib/config"; import { mediaSizes, isSafari } from "../lib/config";
import { Download } from "../lib/appManagers/appDownloadManager"; import { Download } from "../lib/appManagers/appDownloadManager";
import { deferredPromise, CancellablePromise } from "../lib/polyfill";
// https://github.com/LonamiWebs/Telethon/blob/4393ec0b83d511b6a20d8a20334138730f084375/telethon/utils.py#L1285 // https://github.com/LonamiWebs/Telethon/blob/4393ec0b83d511b6a20d8a20334138730f084375/telethon/utils.py#L1285
export function decodeWaveform(waveform: Uint8Array | number[]) { export function decodeWaveform(waveform: Uint8Array | number[]) {
@ -312,8 +313,8 @@ export default class AudioElement extends HTMLElement {
const audioTimeDiv = this.querySelector('.audio-time') as HTMLDivElement; const audioTimeDiv = this.querySelector('.audio-time') as HTMLDivElement;
audioTimeDiv.innerHTML = durationStr; audioTimeDiv.innerHTML = durationStr;
const onLoad = () => { const onLoad = (autoload = true) => {
const audio = this.audio = appMediaPlaybackController.addMedia(doc, mid); const audio = this.audio = appMediaPlaybackController.addMedia(doc, mid, autoload);
this.onTypeDisconnect = onTypeLoad(); this.onTypeDisconnect = onTypeLoad();
@ -333,7 +334,7 @@ export default class AudioElement extends HTMLElement {
} }
toggle.addEventListener('click', () => { toggle.addEventListener('click', () => {
if(audio.paused) audio.play(); if(audio.paused) audio.play().catch(() => {});
else audio.pause(); else audio.pause();
}); });
@ -343,6 +344,7 @@ export default class AudioElement extends HTMLElement {
}); });
this.addAudioListener('timeupdate', () => { this.addAudioListener('timeupdate', () => {
if(appMediaPlaybackController.isSafariBuffering(audio)) return;
audioTimeDiv.innerText = String(audio.currentTime | 0).toHHMMSS(true) + ' / ' + durationStr; audioTimeDiv.innerText = String(audio.currentTime | 0).toHHMMSS(true) + ' / ' + durationStr;
}); });
@ -390,17 +392,25 @@ export default class AudioElement extends HTMLElement {
this.addEventListener('click', onClick); this.addEventListener('click', onClick);
this.click(); this.click();
} else { } else {
if(appMediaPlaybackController.mediaExists(mid)) { // чтобы показать прогресс, если аудио уже было скачано onLoad(false);
onLoad();
} else { //if(appMediaPlaybackController.mediaExists(mid)) { // чтобы показать прогресс, если аудио уже было скачано
//onLoad();
//} else {
const r = () => { const r = () => {
onLoad(); //onLoad();
appMediaPlaybackController.resolveWaitingForLoadMedia(mid);
appMediaPlaybackController.willBePlayed(this.audio); // prepare for loading audio appMediaPlaybackController.willBePlayed(this.audio); // prepare for loading audio
if(!preloader) { if(!preloader) {
preloader = new ProgressivePreloader(null, false); preloader = new ProgressivePreloader(null, false);
} }
if(isSafari) {
this.audio.autoplay = true;
this.audio.play().catch(() => {});
}
preloader.attach(downloadDiv); preloader.attach(downloadDiv);
this.append(downloadDiv); this.append(downloadDiv);
@ -422,7 +432,7 @@ export default class AudioElement extends HTMLElement {
}; };
this.addEventListener('click', r, {once: true}); this.addEventListener('click', r, {once: true});
} //}
} }
} else { } else {
this.preloader.attach(downloadDiv, false); this.preloader.attach(downloadDiv, false);

View File

@ -62,8 +62,10 @@ export function wrapVideo({doc, container, message, boxWidth, boxHeight, withTai
} */ } */
const video = document.createElement('video'); const video = document.createElement('video');
video.muted = true;
video.setAttribute('playsinline', '');
if(doc.type == 'round') { if(doc.type == 'round') {
video.muted = true; //video.muted = true;
const globalVideo = appMediaPlaybackController.addMedia(doc, message.mid); const globalVideo = appMediaPlaybackController.addMedia(doc, message.mid);
video.addEventListener('canplay', () => { video.addEventListener('canplay', () => {
@ -121,6 +123,8 @@ export function wrapVideo({doc, container, message, boxWidth, boxHeight, withTai
globalVideo.addEventListener('pause', onGlobalPause); globalVideo.addEventListener('pause', onGlobalPause);
video.addEventListener('play', onVideoPlay); video.addEventListener('play', onVideoPlay);
video.addEventListener('pause', onVideoPause); video.addEventListener('pause', onVideoPause);
} else {
video.autoplay = true; // для safari
} }
let img: HTMLImageElement; let img: HTMLImageElement;
@ -224,12 +228,10 @@ export function wrapVideo({doc, container, message, boxWidth, boxHeight, withTai
renderImageFromUrl(video, doc.url); renderImageFromUrl(video, doc.url);
//} //}
video.setAttribute('playsinline', '');
/* if(!container.parentElement) { /* if(!container.parentElement) {
container.append(video); container.append(video);
} */ } */
if(doc.type == 'gif'/* || true */) { if(doc.type == 'gif'/* || true */) {
video.muted = true; video.muted = true;
video.loop = true; video.loop = true;

View File

@ -124,11 +124,14 @@ class AppDocsManager {
if((doc.type == 'gif' && doc.size > 8e6) || doc.type == 'audio' || doc.type == 'video') { if((doc.type == 'gif' && doc.size > 8e6) || doc.type == 'audio' || doc.type == 'video') {
doc.supportsStreaming = true; doc.supportsStreaming = true;
if(!doc.url) { if(!doc.url) {
doc.url = this.getFileURL(doc); doc.url = this.getFileURL(doc);
} }
} }
// for testing purposes
//doc.supportsStreaming = false;
if(!doc.file_name) { if(!doc.file_name) {
doc.file_name = ''; doc.file_name = '';

View File

@ -923,16 +923,26 @@ export class AppMediaViewer {
if(isVideo) { if(isVideo) {
////////this.log('will wrap video', media, size); ////////this.log('will wrap video', media, size);
// потому что для safari нужно создать элемент из event'а
const video = document.createElement('video');
setMoverPromise = this.setMoverToTarget(target, false, fromRight).then(({onAnimationEnd}) => { setMoverPromise = this.setMoverToTarget(target, false, fromRight).then(({onAnimationEnd}) => {
//return; // set and don't move //return; // set and don't move
//if(wasActive) return; //if(wasActive) return;
//return; //return;
const div = mover.firstElementChild && mover.firstElementChild.classList.contains('media-viewer-aspecter') ? mover.firstElementChild : mover; const div = mover.firstElementChild && mover.firstElementChild.classList.contains('media-viewer-aspecter') ? mover.firstElementChild : mover;
const video = mover.querySelector('video') || document.createElement('video'); //const video = mover.querySelector('video') || document.createElement('video');
const moverVideo = mover.querySelector('video');
if(moverVideo) {
moverVideo.remove();
}
//video.src = ''; //video.src = '';
video.setAttribute('playsinline', ''); video.setAttribute('playsinline', '');
video.autoplay = true;
if(media.type == 'gif') { if(media.type == 'gif') {
video.muted = true; video.muted = true;
video.autoplay = true; video.autoplay = true;

View File

@ -223,7 +223,7 @@ export class AppSidebarRight extends SidebarSlider {
let ids = Object.keys(this.mediaDivsByIDs).map(k => +k).sort((a, b) => a - b); let ids = Object.keys(this.mediaDivsByIDs).map(k => +k).sort((a, b) => a - b);
let idx = ids.findIndex(i => i == messageID); let idx = ids.findIndex(i => i == messageID);
let targets = ids.map(id => ({element: this.mediaDivsByIDs[id].firstElementChild as HTMLElement, mid: id})); let targets = ids.map(id => ({element: this.mediaDivsByIDs[id]/* .firstElementChild */ as HTMLElement, mid: id}));
appMediaViewer.openMedia(message, target, false, this.sidebarEl, targets.slice(idx + 1).reverse(), targets.slice(0, idx).reverse(), true); appMediaViewer.openMedia(message, target, false, this.sidebarEl, targets.slice(idx + 1).reverse(), targets.slice(0, idx).reverse(), true);
}); });

View File

@ -1,5 +1,6 @@
import { cancelEvent } from "./utils"; import { cancelEvent } from "./utils";
import { touchSupport } from "./config"; import { touchSupport } from "./config";
import appMediaPlaybackController from "../components/appMediaPlaybackController";
export class ProgressLine { export class ProgressLine {
public container: HTMLDivElement; public container: HTMLDivElement;
@ -183,6 +184,7 @@ export class MediaProgressLine extends ProgressLine {
} }
protected setLoadProgress() { protected setLoadProgress() {
if(appMediaPlaybackController.isSafariBuffering(this.media)) return;
const buf = this.media.buffered; const buf = this.media.buffered;
const numRanges = buf.length; const numRanges = buf.length;
@ -214,6 +216,7 @@ export class MediaProgressLine extends ProgressLine {
} }
public setProgress() { public setProgress() {
if(appMediaPlaybackController.isSafariBuffering(this.media)) return;
const currentTime = this.media.currentTime; const currentTime = this.media.currentTime;
super.setProgress(currentTime); super.setProgress(currentTime);
@ -270,7 +273,7 @@ export default class VideoPlayer {
controls.prepend(this.progress.container); controls.prepend(this.progress.container);
} }
if(play && video.paused) { if(play/* && video.paused */) {
const promise = video.play(); const promise = video.play();
promise.catch((err: Error) => { promise.catch((err: Error) => {
if(err.name == 'NotAllowedError') { if(err.name == 'NotAllowedError') {

View File

@ -5,7 +5,7 @@ import type { InputFileLocation, FileLocation, UploadFile, WorkerTaskTemplate }
import { deferredPromise, CancellablePromise } from '../polyfill'; import { deferredPromise, CancellablePromise } from '../polyfill';
import { notifySomeone } from '../../helpers/context'; import { notifySomeone } from '../../helpers/context';
const log = logger('SW', LogLevels.error); const log = logger('SW', LogLevels.error/* | LogLevels.debug | LogLevels.log */);
const ctx = self as any as ServiceWorkerGlobalScope; const ctx = self as any as ServiceWorkerGlobalScope;
const deferredPromises: {[taskID: number]: CancellablePromise<any>} = {}; const deferredPromises: {[taskID: number]: CancellablePromise<any>} = {};
@ -44,10 +44,18 @@ const onFetch = (event: FetchEvent): void => {
switch(scope) { switch(scope) {
case 'stream': { case 'stream': {
const range = parseRange(event.request.headers.get('Range')); const range = parseRange(event.request.headers.get('Range'));
const [offset, end] = range; let [offset, end] = range;
const info: DownloadOptions = JSON.parse(decodeURIComponent(params)); const info: DownloadOptions = JSON.parse(decodeURIComponent(params));
//const fileName = getFileNameByLocation(info.location); //const fileName = getFileNameByLocation(info.location);
const limitPart = STREAM_CHUNK_UPPER_LIMIT;
/* if(info.size > limitPart && isSafari && offset == limitPart) {
//end = info.size - 1;
//offset = info.size - 1 - limitPart;
offset = info.size - (info.size % limitPart);
} */
log.debug('[stream]', url, offset, end); log.debug('[stream]', url, offset, end);
@ -61,10 +69,10 @@ const onFetch = (event: FetchEvent): void => {
return resolve(possibleResponse); return resolve(possibleResponse);
} }
const limit = end && end < STREAM_CHUNK_UPPER_LIMIT ? alignLimit(end - offset + 1) : STREAM_CHUNK_UPPER_LIMIT; const limit = end && end < limitPart ? alignLimit(end - offset + 1) : limitPart;
const alignedOffset = alignOffset(offset, limit); const alignedOffset = alignOffset(offset, limit);
log.debug('[stream] requestFilePart:', info.dcID, info.location, alignedOffset, limit); log.debug('[stream] requestFilePart:', /* info.dcID, info.location, */ alignedOffset, limit);
const task: ServiceWorkerTask = { const task: ServiceWorkerTask = {
type: 'requestFilePart', type: 'requestFilePart',
@ -77,7 +85,7 @@ const onFetch = (event: FetchEvent): void => {
deferred.then(result => { deferred.then(result => {
let ab = result.bytes; let ab = result.bytes;
//log.debug('[stream] requestFilePart result:', result); log.debug('[stream] requestFilePart result:', result);
const headers: Record<string, string> = { const headers: Record<string, string> = {
'Accept-Ranges': 'bytes', 'Accept-Ranges': 'bytes',

View File

@ -6,7 +6,7 @@ const postcssPresetEnv = require('postcss-preset-env');
const ServiceWorkerWebpackPlugin = require('serviceworker-webpack-plugin'); const ServiceWorkerWebpackPlugin = require('serviceworker-webpack-plugin');
const fs = require('fs'); const fs = require('fs');
const allowedIPs = ['195.66.140.39', '192.168.31.144', '127.0.0.1', '192.168.31.1', '192.168.31.192', '176.100.18.181', '46.219.250.22', '193.42.119.184']; const allowedIPs = ['195.66.140.39', '192.168.31.144', '127.0.0.1', '192.168.31.1', '192.168.31.192', '176.100.18.181', '46.219.250.22', '193.42.119.184', '46.133.168.67'];
const devMode = process.env.NODE_ENV !== 'production'; const devMode = process.env.NODE_ENV !== 'production';
const useLocal = false; const useLocal = false;
@ -110,8 +110,9 @@ module.exports = {
} else { } else {
IP = req.connection.remoteAddress.split(':').pop(); IP = req.connection.remoteAddress.split(':').pop();
} }
if(!allowedIPs.includes(IP) && !/^192\.168\.\d{1,3}\.\d{1,3}$/.test(IP)) { // last is ODESSA
if(!allowedIPs.includes(IP) && !/^192\.168\.\d{1,3}\.\d{1,3}$/.test(IP) && !/^88\.155\.57\.\d{1,3}$/.test(IP)) {
console.log('Bad IP connecting: ' + IP, req.url); console.log('Bad IP connecting: ' + IP, req.url);
res.status(404).send('Nothing interesting here.'); res.status(404).send('Nothing interesting here.');
} else { } else {