]> git.localhorst.tv Git - alttp.git/blob - resources/js/components/episodes/Item.js
d7c4bab4ff7d68245e87ff84f5a3cabe243fb554
[alttp.git] / resources / js / components / episodes / Item.js
1 import moment from 'moment';
2 import PropTypes from 'prop-types';
3 import React from 'react';
4 import { Button } from 'react-bootstrap';
5 import { useTranslation } from 'react-i18next';
6 import { Link } from 'react-router-dom';
7
8 import Channels from './Channels';
9 import Crew from './Crew';
10 import MultiLink from './MultiLink';
11 import Players from './Players';
12 import Icon from '../common/Icon';
13 import { canApplyForEpisode, canRestreamEpisode } from '../../helpers/permissions';
14 import { withUser } from '../../helpers/UserContext';
15
16 const isActive = episode => {
17         if (!episode.start) return false;
18         const now = moment();
19         const start = moment(episode.start).subtract(10, 'minutes');
20         const end = moment(episode.start).add(episode.estimate, 'seconds');
21         return start.isBefore(now) && end.isAfter(now);
22 };
23
24 const Item = ({ episode, onAddRestream, onApply, onEditRestream, user }) => {
25         const { t } = useTranslation();
26
27         const classNames = [
28                 'episodes-item',
29                 'd-flex',
30                 'align-items-stretch',
31                 'my-3',
32                 'p-2',
33                 'border',
34                 'rounded',
35         ];
36         if (isActive(episode)) {
37                 classNames.push('is-active');
38         }
39
40         const style = React.useMemo(() => {
41                 if (episode.event && episode.event.corner) {
42                         return {
43                                 backgroundImage: `url(${episode.event.corner})`,
44                         };
45                 }
46                 return null;
47         }, [episode.event && episode.event.corner]);
48
49         const hasChannels = episode.channels && episode.channels.length;
50         const hasPlayers = episode.players && episode.players.length;
51
52         return <div className={classNames.join(' ')} style={style}>
53                 <div className="episode-start me-3 fs-4 text-end">
54                         {t('schedule.startTime', { date: new Date(episode.start) })}
55                 </div>
56                 <div className="d-flex flex-column flex-fill">
57                         <div className="d-flex align-items-start justify-content-between">
58                                 <div>
59                                         {episode.title || episode.event ?
60                                                 <div className="episode-title fs-4">
61                                                         {episode.title || episode.event.title}
62                                                 </div>
63                                         : null}
64                                         {episode.comment ?
65                                                 <div className="episode-comment">
66                                                         {episode.comment}
67                                                 </div>
68                                         : null}
69                                 </div>
70                                 <div className="episode-channel-links text-end">
71                                         {hasChannels ?
72                                                 <Channels
73                                                         channels={episode.channels}
74                                                         episode={episode}
75                                                         onEditRestream={onEditRestream}
76                                                 />
77                                         : null}
78                                         {!hasChannels && hasPlayers ?
79                                                 <MultiLink players={episode.players} />
80                                         : null}
81                                         {onAddRestream && canRestreamEpisode(user, episode) ?
82                                                 <div>
83                                                         <Button
84                                                                 onClick={() => onAddRestream(episode)}
85                                                                 title={t('episodes.addRestream')}
86                                                                 variant="outline-secondary"
87                                                         >
88                                                                 <Icon.ADD title="" />
89                                                         </Button>
90                                                 </div>
91                                         : null}
92                                 </div>
93                         </div>
94                         {hasPlayers ?
95                                 <Players players={episode.players} />
96                         : null}
97                         {(episode.crew && episode.crew.length)
98                                         || canApplyForEpisode(user, episode, 'commentary')
99                                         || canApplyForEpisode(user, episode, 'tracking') ?
100                                 <div className="mb-3">
101                                         <Crew episode={episode} onApply={onApply} />
102                                 </div>
103                         : null}
104                         {episode.event ?
105                                 <div className="episode-event mt-auto">
106                                         <Link className="event-link" to={`/events/${episode.event.name}`}>
107                                                 {episode.event.title}
108                                         </Link>
109                                 </div>
110                         : null}
111                 </div>
112         </div>;
113 };
114
115 Item.propTypes = {
116         episode: PropTypes.shape({
117                 channels: PropTypes.arrayOf(PropTypes.shape({
118                 })),
119                 comment: PropTypes.string,
120                 crew: PropTypes.arrayOf(PropTypes.shape({
121                 })),
122                 event: PropTypes.shape({
123                         corner: PropTypes.string,
124                         name: PropTypes.string,
125                         title: PropTypes.string,
126                 }),
127                 players: PropTypes.arrayOf(PropTypes.shape({
128                 })),
129                 start: PropTypes.string,
130                 title: PropTypes.string,
131         }),
132         onAddRestream: PropTypes.func,
133         onApply: PropTypes.func,
134         onEditRestream: PropTypes.func,
135         user: PropTypes.shape({
136         }),
137 };
138
139 export default withUser(Item);