* Move min-width to RaidCombobox, not Popover This fixes #318 * Use snapshots to make tokens reactive This fixes #319 * Revert ChatGPT refactor of this method Oops. This code was nice, but it didn't actually assign `false` to keys to be sent to the server. We will revisit this, but it needs to be fixed right now. This fixes #325 * Ignore gacha directory We will probably scrape these images soon. * Add translation for Auto Summon token * Add auto summon token to app state * Set battle settings in state on update Also renames PartyDetails to PartyFooter and makes description reactive * Stop 1password icon from appearing in name field * Use snapshot for reactive Edit party modal * Fix Edit modal placeholder colors * Fix bug with RaidCombobox and Farming Selecting farming then opening the raid combobox *twice* consecutively would put you in no segment, so no raids appeared Fixes #323 * Fix values staying in Edit team even if not saved The values a user entered in the Edit team modal would persist even if the user hit cancel to close the modal. They wouldn't save to the server, but very confusing nonetheless. Now fixed. * Fix unreadable colors in ElementToggle * Fix button alignment in weapon modal * Add text to filters button on small screens The FilterBar showed a left aligned filter icon on mobile for months and it was driving me insane * Remove extraneous code from Header Including the party name, since it's at the top now * Fix Alert at small sizes * Make copy link toast work again * Remove stylesheet links * Fix remix toasts and alerts from both locations The remix toast and alert was barely hooked up and not showing up when invoked from PartyHeader. It now shows up whether you remix your own team (from PartyDropdown) or if you remix another person's team (from PartyHeader).
194 lines
5.2 KiB
TypeScript
194 lines
5.2 KiB
TypeScript
import React, { useEffect, useState } from 'react'
|
|
import { useRouter } from 'next/router'
|
|
import { useSnapshot } from 'valtio'
|
|
import { useTranslation } from 'next-i18next'
|
|
import clonedeep from 'lodash.clonedeep'
|
|
|
|
import Linkify from 'react-linkify'
|
|
import LiteYouTubeEmbed from 'react-lite-youtube-embed'
|
|
import classNames from 'classnames'
|
|
import reactStringReplace from 'react-string-replace'
|
|
|
|
import GridRepCollection from '~components/GridRepCollection'
|
|
import GridRep from '~components/GridRep'
|
|
|
|
import api from '~utils/api'
|
|
import { appState } from '~utils/appState'
|
|
import { youtube } from '~utils/youtube'
|
|
|
|
import type { DetailsObject } from 'types'
|
|
|
|
import './index.scss'
|
|
|
|
// Props
|
|
interface Props {
|
|
party?: Party
|
|
new: boolean
|
|
editable: boolean
|
|
updateCallback: (details: DetailsObject) => void
|
|
}
|
|
|
|
const PartyFooter = (props: Props) => {
|
|
const { t } = useTranslation('common')
|
|
const router = useRouter()
|
|
|
|
const { party } = useSnapshot(appState)
|
|
|
|
const youtubeUrlRegex =
|
|
/(?:https:\/\/www\.youtube\.com\/watch\?v=|https:\/\/youtu\.be\/)([\w-]+)/g
|
|
|
|
const [open, setOpen] = useState(false)
|
|
|
|
const [remixes, setRemixes] = useState<Party[]>([])
|
|
const [embeddedDescription, setEmbeddedDescription] =
|
|
useState<React.ReactNode>()
|
|
|
|
useEffect(() => {
|
|
// Extract the video IDs from the description
|
|
if (party.description) {
|
|
const videoIds = extractYoutubeVideoIds(party.description)
|
|
|
|
// Fetch the video titles for each ID
|
|
const fetchPromises = videoIds.map(({ id }) => fetchYoutubeData(id))
|
|
|
|
// Wait for all the video titles to be fetched
|
|
Promise.all(fetchPromises).then((videoTitles) => {
|
|
// Replace the video URLs in the description with LiteYoutubeEmbed elements
|
|
const newDescription = reactStringReplace(
|
|
party.description,
|
|
youtubeUrlRegex,
|
|
(match, i) => (
|
|
<LiteYouTubeEmbed
|
|
key={`${match}-${i}`}
|
|
id={match}
|
|
title={videoTitles[i]}
|
|
wrapperClass="YoutubeWrapper"
|
|
playerClass="PlayerButton"
|
|
/>
|
|
)
|
|
)
|
|
|
|
// Update the state with the new description
|
|
setEmbeddedDescription(newDescription)
|
|
})
|
|
} else {
|
|
setEmbeddedDescription('')
|
|
}
|
|
}, [party.description])
|
|
|
|
async function fetchYoutubeData(videoId: string) {
|
|
return await youtube
|
|
.getVideoById(videoId, { maxResults: 1 })
|
|
.then((data) => data.items[0].snippet.localized.title)
|
|
}
|
|
|
|
// Methods: Navigation
|
|
function goTo(shortcode?: string) {
|
|
if (shortcode) router.push(`/p/${shortcode}`)
|
|
}
|
|
|
|
function extractYoutubeVideoIds(text: string) {
|
|
// Initialize an array to store the video IDs
|
|
const videoIds = []
|
|
|
|
// Use the regular expression to find all the Youtube URLs in the text
|
|
let match
|
|
while ((match = youtubeUrlRegex.exec(text)) !== null) {
|
|
// Extract the video ID from the URL
|
|
const videoId = match[1]
|
|
|
|
// Add the video ID to the array, along with the character position of the URL
|
|
videoIds.push({
|
|
id: videoId,
|
|
url: match[0],
|
|
position: match.index,
|
|
})
|
|
}
|
|
|
|
// Return the array of video IDs
|
|
return videoIds
|
|
}
|
|
|
|
// Methods: Favorites
|
|
function toggleFavorite(teamId: string, favorited: boolean) {
|
|
if (favorited) unsaveFavorite(teamId)
|
|
else saveFavorite(teamId)
|
|
}
|
|
|
|
function saveFavorite(teamId: string) {
|
|
api.saveTeam({ id: teamId }).then((response) => {
|
|
if (response.status == 201) {
|
|
const index = remixes.findIndex((p) => p.id === teamId)
|
|
const party = remixes[index]
|
|
|
|
party.favorited = true
|
|
|
|
let clonedParties = clonedeep(remixes)
|
|
clonedParties[index] = party
|
|
|
|
setRemixes(clonedParties)
|
|
}
|
|
})
|
|
}
|
|
|
|
function unsaveFavorite(teamId: string) {
|
|
api.unsaveTeam({ id: teamId }).then((response) => {
|
|
if (response.status == 200) {
|
|
const index = remixes.findIndex((p) => p.id === teamId)
|
|
const party = remixes[index]
|
|
|
|
party.favorited = false
|
|
|
|
let clonedParties = clonedeep(remixes)
|
|
clonedParties[index] = party
|
|
|
|
setRemixes(clonedParties)
|
|
}
|
|
})
|
|
}
|
|
|
|
function renderRemixes() {
|
|
return remixes.map((party, i) => {
|
|
return (
|
|
<GridRep
|
|
id={party.id}
|
|
shortcode={party.shortcode}
|
|
name={party.name}
|
|
createdAt={new Date(party.created_at)}
|
|
raid={party.raid}
|
|
grid={party.weapons}
|
|
user={party.user}
|
|
favorited={party.favorited}
|
|
fullAuto={party.full_auto}
|
|
autoGuard={party.auto_guard}
|
|
key={`party-${i}`}
|
|
displayUser={true}
|
|
onClick={goTo}
|
|
onSave={toggleFavorite}
|
|
/>
|
|
)
|
|
})
|
|
}
|
|
|
|
const remixSection = () => {
|
|
return (
|
|
<section className="Remixes">
|
|
<h3>{t('remixes')}</h3>
|
|
{<GridRepCollection>{renderRemixes()}</GridRepCollection>}
|
|
</section>
|
|
)
|
|
}
|
|
|
|
return (
|
|
<>
|
|
<section className="FooterWrapper">
|
|
<section className="PartyFooter">
|
|
<Linkify>{embeddedDescription}</Linkify>
|
|
</section>
|
|
</section>
|
|
{remixes && remixes.length > 0 ? remixSection() : ''}
|
|
</>
|
|
)
|
|
}
|
|
|
|
export default PartyFooter
|