okay, we can't ditch mutationobserver 5ever

This commit is contained in:
Tamius Han 2021-03-30 01:10:37 +02:00
parent b634e4d2b3
commit 48319acc31

View File

@ -26,6 +26,12 @@ class VideoData {
vdid: string; vdid: string;
video: any; video: any;
observer: ResizeObserver; observer: ResizeObserver;
mutationObserver: MutationObserver;
mutationObserverConf: MutationObserverInit = {
attributes: true,
attributeFilter: ['class', 'style'],
attributeOldValue: true,
};
extensionMode: any; extensionMode: any;
userCssClassName: string; userCssClassName: string;
validationId: number; validationId: number;
@ -106,12 +112,17 @@ class VideoData {
async injectBaseCss() { async injectBaseCss() {
try { try {
if (!this.mutationObserver) {
this.setupMutationObserver();
}
await this.pageInfo.injectCss(` await this.pageInfo.injectCss(`
.uw-ultrawidify-base-wide-screen { .uw-ultrawidify-base-wide-screen {
margin: 0px 0px 0px 0px !important; margin: 0px 0px 0px 0px !important;
width: initial !important; width: initial !important;
align-self: start !important; align-self: start !important;
justify-self: start !important; justify-self: start !important;
max-height: initial !important;
max-width: initial !important;
} }
`); `);
} catch (e) { } catch (e) {
@ -120,6 +131,8 @@ class VideoData {
} }
unsetBaseClass() { unsetBaseClass() {
this.mutationObserver.disconnect();
this.mutationObserver = undefined;
this.video.classList.remove('uw-ultrawidify-base-wide-screen'); this.video.classList.remove('uw-ultrawidify-base-wide-screen');
} }
@ -163,14 +176,6 @@ class VideoData {
async setupStageTwo() { async setupStageTwo() {
// POZOR: VRSTNI RED JE POMEMBEN (arDetect mora bit zadnji) // POZOR: VRSTNI RED JE POMEMBEN (arDetect mora bit zadnji)
// NOTE: ORDERING OF OBJ INITIALIZATIONS IS IMPORTANT (arDetect needs to go last) // NOTE: ORDERING OF OBJ INITIALIZATIONS IS IMPORTANT (arDetect needs to go last)
// NOTE: We only init observers once player is confirmed valid
const observerConf = {
attributes: true,
// attributeFilter: ['style', 'class'],
attributeOldValue: true,
};
this.player = new PlayerData(this); this.player = new PlayerData(this);
if (this.player.invalid) { if (this.player.invalid) {
this.invalid = true; this.invalid = true;
@ -255,6 +260,41 @@ class VideoData {
} }
} }
setupMutationObserver() {
try {
if (BrowserDetect.firefox) {
this.mutationObserver = new MutationObserver(
_.debounce(
this.onVideoMutation,
250,
{
leading: true,
trailing: true
}
)
)
} else {
// Chrome for some reason insists that this.onPlayerDimensionsChanged is not a function
// when it's not wrapped into an anonymous function
this.mutationObserver = new MutationObserver(
_.debounce(
(m, o) => {
this.onVideoMutation(m, o)
},
250,
{
leading: true,
trailing: true
}
)
)
}
} catch (e) {
console.error('[VideoData] Observer setup failed:', e);
}
this.mutationObserver.observe(this.video, this.mutationObserverConf);
}
/** /**
* cleans up handlers and stuff when the show is over * cleans up handlers and stuff when the show is over
*/ */
@ -337,6 +377,37 @@ class VideoData {
this.validateVideoOffsets(); this.validateVideoOffsets();
} }
onVideoMutation(mutationList?: MutationRecord[], observer?) {
// verify that mutation didn't remove our class. Some pages like to do that.
let confirmAspectRatioRestore = false;
for(const mutation of mutationList) {
if (mutation.type === 'attributes') {
if( mutation.attributeName === 'class'
&& mutation.oldValue.indexOf('uw-ultrawidify-base-wide-screen') !== -1
&& mutation.oldValue.indexOf('uw-ultrawidify-base-wide-screen') === -1
) {
// force the page to include our class in classlist, if the classlist has been removed
// while classList.add() doesn't duplicate classes (does nothing if class is already added),
// we still only need to make sure we're only adding our class to classlist if it has been
// removed. classList.add() will _still_ trigger mutation (even if classlist wouldn't change).
// This is a problem because INFINITE RECURSION TIME, and we _really_ don't want that.
confirmAspectRatioRestore = true;
this.video.classList.add(this.userCssClassName);
this.video.classList.add('uw-ultrawidify-base-wide-screen');
console.warn('our classname was removed by site! undoign!')
} else if (mutation.attributeName === 'style') {
confirmAspectRatioRestore = true;
}
}
}
this.processDimensionsChanged();
// console.info('video mutated. mutation list?', mutationList);
}
onVideoDimensionsChanged(mutationList, observer) { onVideoDimensionsChanged(mutationList, observer) {
if (!mutationList || this.video === undefined) { // something's wrong if (!mutationList || this.video === undefined) { // something's wrong
if (observer && this.video) { if (observer && this.video) {
@ -344,34 +415,15 @@ class VideoData {
} }
return; return;
} }
let confirmAspectRatioRestore = false;
for (let mutation of mutationList) { this.processDimensionsChanged();
if (mutation.type === 'attributes') {
if (mutation.attributeName === 'class') {
if(!this.video.classList.contains(this.userCssClassName) ) {
// force the page to include our class in classlist, if the classlist has been removed
// while classList.add() doesn't duplicate classes (does nothing if class is already added),
// we still only need to make sure we're only adding our class to classlist if it has been
// removed. classList.add() will _still_ trigger mutation (even if classlist wouldn't change).
// This is a problem because INFINITE RECURSION TIME, and we _really_ don't want that.
this.video.classList.add(this.userCssClassName);
this.video.classList.add('uw-ultrawidify-base-wide-screen');
}
// always trigger refresh on class changes, since change of classname might trigger change
// of the player size as well.
confirmAspectRatioRestore = true;
}
if (mutation.attributeName === 'style') {
confirmAspectRatioRestore = true;
}
}
}
if (!confirmAspectRatioRestore) {
return;
} }
/**
* Forces Ultrawidify to resotre aspect ratio. You should never call this method directly,
* instead you should be calling processDimensionChanged() wrapper function.
*/
private _processDimensionsChanged() {
// adding player observer taught us that if element size gets triggered by a class, then // adding player observer taught us that if element size gets triggered by a class, then
// the 'style' attributes don't necessarily trigger. This means we also need to trigger // the 'style' attributes don't necessarily trigger. This means we also need to trigger
// restoreAr here, in case video size was changed this way // restoreAr here, in case video size was changed this way
@ -385,6 +437,21 @@ class VideoData {
}, 100); }, 100);
} }
/**
* Restores aspect ratio and validates video offsets after the restore. Execution uses
* debounce to limit how often the function executes.
*/
private processDimensionsChanged() {
_.debounce(
this._processDimensionsChanged,
250,
{
leading: true,
trailing: true
}
);
}
validateVideoOffsets() { validateVideoOffsets() {
// validate if current video still exists. If not, we destroy current object // validate if current video still exists. If not, we destroy current object
try { try {