Browse Source

fix(status): Content jump due to height changes (#3734)

closed-social-glitch-2
Sorin Davidoi 6 years ago
committed by Eugen Rochko
parent
commit
0f52e42c2d
2 changed files with 32 additions and 8 deletions
  1. +14
    -6
      app/javascript/mastodon/components/status.js
  2. +18
    -2
      app/javascript/mastodon/components/status_content.js

+ 14
- 6
app/javascript/mastodon/components/status.js View File

@ -41,6 +41,7 @@ class Status extends ImmutablePureComponent {
}; };
state = { state = {
isExpanded: false,
isIntersecting: true, // assume intersecting until told otherwise isIntersecting: true, // assume intersecting until told otherwise
isHidden: false, // set to true in requestIdleCallback to trigger un-render isHidden: false, // set to true in requestIdleCallback to trigger un-render
} }
@ -57,7 +58,7 @@ class Status extends ImmutablePureComponent {
'muted', 'muted',
] ]
updateOnStates = []
updateOnStates = ['isExpanded']
shouldComponentUpdate (nextProps, nextState) { shouldComponentUpdate (nextProps, nextState) {
if (!nextState.isIntersecting && nextState.isHidden) { if (!nextState.isIntersecting && nextState.isHidden) {
@ -112,12 +113,15 @@ class Status extends ImmutablePureComponent {
this.setState((prevState) => ({ isHidden: !prevState.isIntersecting })); this.setState((prevState) => ({ isHidden: !prevState.isIntersecting }));
} }
saveHeight = () => {
if (this.node && this.node.children.length !== 0) {
this.height = this.node.clientHeight;
}
}
handleRef = (node) => { handleRef = (node) => {
this.node = node; this.node = node;
if (node && node.children.length !== 0) {
this.height = node.clientHeight;
}
this.saveHeight();
} }
handleClick = () => { handleClick = () => {
@ -133,11 +137,15 @@ class Status extends ImmutablePureComponent {
} }
} }
handleExpandedToggle = () => {
this.setState({ isExpanded: !this.state.isExpanded });
};
render () { render () {
let media = null; let media = null;
let statusAvatar; let statusAvatar;
const { status, account, ...other } = this.props; const { status, account, ...other } = this.props;
const { isIntersecting, isHidden } = this.state;
const { isExpanded, isIntersecting, isHidden } = this.state;
if (status === null) { if (status === null) {
return null; return null;
@ -203,7 +211,7 @@ class Status extends ImmutablePureComponent {
</a> </a>
</div> </div>
<StatusContent status={status} onClick={this.handleClick} />
<StatusContent status={status} onClick={this.handleClick} expanded={isExpanded} onExpandedToggle={this.handleExpandedToggle} onHeightUpdate={this.saveHeight} />
{media} {media}

+ 18
- 2
app/javascript/mastodon/components/status_content.js View File

@ -15,6 +15,9 @@ class StatusContent extends React.PureComponent {
static propTypes = { static propTypes = {
status: ImmutablePropTypes.map.isRequired, status: ImmutablePropTypes.map.isRequired,
expanded: PropTypes.bool,
onExpandedToggle: PropTypes.func,
onHeightUpdate: PropTypes.func,
onClick: PropTypes.func, onClick: PropTypes.func,
}; };
@ -44,6 +47,12 @@ class StatusContent extends React.PureComponent {
} }
} }
componentDidUpdate () {
if (this.props.onHeightUpdate) {
this.props.onHeightUpdate();
}
}
onMentionClick = (mention, e) => { onMentionClick = (mention, e) => {
if (e.button === 0) { if (e.button === 0) {
e.preventDefault(); e.preventDefault();
@ -85,7 +94,13 @@ class StatusContent extends React.PureComponent {
handleSpoilerClick = (e) => { handleSpoilerClick = (e) => {
e.preventDefault(); e.preventDefault();
this.setState({ hidden: !this.state.hidden });
if (this.props.onExpandedToggle) {
// The parent manages the state
this.props.onExpandedToggle();
} else {
this.setState({ hidden: !this.state.hidden });
}
} }
setRef = (c) => { setRef = (c) => {
@ -94,7 +109,8 @@ class StatusContent extends React.PureComponent {
render () { render () {
const { status } = this.props; const { status } = this.props;
const { hidden } = this.state;
const hidden = this.props.onExpandedToggle ? !this.props.expanded : this.state.hidden;
const content = { __html: emojify(status.get('content')) }; const content = { __html: emojify(status.get('content')) };
const spoilerContent = { __html: emojify(escapeTextContentForBrowser(status.get('spoiler_text', ''))) }; const spoilerContent = { __html: emojify(escapeTextContentForBrowser(status.get('spoiler_text', ''))) };

Loading…
Cancel
Save