Skip to content

Commit

Permalink
Fix browser glitch caused by two overlapping scroll animations in web…
Browse files Browse the repository at this point in the history
  • Loading branch information
Gargron authored Sep 19, 2024
1 parent efdc175 commit ef4d6ab
Show file tree
Hide file tree
Showing 3 changed files with 26 additions and 41 deletions.
32 changes: 1 addition & 31 deletions app/javascript/mastodon/features/ui/components/columns_area.jsx
Original file line number Diff line number Diff line change
Expand Up @@ -4,8 +4,6 @@ import { Children, cloneElement, useCallback } from 'react';
import ImmutablePropTypes from 'react-immutable-proptypes';
import ImmutablePureComponent from 'react-immutable-pure-component';

import { supportsPassiveEvents } from 'detect-passive-events';

import { scrollRight } from '../../../scroll';
import BundleContainer from '../containers/bundle_container';
import {
Expand Down Expand Up @@ -71,10 +69,6 @@ export default class ColumnsArea extends ImmutablePureComponent {
};

componentDidMount() {
if (!this.props.singleColumn) {
this.node.addEventListener('wheel', this.handleWheel, supportsPassiveEvents ? { passive: true } : false);
}

if (this.mediaQuery) {
if (this.mediaQuery.addEventListener) {
this.mediaQuery.addEventListener('change', this.handleLayoutChange);
Expand All @@ -87,23 +81,7 @@ export default class ColumnsArea extends ImmutablePureComponent {
this.isRtlLayout = document.getElementsByTagName('body')[0].classList.contains('rtl');
}

UNSAFE_componentWillUpdate(nextProps) {
if (this.props.singleColumn !== nextProps.singleColumn && nextProps.singleColumn) {
this.node.removeEventListener('wheel', this.handleWheel);
}
}

componentDidUpdate(prevProps) {
if (this.props.singleColumn !== prevProps.singleColumn && !this.props.singleColumn) {
this.node.addEventListener('wheel', this.handleWheel, supportsPassiveEvents ? { passive: true } : false);
}
}

componentWillUnmount () {
if (!this.props.singleColumn) {
this.node.removeEventListener('wheel', this.handleWheel);
}

if (this.mediaQuery) {
if (this.mediaQuery.removeEventListener) {
this.mediaQuery.removeEventListener('change', this.handleLayoutChange);
Expand All @@ -116,22 +94,14 @@ export default class ColumnsArea extends ImmutablePureComponent {
handleChildrenContentChange() {
if (!this.props.singleColumn) {
const modifier = this.isRtlLayout ? -1 : 1;
this._interruptScrollAnimation = scrollRight(this.node, (this.node.scrollWidth - window.innerWidth) * modifier);
scrollRight(this.node, (this.node.scrollWidth - window.innerWidth) * modifier);
}
}

handleLayoutChange = (e) => {
this.setState({ renderComposePanel: !e.matches });
};

handleWheel = () => {
if (typeof this._interruptScrollAnimation !== 'function') {
return;
}

this._interruptScrollAnimation();
};

setRef = (node) => {
this.node = node;
};
Expand Down
27 changes: 17 additions & 10 deletions app/javascript/mastodon/scroll.ts
Original file line number Diff line number Diff line change
Expand Up @@ -38,13 +38,20 @@ const scroll = (
const isScrollBehaviorSupported =
'scrollBehavior' in document.documentElement.style;

export const scrollRight = (node: Element, position: number) => {
if (isScrollBehaviorSupported)
node.scrollTo({ left: position, behavior: 'smooth' });
else scroll(node, 'scrollLeft', position);
};

export const scrollTop = (node: Element) => {
if (isScrollBehaviorSupported) node.scrollTo({ top: 0, behavior: 'smooth' });
else scroll(node, 'scrollTop', 0);
};
export const scrollRight = (node: Element, position: number) =>
requestIdleCallback(() => {
if (isScrollBehaviorSupported) {
node.scrollTo({ left: position, behavior: 'smooth' });
} else {
scroll(node, 'scrollLeft', position);
}
});

export const scrollTop = (node: Element) =>
requestIdleCallback(() => {
if (isScrollBehaviorSupported) {
node.scrollTo({ top: 0, behavior: 'smooth' });
} else {
scroll(node, 'scrollTop', 0);
}
});
8 changes: 8 additions & 0 deletions app/javascript/mastodon/test_helpers.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,14 @@ import { render as rtlRender } from '@testing-library/react';

import { IdentityContext } from './identity_context';

beforeEach(() => {
global.requestIdleCallback = jest
.fn()
.mockImplementation((fn: () => void) => {
fn();
});
});

function render(
ui: React.ReactElement,
{
Expand Down

0 comments on commit ef4d6ab

Please sign in to comment.