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