2023-05-23 17:15:17 +02:00
|
|
|
import PropTypes from 'prop-types';
|
2023-05-23 11:47:36 +02:00
|
|
|
import { PureComponent } from 'react';
|
2023-05-22 15:48:01 +02:00
|
|
|
import { createPortal } from 'react-dom';
|
2023-05-23 17:15:17 +02:00
|
|
|
|
2020-12-30 23:18:11 +01:00
|
|
|
import { fromJS } from 'immutable';
|
2023-05-23 17:15:17 +02:00
|
|
|
|
2021-10-14 20:44:59 +02:00
|
|
|
import { ImmutableHashtag as Hashtag } from 'mastodon/components/hashtag';
|
2023-05-23 17:15:17 +02:00
|
|
|
import MediaGallery from 'mastodon/components/media_gallery';
|
2019-09-29 14:30:58 +02:00
|
|
|
import ModalRoot from 'mastodon/components/modal_root';
|
2023-05-23 17:15:17 +02:00
|
|
|
import Poll from 'mastodon/components/poll';
|
|
|
|
import Audio from 'mastodon/features/audio';
|
|
|
|
import Card from 'mastodon/features/status/components/card';
|
2019-09-29 14:30:58 +02:00
|
|
|
import MediaModal from 'mastodon/features/ui/components/media_modal';
|
|
|
|
import Video from 'mastodon/features/video';
|
2023-06-02 15:00:27 +02:00
|
|
|
import { IntlProvider } from 'mastodon/locales';
|
2023-05-23 17:15:17 +02:00
|
|
|
import { getScrollbarWidth } from 'mastodon/utils/scrollbar';
|
2018-03-24 12:52:26 +01:00
|
|
|
|
2019-08-23 22:38:02 +02:00
|
|
|
const MEDIA_COMPONENTS = { MediaGallery, Video, Card, Poll, Hashtag, Audio };
|
2018-05-12 15:30:06 +02:00
|
|
|
|
|
|
|
export default class MediaContainer extends PureComponent {
|
2018-03-24 12:52:26 +01:00
|
|
|
|
|
|
|
static propTypes = {
|
2018-05-12 15:30:06 +02:00
|
|
|
components: PropTypes.object.isRequired,
|
2018-03-24 12:52:26 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
state = {
|
|
|
|
media: null,
|
|
|
|
index: null,
|
2023-05-11 12:41:55 +02:00
|
|
|
lang: null,
|
2018-05-13 13:48:14 +02:00
|
|
|
time: null,
|
2020-11-27 15:41:58 +01:00
|
|
|
backgroundColor: null,
|
2020-12-30 23:18:11 +01:00
|
|
|
options: null,
|
2018-03-24 12:52:26 +01:00
|
|
|
};
|
|
|
|
|
2023-05-11 12:41:55 +02:00
|
|
|
handleOpenMedia = (media, index, lang) => {
|
2018-07-31 01:14:33 +02:00
|
|
|
document.body.classList.add('with-modals--active');
|
2019-07-19 09:25:22 +02:00
|
|
|
document.documentElement.style.marginRight = `${getScrollbarWidth()}px`;
|
|
|
|
|
2023-05-11 12:41:55 +02:00
|
|
|
this.setState({ media, index, lang });
|
2023-01-30 01:45:35 +01:00
|
|
|
};
|
2018-03-24 12:52:26 +01:00
|
|
|
|
2023-05-11 12:41:55 +02:00
|
|
|
handleOpenVideo = (lang, options) => {
|
2020-12-30 23:18:11 +01:00
|
|
|
const { components } = this.props;
|
2022-03-06 22:51:40 +01:00
|
|
|
const { media } = JSON.parse(components[options.componentIndex].getAttribute('data-props'));
|
2020-12-30 23:18:11 +01:00
|
|
|
const mediaList = fromJS(media);
|
2018-05-13 13:48:14 +02:00
|
|
|
|
2018-07-31 01:14:33 +02:00
|
|
|
document.body.classList.add('with-modals--active');
|
2019-07-19 09:25:22 +02:00
|
|
|
document.documentElement.style.marginRight = `${getScrollbarWidth()}px`;
|
|
|
|
|
2023-05-11 12:41:55 +02:00
|
|
|
this.setState({ media: mediaList, lang, options });
|
2023-01-30 01:45:35 +01:00
|
|
|
};
|
2018-05-13 13:48:14 +02:00
|
|
|
|
2018-03-24 12:52:26 +01:00
|
|
|
handleCloseMedia = () => {
|
2018-07-31 01:14:33 +02:00
|
|
|
document.body.classList.remove('with-modals--active');
|
2023-04-05 10:57:36 +02:00
|
|
|
document.documentElement.style.marginRight = '0';
|
2019-07-19 09:25:22 +02:00
|
|
|
|
2020-11-27 15:41:58 +01:00
|
|
|
this.setState({
|
|
|
|
media: null,
|
|
|
|
index: null,
|
|
|
|
time: null,
|
|
|
|
backgroundColor: null,
|
2020-12-30 23:18:11 +01:00
|
|
|
options: null,
|
2020-11-27 15:41:58 +01:00
|
|
|
});
|
2023-01-30 01:45:35 +01:00
|
|
|
};
|
2020-11-27 15:41:58 +01:00
|
|
|
|
|
|
|
setBackgroundColor = color => {
|
|
|
|
this.setState({ backgroundColor: color });
|
2023-01-30 01:45:35 +01:00
|
|
|
};
|
2018-03-24 12:52:26 +01:00
|
|
|
|
|
|
|
render () {
|
2023-06-02 15:00:27 +02:00
|
|
|
const { components } = this.props;
|
2018-03-24 12:52:26 +01:00
|
|
|
|
2023-05-23 14:28:21 +02:00
|
|
|
let handleOpenVideo;
|
|
|
|
|
|
|
|
// Don't offer to expand the video in a lightbox if we're in a frame
|
|
|
|
if (window.self === window.top) {
|
|
|
|
handleOpenVideo = this.handleOpenVideo;
|
|
|
|
}
|
|
|
|
|
2018-03-24 12:52:26 +01:00
|
|
|
return (
|
2023-06-02 15:00:27 +02:00
|
|
|
<IntlProvider>
|
2023-05-23 11:47:36 +02:00
|
|
|
<>
|
2024-02-27 16:42:05 +01:00
|
|
|
{Array.from(components).map((component, i) => {
|
2018-05-12 15:30:06 +02:00
|
|
|
const componentName = component.getAttribute('data-component');
|
|
|
|
const Component = MEDIA_COMPONENTS[componentName];
|
2019-08-18 14:55:32 +02:00
|
|
|
const { media, card, poll, hashtag, ...props } = JSON.parse(component.getAttribute('data-props'));
|
2018-05-12 15:30:06 +02:00
|
|
|
|
|
|
|
Object.assign(props, {
|
2019-08-18 14:55:32 +02:00
|
|
|
...(media ? { media: fromJS(media) } : {}),
|
|
|
|
...(card ? { card: fromJS(card) } : {}),
|
|
|
|
...(poll ? { poll: fromJS(poll) } : {}),
|
|
|
|
...(hashtag ? { hashtag: fromJS(hashtag) } : {}),
|
2018-05-13 13:48:14 +02:00
|
|
|
|
|
|
|
...(componentName === 'Video' ? {
|
2022-03-06 22:51:40 +01:00
|
|
|
componentIndex: i,
|
2023-05-23 14:28:21 +02:00
|
|
|
onOpenVideo: handleOpenVideo,
|
2018-05-13 13:48:14 +02:00
|
|
|
} : {
|
|
|
|
onOpenMedia: this.handleOpenMedia,
|
|
|
|
}),
|
2018-05-12 15:30:06 +02:00
|
|
|
});
|
2018-03-24 12:52:26 +01:00
|
|
|
|
2023-05-22 15:48:01 +02:00
|
|
|
return createPortal(
|
2018-05-13 13:48:14 +02:00
|
|
|
<Component {...props} key={`media-${i}`} />,
|
2018-05-12 15:30:06 +02:00
|
|
|
component,
|
2018-03-24 12:52:26 +01:00
|
|
|
);
|
|
|
|
})}
|
2019-08-18 14:55:32 +02:00
|
|
|
|
2020-11-27 15:41:58 +01:00
|
|
|
<ModalRoot backgroundColor={this.state.backgroundColor} onClose={this.handleCloseMedia}>
|
2018-05-13 13:48:14 +02:00
|
|
|
{this.state.media && (
|
2018-03-24 12:52:26 +01:00
|
|
|
<MediaModal
|
|
|
|
media={this.state.media}
|
2018-05-13 13:48:14 +02:00
|
|
|
index={this.state.index || 0}
|
2023-05-11 12:41:55 +02:00
|
|
|
lang={this.state.lang}
|
2020-12-30 23:18:11 +01:00
|
|
|
currentTime={this.state.options?.startTime}
|
|
|
|
autoPlay={this.state.options?.autoPlay}
|
|
|
|
volume={this.state.options?.defaultVolume}
|
2018-03-24 12:52:26 +01:00
|
|
|
onClose={this.handleCloseMedia}
|
2020-11-27 15:41:58 +01:00
|
|
|
onChangeBackgroundColor={this.setBackgroundColor}
|
2018-03-24 12:52:26 +01:00
|
|
|
/>
|
|
|
|
)}
|
|
|
|
</ModalRoot>
|
2023-05-23 11:47:36 +02:00
|
|
|
</>
|
2018-03-24 12:52:26 +01:00
|
|
|
</IntlProvider>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|