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 { canApplyForEpisode, canRestreamEpisode } from '../../helpers/permissions';
+import { withUser } from '../../helpers/UserContext';
-const Item = ({ episode }) => {
+const Item = ({ episode, onAddRestream, onApply, onEditRestream, user }) => {
const { t } = useTranslation();
- return <div className="episodes-item d-flex align-items-start my-3 p-2 border rounded">
+ const classNames = [
+ 'episodes-item',
+ 'd-flex',
+ 'align-items-stretch',
+ '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="episode-start me-3 fs-4 text-end">
{t('schedule.startTime', { date: new Date(episode.start) })}
</div>
- <div className="flex-fill">
- {episode.title ?
- <div className="episode-title fs-4">
- {episode.title}
+ <div className="d-flex flex-column flex-fill">
+ <div className="d-flex align-items-start justify-content-between">
+ <div>
+ {episode.title || episode.event ?
+ <div className="episode-title fs-4">
+ {episode.title || episode.event.title}
+ </div>
+ : null}
+ {episode.comment ?
+ <div className="episode-comment">
+ {episode.comment}
+ </div>
+ : null}
+ </div>
+ <div className="episode-channel-links text-end">
+ {hasChannels ?
+ <Channels
+ channels={episode.channels}
+ episode={episode}
+ onEditRestream={onEditRestream}
+ />
+ : null}
+ {!hasChannels && hasPlayers ?
+ <MultiLink players={episode.players} />
+ : 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>
+ {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={`/events/${episode.event.name}`}>
+ {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({
start: PropTypes.string,
title: PropTypes.string,
}),
+ onAddRestream: PropTypes.func,
+ onApply: PropTypes.func,
+ onEditRestream: PropTypes.func,
+ user: PropTypes.shape({
+ }),
};
-export default Item;
+export default withUser(Item);