A suite to track Project Diva score statistics and ratings / D4DJ event data.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
projectdivar/frontend/src/App.js

1600 lines
142 KiB

import React, {useState,useEffect,useRef} from 'react';
import logo from './logo.svg';
import './App.css';
import {
BrowserRouter as Router,
Switch,
Route,
Link,
useRouteMatch,
useParams,
useHistory
} from "react-router-dom";
import {
Modal,
Button,
Form
} from "react-bootstrap";
const REMOTE_ADDR = "http://45.33.13.215:4502";
const axios = require('axios');
var IMAGE_CAMERA=(p)=>{
return(
<svg {...p} width="1em" height="1em" viewBox="0 0 16 16" className="bi bi-camera-fill link cursor" fill="currentColor" xmlns="http://www.w3.org/2000/svg">
<path d="M10.5 8.5a2.5 2.5 0 1 1-5 0 2.5 2.5 0 0 1 5 0z"/>
<path fillRule="evenodd" d="M2 4a2 2 0 0 0-2 2v6a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V6a2 2 0 0 0-2-2h-1.172a2 2 0 0 1-1.414-.586l-.828-.828A2 2 0 0 0 9.172 2H6.828a2 2 0 0 0-1.414.586l-.828.828A2 2 0 0 1 3.172 4H2zm.5 2a.5.5 0 1 0 0-1 .5.5 0 0 0 0 1zm9 2.5a3.5 3.5 0 1 1-7 0 3.5 3.5 0 0 1 7 0z"/>
</svg>
)
}
var RATING_cool=new Image();
RATING_cool.src="
var RATING_fine=new Image();
RATING_fine.src="
var RATING_safe=new Image();
RATING_safe.src="
var RATING_sad=new Image();
RATING_sad.src="
var RATING_worst=new Image();
RATING_worst.src="
function Logo(){
var note_arrow=new Image();
note_arrow.src=""
var note_circle=new Image();
var note_circle_shadow=new Image();
var notes=[new Image(),new Image(),new Image(),new Image()]
notes[0].src="
notes[1].src="
notes[2].src=""
notes[3].src="
var shadownotes=[new Image(),new Image(),new Image(),new Image()]
shadownotes[0].src="
shadownotes[1].src=""
shadownotes[2].src=""
shadownotes[3].src=""
var divar=new Image();
divar.src="
var logo;
var timer=0;
var startpoint = [Math.random()*1000-500,Math.random()*1000-500]
var scorepoint = [Math.random()*10-5,Math.random()*10-5]
var pos = [...startpoint];
var hitrating=-1;
var timer2=0;
var notetype=0;
setTimeout(()=>{
logo=document.getElementById("logo");
var draw = logo.getContext("2d");
draw.font="64px Open Sans Condensed";
var size=draw.measureText("Pr ject")
logo.width=size.width
setInterval(()=>{
if (hitrating==-1) {
pos[0]-=startpoint[0]/20;
pos[1]-=startpoint[1]/20;
if ((Math.abs(pos[0])<32 && Math.abs(pos[1])<32 && Math.random()*10<1)
|| (Math.abs(pos[0])<16 && Math.abs(pos[1])<16 && Math.random()*4<1)
|| (Math.abs(pos[0])<4 && Math.abs(pos[1])<4 && Math.random()*2<1)) {
scorepoint = [Math.random()*30-15,Math.random()*30-15]
var distance = Math.abs(pos[0])+Math.abs(pos[1])
if (distance>32) {
hitrating=4;
} else
if (distance>16) {
hitrating=3;
} else
if (distance>4) {
hitrating=2;
} else
if (distance>2) {
hitrating=1;
} else {
hitrating=0;
}
timer2=Math.random()*400+350;
}
if (timer>1300) {
hitrating=4;
timer2=350;
}
}
if (hitrating!=-1 && timer2<=0) {
hitrating=-1;
notetype=Math.floor(Math.random()*4)
startpoint = [Math.random()*1000-500,Math.random()*1000-500]
pos = [...startpoint];
timer=0;
}
var draw = logo.getContext("2d");
draw.clearRect(0, 0, logo.width, logo.height);
draw.font="64px Open Sans Condensed";
var size=draw.measureText("Pr ject")
draw.fillStyle="#000000";
draw.fillText("Pr ject",0,48);
draw.font="28px Open Sans Condensed";
draw.fillStyle="#FFFFFF";
draw.drawImage(divar,14,45,152,32);
if (hitrating==-1) {
draw.drawImage(shadownotes[notetype],70-19,14,36,36);
draw.drawImage(notes[notetype],70-19+pos[0],14+pos[1],36,36);
draw.save();
drawImage(note_arrow,70+22-4-19,14-19+38,0.5,((timer/50)*18)*(Math.PI/180));
draw.restore();
} else {
switch (hitrating) {
case 0:{
draw.drawImage(RATING_cool,70-19+scorepoint[0],14+scorepoint[1]);
}break;
case 1:{
draw.drawImage(RATING_fine,70-19+scorepoint[0],14+scorepoint[1]);
}break;
case 2:{
draw.drawImage(RATING_safe,70-19+scorepoint[0],14+scorepoint[1]);
}break;
case 3:{
draw.drawImage(RATING_sad,70-19+scorepoint[0],14+scorepoint[1]);
}break;
case 4:{
draw.drawImage(RATING_worst,70-19+scorepoint[0],14+scorepoint[1]);
}break;
}
}
timer+=50;
timer2-=50;
//p.setSiteCounter(p.siteCounter+1);
},50)
},300);
function drawImage(image, x, y, scale, rotation){
var draw = logo.getContext("2d");
draw.setTransform(scale, 0, 0, scale, x, y); // sets scale and origin
draw.rotate(rotation);
draw.drawImage(image, -image.width / 2, -image.height / 2);
}
return (<>
<canvas className="homelink" id="logo" width="0" height="84"/>
</>);
}
function Sort(p){
let { sort,sortOrder } = useParams();
return (
<>
<Link to={"/rankings/"+p.order+"/"+((p.order===sort)?(sortOrder==="desc")?"asc":"desc":"desc")} onClick={()=>{p.setIsLoading(true);p.setUpdateUsers(!p.updateUsers)}}>{p.label}</Link>{(sort===p.order?<>{(sortOrder==="desc")?<>▼</>:<></>}</>:<></>)}
</>
)
}
function ProfileDataContainer(p){
return (
<div className={"col-md-"+p.width+" border"} onMouseOver={()=>{if (p.setMouseOver) {p.setMouseOver(true)}}} onMouseOut={()=>{if(p.setMouseOver){p.setMouseOver(false)}}}>
<div className="row">
<div className="text-center label col-6 col-md-12">
{p.label}
</div>
<div className="data col-6 col-md-12 text-center">
{p.data}
</div>
</div>
</div>
);
}
function StatisticsPanel(p) {
return (
<>
<div className="row">
<div className="col-md-1"></div>
<ProfileDataContainer label="Play Count" data={p.playcount} width="2"/>
<ProfileDataContainer label="FC Count" data={p.fccount} width="2"/>
<ProfileDataContainer setMouseOver={p.setMouseOver} label="Cleared" data={p.cleared} width="4"/>
<ProfileDataContainer label="Accuracy" data={p.accuracy} width="2"/>
<div className="col-md-1"></div>
</div>
</>
)
}
function HitCountsPanel(p) {
return (
<>
<div className="row text-center">
<div className="col-md-1">
</div>
<div className="col-md-2 border">
<div className="row">
<div className="col-6 col-md-12">
<img src={RATING_cool.src} height="16"/>
</div>
<div className="col-6 col-md-12">
{p.user.cool}
</div>
</div>
</div>
<div className="col-md-2 border">
<div className="row">
<div className="col-6 col-md-12">
<img src={RATING_fine.src} height="16"/>
</div>
<div className="col-6 col-md-12">
{p.user.fine}
</div>
</div>
</div>
<div className="col-md-2 border">
<div className="row">
<div className="col-6 col-md-12">
<img src={RATING_safe.src} height="16"/>
</div>
<div className="col-6 col-md-12">
{p.user.safe}
</div>
</div>
</div>
<div className="col-md-2 border">
<div className="row">
<div className="col-6 col-md-12">
<img src={RATING_sad.src} height="16"/>
</div>
<div className="col-6 col-md-12">
{p.user.sad}
</div>
</div>
</div>
<div className="col-md-2 border">
<div className="row">
<div className="col-6 col-md-12">
<img src={RATING_worst.src} height="16"/>
</div>
<div className="col-6 col-md-12">
{p.user.worst}
</div>
</div>
</div>
<div className="col-md-1">
</div>
</div>
</>
)
}
function SongName(p){
//console.log(p.song)
return (
<>
{(p.song)?(p.song.english_name===p.song.name)?
<>{p.song.name}</>
:
<>{p.song.name}<br/>{(p.song.romanized_name.length>0)?p.song.romanized_name:p.song.english_name}</>
:<></>
}
</>
)
}
function CalculateBadge(difficulty) {
switch (difficulty) {
case "E":{
return "primary";
}break;
case "N":{
return "info";
}break;
case "H":{
return "success";
}break;
case "EX":{
return "warning";
}break;
case "EXEX":{
return "danger";
}break;
}
}
function FormatRating(rating) {
if (rating) {
if (rating.includes(".5")) {
return rating;
} else {
return Math.floor(rating)
}
}
}
function Difficulty(p) {
return (
<>
<span className={"badge badge-"+CalculateBadge(p.play.difficulty)}>{(p.song && p.song.rating && p.song.rating[p.play.difficulty])?FormatRating(p.song.rating[p.play.difficulty]):<></>} {p.play.difficulty}</span>
</>
);
}
function ImageDisplayer(p) {
var [zoom,setZoom] = useState(false)
//p.play
return(
<Modal
size="xl"
aria-labelledby="contained-modal-title-vcenter"
centered
scrollable
show={p.modalVisible}
onHide={()=>{p.setModalVisible(false)}}
>
{(p.play!==undefined&&p.songs[p.play.songid])?<><Modal.Header closeButton>
<Modal.Title id="contained-modal-title-vcenter">
{`${p.username}'s ${p.songs[p.play.songid].name} Play - [${p.play.difficulty}] ${p.play.percent}%`}
</Modal.Title>
</Modal.Header>
<Modal.Body>
<img src={p.play.src} width={(zoom)?"auto":"100%"} className={(zoom)?"zoomout":"zoomin"} onClick={()=>{setZoom(!zoom)}}/>
</Modal.Body></>:<></>}
</Modal>
)
}
function Play(p) {
function GetModifiedDiff(name) {
switch (name) {
case "E":{
return "easy";
}
case "N":{
return "normal";
}
case "H":{
return "hard";
}
case "EX":{
return "ex";
}
case "EXEX":{
return "exex";
}
}
}
function GetDateDiff() {
var hours = Math.floor((Date.now()-new Date(p.play.date))/1000/60/60);
var days = Math.floor(hours/24)
if (hours<24) {return <>{hours} {"hour"+((hours!==1)?"s":"")} ago</>}
return <>{days} {"day"+((days!==1)?"s":"")} ago</>
}
function GetDateDisplay(timeStampBelow) {
var date = new Date(p.play.date);
var months = ["Jan","Feb","Mar","Apr","May","Jun","Jul","Aug","Sep","Oct","Nov","Dec"]
return <>{months[date.getMonth()]+" "+date.getDate()+" "+date.getFullYear()+" "+date.getHours()+":"+((date.getMinutes()<10)?"0"+date.getMinutes():date.getMinutes())}<span className={"tinytime infront "+((timeStampBelow)?"moveDown":"")}>{GetDateDiff()}</span></>
}
if (p.mini) {
return (
<>
<div className={"d-none d-md-block below border border-bottom "+((p.play.src)?"background-songs-click":"")} onClick={(p.play.src)?()=>{p.setModalSrc(p.play)
p.setModalVisible(true)}:null}>
<div className={((p.play.fine==0&&p.play.safe==0&&p.play.sad==0&&p.play.worst==0)?"pfchighlight":(p.play.safe==0&&p.play.sad==0&&p.play.worst==0)?"fchighlight":"")}>
<div className={"row align-middle"}>
<div className="col-md-2 order-1 order-md-1 text-center border-right align-middle text-nowrap overflow-hidden">{(p.title&&p.song)?<>{p.song.name}<br/></>:<></>}{((p.play.fine==0&&p.play.safe==0&&p.play.sad==0&&p.play.worst==0)?<span className="badge pfc">PFC</span>:(p.play.safe==0&&p.play.sad==0&&p.play.worst==0)?<span className="badge fc">FC</span>:<></>)} <Difficulty play={p.play} song={p.song}/> {Math.floor(p.play.score)} pts</div>
<div className="col-md-3 order-3 order-md-2 text-center border-right align-middle text-nowrap overflow-hidden">{GetDateDisplay()}</div>
<div className="col-md-7 order-2 order-md-3">
<div className="row">
<div className="col-12 order-1 order-md-1 col-md-5 text-center">
{p.play.cool+"/"+p.play.fine+"/"+p.play.safe+"/"+p.play.sad+"/"+p.play.worst}
</div>
<div className="col-6 order-3 order-md-2 col-md-2 text-left text-md-left">
{(p.play.mod!==null&&p.play.mod.length>0)?
<ModDisplay side={true} badge={CalculateBadge(p.play.difficulty)} diff={GetModifiedDiff(p.play.difficulty)}
hs={p.play.mod=="HS"?1:0} hd={p.play.mod=="HD"?1:0} sd={p.play.mod=="SD"?1:0}/>
:<></>
}
</div>
<div className="col-6 order-2 order-md-3 col-md-3 text-right text-md-left">
<b>{p.play.percent}%</b>
</div>
<div className="col-6 order-2 order-md-3 col-md-2 text-right text-md-left">
{(p.play.src)?<IMAGE_CAMERA/>:<></>}
</div>
</div>
</div>
</div>
</div>
</div>
<div className={"d-block d-small d-md-none "+((p.play.src)?"background-songs-click":"")} onClick={(p.play.src)?()=>{p.setModalSrc(p.play)
p.setModalVisible(true)}:null}>
<div className={((p.play.fine==0&&p.play.safe==0&&p.play.sad==0&&p.play.worst==0)?"pfchighlight":(p.play.safe==0&&p.play.sad==0&&p.play.worst==0)?"fchighlight":"")}>{(p.title&&p.song)?<>{p.song.name}<br/></>:<></>}
{<div className="row"><div className="offset-4 col-4 text-center">{((p.play.fine==0&&p.play.safe==0&&p.play.sad==0&&p.play.worst==0)?<span className="badge pfc">PFC</span>:(p.play.safe==0&&p.play.sad==0&&p.play.worst==0)?<span className="badge fc">FC</span>:<></>)}<Difficulty play={p.play} song={p.song}/></div></div>}
<div className="row">
<div className="offset-2 col-4 text-center">
{Math.floor(p.play.score)} pts
<br/>
{p.play.cool+"/"+p.play.fine+"/"+p.play.safe+"/"+p.play.sad+"/"+p.play.worst}
</div>
<div className="col-4 text-center">
<b>{p.play.percent}%</b> {(p.play.mod!==null&&p.play.mod.length>0)?<ModDisplay side={true} badge={CalculateBadge(p.play.difficulty)} diff={GetModifiedDiff(p.play.difficulty)}
hs={p.play.mod=="HS"?1:0} hd={p.play.mod=="HD"?1:0} sd={p.play.mod=="SD"?1:0}/>:<></>}
<span className="pl-2"/>
{(p.play.src)?<IMAGE_CAMERA/>:<></>}
<br/>
{GetDateDisplay(true)}
</div>
</div>
</div>
</div>
</>
);
} else {
return (
<>
<div className={"row align-middle "+((p.play.fine==0&&p.play.safe==0&&p.play.sad==0&&p.play.worst==0)?"pfchighlight":(p.play.safe==0&&p.play.sad==0&&p.play.worst==0)?"fchighlight":"")+((p.play.src)?" background-songs-click":"")} onClick={(p.play.src)?()=>{p.setModalSrc(p.play)
p.setModalVisible(true)}:null}>
{(p.index!==undefined)?<div className=" col-md-1 text-center border-right align-middle text-nowrap overflow-hidden"><span className="d-none d-md-block">{p.index+1}</span>{((p.play.fine==0&&p.play.safe==0&&p.play.sad==0&&p.play.worst==0)?<span className="badge pfc">PFC</span>:(p.play.safe==0&&p.play.sad==0&&p.play.worst==0)?<span className="badge fc">FC</span>:<></>)}</div>:<></>}
<div className="col-md-3 text-center border-right align-middle text-nowrap overflow-hidden"><SongName song={p.song}/><span className="tinytime">{GetDateDiff()}</span></div>
<div className="col-md-2 text-center border-right align-middle text-nowrap overflow-hidden">{Math.floor(p.play.score)} pts<br/><Difficulty play={p.play} song={p.song}/></div>
<div className="col-md-6">
<div className="row">
<div className="order-1 order-md-1 col-md-4 numbers text-center">
<div className="row justify-content-center">
<div className="col-4 col-md-5">
<img src={RATING_cool.src} className="pr-2" height="16"/>
</div>
<div className="col-4 col-md-7">
{p.play.cool}
</div>
</div>
</div>
<div className="order-3 order-md-2 col-md-4 numbers text-center">
<div className="row justify-content-center">
<div className="col-4 col-md-5">
<img src={RATING_safe.src} className="pr-2" height="16"/>
</div>
<div className="col-4 col-md-7">
{p.play.safe}
</div>
</div></div>
<div className="order-5 order-md-3 col-md-4 numbers text-center">
<div className="row justify-content-center">
<div className="col-4 col-md-5">
<img src={RATING_worst.src} className="pr-2" height="16"/>
</div>
<div className="col-4 col-md-7">
{p.play.worst}
</div>
</div></div>
<div className="order-2 order-md-4 order-sm-2 col-md-4 numbers text-center">
<div className="row justify-content-center">
<div className="col-4 col-md-5">
<img src={RATING_fine.src} className="pr-2" height="16"/>
</div>
<div className="col-4 col-md-7">
{p.play.fine}
</div>
</div></div>
<div className="order-4 order-md-5 col-md-4 numbers text-center">
<div className="row justify-content-center">
<div className="col-4 col-md-5">
<img src={RATING_sad.src} className="pr-2" height="16"/>
</div>
<div className="col-4 col-md-7">
{p.play.sad}
</div>
</div></div>
<div className="order-6 order-md-6 col-md-4 numbers text-center"><b>{p.play.percent}% {(p.play.src)?<span className="pl-4"><IMAGE_CAMERA onClick={()=>{p.setModalSrc(p.play)
p.setModalVisible(true)}}/></span>:<></>}</b></div>
</div>
</div>
</div>
</>
);
}
}
function LoadMore(p) {
var [offset,setOffset] = useState(p.params.limit)
var [params,setParams] = useState(p.params)
var [loading,setLoading] = useState(false)
var [visible,setVisible] = useState(false)
var [update,setUpdate] = useState(false)
useEffect(()=>{
axios.get(p.url+constructParams(params))
.then((data)=>{
if (data.data.length>0) {
setVisible(true)
}
})
},[update])
function constructParams(params) {
var st = Object.keys(params).reduce((str,key)=>{
if (str==="") {
return str+="?"+key+"="+params[key]
} else {
return str+="&"+key+"="+params[key]
}
},"")
return st
}
if (!visible){
return (<></>)
}
else
if (p.listItem) {
return (
<li className="list-group-item list-group-item-action homelink" onClick={()=>{
setLoading(true)
axios.get(p.url+constructParams(params))
.then((data)=>{
if (data.data.length>0) {
var newVals = [...p.value]
data.data.forEach((val)=>{
newVals.push(val)
})
p.setValue(newVals)
//console.log(newVals)
var obj = {...params}
obj.offset+=offset
setParams(obj)
setLoading(false)
if (data.data.length<p.params.limit) {
setVisible(false)
}
} else {
setVisible(false)
setLoading(false)
}
})
}
}>
<div className="row text-center">
<div className="col-12">
{(!loading)?<h6 className="link"><i>- Load More -</i></h6>:<div className="spinner-grow text-primary" role="status"><span className="sr-only">Loading...</span></div>
}
</div>
</div>
</li>
)
} else
return (
<div className="row text-center homelink" onClick={()=>{
setLoading(true)
axios.get(p.url+constructParams(params))
.then((data)=>{
if (!data.data) {
setVisible(false)
} else
if (data.data.length>0) {
var newVals = [...p.value]
data.data.forEach((val)=>{
newVals.push(val)
})
p.setValue(newVals)
//console.log(newVals)
var obj = {...params}
obj.offset+=offset
setParams(obj)
setLoading(false)
if (data.data.length<p.params.limit) {
setVisible(false)
}
} else {
setVisible(false)
setLoading(false)
}
})
}
}>
<div className="col-12">
{(!loading)?<h6 className="link"><i>- Load More -</i></h6>:<div className="spinner-grow text-primary" role="status"><span className="sr-only">Loading...</span></div>
}
</div>
</div>
)
}
function BestPlaysPanel(p) {
var [bestPlays,setBestPlays] = useState([])
var [update,setUpdate] = useState(false)
useEffect(()=>{
axios.get("http://www.projectdivar.com/bestplays/"+p.username+"?fails=false&limit=5&offset=0")
.then((data)=>{setBestPlays(data.data);})
},[update])
var content=<div className="col-md-12 mt-3 mb-3">
<ul className="list-group list-group-flush border border-danger rounded-lg">
{bestPlays.map((play,i)=>{return <li key={play.id} className={"list-group-item list-group-item-action "+(i%2==0?"background-list-1":"background-list-2")}>
<Play setModalSrc={p.setModalSrc} setModalVisible={p.setModalVisible} index={i} play={play} song={p.songs[play.songid]}/>
</li>})}
<LoadMore listItem={true} url={"http://www.projectdivar.com/bestplays/"+p.username} params={{fails:false,limit:15,offset:5}} value={bestPlays} setValue={setBestPlays}/>
</ul>
</div>
return (
<>
<div className="row ml-3 mr-3">
{content}
</div>
</>
);
}
function ModDisplay(p) {
const[tooltip,setTooltip] = useState("")
const[visibility,setVisibility] = useState(false)
return (
<span className={"border border-"+p.badge+" rounded "+p.diff+"-background"} style={{fontSize:"18px"}}>
{(p.hs>0)?<span style={{color:"#b33"}} onMouseOver={()=>{setTooltip("High Speed - "+p.diff.toUpperCase());setVisibility(true)}} onMouseOut={()=>{setVisibility(false)}}></span>:<></>}
{(p.hd>0)?<span onMouseOver={()=>{setTooltip("Hidden - "+p.diff.toUpperCase());setVisibility(true)}} onMouseOut={()=>{setVisibility(false)}} style={{color:"#968a0e"}}></span>:<></>}
{(p.sd>0)?<span onMouseOver={()=>{setTooltip("Sudden - "+p.diff.toUpperCase());setVisibility(true)}} onMouseOut={()=>{setVisibility(false)}} style={{color:"#49b"}}></span>:<></>}
{(visibility)?<span style={{position:"absolute"}} className={((p.side)?"display-tooltipside":"display-tooltip")+" alert alert-dark "+p.diff+"-background"}>{tooltip}</span>:<></>}
</span>
)
}
function PlayDetail(p) {
const[tooltip,setTooltip] = useState("")
const[visibility,setVisibility] = useState(false)
const[style,setStyle] = useState("")
return (
<>
<td>
{(p.song.report.rank>0)?p.song.report.rank:<i>Not Played</i>}
</td>
<td>
{(p.song.report.rank>0)?<>{p.song.report.score}</>:""}
</td>
<td>
{(p.song.report.rank>0)?<>{p.song.report.percent}%</>:""}
</td>
<td>
{p.song.report.ecount>0?<span className="badge badge-primary" onMouseOver={()=>{setTooltip(<>{p.song.report.eclearcount+" / "+p.song.report.ecount+" ("+(Math.floor((p.song.report.eclearcount/p.song.report.ecount)*100))+"% pass rate)"}{(p.song.report.efccount>0)?<><br/>{"★FC'd "+p.song.report.efccount+" time"+(p.song.report.efccount!=1?"s":"")}</>:<></>}{(p.song.report.epfccount>0)?<><br/>{"✪Perfected "+p.song.report.epfccount+" time"+(p.song.report.epfccount!=1?"s":"")}</>:<></>}</>);setStyle("easy");setVisibility(true)}} onMouseOut={()=>{setVisibility(false)}}>{p.song.report.epfccount>0?"":p.song.report.efccount>0?"":""}{p.song.report.ecount}</span>:<></>}
{p.song.report.ncount>0?<span className="badge badge-info" onMouseOver={()=>{setTooltip(<>{p.song.report.nclearcount+" / "+p.song.report.ncount+" ("+(Math.floor((p.song.report.nclearcount/p.song.report.ncount)*100))+"% pass rate)"}{(p.song.report.nfccount>0)?<><br/>{"★FC'd "+p.song.report.nfccount+" time"+(p.song.report.nfccount!=1?"s":"")}</>:<></>}{(p.song.report.npfccount>0)?<><br/>{"✪Perfected "+p.song.report.npfccount+" time"+(p.song.report.npfccount!=1?"s":"")}</>:<></>}</>);setStyle("normal");setVisibility(true)}} onMouseOut={()=>{setVisibility(false)}}>{p.song.report.npfccount>0?"":p.song.report.nfccount>0?"":""}{p.song.report.ncount}</span>:<></>}
{p.song.report.hcount>0?<span className="badge badge-success" onMouseOver={()=>{setTooltip(<>{p.song.report.hclearcount+" / "+p.song.report.hcount+" ("+(Math.floor((p.song.report.hclearcount/p.song.report.hcount)*100))+"% pass rate)"}{(p.song.report.hfccount>0)?<><br/>{"★FC'd "+p.song.report.hfccount+" time"+(p.song.report.hfccount!=1?"s":"")}</>:<></>}{(p.song.report.hpfccount>0)?<><br/>{"✪Perfected "+p.song.report.hpfccount+" time"+(p.song.report.hpfccount!=1?"s":"")}</>:<></>}</>);setStyle("hard");setVisibility(true)}} onMouseOut={()=>{setVisibility(false)}}>{p.song.report.hpfccount>0?"":p.song.report.hfccount>0?"":""}{p.song.report.hcount}</span>:<></>}
{p.song.report.excount>0?<span className="badge badge-warning" onMouseOver={()=>{setTooltip(<>{p.song.report.exclearcount+" / "+p.song.report.excount+" ("+(Math.floor((p.song.report.exclearcount/p.song.report.excount)*100))+"% pass rate)"}{(p.song.report.exfccount>0)?<><br/>{"★FC'd "+p.song.report.exfccount+" time"+(p.song.report.exfccount!=1?"s":"")}</>:<></>}{(p.song.report.expfccount>0)?<><br/>{"✪Perfected "+p.song.report.expfccount+" time"+(p.song.report.expfccount!=1?"s":"")}</>:<></>}</>);setStyle("ex");setVisibility(true)}} onMouseOut={()=>{setVisibility(false)}}>{p.song.report.expfccount>0?"":p.song.report.exfccount>0?"":""}{p.song.report.excount}</span>:<></>}
{p.song.report.exexcount>0?<span className="badge badge-danger" onMouseOver={()=>{setTooltip(<>{p.song.report.exexclearcount+" / "+p.song.report.exexcount+" ("+(Math.floor((p.song.report.exexclearcount/p.song.report.exexcount)*100))+"% pass rate)"}{(p.song.report.exexfccount>0)?<><br/>{"★FC'd "+p.song.report.exexfccount+" time"+(p.song.report.exexfccount!=1?"s":"")}</>:<></>}{(p.song.report.exexpfccount>0)?<><br/>{"✪Perfected "+p.song.report.exexpfccount+" time"+(p.song.report.exexpfccount!=1?"s":"")}</>:<></>}</>);setStyle("exex");setVisibility(true)}} onMouseOut={()=>{setVisibility(false)}}>{p.song.report.exexpfccount>0?"":p.song.report.exexfccount>0?"":""}{p.song.report.exexcount}</span>:<></>}
{(visibility)?<span style={{position:"absolute"}} className={"display-tooltip alert alert-dark "+style+"-background"}>{tooltip}</span>:<></>}
</td>
<td>
{(p.song.report.ehscount>0||p.song.report.ehdcount>0||p.song.report.esdcount>0)?
<ModDisplay badge="primary" diff="easy"
hs={p.song.report.ehscount} hd={p.song.report.ehdcount} sd={p.song.report.esdcount}/>
:<></>
}
{(p.song.report.nhscount>0||p.song.report.nhdcount>0||p.song.report.nsdcount>0)?
<ModDisplay badge="info" diff="normal"
hs={p.song.report.nhscount} hd={p.song.report.nhdcount} sd={p.song.report.nsdcount}/>
:<></>
}
{(p.song.report.hhscount>0||p.song.report.hhdcount>0||p.song.report.hsdcount>0)?
<ModDisplay badge="success" diff="hard"
hs={p.song.report.hhscount} hd={p.song.report.hhdcount} sd={p.song.report.hsdcount}/>
:<></>
}
{(p.song.report.exhscount>0||p.song.report.exhdcount>0||p.song.report.exsdcount>0)?
<ModDisplay badge="warning" diff="ex"
hs={p.song.report.exhscount} hd={p.song.report.exhdcount} sd={p.song.report.exsdcount}/>
:<></>
}
{(p.song.report.exexhscount>0||p.song.report.exexhdcount>0||p.song.report.exexsdcount>0)?
<ModDisplay badge="danger" diff="exex"
hs={p.song.report.exexhscount} hd={p.song.report.exexhdcount} sd={p.song.report.exexsdcount}/>
:<></>
}
</td>
</>
);
}
function PlayData(p) {
var [data,setData] = useState([])
var [update,setUpdate] = useState(false)
useEffect(()=>{
axios.get("http://projectdivar.com/plays/"+p.username+"/"+p.song.id)
.then((data)=>{setData(data.data);})
},[update])
return (
<>
<div className="text-center background-songs below">
<h5>Individual Plays for {p.song.name} from {p.username}</h5>
<div className="border rounded">
{data.map((play,i)=><Play setModalSrc={p.setModalSrc} setModalVisible={p.setModalVisible} key={i} play={play} mini={true} song={p.song}/>)}
<LoadMore url={"http://www.projectdivar.com/plays/"+p.username+"/"+p.song.id} params={{limit:15,offset:5}} value={data} setValue={setData}/>
</div>
</div>
</>
)
}
var cumulativeOffset = function(element) {
var top = 0, left = 0;
do {
top += element.offsetTop || 0;
left += element.offsetLeft || 0;
element = element.offsetParent;
} while(element);
return {
top: top,
left: left
};
};
function HoverSongName(p) {
let match = useRouteMatch({
path: "/user/"+p.username+"/"+p.to,
exact: true
});
var [name,setName] = useState(p.song.name)
var [expand,setExpand] = useState(<></>)
var [toggle,setToggle] = useState(match)
let history = useHistory();
useEffect(()=>{
if ((p.song.report.ecount+p.song.report.ncount+p.song.report.hcount+p.song.report.excount+p.song.report.exexcount>0)) {
if (toggle) {
setExpand(<tr className="fadein">
<td colSpan="6"><PlayData setModalSrc={p.setModalSrc} setModalVisible={p.setModalVisible} song={p.song} username={p.username}/></td>
</tr>)
window.scroll(0,cumulativeOffset(document.getElementById("songRow_"+p.song.name)).top-document.getElementById("songRow_"+p.song.name).getBoundingClientRect().height);
} else {
setExpand(<></>)
}
}
},[toggle])
return (
<>
<tr id={"songRow_"+p.song.name} key={p.song.id} className="lighthover cursor" onClick={(e)=>{
if ((p.song.report.ecount+p.song.report.ncount+p.song.report.hcount+p.song.report.excount+p.song.report.exexcount>0)) {
if (!toggle) {
setToggle(true)
history.push("/user/"+p.username+"/"+p.to)
} else {
setToggle(false)
}
}
//console.log(e.target.getBoundingClientRect()+"/"+window.pageYOffset)
}
}>
<td className="overflow-hidden">
{(p.song.report.ecount+p.song.report.ncount+p.song.report.hcount+p.song.report.excount+p.song.report.exexcount>0)?((toggle)?<></>:<>⯈</>):<></>} {name} <span className="tinytext">{(p.song.romanized_name.length>0)?<>{(p.song.romanized_name!==p.song.name)?<>{p.song.romanized_name}</>:<></>}</>:<>{(p.song.english_name!==p.song.name)?<>{p.song.english_name}</>:<></>}</>}</span>
</td>
<PlayDetail song={p.song}/>
</tr>
{expand}
</>
)
}
function CompletionPanel(p) {
const [report,setReport] = useState([])
const [song,setSong] = useState("")
const [filter,setFilter] = useState({})
const [style,setStyle] = useState(true)
const [update,setUpdate] = useState(false)
useEffect(()=>{
axios.get("http://projectdivar.com/completionreport/"+p.username)
.then((data)=>{setReport(data.data)})
.catch((err)=>{console.log(err.message)})
},[update])
return (
<>
<SongSearch songs={p.songs} song={song} setSong={setSong} setStyle={setStyle} filteredSongs={filter} setFilteredSongs={setFilter}/>
<table className="table table-sm">
<thead>
<tr id="headerbar">
<th scope="col" className={(style)?"scrollingHeader":""}>
Song Name
</th>
<th className={(style)?"scrollingHeader":""}>
Ranking
</th>
<th className={(style)?"scrollingHeader":""}>
Score
</th>
<th className={(style)?"scrollingHeader":""}>
%
</th>
<th className={(style)?"scrollingHeader":""}>
Play Count
</th>
<th className={(style)?"scrollingHeader":""}>
Mods
</th>
</tr>
</thead>
<tbody>
{report.filter((report)=>Object.keys(filter).length==0||report.id in filter).map((song,i)=>{return <HoverSongName setModalSrc={p.setModalSrc} setModalVisible={p.setModalVisible} to={song.name} song={song} key={song.id} username={p.username}/>
})}
</tbody>
<tfoot>
<tr><td colSpan="8" id="footer" className={(style)?"scrollingFooter":""}>
<span className="badge badge-primary">Easy</span> <span className="badge badge-info">Normal</span> <span className="badge badge-success">Hard</span> <span className="badge badge-warning">Extreme</span> <span className="badge badge-danger">Extra Extreme</span> <span className="badge badge-light"> = FC</span> <span className="badge badge-light"> = Perfect FCs</span>
</td></tr>
</tfoot>
</table>
</>
);
}
function Panel() {
return (
<>
[Placeholder Panel]
</>
);
}
const CalculateAccuracy=(cool,fine,safe,sad,worst)=>{
var noteCount = cool+fine+safe+sad+worst;
var sum = cool+(fine*0.5)+(safe*0.1)+(sad*0.05);
return Math.round((sum/noteCount)*10000)/100+"%";
}
function ClearBadge(p) {
var [display,setDisplay] = useState(<></>)
/*<span className="badge badge-primary">{easy}/{diffs.E}
{(fcdata&&fcdata.E>0)?<><br/>{fcdata.E}</>:<></>}
{(pfcdata&&pfcdata.E>0)?<><br/>{pfcdata.E}</>:<></>}</span>*/
function toggle(state) {
if (state) {
setDisplay(<>{((p.fcdata&&p.fcdata[p.diff]>0)?<><br/> {p.fcdata[p.diff]}</>:<></>)}
{((p.pfcdata&&p.pfcdata[p.diff]>0)?<><br/> {p.pfcdata[p.diff]}</>:<></>)}</>)
} else {
setDisplay(<></>)
}
}
useEffect(()=>{
toggle(p.mouseOver)
},[p.mouseOver])
return(
<>
<span className={"badge badge-"+CalculateBadge(p.diff)} onTouchStart={()=>{toggle(true)}}>{p.count}/{p.diffs[p.diff]}{display}</span>
</>
)
}
function Profile(p){
let { username } = useParams();
let match = useRouteMatch();
var [updateProfile,setProfile] = useState(false);
var [playcount,setPlayCount] = useState(0);
var [fccount,setFCCount] = useState(0);
var [cleared,setClear] = useState("");
var [accuracy,setAccuracy] = useState("-%");
var [rating,setRating] = useState(0);
var [lastPlayed,setLastPlayed] = useState(new Date());
var [update,setUpdate] = useState(false);
var [diffs,setDiffs] = useState({});
var [user,setUserData] = useState({});
var [render,setRender] = useState(false);
var [modalsrc,setModalSrc] = useState({})
var [modalVisible,setModalVisible] = useState(false);
var [mouseOver,setMouseOver] = useState(false);
function CalculateClear(easy,normal,hard,ex,exex,fcdata,pfcdata) {
return <>
<ClearBadge mouseOver={mouseOver} diff="E" count={easy} diffs={diffs} fcdata={fcdata} pfcdata={pfcdata}/>
<ClearBadge mouseOver={mouseOver} diff="N" count={normal} diffs={diffs} fcdata={fcdata} pfcdata={pfcdata}/>
<ClearBadge mouseOver={mouseOver} diff="H" count={hard} diffs={diffs} fcdata={fcdata} pfcdata={pfcdata}/>
<ClearBadge mouseOver={mouseOver} diff="EX" count={ex} diffs={diffs} fcdata={fcdata} pfcdata={pfcdata}/>
<ClearBadge mouseOver={mouseOver} diff="EXEX" count={exex} diffs={diffs} fcdata={fcdata} pfcdata={pfcdata}/>
</>
}
useEffect(()=>{
axios.get("http://projectdivar.com:4501/userdata/"+username)
.then((data)=>{setUserData(data.data);setPlayCount(data.data.playcount);setFCCount(data.data.fccount);setRating(data.data.rating);setLastPlayed(data.data.last_played);setAccuracy(CalculateAccuracy(data.data.cool,data.data.fine,data.data.safe,data.data.sad,data.data.worst))});
axios.get("http://projectdivar.com:4501/songdiffs")
.then((data)=>{setDiffs(data.data)})
},[update])
useEffect(()=>{
if (user!={}) {
setClear(CalculateClear(user.eclear,user.nclear,user.hclear,user.exclear,user.exexclear,user.fcdata,user.pfcdata))
}
},[diffs,user,mouseOver])
useEffect(()=>{
setRender(user!==undefined&&playcount!==undefined&&fccount!==undefined&&rating!==undefined&&lastPlayed!==undefined&&accuracy!==undefined&&diffs!==undefined&&cleared!==undefined)
},[user,playcount,fccount,rating,lastPlayed,accuracy,diffs,cleared])
return (
<>
<ImageDisplayer username={username} songs={p.songs} play={modalsrc} modalVisible={modalVisible} setModalVisible={setModalVisible}></ImageDisplayer>
<h2>{username+"'s Profile"}</h2>
{(render)?<>
<StatisticsPanel name="Statistics" setMouseOver={setMouseOver} username={username} playcount={playcount} fccount={fccount} cleared={cleared} accuracy={accuracy}/>
<HitCountsPanel name="Hit Counts" username={username} user={user}/>
<BestPlaysPanel name="Best Plays" setModalVisible={setModalVisible} setModalSrc={setModalSrc} username={username} songs={p.songs}/>
<CompletionPanel name="Progress" setModalVisible={setModalVisible} setModalSrc={setModalSrc} username={username} songs={p.songs}/>
<Panel name="Activity" username={username}/>
</>
:<></>
}
</>
);
}
function Rankings(){
let { sort,sortOrder } = useParams();
let match = useRouteMatch();
var [users,setUsers] = useState([]);
var [updateUsers,setUpdateUsers] = useState(false);
var [isLoading,setIsLoading] = useState(true);
useEffect(()=>{
axios.get("http://projectdivar.com:4501/users/"+sort+"/"+sortOrder+"?limit=100&offset=0")
.then((data)=>{setUsers(data.data)
setIsLoading(false);})
//.then(()=>{console.log(users)})
},[updateUsers])
return (
<>
<table>
<thead>
<tr>
<th className="header"><Sort setIsLoading={setIsLoading} updateUsers={updateUsers} setUpdateUsers={setUpdateUsers} label="Username" order="username"/></th>
<th className="header"><Sort setIsLoading={setIsLoading} updateUsers={updateUsers} setUpdateUsers={setUpdateUsers} label="Last Played" order="last_played"/></th>
<th className="header"><Sort setIsLoading={setIsLoading} updateUsers={updateUsers} updateUsers={updateUsers} setUpdateUsers={setUpdateUsers} setUpdateUsers={setUpdateUsers} label="Rating" order="rating"/></th>
<th className="header"><Sort setIsLoading={setIsLoading} updateUsers={updateUsers} setUpdateUsers={setUpdateUsers} label="Play Count" order="playcount"/></th>
<th className="header"><Sort setIsLoading={setIsLoading} updateUsers={updateUsers} setUpdateUsers={setUpdateUsers} label="FC Count" order="fccount"/></th>
</tr>
</thead>
{users.map((user,i)=>
<tbody key={i}>
<tr>
<td className={(isLoading)?"loading":""}><Link to={"/user/"+user.username}>{user.username}</Link></td>
<td className={(isLoading)?"loading":""} className={(isLoading)?"loading":""} className={(isLoading)?"loading":""}>{user.last_played}</td>
<td className={(isLoading)?"loading":""} className={(isLoading)?"loading":""}>{user.rating}</td>
<td className={(isLoading)?"loading":""}>{user.playcount}</td>
<td className={(isLoading)?"loading":""}>{user.fccount}</td>
</tr>
</tbody>)}
</table>
</>
);
}
function ImageUpload(p) {
var [file,setFile] = useState(null);
var [fileProcess,setFileProcess] = useState(0);
var [error,setError] = useState(null);
var [success,setSuccess] = useState(null);
var [fileProgress,setFileProgress] = useState(0);
var prepFile = (e)=>{
setFile(e.currentTarget.files[0])
setError(null);
}
var uploadFile = (e)=>{
setError(null);
if (!file) {setError("No file selected!");return;}
if (file.type!=="application/x-zip-compressed" &&
file.type!=="image/jpeg" && file.type!=="image/png") {
setError("File type is invalid! Please provide a .zip/.jpg/.png file!")
setFileProcess(0)
return;
}
const data = new FormData()
data.append('file', file)
data.append("username","GOD");
data.append("authentication_token","GOD");
if (!data.has("username") || !data.has("authentication_token")) {setError("Authentication failed!");return;}
if (file.size>15*1024*1024) {
setError("File size is too large! Max is 15MB! Consider splitting your plays into chunks (Recommended 50 files per zip).");return;
}
//console.log(file)
setFileProcess(1);
axios.post("http://projectdivar.com/upload", data, {
onUploadProgress: function(progressEvent) {
//console.log(progressEvent)
setFileProgress(Math.round((progressEvent.loaded * 100) / progressEvent.total))
}})
.then(res => {
setSuccess(res.data);
setFileProgress(100)
setFileProcess(0)
})
.catch((err)=>{setError(err.message);setFileProgress(0);setFileProcess(0)})
}
switch (fileProcess) {
default:{
return (
<form method="post" action="#" id="#">
<div className="files form-group color">
<h3>Submit your play</h3>
<i>Plays can be a single image or a bunch of images in a zip file!</i>
<hr/>
{(success!=null)?<h4 className="success">{success}</h4>:<></>}
<div style={{position:"relative",top:"0px"}}>
<input type="file" name="file" className="form-control" onChange={(e)=>{prepFile(e)}} disabled={fileProcess===1}/>
<div className="uploadicon"/>
<div className="dragText">or drag it here</div>
</div>
{(error!==null)?<div className="error">{error}</div>:<></>}
<button type="button" className="btn btn-primary btn-block" onClick={(e)=>{uploadFile(e)}} disabled={fileProcess===1}>
{fileProcess===1?<>Uploading...<span className="spinner-border"/>
<div className="progress" style={{position:"relative"}}>
<div className={"progress-bar"} style={{width:fileProgress+"%"}} role="progressbar" aria-valuemin="0" aria-valuemax="100"></div>
<div style={{position:"relative"}}>{fileProgress+"%"}</div>
</div>
</>:<>Upload</>}</button>
</div>
</form>);
}
}
}
function SongSearch(p) {
//Requires: p.songs / p.song / p.setSong / p.filteredSongs / p.setFilteredSongs
const [song,setSong] = useState("")
const [focused,setFocused] = useState(false)
useEffect(()=>{
if (p.setStyle) {
if (focused) {
p.setStyle(false)
} else {
p.setStyle(true)
}
}
},[focused])
return (
<>
<input className="form-control form-control-lg" value={song} placeholder={"🔍 Search by Song,Artist,Vocaloid"} onKeyDown={(e)=>{if (e.key==="Enter") {setFocused(false);e.target.blur()}}} onFocus={()=>{setFocused(true)}} onChange={(e)=>{
if (p.setFilteredSongs){p.setFilteredSongs({})}
setSong(e.target.value)
}
} onBlur={()=>{
setTimeout(()=>{setFocused(false)},250)
}
}/>
{(focused)?
<div className="overflow-auto rounded-lg" style={{background:"#eef",position:"absolute",width:"95%",height:"240px"}}>{
Object.keys(p.songs).filter((key)=>
{
var s = p.songs[key]
return s.name.toLowerCase().includes(song.toLowerCase())||s.romanized_name.toLowerCase().includes(song.toLowerCase())||s.english_name.toLowerCase().includes(song.toLowerCase())||s.artist.toLowerCase().includes(song.toLowerCase())||s.vocaloid.toLowerCase().includes(song.toLowerCase())
}).map((key)=>{
if (p.setFilteredSongs&&p.filteredSongs) {
var obj = p.filteredSongs;
obj[key]=true
//console.log(obj)
p.setFilteredSongs(obj)
}
return <div key={key} className="pb-1 homelink highest" onClick={()=>{setSong(p.songs[key].name);p.setSong(p.songs[key].name);setFocused(false);if(p.setFilteredSongs&&p.filteredSongs){var obj={};obj[key]=true;p.setFilteredSongs(obj)}}}>
<div className="d-flex flex-row">
<div className="p-2">
<img className="centered" src={p.songs[key].album_art.replace("album_art","album_art/small")}/>
</div>
<div className="p-8">
<h4>{p.songs[key].name}</h4>{(p.songs[key].romanized_name)?p.songs[key].romanized_name:p.songs[key].english_name}
</div>
</div>
</div>})}</div>:<></>}
More stuff goes here.
</>
)
}
function SimpleUpload(p){
const [song,setSong] = useState("")
const [percentage,setPercentage] = useState("")
return (
<>
<SongSearch song={song} setSong={setSong} songs={p.songs}/>
{(song.length>0?<>
<div className="input-group">
<div className="input-group-prepend">
<label htmlFor="validationCustomUsername">
<span className="input-group-text" id= "inputGroupPrepend">Song %</span>
</label>
<input type="text" className="form-control" id="validationCustomUsername" value={percentage} placeholder="101.46%" aria-describedby="inputGroupPrepend"
onChange={(e)=>{
if (!e.target.value.includes("%")) {
e.target.value+="%"
e.target.selectionStart=e.target.selectionEnd=e.target.value.length-1
}
setPercentage(e.target.value)}}
required/>
<small className="text-muted">
Input the % you got on the results screen.
</small>
</div>
</div>
</>:<></>)}
</>
)
}
function SendMail() {
//process.env.REACT_APP_FRONTEND_AUTH
return (
<>
</>
)
}
function Submit(p) {
return (
<div className="row">
<div className="col-12 col-md-8">
<Switch>
<Route path="/submitplay/simple">
<SimpleUpload songs={p.songs}/>
</Route>
<Route path="/submitplay/detail">
Detailed
</Route>
<Route path="/submitplay/switch">
Switch
</Route>
<Route path="/submitplay/image">
<ImageUpload/>
</Route>
<Route path="/submitplay">
<h2>Select a submission method</h2>
<div className="card">
<Link to="/submitplay/simple" className="nostyle">
<div className="card-body">
<h5 className="card-title">Manual Submit</h5>
<p className="card-text">Submit your plays by entering the clear % of a song</p>
<p className="card-text"><small className="text-muted">The simplest way to submit plays, type in a %, tweak the other values quickly, then submit your play! Optionally include a screenshot.</small></p>
</div>
</Link>
</div>
<br/>
<div className="card">
<Link to="/submitplay/image" className="nostyle">
<div className="card-body">
<h5 className="card-title">Image Upload</h5>
<p className="card-text">Upload images from your Nintendo Switch for automatic processing/scoring!</p>
<p className="card-text"><small className="text-muted">Put up to 50 images in a zip file to mass-upload your screenshotted plays to your profile. You will need to extract them from your microSD card from your Nintendo Switch.</small></p>
</div>
</Link>
</div>
<br/>
<div className="card">
<Link to="/submitplay/switch" className="nostyle">
<div className="card-body">
<h5 className="card-title">Nintendo Switch/Twitter Upload</h5>
<p className="card-text">Setup your account for uploading through Twitter using your Nintendo Switch!</p>
<p className="card-text"><small className="text-muted">You can select up to 4 images to post to Twitter at one time.</small></p>
</div>
</Link>
</div>
</Route>
</Switch>
</div>
</div>
)
}
function RecentPlays(p) {
const [update,setUpdate] = useState(false)
const [recentPlayData,setRecentPlayData] = useState([])
useEffect(()=>{
const interval = setInterval(()=>{
axios.get("http://projectdivar.com/recentplays/sigonasr2")
.then((data)=>{
setRecentPlayData(data.data);
})
.catch((err)=>{})
},5000);
return ()=>{clearInterval(interval)}
},[update])
return (
<>
{recentPlayData.map((play,i)=><Play index={i} play={play} song={p.songs[play.songid]} title={true} mini={true}/>)}
</>
)
}
function LoginInfo(p) {
const [username,setUsername] = useState(undefined)
const [authToken,setAuthToken] = useState(undefined)
const [loggedIn,setLoggedIn] = useState(false)
//Load our storage data if exists.
useEffect(()=>{
try {
setUsername(localStorage.getItem("username"))
setAuthToken(localStorage.getItem("authToken"))
}catch{
console.log("Not logged in!");
}
},[p.update])
useEffect(()=>{
if (username!==undefined&&authToken!==undefined) {
axios.post("http://projectdivar.com/authenticate/login",{username:username,authCode:authToken})
.then((data)=>{
setLoggedIn(true);
p.setUsername(username);
})
.catch((err)=>{
setLoggedIn(false);
})
}
},[username,authToken])
return (
<>
{loggedIn?<>
Welcome, <b>{username}</b>!<br/>
<Link to="/auth">App Auth Code</Link><br/>
</>:<>
<Link to="/login">Login</Link><br/>
<Link to="/register">Register</Link>
</>}
</>
)
}
function Login(p) {
const [username,setUsername] = useState("")
const [authCode,setAuthCode] = useState("")
const [authCodeVisible,setAuthCodeVisible] = useState(false)
const [error,setError] = useState(false)
const [disabled,setDisabled] = useState(false)
let history = useHistory();
if (p.isLoggedIn) {
history.push("/")
}
return (
<>
<Form>
{error&&<h3 style={{color:"red"}}>{error}</h3>}
{authCodeVisible&&<><h3>We have sent you an email containing your login code!</h3>
<br/>
Please submit it to finish the login process.
</>}
<Form.Group controlId="formUsername">
<Form.Label>Username</Form.Label>
<Form.Control disabled={authCodeVisible} isInvalid={username.length<1} onChange={(e)=>{setUsername(e.currentTarget.value)}} placeholder="MikuMiku" value={username} />
<Form.Text className="text-muted">
We will send an email to the registered email of this account.
</Form.Text>
</Form.Group>
{authCodeVisible&&<>
<div className="row">
<div className="col-4">
<Form.Group controlId="formCode">
<Form.Label>Authentication Code</Form.Label>
<Form.Control isInvalid={authCode.length!=5} onChange={(e)=>{setAuthCode(e.currentTarget.value)}} placeholder="XXXXX" value={authCode} />
<Form.Text className="text-muted">
Please enter the code you received in your email here. Then submit again.
</Form.Text>
</Form.Group>
</div>
</div></>
}
<Button disabled={disabled} variant="primary" type="submit" onClick={(e)=>{e.preventDefault()
if (username.length>=1) {
setDisabled(true)
setError(false)
if (authCode.length===5) {
axios.post("http://projectdivar.com/authenticate/login",{username:username,authCode:authCode})
.then((data)=>{
localStorage.setItem("username",username)
localStorage.setItem("authToken",authCode)
p.setLoginPanelUpdate(true)
history.push("/")
})
.catch((err)=>{
if (err) {
setError("Invalid Authentication Code!")
setDisabled(false)
}
})
} else {
axios.post("http://projectdivar.com/sendemail/login",{username:username})
.then((data)=>{
setAuthCodeVisible(true)
setDisabled(false)
})
.catch((err)=>{
if (err) {
setError("Invalid credentials provided!")
setDisabled(false)
}
})
}
}
}}>
{authCodeVisible?"Submit Code":"Login"}
</Button>
</Form>
</>
)
}
function Register(p) {
const [username,setUsername] = useState("")
const [email,setEmail] = useState("")
const [authCode,setAuthCode] = useState("")
const [authCodeVisible,setAuthCodeVisible] = useState(false)
const [error,setError] = useState(false)
const [disabled,setDisabled] = useState(false)
let history = useHistory();
if (p.isLoggedIn) {
history.push("/")
}
return (
<>
<Form>
{error&&<h3 style={{color:"red"}}>{error}</h3>}
{authCodeVisible&&<><h3>We have sent you an email containing your registration code!</h3>
<br/>
Please submit it to finish the registration process.
</>}
<Form.Group controlId="formUsername">
<Form.Label>Username</Form.Label>
<Form.Control disabled={authCodeVisible} isInvalid={username.length<1} onChange={(e)=>{setUsername(e.currentTarget.value)}} placeholder="MikuMiku" value={username} />
</Form.Group>
<Form.Group controlId="formEmail">
<Form.Label>Email Address</Form.Label>
<Form.Control disabled={authCodeVisible} type="email" disabled={authCodeVisible} isInvalid={email.length<1} onChange={(e)=>{setEmail(e.currentTarget.value)}} placeholder="MikuMiku@39.net" value={email} />
<Form.Text className="text-muted">
Please provide a valid email address! We use your email account as your "password", so it is a requirement for this site.
</Form.Text>
</Form.Group>
{authCodeVisible&&<>
<div className="row">
<div className="col-4">
<Form.Group controlId="formAuthCode">
<Form.Label>Authentication Code</Form.Label>
<Form.Control isInvalid={authCode.length!=5} onChange={(e)=>{setAuthCode(e.currentTarget.value)}} placeholder="XXXXX" value={authCode} />
<Form.Text className="text-muted">
Please enter the code you received in your email here. Then submit again.
</Form.Text>
</Form.Group>
</div>
</div></>
}
<Button disabled={disabled} variant="primary" type="submit" onClick={(e)=>{e.preventDefault()
if (username.length>=1&&email.length>=1) {
setDisabled(true)
setError(false)
if (authCode.length===5) {
axios.post("http://projectdivar.com/authenticate/login",{username:username,authCode:authCode})
.then((data)=>{
localStorage.setItem("username",username)
localStorage.setItem("authToken",authCode)
p.setLoginPanelUpdate(true)
return axios.patch("http://projectdivar.com/updateRegisteredState",{username:username,authCode:authCode})
})
.then((data)=>{
//console.log(data)
history.push("/")
})
.catch((err)=>{
if (err) {
setError("Invalid Authentication Code!")
setDisabled(false)
}
})
} else {
axios.post("http://projectdivar.com/sendemail/register",{username:username,email:email})
.then((data)=>{
setAuthCodeVisible(true)
setDisabled(false)
})
.catch((err)=>{
if (err) {
setError("Username or Email already in use!")
setDisabled(false)
}
})
}
}
}}>
{authCodeVisible?"Submit Code":"Register"}
</Button>
</Form>
</>
)
}
function UserAuth(p) {
const[showAuthCode,setShowAuthCode] = useState(false)
const[authToken,setAuthToken] = useState("")
let history = useHistory();
if (!p.isLoggedIn) {
history.push("/")
}
return(<>
Your <b>App Authentication Code</b> is used for verifying your identity when using apps such as <b>DivaBot</b>. By clicking the <b>{"<Reveal Code>"}</b> button, you understand that you should not share this code or show it to anyone!
<br/><br/>
{showAuthCode?
<input readOnly value={authToken}></input>
:<button onClick={()=>{
axios.post("http://projectdivar.com/authenticate/authToken",{username:localStorage.getItem("username"),
authCode:localStorage.getItem("authToken")})
.then((data)=>{
setAuthToken(data.data.authentication_token)
setShowAuthCode(true)
})
.catch((err)=>{
setAuthToken(err.message)
})
}
}>
{"<Click to reveal App Authentication Code>"}
</button>
}
</>)
}
function ReleaseList(p) {
return(
<>
<ul className="list-group">
{p.releases.map((release,i)=>
<li key={i} className="list-group-item">
<b>{release[0]}</b> - <a href={release[1]}>{release[1].replace("http://projectdivar.com/files/releases/","")}</a> <i>(Released {release[2]})</i>
</li>
)}
</ul>
</>
)
}
function DivaBot() {
const releases=[
["01B","http://projectdivar.com/files/releases/DivaBot01B.zip","13 Sep 2020"],
["01A","http://projectdivar.com/files/releases/DivaBot01A.zip","13 Sep 2020"],
["01","http://projectdivar.com/files/releases/DivaBot01.zip","13 Sep 2020"]]
return (
<>
<b>DivaBot</b> was created by <b>sigonasr2</b> in order to allow Project Diva streamers to personalize their stream setups with their personal scores and achievements into their game.
<br/><br/>
The app works by monitoring your game's capture area as you are streaming in order to identify what song you are playing, and what scores you achieve. It is used with this website to make score
submitting and tracking easier.
<br/>
The app currently supports <b>Megamix</b>. Other games will be supported in the future.
<hr/>
<h3>Setup Instructions</h3>
{<iframe width="100%" height="480" src="https://www.youtube.com/embed/IAPpbp5EFto" frameBorder="0" allow="accelerometer; autoplay; encrypted-media; gyroscope; picture-in-picture" allowFullScreen></iframe>}
<hr/>
<h3>Downloads</h3>
<ReleaseList releases={releases}/>
</>
)
}
function Website() {
const [songs,setSongs] = useState([])
const [update,setUpdate] = useState(false)
const [tooltip,setTooltip] = useState("")
const [loginPanelUpdate,setLoginPanelUpdate] = useState(false)
const [username,setUsername] = useState(undefined)
useEffect(()=>{
axios.get("http://www.projectdivar.com/songs")
.then((data)=>{
setSongs(data.data)
})
},[update])
return (
<div className="row">
<div className="col-md-2 pt-3 pb-3 overflow-hidden text-center">
<h3 className="d-none d-md-block">Menu</h3>
<LoginInfo setUsername={setUsername} update={loginPanelUpdate}/>
<br/><br/>
<Link to="/rankings/rating/desc">Rankings</Link><br/>
<Link to="/divabot">DivaBot</Link><br/>
</div>
<div className="col-md-10 pt-3 pb-3">
<Switch>
<Route path="/rankings/:sort/:sortOrder">
<Rankings/>
</Route>
<Route path="/user/:username">
{(songs)?
<Profile songs={songs}/>:<></>
}
</Route>
<Route path="/submitplay">
<Submit songs={songs}/>
</Route>
<Route path="/divabot">
<h1 className="title">DivaBot</h1>
<DivaBot/>
</Route>
<Route path="/auth">
<h1 className="title">App Authentication Token</h1>
<UserAuth username={username} isLoggedIn={username!==undefined}/>
</Route>
<Route path="/recentplays">
<h1 className="title">Project DivaR</h1>
<RecentPlays songs={songs}/>
</Route>
<Route path="/login">
<h1 className="title">Login to Project DivaR</h1>
<Login isLoggedIn={username!==undefined} setLoginPanelUpdate={setLoginPanelUpdate}/>
</Route>
<Route path="/register">
<h1 className="title">Register New Account</h1>
<Register isLoggedIn={username!==undefined} setLoginPanelUpdate={setLoginPanelUpdate}/>
</Route>
<Route path="/">
<h1 className="title">Project DivaR</h1>
Under construction!
</Route>
</Switch>
</div>
</div>
);
}
/*window.onmousemove = function(e) {
var obj = document.getElementById("display-tooltip")
if (obj!=null) {
//var offset = obj.parentElement.getBoundingClientRect();
var tipDist = 15;
obj.style.top = (e.clientY + tipDist) + 'px';
obj.style.left = (e.clientX + tipDist) + 'px';
}
}*/
function App() {
return (
<Router>
<div className="container-fluid content">
<div className="row">
<div className="topbar col-md-12 pt-1 overflow-hidden border rounded text-center">
<div>
<Link to="/">
<Logo/>
</Link>
</div>
</div>
</div>
<Website/>
</div>
</Router>
)
}
export default App;