|
|
|
import React, {useState,useEffect} from 'react';
|
|
|
|
import logo from './logo.svg';
|
|
|
|
import './App.css';
|
|
|
|
import {
|
|
|
|
BrowserRouter as Router,
|
|
|
|
Switch,
|
|
|
|
Route,
|
|
|
|
Link,
|
|
|
|
useRouteMatch,
|
|
|
|
useParams
|
|
|
|
} from "react-router-dom";
|
|
|
|
|
|
|
|
const REMOTE_ADDR = "http://45.33.13.215:4502";
|
|
|
|
|
|
|
|
const axios = require('axios');
|
|
|
|
|
|
|
|
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"}>
|
|
|
|
<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 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 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() {
|
|
|
|
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">{GetDateDiff()}</span></>
|
|
|
|
}
|
|
|
|
if (p.mini) {
|
|
|
|
return (
|
|
|
|
<>
|
|
|
|
<div className={"row align-middle "+((p.play.fine==0&&p.play.safe==0&&p.play.sad==0&&p.play.worst==0)?"pfc":(p.play.safe==0&&p.play.sad==0&&p.play.worst==0)?"fc":"")}>
|
|
|
|
<div className="col-md-2 order-1 order-md-1 text-center border-right align-middle text-nowrap overflow-hidden">{Math.floor(p.play.score)} pts<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}/></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-5 order-2 order-md-3">
|
|
|
|
<div className="row">
|
|
|
|
<div className="col-12 order-1 order-md-1 col-md-6 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-3 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>
|
|
|
|
</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)?"pfc":(p.play.safe==0&&p.play.sad==0&&p.play.worst==0)?"fc":"")}>
|
|
|
|
{(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}%</b></div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function BestPlaysPanel(p) {
|
|
|
|
var [bestPlays,setBestPlays] = useState([])
|
|
|
|
|
|
|
|
useEffect(()=>{
|
|
|
|
axios.get("http://www.projectdivar.com/bestplays/"+p.username+"?fails=false&limit=5&offset=0")
|
|
|
|
.then((data)=>{setBestPlays(data.data);})
|
|
|
|
})
|
|
|
|
|
|
|
|
return (
|
|
|
|
<>
|
|
|
|
<div className="d-none d-md-block row">
|
|
|
|
<div className="col-md-12 mt-3 mb-3">
|
|
|
|
<ul className="list-group list-group-flush overflow-auto border border-danger rounded-lg" style={{height:"320px"}}>
|
|
|
|
{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 index={i} play={play} song={p.songs[play.songid]}/>
|
|
|
|
</li>})}
|
|
|
|
</ul>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div className="d-block d-sm-block d-md-none row ml-3 mr-3">
|
|
|
|
<div className="col-md-12 mt-3 mb-3">
|
|
|
|
<ul className="list-group list-group-flush overflow-auto border border-danger rounded-lg" style={{height:"320px"}}>
|
|
|
|
{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 index={i} play={play} song={p.songs[play.songid]}/>
|
|
|
|
</li>})}
|
|
|
|
</ul>
|
|
|
|
</div>
|
|
|
|
</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="overflow-auto text-center" style={{height:"160px",width:"90%"}}>
|
|
|
|
{data.map((play,i)=><Play key={i} play={play} mini={true} song={p.song}/>)}
|
|
|
|
</div>
|
|
|
|
</>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
function HoverSongName(p) {
|
|
|
|
var [name,setName] = useState(p.song.name)
|
|
|
|
var [expand,setExpand] = useState(<></>)
|
|
|
|
var [toggle,setToggle] = useState(false)
|
|
|
|
return (
|
|
|
|
<>
|
|
|
|
<tr key={p.song.id} className="lighthover" onMouseOver={()=>{setName((p.song.romanized_name.length>0)?p.song.romanized_name:p.song.english_name)}} onMouseOut={()=>{setName(p.song.name)}}
|
|
|
|
data-toggle="collapse" data-target={"#collapse"+p.song.id} aria-expanded="false" aria-controls="collapseExample" onClick={()=>{
|
|
|
|
if (!toggle) {
|
|
|
|
setExpand(<PlayData song={p.song} username={p.username}/>)
|
|
|
|
setToggle(true)
|
|
|
|
} else {
|
|
|
|
setToggle(false)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}>
|
|
|
|
<td>
|
|
|
|
{name}
|
|
|
|
</td>
|
|
|
|
<PlayDetail song={p.song}/>
|
|
|
|
</tr>
|
|
|
|
<tr className="collapse" id={"collapse"+p.song.id}>
|
|
|
|
<td colSpan="6">{expand}</td>
|
|
|
|
</tr>
|
|
|
|
</>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
function CompletionPanel(p) {
|
|
|
|
const [report,setReport] = useState([])
|
|
|
|
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 (
|
|
|
|
<>
|
|
|
|
<table className="table table-sm">
|
|
|
|
<thead>
|
|
|
|
<tr id="headerbar">
|
|
|
|
<th scope="col">
|
|
|
|
Song Name
|
|
|
|
</th>
|
|
|
|
<th>
|
|
|
|
Ranking
|
|
|
|
</th>
|
|
|
|
<th>
|
|
|
|
Score
|
|
|
|
</th>
|
|
|
|
<th>
|
|
|
|
%
|
|
|
|
</th>
|
|
|
|
<th>
|
|
|
|
Play Count
|
|
|
|
</th>
|
|
|
|
<th>
|
|
|
|
Mods
|
|
|
|
</th>
|
|
|
|
</tr>
|
|
|
|
</thead>
|
|
|
|
<tbody>
|
|
|
|
{report.map((song,i)=>{return <HoverSongName song={song} key={song.id} username={p.username}/>
|
|
|
|
})}
|
|
|
|
</tbody>
|
|
|
|
<tfoot>
|
|
|
|
<tr><td colSpan="8" id="footer">
|
|
|
|
<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>*/
|
|
|
|
return(
|
|
|
|
<>
|
|
|
|
<span className={"badge badge-"+CalculateBadge(p.diff)} onMouseOver={()=>{
|
|
|
|
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]}</>:<></>)}</>)
|
|
|
|
}} onMouseOut={()=>{
|
|
|
|
setDisplay(<></>)
|
|
|
|
}}>{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);
|
|
|
|
|
|
|
|
function CalculateClear(easy,normal,hard,ex,exex,fcdata,pfcdata) {
|
|
|
|
return <>
|
|
|
|
<ClearBadge diff="E" count={easy} diffs={diffs} fcdata={fcdata} pfcdata={pfcdata}/>
|
|
|
|
<ClearBadge diff="N" count={normal} diffs={diffs} fcdata={fcdata} pfcdata={pfcdata}/>
|
|
|
|
<ClearBadge diff="H" count={hard} diffs={diffs} fcdata={fcdata} pfcdata={pfcdata}/>
|
|
|
|
<ClearBadge diff="EX" count={ex} diffs={diffs} fcdata={fcdata} pfcdata={pfcdata}/>
|
|
|
|
<ClearBadge 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])
|
|
|
|
|
|
|
|
useEffect(()=>{
|
|
|
|
setRender(user&&playcount&&fccount&&rating&&lastPlayed&&accuracy&&diffs&&cleared)
|
|
|
|
},[user,playcount,fccount,rating,lastPlayed,accuracy,diffs,cleared])
|
|
|
|
|
|
|
|
return (
|
|
|
|
<>
|
|
|
|
<h2>{username+"'s Profile"}</h2>
|
|
|
|
{(render)?<>
|
|
|
|
<StatisticsPanel name="Statistics" username={username} playcount={playcount} fccount={fccount} cleared={cleared} accuracy={accuracy}/>
|
|
|
|
<HitCountsPanel name="Hit Counts" username={username} user={user}/>
|
|
|
|
<BestPlaysPanel name="Best Plays" username={username} songs={p.songs}/>
|
|
|
|
<CompletionPanel name="Progress" 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)=>
|
|
|
|
<tbody>
|
|
|
|
<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","The Internet");
|
|
|
|
data.append("authentication_token","sig");
|
|
|
|
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.song / p.setSong
|
|
|
|
const [song,setSong] = useState("")
|
|
|
|
const [focused,setFocused] = useState(false)
|
|
|
|
|
|
|
|
return (
|
|
|
|
<>
|
|
|
|
<input className="form-control form-control-lg" value={song} placeholder={p.song} onFocus={()=>{setFocused(true)}} onChange={(e)=>{
|
|
|
|
setSong(e.target.value)
|
|
|
|
}
|
|
|
|
}/>
|
|
|
|
{(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)=><div className="pb-1 homelink" onClick={()=>{setSong(p.songs[key].name);setFocused(false)}}><h4>{p.songs[key].name}</h4>{(p.songs[key].romanized_name)?p.songs[key].romanized_name:p.songs[key].english_name}</div>)}</div>:<></>}
|
|
|
|
|
|
|
|
More stuff goes here.
|
|
|
|
</>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
function SimpleUpload(p){
|
|
|
|
const [song,setSong] = useState("Catch the Wave")
|
|
|
|
|
|
|
|
return (
|
|
|
|
<>
|
|
|
|
<SongSearch song={song} setSong={setSong} songs={p.songs}/>
|
|
|
|
</>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
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">Simple 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, it won't be entirely accurate, but it lets you submit plays very quickly.</small></p>
|
|
|
|
</div>
|
|
|
|
</Link>
|
|
|
|
</div>
|
|
|
|
<br/>
|
|
|
|
<div className="card">
|
|
|
|
<Link to="/submitplay/detail" className="nostyle">
|
|
|
|
<div className="card-body">
|
|
|
|
<h5 className="card-title">Detailed Submit</h5>
|
|
|
|
<p className="card-text">Submit your plays by entering all the information about a play</p>
|
|
|
|
<p className="card-text"><small className="text-muted">You can submit as many songs as you like, but you will have to provide details for each play.</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 Website() {
|
|
|
|
const [songs,setSongs] = useState([])
|
|
|
|
const [update,setUpdate] = useState(false)
|
|
|
|
const [tooltip,setTooltip] = useState("")
|
|
|
|
|
|
|
|
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">Sidebar Contents</h3>
|
|
|
|
<Link to="/rankings/rating/desc">Rankings</Link><br/>
|
|
|
|
Item 2<br/>
|
|
|
|
Item 3<br/>
|
|
|
|
Item 4<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="/">
|
|
|
|
<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;
|