]> git.localhorst.tv Git - alttp.git/blobdiff - resources/js/components/episodes/Item.js
improved user context
[alttp.git] / resources / js / components / episodes / Item.js
index 8f1bafcb240508500e0336768c3a4af99d70b4ef..612c5ea85f6f76db13ca777f85a93fbebae70bf2 100644 (file)
 import PropTypes from 'prop-types';
 import React from 'react';
+import { Button } from 'react-bootstrap';
 import { useTranslation } from 'react-i18next';
+import { Link } from 'react-router-dom';
 
+import Channels from './Channels';
+import Crew from './Crew';
+import MultiLink from './MultiLink';
 import Players from './Players';
+import Icon from '../common/Icon';
+import { hasPassed, hasSGRestream, isActive } from '../../helpers/Episode';
+import { getLink } from '../../helpers/Event';
+import { canApplyForEpisode, canRestreamEpisode } from '../../helpers/permissions';
+import { useUser } from '../../hooks/user';
 
-const Item = ({ episode }) => {
+const Item = ({ episode, onAddRestream, onApply, onEditRestream }) => {
        const { t } = useTranslation();
+       const { user } = useUser();
 
-       return <div className="episodes-item d-flex align-items-start my-3 p-2 border rounded">
-               <div className="episode-start me-3 fs-4 text-end">
-                       {t('schedule.startTime', { date: new Date(episode.start) })}
+       const classNames = [
+               'episodes-item',
+               'my-3',
+               'p-2',
+               'border',
+               'rounded',
+       ];
+       if (isActive(episode)) {
+               classNames.push('is-active');
+       }
+
+       const style = React.useMemo(() => {
+               if (episode.event && episode.event.corner) {
+                       return {
+                               backgroundImage: `url(${episode.event.corner})`,
+                       };
+               }
+               return null;
+       }, [episode.event && episode.event.corner]);
+
+       const hasChannels = episode.channels && episode.channels.length;
+       const hasPlayers = episode.players && episode.players.length;
+
+       return <div className={classNames.join(' ')} style={style}>
+               <div className="d-flex align-items-stretch">
+                       <div className="episode-start me-3 fs-5 fs-md-4 text-end">
+                               {t('schedule.startTime', { date: new Date(episode.start) })}
+                       </div>
+                       <div className="episode-titlebar">
+                               {episode.title || episode.event ?
+                                       <div className="episode-title fs-5 fs-md-4">
+                                               {episode.title || episode.event.title}
+                                       </div>
+                               : null}
+                               {episode.comment ?
+                                       <div className="episode-comment">
+                                               {episode.comment}
+                                       </div>
+                               : null}
+                       </div>
+                       <div className="episode-channel-links ms-auto text-end">
+                               {hasChannels ?
+                                       <Channels
+                                               channels={episode.channels}
+                                               episode={episode}
+                                               onEditRestream={onEditRestream}
+                                       />
+                               : null}
+                               {!hasChannels && hasPlayers ?
+                                       <MultiLink players={episode.players} />
+                               : null}
+                               {episode.raceroom ?
+                                       <div>
+                                               <Button
+                                                       href={episode.raceroom}
+                                                       target="_blank"
+                                                       title={t('episodes.raceroom')}
+                                                       variant="outline-secondary"
+                                               >
+                                                       <Icon.RACETIME title="" />
+                                                       {' '}
+                                                       {t('episodes.raceroom')}
+                                               </Button>
+                                       </div>
+                               : null}
+                               {onAddRestream && canRestreamEpisode(user, episode) ?
+                                       <div>
+                                               <Button
+                                                       onClick={() => onAddRestream(episode)}
+                                                       title={t('episodes.addRestream')}
+                                                       variant="outline-secondary"
+                                               >
+                                                       <Icon.ADD title="" />
+                                               </Button>
+                                       </div>
+                               : null}
+                       </div>
                </div>
-               <div className="flex-fill">
-                       {episode.title ?
-                               <div className="episode-title fs-4">
-                                       {episode.title}
+               <div className="episode-body d-flex flex-column flex-fill">
+                       {hasPlayers ?
+                               <Players players={episode.players} />
+                       : null}
+                       {(episode.crew && episode.crew.length) || (!hasPassed(episode) && (
+                                       hasSGRestream(episode)
+                                       || canApplyForEpisode(user, episode, 'commentary')
+                                       || canApplyForEpisode(user, episode, 'tracking')
+                       )) ?
+                               <div className="mb-3">
+                                       <Crew episode={episode} onApply={onApply} />
                                </div>
                        : null}
                        {episode.event ?
-                               <div className="episode-event">
-                                       {episode.event.title}
+                               <div className="episode-event mt-auto">
+                                       {episode.event.description_id ?
+                                               <Link className="event-link" to={getLink(episode.event)}>
+                                                       {episode.event.title}
+                                               </Link>
+                                       :
+                                               episode.event.title
+                                       }
                                </div>
                        : null}
-                       <Players players={episode.players} />
                </div>
        </div>;
 };
 
 Item.propTypes = {
        episode: PropTypes.shape({
+               channels: PropTypes.arrayOf(PropTypes.shape({
+               })),
+               comment: PropTypes.string,
+               crew: PropTypes.arrayOf(PropTypes.shape({
+               })),
                event: PropTypes.shape({
+                       corner: PropTypes.string,
+                       description_id: PropTypes.number,
+                       name: PropTypes.string,
                        title: PropTypes.string,
                }),
                players: PropTypes.arrayOf(PropTypes.shape({
                })),
+               raceroom: PropTypes.string,
                start: PropTypes.string,
                title: PropTypes.string,
        }),
+       onAddRestream: PropTypes.func,
+       onApply: PropTypes.func,
+       onEditRestream: PropTypes.func,
 };
 
 export default Item;