diff --git a/client/database/index.jsx b/client/database/index.jsx index c77e75c1..586e4385 100644 --- a/client/database/index.jsx +++ b/client/database/index.jsx @@ -9,7 +9,7 @@ import DifficultyDropdown from '../scripts/components/DifficultyDropdown.min.js' import Star from '../scripts/components/Star.min.js'; import CategoryManager from '../scripts/utilities/category-manager.js'; import { getDropdownValues } from '../scripts/utilities/dropdown-checklist.js'; -import { insertTokensIntoHTML } from '../scripts/utilities/insert-tokens-into-html.js'; +import { insertTokensIntoHTML } from '../insert-tokens-into-html.js'; const starredTossupIds = new Set(await star.getStarredTossupIds()); const starredBonusIds = new Set(await star.getStarredBonusIds()); diff --git a/client/database/index.min.js b/client/database/index.min.js index 4201b405..c1a603c7 100644 --- a/client/database/index.min.js +++ b/client/database/index.min.js @@ -1,4 +1,4 @@ -import{downloadQuestionsAsText,downloadBonusesAsCSV,downloadTossupsAsCSV,downloadQuestionsAsJSON}from"./download.js";import api from"../scripts/api/index.js";import star from"../scripts/auth/star.js";import TossupCard from"../scripts/components/TossupCard.min.js";import BonusCard from"../scripts/components/BonusCard.min.js";import CategoryModal from"../scripts/components/CategoryModal.min.js";import DifficultyDropdown from"../scripts/components/DifficultyDropdown.min.js";import Star from"../scripts/components/Star.min.js";import CategoryManager from"../scripts/utilities/category-manager.js";import{getDropdownValues}from"../scripts/utilities/dropdown-checklist.js";import{insertTokensIntoHTML}from"../scripts/utilities/insert-tokens-into-html.js";const starredTossupIds=new Set(await star.getStarredTossupIds()),starredBonusIds=new Set(await star.getStarredBonusIds()),paginationShiftLength=992""!==a).map(a=>new RegExp(a,"ig")):[b];for(const g of f){if("question"===e||"all"===e){const{starts:b,ends:c}=getMatchIndices(a.question_sanitized,g);a.question=insertTokensIntoHTML(a.question,a.question_sanitized,[b,c])}if("answer"===e||"all"===e){const{starts:b,ends:c}=getMatchIndices(a.answer_sanitized,g);a.answer=insertTokensIntoHTML(a.answer,a.answer_sanitized,[b,c])}}return a}function highlightBonusQuery({bonus:a,regExp:b,searchType:e="all",ignoreWordOrder:c,queryString:d}){const f=c?d.split(" ").filter(a=>""!==a).map(a=>new RegExp(a,"ig")):[b];for(const g of f){if("question"===e||"all"===e){{const{starts:b,ends:c}=getMatchIndices(a.leadin_sanitized,g);a.leadin=insertTokensIntoHTML(a.leadin,a.leadin_sanitized,[b,c])}for(let b=0;ba+c)}function b(){return Math.floor(1e4/(r||25))}function c(a,c){switch(a.preventDefault(),c){case"first":R=1;break;case"previous":R=Math.max(1,R-1);break;case"next":R=Math.min(V,R+1,b());break;case"last":R=Math.min(V,b());break;default:R=c}S(R),$(paginationShiftLength*Math.floor((R-1)/paginationShiftLength)),e(a,!1,!0),window.scrollTo({top:document.getElementById("tossups").offsetTop-100,behavior:"smooth"})}function d(a,c){switch(a.preventDefault(),c){case"first":T=1;break;case"previous":T=Math.max(1,T-1);break;case"next":T=Math.min(X,T+1,b());break;case"last":T=Math.min(X,b());break;default:T=c}U(T),aa(paginationShiftLength*Math.floor((T-1)/paginationShiftLength)),e(a,!1,!0),window.scrollTo({top:document.getElementById("bonuses").offsetTop-100,behavior:"smooth"})}function e(a,b=!1,c=!1){const d=performance.now();a.preventDefault(),Q(!0),(b||!c)&&(R=1,T=1,S(R),U(T));const e=new URLSearchParams({queryString:t,...categoryManager.export(),difficulties:getDropdownValues("difficulties"),maxReturnLength:r,questionType:v,randomize:b,exactPhrase:H,powermarkOnly:J,regex:D,ignoreWordOrder:F,searchType:x,setName:document.getElementById("set-name").value,tossupPagination:R,bonusPagination:T,minYear:z,maxYear:B}).toString();fetch(`/api/query?${e}`).then(a=>{if(400===a.status)throw new Error("Invalid query");return a}).then(a=>a.json()).then(a=>{const{tossups:c,bonuses:f,queryString:h}=a,j=RegExp(h,"ig"),l=Math.max(1,r||25),{count:n,questionArray:p}=c,{count:s,questionArray:u}=f,v=JSON.parse(JSON.stringify(p)),w=JSON.parse(JSON.stringify(u));// create deep copy to highlight +import{downloadQuestionsAsText,downloadBonusesAsCSV,downloadTossupsAsCSV,downloadQuestionsAsJSON}from"./download.js";import api from"../scripts/api/index.js";import star from"../scripts/auth/star.js";import TossupCard from"../scripts/components/TossupCard.min.js";import BonusCard from"../scripts/components/BonusCard.min.js";import CategoryModal from"../scripts/components/CategoryModal.min.js";import DifficultyDropdown from"../scripts/components/DifficultyDropdown.min.js";import Star from"../scripts/components/Star.min.js";import CategoryManager from"../scripts/utilities/category-manager.js";import{getDropdownValues}from"../scripts/utilities/dropdown-checklist.js";import{insertTokensIntoHTML}from"../insert-tokens-into-html.js";const starredTossupIds=new Set(await star.getStarredTossupIds()),starredBonusIds=new Set(await star.getStarredBonusIds()),paginationShiftLength=992""!==a).map(a=>new RegExp(a,"ig")):[b];for(const g of f){if("question"===e||"all"===e){const{starts:b,ends:c}=getMatchIndices(a.question_sanitized,g);a.question=insertTokensIntoHTML(a.question,a.question_sanitized,[b,c])}if("answer"===e||"all"===e){const{starts:b,ends:c}=getMatchIndices(a.answer_sanitized,g);a.answer=insertTokensIntoHTML(a.answer,a.answer_sanitized,[b,c])}}return a}function highlightBonusQuery({bonus:a,regExp:b,searchType:e="all",ignoreWordOrder:c,queryString:d}){const f=c?d.split(" ").filter(a=>""!==a).map(a=>new RegExp(a,"ig")):[b];for(const g of f){if("question"===e||"all"===e){{const{starts:b,ends:c}=getMatchIndices(a.leadin_sanitized,g);a.leadin=insertTokensIntoHTML(a.leadin,a.leadin_sanitized,[b,c])}for(let b=0;ba+c)}function b(){return Math.floor(1e4/(r||25))}function c(a,c){switch(a.preventDefault(),c){case"first":R=1;break;case"previous":R=Math.max(1,R-1);break;case"next":R=Math.min(V,R+1,b());break;case"last":R=Math.min(V,b());break;default:R=c}S(R),$(paginationShiftLength*Math.floor((R-1)/paginationShiftLength)),e(a,!1,!0),window.scrollTo({top:document.getElementById("tossups").offsetTop-100,behavior:"smooth"})}function d(a,c){switch(a.preventDefault(),c){case"first":T=1;break;case"previous":T=Math.max(1,T-1);break;case"next":T=Math.min(X,T+1,b());break;case"last":T=Math.min(X,b());break;default:T=c}U(T),aa(paginationShiftLength*Math.floor((T-1)/paginationShiftLength)),e(a,!1,!0),window.scrollTo({top:document.getElementById("bonuses").offsetTop-100,behavior:"smooth"})}function e(a,b=!1,c=!1){const d=performance.now();a.preventDefault(),Q(!0),(b||!c)&&(R=1,T=1,S(R),U(T));const e=new URLSearchParams({queryString:t,...categoryManager.export(),difficulties:getDropdownValues("difficulties"),maxReturnLength:r,questionType:v,randomize:b,exactPhrase:H,powermarkOnly:J,regex:D,ignoreWordOrder:F,searchType:x,setName:document.getElementById("set-name").value,tossupPagination:R,bonusPagination:T,minYear:z,maxYear:B}).toString();fetch(`/api/query?${e}`).then(a=>{if(400===a.status)throw new Error("Invalid query");return a}).then(a=>a.json()).then(a=>{const{tossups:c,bonuses:f,queryString:h}=a,j=RegExp(h,"ig"),l=Math.max(1,r||25),{count:n,questionArray:p}=c,{count:s,questionArray:u}=f,v=JSON.parse(JSON.stringify(p)),w=JSON.parse(JSON.stringify(u));// create deep copy to highlight if(""!==t){for(let a=0;a{console.error("Error:",a),window.alert("Invalid query. Please check your search parameters and try again.")}).finally(()=>{document.querySelectorAll("b.collapsed[data-bs-toggle=\"collapse\"]").forEach(a=>a.classList.remove("collapsed")),document.querySelectorAll("div.card-container.collapse:not(.show)").forEach(a=>a.classList.add("show")),Q(!1)})}const[f,g]=React.useState([]),[h,i]=React.useState([]),[j,k]=React.useState([]),[l,m]=React.useState([]),[n,o]=React.useState(0),[p,q]=React.useState(0),[r,s]=React.useState(""),[t,u]=React.useState(""),[v,w]=React.useState("all"),[x,y]=React.useState("all"),[z,A]=React.useState(""),[B,C]=React.useState(""),[D,E]=React.useState(!1),[F,G]=React.useState(!1),[H,I]=React.useState(!1),[J,K]=React.useState(!1),[L,M]=React.useState(!1),[N,O]=React.useState(!0),[P,Q]=React.useState(!1);let[R,S]=React.useState(1),[T,U]=React.useState(1);const[V,W]=React.useState(1),[X,Y]=React.useState(1),[Z,$]=React.useState(0),[_,aa]=React.useState(0),[ba,ca]=React.useState(0),da="true"===window.localStorage.getItem("database-font-size")?window.localStorage.getItem("font-size")??16:16,ea=[];for(let a=0;a{document.getElementById("report-question-submit").addEventListener("click",function(){api.reportQuestion(document.getElementById("report-question-id").value,document.getElementById("report-question-reason").value,document.getElementById("report-question-description").value)}),window.addEventListener("popstate",a=>{if(null===a.state)return o(0),g([]),k([]),q(0),i([]),m([]),W(1),Y(1),$(0),void aa(0);const{tossups:b,highlightedTossupArray:c,bonuses:d,highlightedBonusArray:e,timeElapsed:f,workingMaxReturnLength:h,randomize:j}=a.state,{count:l,questionArray:n}=b,{count:p,questionArray:r}=d;o(l),g(n),k(c),q(p),i(r),m(e),j?(W(1),Y(1)):(W(Math.ceil(l/h)),Y(Math.ceil(p/h))),$(paginationShiftLength*Math.floor((R-1)/paginationShiftLength)),aa(paginationShiftLength*Math.floor((T-1)/paginationShiftLength)),ca(f)}),document.getElementById("set-list").innerHTML=api.getSetList().map(a=>``).join("")},[]),/*#__PURE__*/React.createElement("div",null,/*#__PURE__*/React.createElement(CategoryModal,{categoryManager:categoryManager,disablePercentView:!0}),/*#__PURE__*/React.createElement("form",{className:"mt-3",onSubmit:a=>{e(a)}},/*#__PURE__*/React.createElement("div",{className:"input-group mb-2"},/*#__PURE__*/React.createElement("input",{type:"text",className:"form-control",id:"query",placeholder:"Query",value:t,onChange:a=>{u(a.target.value)}}),/*#__PURE__*/React.createElement("button",{type:"submit",className:"btn btn-info"},"Search"),/*#__PURE__*/React.createElement("button",{id:"randomize",className:"btn btn-success",onClick:a=>{e(a,!0)}},"Random")),/*#__PURE__*/React.createElement("div",{className:"row"},/*#__PURE__*/React.createElement("div",{className:"col-6 col-xl-3 mb-2"},/*#__PURE__*/React.createElement(DifficultyDropdown,null)),/*#__PURE__*/React.createElement("div",{className:"col-6 col-xl-3 mb-2"},/*#__PURE__*/React.createElement("input",{type:"number",className:"form-control",id:"max-return-length",placeholder:"# to Display",value:r,onChange:a=>{s(a.target.value)}})),/*#__PURE__*/React.createElement("div",{className:"input-group col-12 col-xl-6 mb-2"},/*#__PURE__*/React.createElement("input",{type:"text",className:"form-control",id:"set-name",placeholder:"Set Name",list:"set-list"}),/*#__PURE__*/React.createElement("datalist",{id:"set-list"}),/*#__PURE__*/React.createElement("button",{type:"button",className:"btn btn-danger",id:"category-select-button","data-bs-toggle":"modal","data-bs-target":"#category-modal"},"Categories"))),/*#__PURE__*/React.createElement("div",{className:"row"},/*#__PURE__*/React.createElement("div",{className:"col-6 col-md-3 mb-2"},/*#__PURE__*/React.createElement("select",{className:"form-select",id:"search-type",value:x,onChange:a=>{y(a.target.value)}},/*#__PURE__*/React.createElement("option",{value:"all"},"All text"),/*#__PURE__*/React.createElement("option",{value:"question"},"Question"),/*#__PURE__*/React.createElement("option",{value:"answer"},"Answer"))),/*#__PURE__*/React.createElement("div",{className:"col-6 col-md-3 mb-2"},/*#__PURE__*/React.createElement("select",{className:"form-select",id:"question-type",value:v,onChange:a=>{w(a.target.value)}},/*#__PURE__*/React.createElement("option",{value:"all"},"All questions"),/*#__PURE__*/React.createElement("option",{value:"tossup"},"Tossups"),/*#__PURE__*/React.createElement("option",{value:"bonus"},"Bonuses"))),/*#__PURE__*/React.createElement("div",{className:"col-6 col-md-3 mb-2"},/*#__PURE__*/React.createElement("input",{type:"number",className:"form-control",id:"min-year",placeholder:"Min Year",value:z,onChange:a=>{A(a.target.value)}})),/*#__PURE__*/React.createElement("div",{className:"col-6 col-md-3 mb-2"},/*#__PURE__*/React.createElement("input",{type:"number",className:"form-control",id:"max-year",placeholder:"Max Year",value:B,onChange:a=>{C(a.target.value)}}))),/*#__PURE__*/React.createElement("div",{className:"row"},/*#__PURE__*/React.createElement("div",{className:"col-12"},/*#__PURE__*/React.createElement("div",{className:"form-check form-switch"},/*#__PURE__*/React.createElement("input",{className:"form-check-input",type:"checkbox",role:"switch",id:"toggle-regex",checked:D,onChange:()=>{E(!D)}}),/*#__PURE__*/React.createElement("label",{className:"form-check-label",htmlFor:"toggle-regex"},"Search using regular expression"),/*#__PURE__*/React.createElement("a",{href:"https://www.sitepoint.com/learn-regex/"}," What's this?")),/*#__PURE__*/React.createElement("div",{className:"form-check form-switch"},/*#__PURE__*/React.createElement("input",{className:"form-check-input",type:"checkbox",role:"switch",id:"toggle-ignore-word-order",checked:!D&&F,disabled:D,onChange:()=>{G(!F)}}),/*#__PURE__*/React.createElement("label",{className:"form-check-label",htmlFor:"toggle-ignore-word-order"},"Ignore word order")),/*#__PURE__*/React.createElement("div",{className:"form-check form-switch"},/*#__PURE__*/React.createElement("input",{className:"form-check-input",type:"checkbox",role:"switch",id:"toggle-exact-phrase",checked:!D&&H,disabled:D,onChange:()=>{I(!H)}}),/*#__PURE__*/React.createElement("label",{className:"form-check-label",htmlFor:"toggle-exact-phrase"},"Search for exact phrase")),/*#__PURE__*/React.createElement("div",{className:"form-check form-switch"},/*#__PURE__*/React.createElement("input",{className:"form-check-input",type:"checkbox",role:"switch",id:"toggle-powermark-only",checked:J,onChange:()=>{K(!J)}}),/*#__PURE__*/React.createElement("label",{className:"form-check-label",htmlFor:"toggle-powermark-only"},"Powermarked tossups only")),/*#__PURE__*/React.createElement("div",{className:"form-check form-switch"},/*#__PURE__*/React.createElement("input",{className:"form-check-input",type:"checkbox",role:"switch",id:"toggle-hide-answerlines",checked:L,onChange:()=>{M(!L)}}),/*#__PURE__*/React.createElement("label",{className:"form-check-label",htmlFor:"toggle-hide-answerlines"},"Hide answerlines")),/*#__PURE__*/React.createElement("div",{className:"form-check form-switch"},/*#__PURE__*/React.createElement("input",{className:"form-check-input",type:"checkbox",role:"switch",id:"toggle-show-card-footers",checked:N,onChange:()=>{O(!N)}}),/*#__PURE__*/React.createElement("label",{className:"form-check-label",htmlFor:"toggle-show-card-footers"},"Show card footers")),/*#__PURE__*/React.createElement("div",{className:"float-end"},/*#__PURE__*/React.createElement("b",null,"Download this page:"),/*#__PURE__*/React.createElement("a",{className:"ms-2 clickable",onClick:()=>{downloadQuestionsAsText(f,h)}},"TXT"),/*#__PURE__*/React.createElement("a",{className:"ms-2 clickable",onClick:()=>{downloadTossupsAsCSV(f),downloadBonusesAsCSV(h)}},"CSV"),/*#__PURE__*/React.createElement("a",{className:"ms-2 clickable",onClick:()=>{downloadQuestionsAsJSON(f,h)}},"JSON"))))),P&&/*#__PURE__*/React.createElement("div",{className:"d-block mx-auto mt-3 spinner-border",role:"status"},/*#__PURE__*/React.createElement("span",{className:"d-none"},"Loading...")),/*#__PURE__*/React.createElement("div",{className:"row text-center mt-2 mt-sm-0"},/*#__PURE__*/React.createElement("h3",{id:"tossups"},"Tossups")),0window.scrollTo({top:document.getElementById("bonuses").offsetTop,behavior:"smooth"})},"Jump to bonuses")))// eslint-disable-line :/*#__PURE__*/React.createElement("div",{className:"text-muted"},"No tossups found"),/*#__PURE__*/React.createElement("div",null,ea),1{c(a,"first")}},"\xAB")),/*#__PURE__*/React.createElement("li",{className:"page-item"},/*#__PURE__*/React.createElement("a",{className:"page-link",href:"#","aria-label":"Previous",onClick:a=>{c(a,"previous")}},"\u2039")),a(Math.min(Z),Math.min(Z+paginationShiftLength,V)).map(a=>{const b=R===a+1;return/*#__PURE__*/React.createElement("li",{key:`tossup-pagination-${a+1}`,className:"page-item"},/*#__PURE__*/React.createElement("a",{className:`page-link ${b&&"active"}`,href:"#",onClick:b=>{c(b,a+1)}},a+1))}),/*#__PURE__*/React.createElement("li",{className:"page-item"},/*#__PURE__*/React.createElement("a",{className:"page-link",href:"#","aria-label":"Next",onClick:a=>{c(a,"next")}},"\u203A")),/*#__PURE__*/React.createElement("li",{className:"page-item"},/*#__PURE__*/React.createElement("a",{className:"page-link",href:"#","aria-label":"Last",onClick:a=>{c(a,"last")}},/*#__PURE__*/React.createElement("span",{"aria-hidden":"true"},"\xBB"))))),/*#__PURE__*/React.createElement("div",{className:"mb-5"}),/*#__PURE__*/React.createElement("div",{className:"row text-center"},/*#__PURE__*/React.createElement("h3",{id:"bonuses"},"Bonuses")),0window.scrollTo({top:document.getElementById("tossups").offsetTop,behavior:"smooth"})},"Jump to tossups")))// eslint-disable-line :/*#__PURE__*/React.createElement("div",{className:"text-muted"},"No bonuses found"),/*#__PURE__*/React.createElement("div",null,fa),1{d(a,"first")}},"\xAB")),/*#__PURE__*/React.createElement("li",{className:"page-item"},/*#__PURE__*/React.createElement("a",{className:"page-link",href:"#","aria-label":"Previous",onClick:a=>{d(a,"previous")}},"\u2039")),a(Math.min(_),Math.min(_+paginationShiftLength,X)).map(a=>{const b=T===a+1;return/*#__PURE__*/React.createElement("li",{key:`bonus-pagination-${a+1}`,className:"page-item"},/*#__PURE__*/React.createElement("a",{className:`page-link ${b&&"active"}`,href:"#",onClick:b=>{d(b,a+1)}},a+1))}),/*#__PURE__*/React.createElement("li",{className:"page-item"},/*#__PURE__*/React.createElement("a",{className:"page-link",href:"#","aria-label":"Next",onClick:a=>{d(a,"next")}},"\u203A")),/*#__PURE__*/React.createElement("li",{className:"page-item"},/*#__PURE__*/React.createElement("a",{className:"page-link",href:"#","aria-label":"Last",onClick:a=>{d(a,"last")}},/*#__PURE__*/React.createElement("span",{"aria-hidden":"true"},"\xBB"))))),/*#__PURE__*/React.createElement("div",{className:"mb-5"}))}const root=ReactDOM.createRoot(document.getElementById("root"));root.render(/*#__PURE__*/React.createElement(QueryForm,null)); \ No newline at end of file diff --git a/client/multiplayer/room.jsx b/client/multiplayer/room.jsx index 812359b8..7b9ae0f3 100644 --- a/client/multiplayer/room.jsx +++ b/client/multiplayer/room.jsx @@ -452,7 +452,7 @@ function lostBuzzerRace ({ username, userId }) { if (userId === USER_ID) { document.getElementById('answer-input-group').classList.add('d-none'); } } -function next ({ tossup: nextTossup, type, username }) { +function next ({ oldTossup, tossup: nextTossup, type, username }) { switch (type) { case 'next': logEvent(username, 'went to the next question'); @@ -468,9 +468,7 @@ function next ({ tossup: nextTossup, type, username }) { } if (type === 'next' || type === 'skip') { - tossup.question = document.getElementById('question').innerHTML; - tossup.answer = document.getElementById('answer').innerHTML.replace('ANSWER: ', ''); - createTossupCard(tossup); + createTossupCard(oldTossup); } else if (type === 'start') { document.getElementById('next').classList.add('btn-primary'); document.getElementById('next').classList.remove('btn-success'); diff --git a/client/multiplayer/room.min.js b/client/multiplayer/room.min.js index da2ff746..84cdfd51 100644 --- a/client/multiplayer/room.min.js +++ b/client/multiplayer/room.min.js @@ -1,7 +1,7 @@ /* globals WebSocket */import account from"../scripts/accounts.js";import questionStats from"../scripts/auth/question-stats.js";import api from"../scripts/api/index.js";import audio from"../audio/index.js";import CategoryManager from"../scripts/utilities/category-manager.js";import{getDropdownValues}from"../scripts/utilities/dropdown-checklist.js";import{arrayToRange,createTossupCard,rangeToArray}from"../scripts/utilities/index.js";import{escapeHTML}from"../scripts/utilities/strings.js";import CategoryModal from"../scripts/components/CategoryModal.min.js";import DifficultyDropdown from"../scripts/components/DifficultyDropdown.min.js";const categoryManager=new CategoryManager;let oldCategories=JSON.stringify(categoryManager.export()),maxPacketNumber=24;/** * userId to player object */const players={},ROOM_NAME=decodeURIComponent(window.location.pathname.substring(13));let tossup={},USER_ID=window.localStorage.getItem("USER_ID")||"unknown",username=window.localStorage.getItem("multiplayer-username")||(await api.getRandomName());const socket=new WebSocket(window.location.href.replace("http","ws")+(window.location.href.endsWith("?private=true")?"&":"?")+new URLSearchParams({roomName:ROOM_NAME,userId:USER_ID,username}).toString()),PING_INTERVAL_ID=setInterval(()=>socket.send(JSON.stringify({type:"ping"})),45e3);// Ping server every 45 seconds to prevent socket disconnection -socket.onclose=function(a){const{code:b}=a;3e3!==b&&window.alert("Disconnected from server"),clearInterval(PING_INTERVAL_ID)},socket.onmessage=function(a){const b=JSON.parse(a.data);switch(b.type){case"buzz":return buzz(b);case"force-username":return forceUsername(b);case"chat":return chat(b,!1);case"chat-live-update":return chat(b,!0);case"clear-stats":return clearStats(b);case"connection-acknowledged":return connectionAcknowledged(b);case"connection-acknowledged-query":return connectionAcknowledgedQuery(b);case"connection-acknowledged-tossup":return connectionAcknowledgedTossup(b);case"end-of-set":return endOfSet(b);case"error":return handleError(b);case"give-answer":return giveAnswer(b);case"give-answer-live-update":return logGiveAnswer(b,!0);case"join":return join(b);case"leave":return leave(b);case"lost-buzzer-race":return lostBuzzerRace(b);case"next":return next(b);case"no-questions-found":return noQuestionsFound(b);case"pause":return pause(b);case"reveal-answer":return revealAnswer(b);case"set-categories":return setCategories(b);case"set-difficulties":return setDifficulties(b);case"set-reading-speed":return setReadingSpeed(b);case"set-packet-numbers":return setPacketNumbers(b);case"set-set-name":return setSetName(b);case"set-username":return setUsername(b);case"set-year-range":return setYearRange(b);case"skip":return next(b);case"start":return next(b);case"timer-update":return updateTimerDisplay(b.timeRemaining);case"toggle-lock":return toggleLock(b);case"toggle-login-required":return toggleLoginRequired(b);case"toggle-powermark-only":return togglePowermarkOnly(b);case"toggle-public":return togglePublic(b);case"toggle-rebuzz":return toggleRebuzz(b);case"toggle-select-by-set-name":return toggleSelectBySetName(b);case"toggle-skip":return toggleSkip(b);case"toggle-standard-only":return toggleStandardOnly(b);case"toggle-timer":return toggleTimer(b);case"update-question":return updateQuestion(b)}};function buzz({userId:a,username:b}){logEvent(b,"buzzed"),document.getElementById("buzz").disabled=!0,document.getElementById("pause").disabled=!0,document.getElementById("next").disabled=!0,document.getElementById("skip").disabled=!0,a===USER_ID&&(document.getElementById("answer-input-group").classList.remove("d-none"),document.getElementById("answer-input").focus())}function chat({message:a,userId:c,username:d},e=!1){if(!e&&""===a)return void document.getElementById("live-chat-"+c).parentElement.remove();if(!e&&a)return document.getElementById("live-chat-"+c).className="",void(document.getElementById("live-chat-"+c).id="");if(document.getElementById("live-chat-"+c))return void(document.getElementById("live-chat-"+c).textContent=a);const f=document.createElement("b");f.textContent=d;const b=document.createElement("span");b.classList.add("text-muted"),b.id="live-chat-"+c,b.textContent=a;const g=document.createElement("li");g.appendChild(f),g.appendChild(document.createTextNode(" ")),g.appendChild(b),document.getElementById("room-history").prepend(g)}function clearStats({userId:a}){for(const b of["celerity","negs","points","powers","tens","tuh","zeroes"])players[a][b]=0;upsertPlayerItem(players[a]),sortPlayerListGroup()}function connectionAcknowledged({buzzedIn:a,canBuzz:b,isPermanent:c,players:d,questionProgress:e,settings:f,userId:g}){document.getElementById("buzz").disabled=!b,c&&(document.getElementById("category-select-button").disabled=!0,document.getElementById("toggle-public").disabled=!0,document.getElementById("toggle-select-by-set-name").disabled=!0,document.getElementById("private-chat-warning").innerHTML="This is a permanent room. Some settings have been restricted."),Object.keys(d).forEach(a=>{d[a].celerity=d[a].celerity.correct.average,players[a]=d[a],upsertPlayerItem(players[a])}),sortPlayerListGroup();0===e?(document.getElementById("next").textContent="Start",document.getElementById("next").classList.remove("btn-primary"),document.getElementById("next").classList.add("btn-success")):1===e?(showSkipButton(),document.getElementById("settings").classList.add("d-none"),a?(document.getElementById("buzz").disabled=!0,document.getElementById("next").disabled=!0,document.getElementById("pause").disabled=!0):(document.getElementById("buzz").disabled=!1,document.getElementById("pause").disabled=!1)):2===e?(showNextButton(),document.getElementById("settings").classList.add("d-none")):void 0;document.getElementById("toggle-lock").checked=f.lock,document.getElementById("toggle-login-required").checked=f.loginRequired,document.getElementById("chat").disabled=f.public,document.getElementById("toggle-lock").disabled=f.public,document.getElementById("toggle-login-required").disabled=f.public,document.getElementById("toggle-timer").disabled=f.public,document.getElementById("toggle-public").checked=f.public,document.getElementById("reading-speed").value=f.readingSpeed,document.getElementById("reading-speed-display").textContent=f.readingSpeed,document.getElementById("toggle-rebuzz").checked=f.rebuzz,document.getElementById("toggle-skip").checked=f.skip,document.getElementById("timer").classList.toggle("d-none",!f.timer),document.getElementById("toggle-timer").checked=f.timer,USER_ID=g,window.localStorage.setItem("USER_ID",USER_ID)}async function connectionAcknowledgedQuery({difficulties:i=[],minYear:a,maxYear:b,packetNumbers:j=[],powermarkOnly:c,selectBySetName:d,setName:k="",standardOnly:e,validAlternateSubcategories:f,validCategories:g,validSubcategories:h}){setDifficulties({difficulties:i}),$("#slider").slider("values",0,a),$("#slider").slider("values",1,b),document.getElementById("year-range-a").textContent=a,document.getElementById("year-range-b").textContent=b,document.getElementById("packet-number").value=arrayToRange(j),document.getElementById("toggle-powermark-only").checked=c,document.getElementById("difficulty-settings").classList.toggle("d-none",d),document.getElementById("set-settings").classList.toggle("d-none",!d),document.getElementById("toggle-select-by-set-name").checked=d,document.getElementById("toggle-powermark-only").disabled=d,document.getElementById("toggle-standard-only").disabled=d,document.getElementById("set-name").value=k,maxPacketNumber=await api.getNumPackets(k),""!==k&&0===maxPacketNumber&&document.getElementById("set-name").classList.add("is-invalid"),document.getElementById("toggle-standard-only").checked=e,categoryManager.import(g,h,f),categoryManager.loadCategoryModal()}function connectionAcknowledgedTossup({tossup:a}){tossup=a,document.getElementById("set-name-info").textContent=tossup?.set?.name??"",document.getElementById("packet-number-info").textContent=tossup?.packet?.number??"-",document.getElementById("question-number-info").textContent=tossup?.number??"-"}function endOfSet(){window.alert("You have reached the end of the set")}function forceUsername({message:a,username:b}){window.alert(a),window.localStorage.setItem("multiplayer-username",b),document.querySelector("#username").value=b}async function giveAnswer({celerity:a,directive:b,directedPrompt:c,givenAnswer:d,perQuestionCelerity:e,score:f,tossup:g,userId:h,username:i}){document.getElementById("answer-input").value="",document.getElementById("answer-input-group").classList.add("d-none"),document.getElementById("answer-input").blur(),logGiveAnswer({directive:b,message:d,username:i}),"prompt"===b&&c?logEvent(i,`was prompted with "${c}"`):"prompt"===b?logEvent(i,"was prompted"):logEvent(i,`${0{a.textContent=parseInt(a.innerHTML)+1})),"reject"===b&&(document.getElementById("buzz").disabled=!document.getElementById("toggle-rebuzz").checked&&h===USER_ID),10f&&players[h].negs++,players[h].points+=f,players[h].tuh++,players[h].celerity=a,upsertPlayerItem(players[h]),sortPlayerListGroup()),"prompt"!==b&&h===USER_ID&&(await account.getUsername())&&questionStats.recordTossup(g,0{const b=parseInt(document.getElementById("points-"+d.id.substring(f)).innerHTML),e=parseInt(document.getElementById("points-"+a.id.substring(f)).innerHTML);// if points are equal, sort alphabetically by username +socket.onclose=function(a){const{code:b}=a;3e3!==b&&window.alert("Disconnected from server"),clearInterval(PING_INTERVAL_ID)},socket.onmessage=function(a){const b=JSON.parse(a.data);switch(b.type){case"buzz":return buzz(b);case"force-username":return forceUsername(b);case"chat":return chat(b,!1);case"chat-live-update":return chat(b,!0);case"clear-stats":return clearStats(b);case"connection-acknowledged":return connectionAcknowledged(b);case"connection-acknowledged-query":return connectionAcknowledgedQuery(b);case"connection-acknowledged-tossup":return connectionAcknowledgedTossup(b);case"end-of-set":return endOfSet(b);case"error":return handleError(b);case"give-answer":return giveAnswer(b);case"give-answer-live-update":return logGiveAnswer(b,!0);case"join":return join(b);case"leave":return leave(b);case"lost-buzzer-race":return lostBuzzerRace(b);case"next":return next(b);case"no-questions-found":return noQuestionsFound(b);case"pause":return pause(b);case"reveal-answer":return revealAnswer(b);case"set-categories":return setCategories(b);case"set-difficulties":return setDifficulties(b);case"set-reading-speed":return setReadingSpeed(b);case"set-packet-numbers":return setPacketNumbers(b);case"set-set-name":return setSetName(b);case"set-username":return setUsername(b);case"set-year-range":return setYearRange(b);case"skip":return next(b);case"start":return next(b);case"timer-update":return updateTimerDisplay(b.timeRemaining);case"toggle-lock":return toggleLock(b);case"toggle-login-required":return toggleLoginRequired(b);case"toggle-powermark-only":return togglePowermarkOnly(b);case"toggle-public":return togglePublic(b);case"toggle-rebuzz":return toggleRebuzz(b);case"toggle-select-by-set-name":return toggleSelectBySetName(b);case"toggle-skip":return toggleSkip(b);case"toggle-standard-only":return toggleStandardOnly(b);case"toggle-timer":return toggleTimer(b);case"update-question":return updateQuestion(b)}};function buzz({userId:a,username:b}){logEvent(b,"buzzed"),document.getElementById("buzz").disabled=!0,document.getElementById("pause").disabled=!0,document.getElementById("next").disabled=!0,document.getElementById("skip").disabled=!0,a===USER_ID&&(document.getElementById("answer-input-group").classList.remove("d-none"),document.getElementById("answer-input").focus())}function chat({message:a,userId:c,username:d},e=!1){if(!e&&""===a)return void document.getElementById("live-chat-"+c).parentElement.remove();if(!e&&a)return document.getElementById("live-chat-"+c).className="",void(document.getElementById("live-chat-"+c).id="");if(document.getElementById("live-chat-"+c))return void(document.getElementById("live-chat-"+c).textContent=a);const f=document.createElement("b");f.textContent=d;const b=document.createElement("span");b.classList.add("text-muted"),b.id="live-chat-"+c,b.textContent=a;const g=document.createElement("li");g.appendChild(f),g.appendChild(document.createTextNode(" ")),g.appendChild(b),document.getElementById("room-history").prepend(g)}function clearStats({userId:a}){for(const b of["celerity","negs","points","powers","tens","tuh","zeroes"])players[a][b]=0;upsertPlayerItem(players[a]),sortPlayerListGroup()}function connectionAcknowledged({buzzedIn:a,canBuzz:b,isPermanent:c,players:d,questionProgress:e,settings:f,userId:g}){document.getElementById("buzz").disabled=!b,c&&(document.getElementById("category-select-button").disabled=!0,document.getElementById("toggle-public").disabled=!0,document.getElementById("toggle-select-by-set-name").disabled=!0,document.getElementById("private-chat-warning").innerHTML="This is a permanent room. Some settings have been restricted."),Object.keys(d).forEach(a=>{d[a].celerity=d[a].celerity.correct.average,players[a]=d[a],upsertPlayerItem(players[a])}),sortPlayerListGroup();0===e?(document.getElementById("next").textContent="Start",document.getElementById("next").classList.remove("btn-primary"),document.getElementById("next").classList.add("btn-success")):1===e?(showSkipButton(),document.getElementById("settings").classList.add("d-none"),a?(document.getElementById("buzz").disabled=!0,document.getElementById("next").disabled=!0,document.getElementById("pause").disabled=!0):(document.getElementById("buzz").disabled=!1,document.getElementById("pause").disabled=!1)):2===e?(showNextButton(),document.getElementById("settings").classList.add("d-none")):void 0;document.getElementById("toggle-lock").checked=f.lock,document.getElementById("toggle-login-required").checked=f.loginRequired,document.getElementById("chat").disabled=f.public,document.getElementById("toggle-lock").disabled=f.public,document.getElementById("toggle-login-required").disabled=f.public,document.getElementById("toggle-timer").disabled=f.public,document.getElementById("toggle-public").checked=f.public,document.getElementById("reading-speed").value=f.readingSpeed,document.getElementById("reading-speed-display").textContent=f.readingSpeed,document.getElementById("toggle-rebuzz").checked=f.rebuzz,document.getElementById("toggle-skip").checked=f.skip,document.getElementById("timer").classList.toggle("d-none",!f.timer),document.getElementById("toggle-timer").checked=f.timer,USER_ID=g,window.localStorage.setItem("USER_ID",USER_ID)}async function connectionAcknowledgedQuery({difficulties:i=[],minYear:a,maxYear:b,packetNumbers:j=[],powermarkOnly:c,selectBySetName:d,setName:k="",standardOnly:e,validAlternateSubcategories:f,validCategories:g,validSubcategories:h}){setDifficulties({difficulties:i}),$("#slider").slider("values",0,a),$("#slider").slider("values",1,b),document.getElementById("year-range-a").textContent=a,document.getElementById("year-range-b").textContent=b,document.getElementById("packet-number").value=arrayToRange(j),document.getElementById("toggle-powermark-only").checked=c,document.getElementById("difficulty-settings").classList.toggle("d-none",d),document.getElementById("set-settings").classList.toggle("d-none",!d),document.getElementById("toggle-select-by-set-name").checked=d,document.getElementById("toggle-powermark-only").disabled=d,document.getElementById("toggle-standard-only").disabled=d,document.getElementById("set-name").value=k,maxPacketNumber=await api.getNumPackets(k),""!==k&&0===maxPacketNumber&&document.getElementById("set-name").classList.add("is-invalid"),document.getElementById("toggle-standard-only").checked=e,categoryManager.import(g,h,f),categoryManager.loadCategoryModal()}function connectionAcknowledgedTossup({tossup:a}){tossup=a,document.getElementById("set-name-info").textContent=tossup?.set?.name??"",document.getElementById("packet-number-info").textContent=tossup?.packet?.number??"-",document.getElementById("question-number-info").textContent=tossup?.number??"-"}function endOfSet(){window.alert("You have reached the end of the set")}function forceUsername({message:a,username:b}){window.alert(a),window.localStorage.setItem("multiplayer-username",b),document.querySelector("#username").value=b}async function giveAnswer({celerity:a,directive:b,directedPrompt:c,givenAnswer:d,perQuestionCelerity:e,score:f,tossup:g,userId:h,username:i}){document.getElementById("answer-input").value="",document.getElementById("answer-input-group").classList.add("d-none"),document.getElementById("answer-input").blur(),logGiveAnswer({directive:b,message:d,username:i}),"prompt"===b&&c?logEvent(i,`was prompted with "${c}"`):"prompt"===b?logEvent(i,"was prompted"):logEvent(i,`${0{a.textContent=parseInt(a.innerHTML)+1})),"reject"===b&&(document.getElementById("buzz").disabled=!document.getElementById("toggle-rebuzz").checked&&h===USER_ID),10f&&players[h].negs++,players[h].points+=f,players[h].tuh++,players[h].celerity=a,upsertPlayerItem(players[h]),sortPlayerListGroup()),"prompt"!==b&&h===USER_ID&&(await account.getUsername())&&questionStats.recordTossup(g,0{const b=parseInt(document.getElementById("points-"+d.id.substring(f)).innerHTML),e=parseInt(document.getElementById("points-"+a.id.substring(f)).innerHTML);// if points are equal, sort alphabetically by username if(b===e){const b=document.getElementById("username-"+d.id.substring(f)).innerHTML,e=document.getElementById("username-"+a.id.substring(f)).innerHTML;return c?b.localeCompare(e):e.localeCompare(b)}return c?e-b:b-e}).forEach(a=>{d.appendChild(a)})}function setCategories({alternateSubcategories:a,categories:b,subcategories:c,username:d}){logEvent(d,"updated the categories"),categoryManager.import(b,c,a),categoryManager.loadCategoryModal()}function setDifficulties({difficulties:a,username:b=void 0}){b&&logEvent(b,0{const c=b.querySelector("input");a.includes(parseInt(c.value))?(c.checked=!0,b.classList.add("active")):(c.checked=!1,b.classList.remove("active"))})}function setPacketNumbers({username:a,value:b}){b=arrayToRange(b),logEvent(a,0 0 && subcategories.length === 0) { categories.forEach(category => { SUBCATEGORIES[category].forEach(subcategory => { diff --git a/client/singleplayer/ClientTossupRoom.js b/client/singleplayer/ClientTossupRoom.js new file mode 100644 index 00000000..5ea1c5e8 --- /dev/null +++ b/client/singleplayer/ClientTossupRoom.js @@ -0,0 +1,77 @@ +import api from '../scripts/api/index.js'; +// import TossupRoom from '../../quizbowl/TossupRoom.js'; +import TossupRoom from '../TossupRoom.js'; + +export default class ClientTossupRoom extends TossupRoom { + constructor (name, categories = [], subcategories = [], alternateSubcategories = []) { + super(name, categories, subcategories, alternateSubcategories); + + this.previous = { + celerity: 0, + endOfQuestion: false, + isCorrect: true, + inPower: false, + negValue: -5, + powerValue: 15, + tossup: {} + }; + this.settings.skip = true; + + this.checkAnswer = api.checkAnswer; + this.getRandomTossups = async (args) => await api.getRandomTossup({ number: 20, ...args }); + this.getSet = async ({ setName, packetNumbers }) => setName ? await api.getPacketTossups(setName, packetNumbers[0] ?? 1) : []; + this.getSetList = api.getSetList; + this.getNumPackets = api.getNumPackets; + + this.setList = this.getSetList().concat(''); + } + + async message (userId, message) { + switch (message.type) { + case 'toggle-correct': return this.toggleCorrect(userId, message); + default: super.message(userId, message); + } + } + + get liveAnswer () { + return document.getElementById('answer-input').value; + } + + set liveAnswer (value) { + document.getElementById('answer-input').value = value; + } + + async scoreTossup ({ givenAnswer }) { + const { celerity, directive, directedPrompt, endOfQuestion, inPower, points } = await super.scoreTossup({ givenAnswer }); + this.previous.celerity = celerity; + this.previous.endOfQuestion = endOfQuestion; + this.previous.isCorrect = points > 0; + this.previous.inPower = inPower; + this.previous.tossup = this.tossup; + return { celerity, directive, directedPrompt, points }; + } + + toggleCorrect (userId, { correct }) { + const multiplier = correct ? 1 : -1; + + if (this.previous.inPower) { + this.players[userId].powers += multiplier * 1; + this.players[userId].points += multiplier * this.previous.powerValue; + } else { + this.players[userId].tens += multiplier * 1; + this.players[userId].points += multiplier * 10; + } + + if (this.previous.endOfQuestion) { + this.players[userId].dead += multiplier * -1; + } else { + this.players[userId].negs += multiplier * -1; + this.players[userId].points += multiplier * -this.previous.negValue; + } + + this.players[userId].celerity.correct.total += multiplier * this.previous.celerity; + this.players[userId].celerity.correct.average = this.players[userId].celerity.correct.total / (this.players[userId].powers + this.players[userId].tens); + + this.emitMessage({ type: 'toggle-correct', correct, userId }); + } +} diff --git a/client/singleplayer/tossups/index.jsx b/client/singleplayer/tossups/index.jsx index 22f442b4..a805f4e6 100644 --- a/client/singleplayer/tossups/index.jsx +++ b/client/singleplayer/tossups/index.jsx @@ -1,366 +1,130 @@ import account from '../../scripts/accounts.js'; -import questionStats from '../../scripts/auth/question-stats.js'; import api from '../../scripts/api/index.js'; +import questionStats from '../../scripts/auth/question-stats.js'; import audio from '../../audio/index.js'; -import Timer from '../../scripts/Timer.js'; -import { arrayToRange, createTossupCard, rangeToArray } from '../../scripts/utilities/index.js'; +// import Player from '../../../quizbowl/Player.js'; +import Player from '../../Player.js'; +import ClientTossupRoom from '../ClientTossupRoom.js'; import CategoryManager from '../../scripts/utilities/category-manager.js'; +import { createTossupCard, rangeToArray } from '../../scripts/utilities/index.js'; import { getDropdownValues } from '../../scripts/utilities/dropdown-checklist.js'; -import { insertTokensIntoHTML } from '../../insert-tokens-into-html.js'; import CategoryModal from '../../scripts/components/CategoryModal.min.js'; import DifficultyDropdown from '../../scripts/components/DifficultyDropdown.min.js'; -// Functions and variables specific to the tossups page. +let maxPacketNumber = 24; -const ANSWER_TIME_LIMIT = 10; -const DEAD_TIME_LIMIT = 5; +const categoryManager = new CategoryManager(); -// Status variables -let buzzpointIndex = -1; -let currentlyBuzzing = false; -let maxPacketNumber = 24; -let paused = false; -let questionNumber = 0; // WARNING: 1-indexed -const timer = new Timer(); +const USER_ID = 'user'; -/** - * An array of random questions. - * We get 20 random questions at a time so we don't have to make an HTTP request between every question. - */ -let randomTossups = []; -let timeoutID = -1; - -let tossups = [{}]; -let tossupText = ''; -let tossupTextSplit = []; - -const previous = { - isCorrect: true, - inPower: false, - negValue: -5, - powerValue: 15, - endOfQuestion: false, - celerity: 0 -}; +const room = new ClientTossupRoom(); +room.players[USER_ID] = new Player(USER_ID); -const stats = window.sessionStorage.getItem('tossup-stats') - ? JSON.parse(window.sessionStorage.getItem('tossup-stats')) - : { - powers: 0, - tens: 0, - negs: 0, - dead: 0, - points: 0, - totalCorrectCelerity: 0 - }; - -const defaults = { - alternateSubcategories: [], - categories: [], - difficulties: [], - minYear: 2010, - maxYear: 2024, - packetNumbers: [], - powermarkOnly: false, - setName: '', - standardOnly: false, - subcategories: [], - version: '01-06-2024' +const socket = { + send: onmessage, + sendToServer: (message) => room.message(USER_ID, message) }; - -let query; -if (!window.localStorage.getItem('singleplayer-tossup-query')) { - query = defaults; -} else { - query = JSON.parse(window.localStorage.getItem('singleplayer-tossup-query')); - if (query.version !== defaults.version) { - query = defaults; - window.localStorage.setItem('singleplayer-tossup-query', JSON.stringify(query)); +room.sockets[USER_ID] = socket; + +function onmessage (message) { + const data = JSON.parse(message); + console.log(data); + switch (data.type) { + case 'buzz': return buzz(data); + case 'clear-stats': return clearStats(data); + case 'end-of-set': return endOfSet(data); + case 'give-answer': return giveAnswer(data); + case 'next': return next(data); + case 'no-questions-found': return noQuestionsFound(data); + case 'pause': return pause(data); + case 'reveal-answer': return revealAnswer(data); + case 'set-categories': return setCategories(data); + case 'set-difficulties': return setDifficulties(data); + case 'set-reading-speed': return setReadingSpeed(data); + case 'set-packet-numbers': return setPacketNumbers(data); + case 'set-set-name': return setSetName(data); + case 'set-year-range': return setYearRange(data); + case 'skip': return next(data); + case 'start': return next(data); + case 'timer-update': return updateTimerDisplay(data.timeRemaining); + case 'toggle-correct': return toggleCorrect(data); + case 'toggle-powermark-only': return togglePowermarkOnly(data); + case 'toggle-rebuzz': return toggleRebuzz(data); + case 'toggle-select-by-set-name': return toggleSelectBySetName(data); + case 'toggle-standard-only': return toggleStandardOnly(data); + case 'toggle-timer': return toggleTimer(data); + case 'update-question': return updateQuestion(data); } } -const categoryManager = new CategoryManager(query.categories, query.subcategories, query.alternateSubcategories); - -const settings = window.localStorage.getItem('singleplayer-tossup-settings') - ? JSON.parse(window.localStorage.getItem('singleplayer-tossup-settings')) - : { - readingSpeed: 50, - rebuzz: false, - selectBySetName: false, - showHistory: true, - timer: true, - typeToAnswer: true - }; - -// Load query and settings first so user doesn't see the default settings -if (settings.readingSpeed) { - document.getElementById('reading-speed-display').textContent = settings.readingSpeed; - document.getElementById('reading-speed').value = settings.readingSpeed; -} - -if (settings.rebuzz) { - document.getElementById('toggle-rebuzz').checked = true; -} - -if (settings.selectBySetName) { - document.getElementById('difficulty-settings').classList.add('d-none'); - document.getElementById('set-settings').classList.remove('d-none'); - document.getElementById('toggle-select-by-set-name').checked = true; - document.getElementById('toggle-powermark-only').disabled = true; - document.getElementById('toggle-standard-only').disabled = true; -} +function buzz ({ timer, userId, username }) { + if (audio.soundEffects) { audio.buzz.play(); } -if (!settings.showHistory) { - document.getElementById('toggle-show-history').checked = false; - document.getElementById('room-history').classList.add('d-none'); -} - -if (!settings.timer) { - document.getElementById('toggle-timer').checked = false; - document.getElementById('timer').classList.add('d-none'); -} - -if (!settings.typeToAnswer) { - document.getElementById('type-to-answer').checked = false; - document.getElementById('toggle-rebuzz').disabled = true; -} - -if (query.packetNumbers) { - document.getElementById('packet-number').value = arrayToRange(query.packetNumbers); -} - -if (query.powermarkOnly) { - document.getElementById('toggle-powermark-only').checked = true; + const typeToAnswer = document.getElementById('type-to-answer').checked; + if (typeToAnswer) { + document.getElementById('answer-input-group').classList.remove('d-none'); + document.getElementById('answer-input').focus(); + document.getElementById('buzz').disabled = true; + } } -if (query.setName) { - document.getElementById('set-name').value = query.setName; - api.getNumPackets(query.setName).then(numPackets => { - maxPacketNumber = numPackets; - if (maxPacketNumber === 0) { - document.getElementById('set-name').classList.add('is-invalid'); - } else { - document.getElementById('packet-number').placeholder = `Packet Numbers (1-${maxPacketNumber})`; - } - }); +function clearStats ({ userId }) { + if (userId !== USER_ID) { return; } + updateStatDisplay(room.players[USER_ID]); } -updateStatDisplay(); - -function queryLock () { - document.getElementById('question').textContent = 'Fetching questions...'; - document.getElementById('start').disabled = true; - document.getElementById('next').disabled = true; - document.getElementById('pause').disabled = true; +function endOfSet () { + window.alert('No more questions left'); document.getElementById('buzz').disabled = true; -} - -function queryUnlock () { - document.getElementById('question').textContent = ''; - document.getElementById('start').disabled = false; - document.getElementById('next').disabled = false; - document.getElementById('pause').disabled = false; - document.getElementById('buzz').disabled = false; -} - -/** - * @returns {Promise} Whether or not there is a next question - */ -async function advanceQuestion () { - if (settings.selectBySetName) { - // Get the next question if the current one is in the wrong category and subcategory - do { - questionNumber++; - - // Go to the next packet if you reach the end of this packet - if (questionNumber > tossups.length) { - query.packetNumbers.shift(); - if (query.packetNumbers.length === 0) { - window.alert('No more questions left'); - document.getElementById('buzz').disabled = true; - document.getElementById('pause').disabled = true; - document.getElementById('next').disabled = true; - return false; // alert the user if there are no more packets - } - - queryLock(); - try { - tossups = await api.getPacketTossups(query.setName, query.packetNumbers[0]); - } finally { - queryUnlock(); - } - - questionNumber = 1; - } - } while (!categoryManager.isValidCategory(tossups[questionNumber - 1])); - - if (Object.keys(tossups[0]).length > 0) { - tossupText = tossups[questionNumber - 1].question_sanitized; - tossupTextSplit = tossupText.split(' ').filter(word => word !== ''); - document.getElementById('question-number-info').textContent = questionNumber; - } - } else { - queryLock(); - try { - tossups = await getRandomTossup(query, categoryManager); - tossups = [tossups]; - } finally { - queryUnlock(); - } - - if (!tossups[0]) { - window.alert('No questions found'); - return false; - } - - query.setName = tossups[0].set.name; - query.packetNumbers = [tossups[0].packet.number]; - - tossupText = tossups[0].question_sanitized; - tossupTextSplit = tossupText.split(' ').filter(word => word !== ''); - document.getElementById('question-number-info').textContent = tossups[0].number; - questionNumber = 1; - } - - return true; -} - -/** - * Called when the users buzzes. - * The first "buzz" pauses the question, and the second "buzz" reveals the rest of the question - * and updates the score. - */ -function buzz () { - // Stop the question reading - clearTimeout(timeoutID); - currentlyBuzzing = true; - if (audio.soundEffects) audio.buzz.play(); - - buzzpointIndex = document.getElementById('question').textContent.length; - if (!tossupTextSplit.includes('(*)') && tossupText.includes('(*)')) { - buzzpointIndex += 3; - } - - // Include buzzpoint - document.getElementById('question').textContent += '(#) '; - - document.getElementById('buzz').textContent = 'Reveal'; - document.getElementById('next').disabled = true; - document.getElementById('start').disabled = true; document.getElementById('pause').disabled = true; - - if (settings.timer) { - timer.stopTimer(); - timer.startTimer(ANSWER_TIME_LIMIT, () => document.getElementById('answer-submit').click()); - } -} - -/** - * Clears user stats. - */ -function clearStats () { - stats.powers = 0; - stats.tens = 0; - stats.negs = 0; - stats.dead = 0; - stats.points = 0; - stats.totalCorrectCelerity = 0; - - updateStatDisplay(); - window.sessionStorage.removeItem('tossup-stats'); -} - -async function giveAnswer (givenAnswer) { - currentlyBuzzing = false; - - const { directive, directedPrompt } = await api.checkAnswer(tossups[questionNumber - 1].answer, givenAnswer); - - switch (directive) { - case 'accept': { - const points = updateScore(true); - if (audio.soundEffects) { - if (points > 10) { - audio.power.play(); - } else { - audio.correct.play(); - } - } - revealQuestion(); - break; - } - case 'reject': - updateScore(false); - if (audio.soundEffects) audio.incorrect.play(); - if (settings.rebuzz) { - document.getElementById('buzz').disabled = false; - document.getElementById('buzz').textContent = 'Buzz'; - document.getElementById('next').disabled = false; - document.getElementById('pause').disabled = false; - document.getElementById('start').disabled = false; - readQuestion(Date.now()); - } else { - revealQuestion(); - } - break; - case 'prompt': - document.getElementById('answer-input-group').classList.remove('d-none'); - document.getElementById('answer-input').focus(); - document.getElementById('answer-input').placeholder = directedPrompt ? `Prompt: "${directedPrompt}"` : 'Prompt'; - break; - } -} - -function isPace (setName) { - if (!setName) { return false; } - - return setName.includes('PACE'); -} - -async function loadRandomTossups ({ alternateSubcategories, categories, difficulties, maxYear, minYear, number = 1, powermarkOnly, standardOnly, subcategories } = {}) { - randomTossups = []; - randomTossups = await api.getRandomTossup({ alternateSubcategories, categories, difficulties, maxYear, minYear, number, powermarkOnly, standardOnly, subcategories }); + document.getElementById('next').disabled = true; } -/** - * Get a random tossup. - * @returns - */ -async function getRandomTossup ({ alternateSubcategories, categories, difficulties, minYear, maxYear, powermarkOnly, subcategories, standardOnly } = {}, categoryManager = null) { - if (categoryManager?.percentView) { - categories = [categoryManager.getRandomCategory()]; - subcategories = []; - alternateSubcategories = []; - await loadRandomTossups({ alternateSubcategories, categories, difficulties, maxYear, minYear, powermarkOnly, subcategories, standardOnly }); - return randomTossups.pop(); +async function giveAnswer ({ directive, directedPrompt, perQuestionCelerity, score, tossup, userId }) { + if (directive === 'prompt') { + document.getElementById('answer-input-group').classList.remove('d-none'); + document.getElementById('answer-input').focus(); + document.getElementById('answer-input').placeholder = directedPrompt ? `Prompt: "${directedPrompt}"` : 'Prompt'; + return; } - if (randomTossups.length === 0) { - await loadRandomTossups({ alternateSubcategories, categories, difficulties, maxYear, minYear, number: 20, powermarkOnly, subcategories, standardOnly }); + document.getElementById('answer-input').value = ''; + document.getElementById('answer-input').blur(); + document.getElementById('answer-input').placeholder = 'Enter answer'; + document.getElementById('answer-input-group').classList.add('d-none'); + document.getElementById('next').disabled = false; + document.getElementById('pause').disabled = false; + if (room.settings.rebuzz) { + document.getElementById('buzz').disabled = false; + document.getElementById('buzz').textContent = 'Buzz'; } - const randomQuestion = randomTossups.pop(); + updateStatDisplay(room.players[USER_ID]); - // Begin loading the next batch of questions (asynchronously) - if (randomTossups.length === 0) { - loadRandomTossups({ alternateSubcategories, categories, difficulties, maxYear, minYear, number: 20, powermarkOnly, subcategories, standardOnly }); + if (audio.soundEffects && userId === USER_ID) { + if (directive === 'accept' && score > 10) { + audio.power.play(); + } else if (directive === 'accept' && score === 10) { + audio.correct.play(); + } else if (directive === 'reject') { + audio.incorrect.play(); + } } - return randomQuestion; + // if (directive !== 'prompt' && userId === USER_ID && await account.getUsername()) { + // questionStats.recordTossup(tossup, score > 0, score, perQuestionCelerity, true); + // } } -async function next () { - // Stop reading the current question: - clearTimeout(timeoutID); - currentlyBuzzing = false; - if (settings.timer) { - timer.stopTimer(); - timer.tenthsRemaining = 0; - timer.updateDisplay(); +async function next ({ oldTossup, tossup: nextTossup, type }) { + if (type === 'start') { + document.getElementById('next').disabled = false; + document.getElementById('next').textContent = 'Skip'; + document.getElementById('settings').classList.add('d-none'); } - if (await account.getUsername() && document.getElementById('answer').innerHTML) { - const pointValue = previous.isCorrect ? (previous.inPower ? previous.powerValue : 10) : (previous.endOfQuestion ? 0 : previous.negValue); - questionStats.recordTossup(tossups[questionNumber - 1], previous.isCorrect, pointValue, previous.celerity, false); + if (type === 'next' || type === 'skip') { + createTossupCard(oldTossup); } document.getElementById('answer').textContent = ''; @@ -368,74 +132,35 @@ async function next () { document.getElementById('toggle-correct').textContent = 'I was wrong'; document.getElementById('toggle-correct').classList.add('d-none'); - const hasNextQuestion = await advanceQuestion(); - - if (!hasNextQuestion) { - return; - } - document.getElementById('buzz').textContent = 'Buzz'; document.getElementById('buzz').disabled = false; document.getElementById('next').textContent = 'Skip'; - document.getElementById('packet-number-info').textContent = query.packetNumbers[0]; - document.getElementById('packet-length-info').textContent = settings.selectBySetName ? tossups.length : '-'; + document.getElementById('packet-number-info').textContent = nextTossup.packet.number; + console.log(room); + document.getElementById('packet-length-info').textContent = room.query.selectBySetName ? room.tossup.length : '-'; document.getElementById('pause').textContent = 'Pause'; document.getElementById('pause').disabled = false; - document.getElementById('question').textContent = ''; - document.getElementById('set-name-info').textContent = query.setName; - - paused = false; - readQuestion(Date.now()); -} + document.getElementById('set-name-info').textContent = nextTossup.set.name; -/** - * Toggles pausing or resuming the tossup. - */ -function pause () { - if (paused) { - document.getElementById('buzz').removeAttribute('disabled'); - document.getElementById('pause').textContent = 'Pause'; - readQuestion(Date.now()); - } else { - document.getElementById('buzz').setAttribute('disabled', 'disabled'); - document.getElementById('pause').textContent = 'Resume'; - clearTimeout(timeoutID); + if (type === 'next' && await account.getUsername() && document.getElementById('answer').innerHTML) { + const pointValue = room.previous.isCorrect ? (room.previous.inPower ? room.previous.powerValue : 10) : (room.previous.endOfQuestion ? 0 : room.previous.negValue); + questionStats.recordTossup(room.previous.tossup, room.previous.isCorrect, pointValue, room.previous.celerity, false); } - paused = !paused; } -/** - * Recursively reads the question based on the reading speed. - */ -function readQuestion (expectedReadTime) { - if (!currentlyBuzzing && tossupTextSplit.length > 0) { - const word = tossupTextSplit.shift(); - if (word !== '(*)') { - document.getElementById('question').textContent += word + ' '; - } - - // calculate time needed before reading next word - let time = Math.log(word.length) + 1; - if ((word.endsWith('.') && word.charCodeAt(word.length - 2) > 96 && word.charCodeAt(word.length - 2) < 123) || - word.slice(-2) === '.\u201d' || word.slice(-2) === '!\u201d' || word.slice(-2) === '?\u201d') { time += 2; } else if (word.endsWith(',') || word.slice(-2) === ',\u201d') { time += 0.75; } else if (word === '(*)') { time = 0; } - - time = time * 0.9 * (125 - settings.readingSpeed); - const delay = time - Date.now() + expectedReadTime; +function noQuestionsFound () { + window.alert('No questions found'); +} - timeoutID = window.setTimeout(() => { - readQuestion(time + expectedReadTime); - }, delay); - } else { - document.getElementById('pause').disabled = true; - if (settings.timer) { - timer.startTimer(DEAD_TIME_LIMIT, revealQuestion); - } - } +function pause ({ paused }) { + document.getElementById('buzz').disabled = paused; + document.getElementById('pause').textContent = paused ? 'Resume' : 'Pause'; } -function revealQuestion () { - document.getElementById('question').innerHTML = insertTokensIntoHTML(tossups[questionNumber - 1].question, tossups[questionNumber - 1].question_sanitized, [[buzzpointIndex]], [' (#) ']); - document.getElementById('answer').innerHTML = 'ANSWER: ' + tossups[questionNumber - 1].answer; +function revealAnswer ({ answer, question }) { + document.getElementById('question').innerHTML = question; + document.getElementById('answer').innerHTML = 'ANSWER: ' + answer; + document.getElementById('pause').disabled = true; document.getElementById('buzz').disabled = true; document.getElementById('buzz').textContent = 'Buzz'; @@ -444,156 +169,122 @@ function revealQuestion () { document.getElementById('start').disabled = false; document.getElementById('toggle-correct').classList.remove('d-none'); - document.getElementById('toggle-correct').textContent = previous.isCorrect ? 'I was wrong' : 'I was right'; + document.getElementById('toggle-correct').textContent = room.previous.isCorrect ? 'I was wrong' : 'I was right'; } -function toggleCorrect () { - const multiplier = previous.isCorrect ? -1 : 1; - - if (previous.inPower) { - stats.powers += multiplier * 1; - stats.points += multiplier * previous.powerValue; - } else { - stats.tens += multiplier * 1; - stats.points += multiplier * 10; - } - - if (previous.endOfQuestion) { - stats.dead += multiplier * -1; - } else { - stats.negs += multiplier * -1; - stats.points += multiplier * -previous.negValue; - } +function setCategories ({ alternateSubcategories, categories, subcategories }) { + categoryManager.import(categories, subcategories, alternateSubcategories); + categoryManager.loadCategoryModal(); +} - stats.totalCorrectCelerity += multiplier * previous.celerity; +function setDifficulties ({ value }) {} - previous.isCorrect = !previous.isCorrect; - document.getElementById('toggle-correct').textContent = previous.isCorrect ? 'I was wrong' : 'I was right'; +function setPacketNumbers ({ value }) {} - updateStatDisplay(); - window.sessionStorage.setItem('tossup-stats', JSON.stringify(stats)); +function setReadingSpeed ({ value }) { + document.getElementById('reading-speed').value = value; + document.getElementById('reading-speed-display').textContent = value; } -function updateScore (isCorrect) { - const endOfQuestion = (tossupTextSplit.length === 0); - const inPower = tossupTextSplit.includes('(*)') && tossupText.includes('(*)'); - const powerValue = isPace(query.setName) ? 20 : 15; - const negValue = isPace(query.setName) ? 0 : -5; - const points = isCorrect ? (inPower ? powerValue : 10) : (endOfQuestion ? 0 : negValue); +function setSetName ({ value }) {} + +function setYearRange ({ minYear, maxYear }) {} - const characterCount = tossupTextSplit.join(' ').length; - const celerity = characterCount / tossupText.length; +function toggleCorrect ({ correct, userId }) { + updateStatDisplay(room.players[USER_ID]); + document.getElementById('toggle-correct').textContent = correct ? 'I was wrong' : 'I was right'; +} - let result; +function togglePowermarkOnly ({ powermarkOnly }) {} - if (isCorrect) { - result = inPower ? 'powers' : 'tens'; - stats.totalCorrectCelerity += celerity; - } else { - result = endOfQuestion ? 'dead' : 'negs'; - } +function toggleRebuzz ({ rebuzz }) {} - stats[result] += 1; - stats.points += points; +function toggleSelectBySetName ({ selectBySetName, setName }) { + document.getElementById('difficulty-settings').classList.toggle('d-none', selectBySetName); + document.getElementById('set-settings').classList.toggle('d-none', !selectBySetName); + document.getElementById('toggle-powermark-only').disabled = selectBySetName; + document.getElementById('toggle-standard-only').disabled = selectBySetName; +} - previous.celerity = celerity; - previous.endOfQuestion = endOfQuestion; - previous.inPower = inPower; - previous.negValue = negValue; - previous.powerValue = powerValue; - previous.isCorrect = isCorrect; +function toggleStandardOnly ({ standardOnly }) {} - updateStatDisplay(); - window.sessionStorage.setItem('tossup-stats', JSON.stringify(stats)); +function toggleTimer ({ timer }) { + document.getElementById('timer').classList.toggle('d-none', !timer); +} - return points; +function updateQuestion ({ word }) { + if (word === '(*)') { return; } + document.getElementById('question').innerHTML += word + ' '; } /** * Updates the displayed stat line. */ -function updateStatDisplay () { - const { powers, tens, negs, dead, points, totalCorrectCelerity } = stats; - const numTossups = powers + tens + negs + dead; - const numCorrectTossups = powers + tens; - let celerity = numCorrectTossups === 0 ? 0 : parseFloat(totalCorrectCelerity) / numCorrectTossups; - celerity = Math.round(1000 * celerity) / 1000; - const includePlural = (numTossups === 1) ? '' : 's'; - document.getElementById('statline').innerHTML = - `${powers}/${tens}/${negs} with ${numTossups} tossup${includePlural} seen (${points} pts, celerity: ${celerity})`; +function updateStatDisplay ({ powers, tens, negs, tuh, points, celerity }) { + const averageCelerity = celerity.correct.average.toFixed(3); + const plural = (tuh === 1) ? '' : 's'; + document.getElementById('statline').innerHTML = `${powers}/${tens}/${negs} with ${tuh} tossup${plural} seen (${points} pts, celerity: ${averageCelerity})`; // disable clear stats button if no stats - document.getElementById('clear-stats').disabled = (numTossups === 0); + document.getElementById('clear-stats').disabled = (tuh === 0); +} + +function updateTimerDisplay (time) { + const seconds = Math.floor(time / 10); + const tenths = time % 10; + document.querySelector('.timer .face').innerText = seconds; + document.querySelector('.timer .fraction').innerText = '.' + tenths; } document.getElementById('answer-form').addEventListener('submit', function (event) { event.preventDefault(); event.stopPropagation(); - - if (settings.timer) { - timer.stopTimer(); - timer.tenthsRemaining = 0; - timer.updateDisplay(); - } - const answer = document.getElementById('answer-input').value; - - document.getElementById('answer-input').value = ''; - document.getElementById('answer-input').blur(); - document.getElementById('answer-input').placeholder = 'Enter answer'; - document.getElementById('answer-input-group').classList.add('d-none'); - - giveAnswer(answer); + socket.sendToServer({ type: 'give-answer', givenAnswer: answer }); }); document.getElementById('buzz').addEventListener('click', function () { this.blur(); - - // reveal answer on second click - // when NOT using type to answer - if (currentlyBuzzing) { - currentlyBuzzing = false; - updateScore(true); - revealQuestion(); - return; - } - - buzz(); - - if (settings.typeToAnswer) { - document.getElementById('answer-input-group').classList.remove('d-none'); - document.getElementById('answer-input').focus(); - this.disabled = true; - } + if (audio.soundEffects) audio.buzz.play(); + socket.sendToServer({ type: 'buzz' }); }); document.getElementById('clear-stats').addEventListener('click', function () { this.blur(); - clearStats(); + socket.sendToServer({ type: 'clear-stats' }); }); document.getElementById('next').addEventListener('click', function () { this.blur(); - createTossupCard(tossups[questionNumber - 1]); - next(); + if (this.innerHTML === 'Skip') { + socket.sendToServer({ type: 'skip' }); + } else { + socket.sendToServer({ type: 'next' }); + } }); document.getElementById('packet-number').addEventListener('change', function () { - // if field is blank, store blank result in `query` - query.packetNumbers = rangeToArray(this.value.trim(), 0); - window.localStorage.setItem('singleplayer-tossup-query', JSON.stringify(query)); - query.packetNumbers = rangeToArray(this.value.trim(), maxPacketNumber); + const range = rangeToArray(this.value.trim(), maxPacketNumber); + + if (range.some((num) => num < 1 || num > maxPacketNumber)) { + document.getElementById('packet-number').classList.add('is-invalid'); + return; + } + + document.getElementById('packet-number').classList.remove('is-invalid'); + socket.sendToServer({ type: 'set-packet-numbers', value: range }); }); document.getElementById('pause').addEventListener('click', function () { this.blur(); - pause(); + const seconds = parseFloat(document.querySelector('.timer .face').innerText); + const tenths = parseFloat(document.querySelector('.timer .fraction').innerText); + const pausedTime = (seconds + tenths) * 10; + socket.sendToServer({ type: 'pause', pausedTime }); }); -document.getElementById('reading-speed').addEventListener('input', function () { - settings.readingSpeed = this.value; - document.getElementById('reading-speed-display').textContent = this.value; - window.localStorage.setItem('singleplayer-tossup-settings', JSON.stringify(settings)); +document.getElementById('reading-speed').addEventListener('change', function () { + socket.sendToServer({ type: 'set-reading-speed', value: this.value }); }); document.getElementById('report-question-submit').addEventListener('click', function () { @@ -605,14 +296,9 @@ document.getElementById('report-question-submit').addEventListener('click', func }); document.getElementById('set-name').addEventListener('change', async function () { - query.setName = this.value.trim(); - // make border red if set name is not in set list - if (api.getSetList().includes(this.value) || this.value.length === 0) { - this.classList.remove('is-invalid'); - } else { - this.classList.add('is-invalid'); - } + const valid = api.getSetList().includes(this.value) || this.value.length === 0; + this.classList.toggle('is-invalid', !valid); maxPacketNumber = await api.getNumPackets(this.value); @@ -622,98 +308,39 @@ document.getElementById('set-name').addEventListener('change', async function () document.getElementById('packet-number').placeholder = `Packet Numbers (1-${maxPacketNumber})`; } - window.localStorage.setItem('singleplayer-tossup-query', JSON.stringify(query)); + socket.sendToServer({ + type: 'set-set-name', + value: this.value.trim(), + packetNumbers: rangeToArray(document.getElementById('packet-number').value) + }); }); -document.getElementById('start').addEventListener('click', async function () { - this.blur(); - if (query.setName.length === 0 && settings.selectBySetName) { - window.alert('Please enter a set name.'); - return false; - } - - if (query.packetNumbers.length === 0 && settings.selectBySetName) { - query.packetNumbers = rangeToArray(document.getElementById('packet-number').value.trim(), maxPacketNumber); - } - - document.getElementById('next').disabled = false; - document.getElementById('next').textContent = 'Skip'; - document.getElementById('settings').classList.add('d-none'); - - if (settings.selectBySetName) { - queryLock(); - questionNumber = 0; - try { - tossups = await api.getPacketTossups(query.setName, query.packetNumbers[0]); - } finally { - queryUnlock(); - } - } - - next(); +document.getElementById('start').addEventListener('click', function () { + socket.sendToServer({ type: 'start' }); }); document.getElementById('toggle-correct').addEventListener('click', function () { this.blur(); - toggleCorrect(); + socket.sendToServer({ type: 'toggle-correct', correct: this.textContent === 'I was right' }); }); document.getElementById('toggle-powermark-only').addEventListener('click', function () { this.blur(); - query.powermarkOnly = this.checked; - loadRandomTossups(query); - window.localStorage.setItem('singleplayer-tossup-query', JSON.stringify(query)); -}); - -document.getElementById('toggle-select-by-set-name').addEventListener('click', function () { - this.blur(); - settings.selectBySetName = this.checked; - document.getElementById('toggle-powermark-only').disabled = this.checked; - document.getElementById('toggle-standard-only').disabled = this.checked; - - if (this.checked) { - document.getElementById('difficulty-settings').classList.add('d-none'); - document.getElementById('set-settings').classList.remove('d-none'); - } else { - document.getElementById('difficulty-settings').classList.remove('d-none'); - document.getElementById('set-settings').classList.add('d-none'); - } - - window.localStorage.setItem('singleplayer-tossup-settings', JSON.stringify(settings)); + socket.sendToServer({ type: 'toggle-powermark-only', powermarkOnly: this.checked }); }); -document.getElementById('toggle-show-history').addEventListener('click', function () { - this.blur(); - settings.showHistory = this.checked; - - if (this.checked) { - document.getElementById('room-history').classList.remove('d-none'); - } else { - document.getElementById('room-history').classList.add('d-none'); - } - - window.localStorage.setItem('singleplayer-tossup-settings', JSON.stringify(settings)); -}); - -document.getElementById('toggle-standard-only').addEventListener('click', function () { - this.blur(); - query.standardOnly = this.checked; - loadRandomTossups(query); - window.localStorage.setItem('singleplayer-tossup-query', JSON.stringify(query)); -}); - -document.getElementById('toggle-timer').addEventListener('click', function () { +document.getElementById('toggle-rebuzz').addEventListener('click', function () { this.blur(); - settings.timer = this.checked; - document.getElementById('timer').classList.toggle('d-none'); - window.localStorage.setItem('singleplayer-tossup-settings', JSON.stringify(settings)); + socket.sendToServer({ type: 'toggle-rebuzz', rebuzz: this.checked }); }); -document.getElementById('type-to-answer').addEventListener('click', function () { +document.getElementById('toggle-select-by-set-name').addEventListener('click', function () { this.blur(); - settings.typeToAnswer = this.checked; - document.getElementById('toggle-rebuzz').disabled = !this.checked; - window.localStorage.setItem('singleplayer-tossup-settings', JSON.stringify(settings)); + socket.sendToServer({ + type: 'toggle-select-by-set-name', + setName: document.getElementById('set-name').value, + selectBySetName: this.checked + }); }); document.getElementById('toggle-settings').addEventListener('click', function () { @@ -726,24 +353,31 @@ document.getElementById('toggle-settings').addEventListener('click', function () document.getElementById('settings').classList.toggle('d-lg-none'); }); -document.getElementById('toggle-rebuzz').addEventListener('click', function () { +document.getElementById('toggle-show-history').addEventListener('click', function () { + this.blur(); + document.getElementById('room-history').classList.toggle('d-none', !this.checked); +}); + +document.getElementById('toggle-standard-only').addEventListener('click', function () { + this.blur(); + socket.sendToServer({ type: 'toggle-standard-only', standardOnly: this.checked }); +}); + +document.getElementById('toggle-timer').addEventListener('click', function () { this.blur(); - settings.rebuzz = this.checked; - window.localStorage.setItem('singleplayer-tossup-settings', JSON.stringify(settings)); + socket.sendToServer({ type: 'toggle-timer', timer: this.checked }); }); document.getElementById('year-range-a').onchange = function () { - query.minYear = $('#slider').slider('values', 0); - query.maxYear = $('#slider').slider('values', 1); - loadRandomTossups(query); - window.localStorage.setItem('singleplayer-tossup-query', JSON.stringify(query)); + const minYear = $('#slider').slider('values', 0); + const maxYear = $('#slider').slider('values', 1); + socket.sendToServer({ type: 'set-year-range', minYear, maxYear }); }; document.getElementById('year-range-b').onchange = function () { - query.minYear = $('#slider').slider('values', 0); - query.maxYear = $('#slider').slider('values', 1); - loadRandomTossups(query); - window.localStorage.setItem('singleplayer-tossup-query', JSON.stringify(query)); + const minYear = $('#slider').slider('values', 0); + const maxYear = $('#slider').slider('values', 1); + socket.sendToServer({ type: 'set-year-range', minYear, maxYear }); }; document.addEventListener('keydown', (event) => { @@ -765,7 +399,7 @@ document.addEventListener('keydown', (event) => { document.getElementsByClassName('star-tossup')[0].click(); break; case 'y': - navigator.clipboard.writeText(tossups[0]?._id ?? ''); + navigator.clipboard.writeText(room.tossup._id ?? ''); break; case 'n': document.getElementById('next').click(); @@ -779,31 +413,22 @@ document.addEventListener('keydown', (event) => { } }); -$(document).ready(function () { - $('#slider').slider('values', 0, query.minYear); - $('#slider').slider('values', 1, query.maxYear); -}); -document.getElementById('year-range-a').textContent = query.minYear; -document.getElementById('year-range-b').textContent = query.maxYear; +// $(document).ready(function () { +// $('#slider').slider('values', 0, query.minYear); +// $('#slider').slider('values', 1, query.maxYear); +// }); +// document.getElementById('year-range-a').textContent = query.minYear; +// document.getElementById('year-range-b').textContent = query.maxYear; ReactDOM.createRoot(document.getElementById('category-modal-root')).render( { - ({ categories: query.categories, subcategories: query.subcategories, alternateSubcategories: query.alternateSubcategories } = categoryManager.export()); - loadRandomTossups(query); - window.localStorage.setItem('singleplayer-tossup-query', JSON.stringify(query)); - }} + onClose={() => socket.sendToServer({ type: 'set-categories', ...categoryManager.export() })} /> ); ReactDOM.createRoot(document.getElementById('difficulty-dropdown-root')).render( { - query.difficulties = getDropdownValues('difficulties'); - loadRandomTossups(query); - window.localStorage.setItem('singleplayer-tossup-query', JSON.stringify(query)); - }} + onChange={() => socket.sendToServer({ type: 'set-difficulties', value: getDropdownValues('difficulties') })} /> ); diff --git a/client/singleplayer/tossups/index.min.js b/client/singleplayer/tossups/index.min.js index 09471d56..0ebca6ef 100644 --- a/client/singleplayer/tossups/index.min.js +++ b/client/singleplayer/tossups/index.min.js @@ -1,36 +1,12 @@ -import account from"../../scripts/accounts.js";import questionStats from"../../scripts/auth/question-stats.js";import api from"../../scripts/api/index.js";import audio from"../../audio/index.js";import Timer from"../../scripts/Timer.js";import{arrayToRange,createTossupCard,rangeToArray}from"../../scripts/utilities/index.js";import CategoryManager from"../../scripts/utilities/category-manager.js";import{getDropdownValues}from"../../scripts/utilities/dropdown-checklist.js";import{insertTokensIntoHTML}from"../../insert-tokens-into-html.js";import CategoryModal from"../../scripts/components/CategoryModal.min.js";import DifficultyDropdown from"../../scripts/components/DifficultyDropdown.min.js";// Functions and variables specific to the tossups page. -const ANSWER_TIME_LIMIT=10,DEAD_TIME_LIMIT=5;// Status variables -let buzzpointIndex=-1,currentlyBuzzing=!1,maxPacketNumber=24,paused=!1,questionNumber=0;// WARNING: 1-indexed -const timer=new Timer;/** - * An array of random questions. - * We get 20 random questions at a time so we don't have to make an HTTP request between every question. - */let randomTossups=[],timeoutID=-1,tossups=[{}],tossupText="",tossupTextSplit=[];const previous={isCorrect:!0,inPower:!1,negValue:-5,powerValue:15,endOfQuestion:!1,celerity:0},stats=window.sessionStorage.getItem("tossup-stats")?JSON.parse(window.sessionStorage.getItem("tossup-stats")):{powers:0,tens:0,negs:0,dead:0,points:0,totalCorrectCelerity:0},defaults={alternateSubcategories:[],categories:[],difficulties:[],minYear:2010,maxYear:2024,packetNumbers:[],powermarkOnly:!1,setName:"",standardOnly:!1,subcategories:[],version:"01-06-2024"};let query;window.localStorage.getItem("singleplayer-tossup-query")?(query=JSON.parse(window.localStorage.getItem("singleplayer-tossup-query")),query.version!==defaults.version&&(query=defaults,window.localStorage.setItem("singleplayer-tossup-query",JSON.stringify(query)))):query=defaults;const categoryManager=new CategoryManager(query.categories,query.subcategories,query.alternateSubcategories),settings=window.localStorage.getItem("singleplayer-tossup-settings")?JSON.parse(window.localStorage.getItem("singleplayer-tossup-settings")):{readingSpeed:50,rebuzz:!1,selectBySetName:!1,showHistory:!0,timer:!0,typeToAnswer:!0};// Load query and settings first so user doesn't see the default settings -settings.readingSpeed&&(document.getElementById("reading-speed-display").textContent=settings.readingSpeed,document.getElementById("reading-speed").value=settings.readingSpeed),settings.rebuzz&&(document.getElementById("toggle-rebuzz").checked=!0),settings.selectBySetName&&(document.getElementById("difficulty-settings").classList.add("d-none"),document.getElementById("set-settings").classList.remove("d-none"),document.getElementById("toggle-select-by-set-name").checked=!0,document.getElementById("toggle-powermark-only").disabled=!0,document.getElementById("toggle-standard-only").disabled=!0),settings.showHistory||(document.getElementById("toggle-show-history").checked=!1,document.getElementById("room-history").classList.add("d-none")),settings.timer||(document.getElementById("toggle-timer").checked=!1,document.getElementById("timer").classList.add("d-none")),settings.typeToAnswer||(document.getElementById("type-to-answer").checked=!1,document.getElementById("toggle-rebuzz").disabled=!0),query.packetNumbers&&(document.getElementById("packet-number").value=arrayToRange(query.packetNumbers)),query.powermarkOnly&&(document.getElementById("toggle-powermark-only").checked=!0),query.setName&&(document.getElementById("set-name").value=query.setName,api.getNumPackets(query.setName).then(a=>{maxPacketNumber=a,0===maxPacketNumber?document.getElementById("set-name").classList.add("is-invalid"):document.getElementById("packet-number").placeholder=`Packet Numbers (1-${maxPacketNumber})`})),updateStatDisplay();function queryLock(){document.getElementById("question").textContent="Fetching questions...",document.getElementById("start").disabled=!0,document.getElementById("next").disabled=!0,document.getElementById("pause").disabled=!0,document.getElementById("buzz").disabled=!0}function queryUnlock(){document.getElementById("question").textContent="",document.getElementById("start").disabled=!1,document.getElementById("next").disabled=!1,document.getElementById("pause").disabled=!1,document.getElementById("buzz").disabled=!1}/** - * @returns {Promise} Whether or not there is a next question - */async function advanceQuestion(){if(settings.selectBySetName){// Get the next question if the current one is in the wrong category and subcategory -do// Go to the next packet if you reach the end of this packet -if(questionNumber++,questionNumber>tossups.length){if(query.packetNumbers.shift(),0===query.packetNumbers.length)return window.alert("No more questions left"),document.getElementById("buzz").disabled=!0,document.getElementById("pause").disabled=!0,document.getElementById("next").disabled=!0,!1;// alert the user if there are no more packets -queryLock();try{tossups=await api.getPacketTossups(query.setName,query.packetNumbers[0])}finally{queryUnlock()}questionNumber=1}while(!categoryManager.isValidCategory(tossups[questionNumber-1]));0""!==a),document.getElementById("question-number-info").textContent=questionNumber)}else{queryLock();try{tossups=await getRandomTossup(query,categoryManager),tossups=[tossups]}finally{queryUnlock()}if(!tossups[0])return window.alert("No questions found"),!1;query.setName=tossups[0].set.name,query.packetNumbers=[tossups[0].packet.number],tossupText=tossups[0].question_sanitized,tossupTextSplit=tossupText.split(" ").filter(a=>""!==a),document.getElementById("question-number-info").textContent=tossups[0].number,questionNumber=1}return!0}/** - * Called when the users buzzes. - * The first "buzz" pauses the question, and the second "buzz" reveals the rest of the question - * and updates the score. - */function buzz(){// Stop the question reading -// Include buzzpoint -clearTimeout(timeoutID),currentlyBuzzing=!0,audio.soundEffects&&audio.buzz.play(),buzzpointIndex=document.getElementById("question").textContent.length,!tossupTextSplit.includes("(*)")&&tossupText.includes("(*)")&&(buzzpointIndex+=3),document.getElementById("question").textContent+="(#) ",document.getElementById("buzz").textContent="Reveal",document.getElementById("next").disabled=!0,document.getElementById("start").disabled=!0,document.getElementById("pause").disabled=!0,settings.timer&&(timer.stopTimer(),timer.startTimer(ANSWER_TIME_LIMIT,()=>document.getElementById("answer-submit").click()))}/** - * Clears user stats. - */function clearStats(){stats.powers=0,stats.tens=0,stats.negs=0,stats.dead=0,stats.points=0,stats.totalCorrectCelerity=0,updateStatDisplay(),window.sessionStorage.removeItem("tossup-stats")}async function giveAnswer(a){currentlyBuzzing=!1;const{directive:b,directedPrompt:c}=await api.checkAnswer(tossups[questionNumber-1].answer,a);switch(b){case"accept":{const a=updateScore(!0);audio.soundEffects&&(10b.charCodeAt(b.length-2)||".\u201D"===b.slice(-2)||"!\u201D"===b.slice(-2)||"?\u201D"===b.slice(-2)?c+=2:b.endsWith(",")||",\u201D"===b.slice(-2)?c+=.75:"(*)"===b&&(c=0),c=.9*c*(125-settings.readingSpeed);const d=c-Date.now()+a;timeoutID=window.setTimeout(()=>{readQuestion(c+a)},d)}else document.getElementById("pause").disabled=!0,settings.timer&&timer.startTimer(DEAD_TIME_LIMIT,revealQuestion)}function revealQuestion(){document.getElementById("question").innerHTML=insertTokensIntoHTML(tossups[questionNumber-1].question,tossups[questionNumber-1].question_sanitized,[[buzzpointIndex]],[" (#) "]),document.getElementById("answer").innerHTML="ANSWER: "+tossups[questionNumber-1].answer,document.getElementById("buzz").disabled=!0,document.getElementById("buzz").textContent="Buzz",document.getElementById("next").disabled=!1,document.getElementById("next").textContent="Next",document.getElementById("start").disabled=!1,document.getElementById("toggle-correct").classList.remove("d-none"),document.getElementById("toggle-correct").textContent=previous.isCorrect?"I was wrong":"I was right"}function toggleCorrect(){const a=previous.isCorrect?-1:1;previous.inPower?(stats.powers+=1*a,stats.points+=a*previous.powerValue):(stats.tens+=1*a,stats.points+=10*a),previous.endOfQuestion?stats.dead+=-1*a:(stats.negs+=-1*a,stats.points+=a*-previous.negValue),stats.totalCorrectCelerity+=a*previous.celerity,previous.isCorrect=!previous.isCorrect,document.getElementById("toggle-correct").textContent=previous.isCorrect?"I was wrong":"I was right",updateStatDisplay(),window.sessionStorage.setItem("tossup-stats",JSON.stringify(stats))}function updateScore(a){const b=0===tossupTextSplit.length,c=tossupTextSplit.includes("(*)")&&tossupText.includes("(*)"),d=isPace(query.setName)?20:15,e=isPace(query.setName)?0:-5,f=a?c?d:10:b?0:e,g=tossupTextSplit.join(" ").length,h=g/tossupText.length;let i;return a?(i=c?"powers":"tens",stats.totalCorrectCelerity+=h):i=b?"dead":"negs",stats[i]+=1,stats.points+=f,previous.celerity=h,previous.endOfQuestion=b,previous.inPower=c,previous.negValue=e,previous.powerValue=d,previous.isCorrect=a,updateStatDisplay(),window.sessionStorage.setItem("tossup-stats",JSON.stringify(stats)),f}/** +import account from"../../scripts/accounts.js";import api from"../../scripts/api/index.js";import questionStats from"../../scripts/auth/question-stats.js";import audio from"../../audio/index.js";// import Player from '../../../quizbowl/Player.js'; +import Player from"../../Player.js";import ClientTossupRoom from"../ClientTossupRoom.js";import CategoryManager from"../../scripts/utilities/category-manager.js";import{createTossupCard,rangeToArray}from"../../scripts/utilities/index.js";import{getDropdownValues}from"../../scripts/utilities/dropdown-checklist.js";import CategoryModal from"../../scripts/components/CategoryModal.min.js";import DifficultyDropdown from"../../scripts/components/DifficultyDropdown.min.js";let maxPacketNumber=24;const categoryManager=new CategoryManager,USER_ID="user",room=new ClientTossupRoom;room.players[USER_ID]=new Player(USER_ID);const socket={send:onmessage,sendToServer:a=>room.message(USER_ID,a)};room.sockets[USER_ID]=socket;function onmessage(a){const b=JSON.parse(a);switch(console.log(b),b.type){case"buzz":return buzz(b);case"clear-stats":return clearStats(b);case"end-of-set":return endOfSet(b);case"give-answer":return giveAnswer(b);case"next":return next(b);case"no-questions-found":return noQuestionsFound(b);case"pause":return pause(b);case"reveal-answer":return revealAnswer(b);case"set-categories":return setCategories(b);case"set-difficulties":return setDifficulties(b);case"set-reading-speed":return setReadingSpeed(b);case"set-packet-numbers":return setPacketNumbers(b);case"set-set-name":return setSetName(b);case"set-year-range":return setYearRange(b);case"skip":return next(b);case"start":return next(b);case"timer-update":return updateTimerDisplay(b.timeRemaining);case"toggle-correct":return toggleCorrect(b);case"toggle-powermark-only":return togglePowermarkOnly(b);case"toggle-rebuzz":return toggleRebuzz(b);case"toggle-select-by-set-name":return toggleSelectBySetName(b);case"toggle-standard-only":return toggleStandardOnly(b);case"toggle-timer":return toggleTimer(b);case"update-question":return updateQuestion(b)}}function buzz({timer:a,userId:b,username:c}){audio.soundEffects&&audio.buzz.play();const d=document.getElementById("type-to-answer").checked;d&&(document.getElementById("answer-input-group").classList.remove("d-none"),document.getElementById("answer-input").focus(),document.getElementById("buzz").disabled=!0)}function clearStats({userId:a}){a!==USER_ID||updateStatDisplay(room.players[USER_ID])}function endOfSet(){window.alert("No more questions left"),document.getElementById("buzz").disabled=!0,document.getElementById("pause").disabled=!0,document.getElementById("next").disabled=!0}async function giveAnswer({directive:a,directedPrompt:b,perQuestionCelerity:c,score:d,tossup:e,userId:f}){return"prompt"===a?(document.getElementById("answer-input-group").classList.remove("d-none"),document.getElementById("answer-input").focus(),void(document.getElementById("answer-input").placeholder=b?`Prompt: "${b}"`:"Prompt")):void(document.getElementById("answer-input").value="",document.getElementById("answer-input").blur(),document.getElementById("answer-input").placeholder="Enter answer",document.getElementById("answer-input-group").classList.add("d-none"),document.getElementById("next").disabled=!1,document.getElementById("pause").disabled=!1,room.settings.rebuzz&&(document.getElementById("buzz").disabled=!1,document.getElementById("buzz").textContent="Buzz"),updateStatDisplay(room.players[USER_ID]),audio.soundEffects&&f===USER_ID&&("accept"===a&&10{if(!["INPUT","TEXTAREA","SELECT"].includes(document.activeElement.tagName))switch(a.key){case" ":document.getElementById("buzz").click(),a.target===document.body&&a.preventDefault();break;case"e":document.getElementById("toggle-settings").click();break;case"k":document.getElementsByClassName("card-header-clickable")[0].click();break;case"t":document.getElementsByClassName("star-tossup")[0].click();break;case"y":navigator.clipboard.writeText(tossups[0]?._id??"");break;case"n":document.getElementById("next").click();break;case"p":document.getElementById("pause").click();break;case"s":document.getElementById("start").click()}}),$(document).ready(function(){$("#slider").slider("values",0,query.minYear),$("#slider").slider("values",1,query.maxYear)}),document.getElementById("year-range-a").textContent=query.minYear,document.getElementById("year-range-b").textContent=query.maxYear,ReactDOM.createRoot(document.getElementById("category-modal-root")).render(/*#__PURE__*/React.createElement(CategoryModal,{categoryManager:categoryManager,onClose:()=>{({categories:query.categories,subcategories:query.subcategories,alternateSubcategories:query.alternateSubcategories}=categoryManager.export()),loadRandomTossups(query),window.localStorage.setItem("singleplayer-tossup-query",JSON.stringify(query))}})),ReactDOM.createRoot(document.getElementById("difficulty-dropdown-root")).render(/*#__PURE__*/React.createElement(DifficultyDropdown,{startingDifficulties:query.difficulties,onChange:()=>{query.difficulties=getDropdownValues("difficulties"),loadRandomTossups(query),window.localStorage.setItem("singleplayer-tossup-query",JSON.stringify(query))}})); \ No newline at end of file + */function updateStatDisplay({powers:a,tens:b,negs:c,tuh:d,points:e,celerity:f}){const g=f.correct.average.toFixed(3),h=1===d?"":"s";// disable clear stats button if no stats +document.getElementById("statline").innerHTML=`${a}/${b}/${c} with ${d} tossup${h} seen (${e} pts, celerity: ${g})`,document.getElementById("clear-stats").disabled=0===d}function updateTimerDisplay(a){const b=Math.floor(a/10);document.querySelector(".timer .face").innerText=b,document.querySelector(".timer .fraction").innerText="."+a%10}// $(document).ready(function () { +// $('#slider').slider('values', 0, query.minYear); +// $('#slider').slider('values', 1, query.maxYear); +// }); +// document.getElementById('year-range-a').textContent = query.minYear; +// document.getElementById('year-range-b').textContent = query.maxYear; +document.getElementById("answer-form").addEventListener("submit",function(a){a.preventDefault(),a.stopPropagation();const b=document.getElementById("answer-input").value;socket.sendToServer({type:"give-answer",givenAnswer:b})}),document.getElementById("buzz").addEventListener("click",function(){this.blur(),audio.soundEffects&&audio.buzz.play(),socket.sendToServer({type:"buzz"})}),document.getElementById("clear-stats").addEventListener("click",function(){this.blur(),socket.sendToServer({type:"clear-stats"})}),document.getElementById("next").addEventListener("click",function(){this.blur(),"Skip"===this.innerHTML?socket.sendToServer({type:"skip"}):socket.sendToServer({type:"next"})}),document.getElementById("packet-number").addEventListener("change",function(){const a=rangeToArray(this.value.trim(),maxPacketNumber);return a.some(a=>1>a||a>maxPacketNumber)?void document.getElementById("packet-number").classList.add("is-invalid"):void(document.getElementById("packet-number").classList.remove("is-invalid"),socket.sendToServer({type:"set-packet-numbers",value:a}))}),document.getElementById("pause").addEventListener("click",function(){this.blur();const a=parseFloat(document.querySelector(".timer .face").innerText),b=parseFloat(document.querySelector(".timer .fraction").innerText);socket.sendToServer({type:"pause",pausedTime:10*(a+b)})}),document.getElementById("reading-speed").addEventListener("change",function(){socket.sendToServer({type:"set-reading-speed",value:this.value})}),document.getElementById("report-question-submit").addEventListener("click",function(){api.reportQuestion(document.getElementById("report-question-id").value,document.getElementById("report-question-reason").value,document.getElementById("report-question-description").value)}),document.getElementById("set-name").addEventListener("change",async function(){// make border red if set name is not in set list +const a=api.getSetList().includes(this.value)||0===this.value.length;this.classList.toggle("is-invalid",!a),maxPacketNumber=await api.getNumPackets(this.value),document.getElementById("packet-number").placeholder=""===this.value||0===maxPacketNumber?"Packet Numbers":`Packet Numbers (1-${maxPacketNumber})`,socket.sendToServer({type:"set-set-name",value:this.value.trim(),packetNumbers:rangeToArray(document.getElementById("packet-number").value)})}),document.getElementById("start").addEventListener("click",function(){socket.sendToServer({type:"start"})}),document.getElementById("toggle-correct").addEventListener("click",function(){this.blur(),socket.sendToServer({type:"toggle-correct",correct:"I was right"===this.textContent})}),document.getElementById("toggle-powermark-only").addEventListener("click",function(){this.blur(),socket.sendToServer({type:"toggle-powermark-only",powermarkOnly:this.checked})}),document.getElementById("toggle-rebuzz").addEventListener("click",function(){this.blur(),socket.sendToServer({type:"toggle-rebuzz",rebuzz:this.checked})}),document.getElementById("toggle-select-by-set-name").addEventListener("click",function(){this.blur(),socket.sendToServer({type:"toggle-select-by-set-name",setName:document.getElementById("set-name").value,selectBySetName:this.checked})}),document.getElementById("toggle-settings").addEventListener("click",function(){this.blur(),document.getElementById("buttons").classList.toggle("col-lg-9"),document.getElementById("buttons").classList.toggle("col-lg-12"),document.getElementById("content").classList.toggle("col-lg-9"),document.getElementById("content").classList.toggle("col-lg-12"),document.getElementById("settings").classList.toggle("d-none"),document.getElementById("settings").classList.toggle("d-lg-none")}),document.getElementById("toggle-show-history").addEventListener("click",function(){this.blur(),document.getElementById("room-history").classList.toggle("d-none",!this.checked)}),document.getElementById("toggle-standard-only").addEventListener("click",function(){this.blur(),socket.sendToServer({type:"toggle-standard-only",standardOnly:this.checked})}),document.getElementById("toggle-timer").addEventListener("click",function(){this.blur(),socket.sendToServer({type:"toggle-timer",timer:this.checked})}),document.getElementById("year-range-a").onchange=function(){const a=$("#slider").slider("values",0),b=$("#slider").slider("values",1);socket.sendToServer({type:"set-year-range",minYear:a,maxYear:b})},document.getElementById("year-range-b").onchange=function(){const a=$("#slider").slider("values",0),b=$("#slider").slider("values",1);socket.sendToServer({type:"set-year-range",minYear:a,maxYear:b})},document.addEventListener("keydown",a=>{if(!["INPUT","TEXTAREA","SELECT"].includes(document.activeElement.tagName))switch(a.key){case" ":document.getElementById("buzz").click(),a.target===document.body&&a.preventDefault();break;case"e":document.getElementById("toggle-settings").click();break;case"k":document.getElementsByClassName("card-header-clickable")[0].click();break;case"t":document.getElementsByClassName("star-tossup")[0].click();break;case"y":navigator.clipboard.writeText(room.tossup._id??"");break;case"n":document.getElementById("next").click();break;case"p":document.getElementById("pause").click();break;case"s":document.getElementById("start").click()}}),ReactDOM.createRoot(document.getElementById("category-modal-root")).render(/*#__PURE__*/React.createElement(CategoryModal,{categoryManager:categoryManager,onClose:()=>socket.sendToServer({type:"set-categories",...categoryManager.export()})})),ReactDOM.createRoot(document.getElementById("difficulty-dropdown-root")).render(/*#__PURE__*/React.createElement(DifficultyDropdown,{onChange:()=>socket.sendToServer({type:"set-difficulties",value:getDropdownValues("difficulties")})})); \ No newline at end of file diff --git a/client/singleplayer/tossups/old-index.jsx b/client/singleplayer/tossups/old-index.jsx new file mode 100644 index 00000000..22f442b4 --- /dev/null +++ b/client/singleplayer/tossups/old-index.jsx @@ -0,0 +1,809 @@ +import account from '../../scripts/accounts.js'; +import questionStats from '../../scripts/auth/question-stats.js'; +import api from '../../scripts/api/index.js'; +import audio from '../../audio/index.js'; +import Timer from '../../scripts/Timer.js'; +import { arrayToRange, createTossupCard, rangeToArray } from '../../scripts/utilities/index.js'; +import CategoryManager from '../../scripts/utilities/category-manager.js'; +import { getDropdownValues } from '../../scripts/utilities/dropdown-checklist.js'; +import { insertTokensIntoHTML } from '../../insert-tokens-into-html.js'; +import CategoryModal from '../../scripts/components/CategoryModal.min.js'; +import DifficultyDropdown from '../../scripts/components/DifficultyDropdown.min.js'; + +// Functions and variables specific to the tossups page. + +const ANSWER_TIME_LIMIT = 10; +const DEAD_TIME_LIMIT = 5; + +// Status variables +let buzzpointIndex = -1; +let currentlyBuzzing = false; +let maxPacketNumber = 24; +let paused = false; +let questionNumber = 0; // WARNING: 1-indexed +const timer = new Timer(); + +/** + * An array of random questions. + * We get 20 random questions at a time so we don't have to make an HTTP request between every question. + */ +let randomTossups = []; +let timeoutID = -1; + +let tossups = [{}]; +let tossupText = ''; +let tossupTextSplit = []; + +const previous = { + isCorrect: true, + inPower: false, + negValue: -5, + powerValue: 15, + endOfQuestion: false, + celerity: 0 +}; + +const stats = window.sessionStorage.getItem('tossup-stats') + ? JSON.parse(window.sessionStorage.getItem('tossup-stats')) + : { + powers: 0, + tens: 0, + negs: 0, + dead: 0, + points: 0, + totalCorrectCelerity: 0 + }; + +const defaults = { + alternateSubcategories: [], + categories: [], + difficulties: [], + minYear: 2010, + maxYear: 2024, + packetNumbers: [], + powermarkOnly: false, + setName: '', + standardOnly: false, + subcategories: [], + version: '01-06-2024' +}; + +let query; +if (!window.localStorage.getItem('singleplayer-tossup-query')) { + query = defaults; +} else { + query = JSON.parse(window.localStorage.getItem('singleplayer-tossup-query')); + if (query.version !== defaults.version) { + query = defaults; + window.localStorage.setItem('singleplayer-tossup-query', JSON.stringify(query)); + } +} + +const categoryManager = new CategoryManager(query.categories, query.subcategories, query.alternateSubcategories); + +const settings = window.localStorage.getItem('singleplayer-tossup-settings') + ? JSON.parse(window.localStorage.getItem('singleplayer-tossup-settings')) + : { + readingSpeed: 50, + rebuzz: false, + selectBySetName: false, + showHistory: true, + timer: true, + typeToAnswer: true + }; + +// Load query and settings first so user doesn't see the default settings +if (settings.readingSpeed) { + document.getElementById('reading-speed-display').textContent = settings.readingSpeed; + document.getElementById('reading-speed').value = settings.readingSpeed; +} + +if (settings.rebuzz) { + document.getElementById('toggle-rebuzz').checked = true; +} + +if (settings.selectBySetName) { + document.getElementById('difficulty-settings').classList.add('d-none'); + document.getElementById('set-settings').classList.remove('d-none'); + document.getElementById('toggle-select-by-set-name').checked = true; + document.getElementById('toggle-powermark-only').disabled = true; + document.getElementById('toggle-standard-only').disabled = true; +} + +if (!settings.showHistory) { + document.getElementById('toggle-show-history').checked = false; + document.getElementById('room-history').classList.add('d-none'); +} + +if (!settings.timer) { + document.getElementById('toggle-timer').checked = false; + document.getElementById('timer').classList.add('d-none'); +} + +if (!settings.typeToAnswer) { + document.getElementById('type-to-answer').checked = false; + document.getElementById('toggle-rebuzz').disabled = true; +} + +if (query.packetNumbers) { + document.getElementById('packet-number').value = arrayToRange(query.packetNumbers); +} + +if (query.powermarkOnly) { + document.getElementById('toggle-powermark-only').checked = true; +} + +if (query.setName) { + document.getElementById('set-name').value = query.setName; + api.getNumPackets(query.setName).then(numPackets => { + maxPacketNumber = numPackets; + if (maxPacketNumber === 0) { + document.getElementById('set-name').classList.add('is-invalid'); + } else { + document.getElementById('packet-number').placeholder = `Packet Numbers (1-${maxPacketNumber})`; + } + }); +} + +updateStatDisplay(); + +function queryLock () { + document.getElementById('question').textContent = 'Fetching questions...'; + document.getElementById('start').disabled = true; + document.getElementById('next').disabled = true; + document.getElementById('pause').disabled = true; + document.getElementById('buzz').disabled = true; +} + +function queryUnlock () { + document.getElementById('question').textContent = ''; + document.getElementById('start').disabled = false; + document.getElementById('next').disabled = false; + document.getElementById('pause').disabled = false; + document.getElementById('buzz').disabled = false; +} + +/** + * @returns {Promise} Whether or not there is a next question + */ +async function advanceQuestion () { + if (settings.selectBySetName) { + // Get the next question if the current one is in the wrong category and subcategory + do { + questionNumber++; + + // Go to the next packet if you reach the end of this packet + if (questionNumber > tossups.length) { + query.packetNumbers.shift(); + if (query.packetNumbers.length === 0) { + window.alert('No more questions left'); + document.getElementById('buzz').disabled = true; + document.getElementById('pause').disabled = true; + document.getElementById('next').disabled = true; + return false; // alert the user if there are no more packets + } + + queryLock(); + try { + tossups = await api.getPacketTossups(query.setName, query.packetNumbers[0]); + } finally { + queryUnlock(); + } + + questionNumber = 1; + } + } while (!categoryManager.isValidCategory(tossups[questionNumber - 1])); + + if (Object.keys(tossups[0]).length > 0) { + tossupText = tossups[questionNumber - 1].question_sanitized; + tossupTextSplit = tossupText.split(' ').filter(word => word !== ''); + document.getElementById('question-number-info').textContent = questionNumber; + } + } else { + queryLock(); + try { + tossups = await getRandomTossup(query, categoryManager); + tossups = [tossups]; + } finally { + queryUnlock(); + } + + if (!tossups[0]) { + window.alert('No questions found'); + return false; + } + + query.setName = tossups[0].set.name; + query.packetNumbers = [tossups[0].packet.number]; + + tossupText = tossups[0].question_sanitized; + tossupTextSplit = tossupText.split(' ').filter(word => word !== ''); + document.getElementById('question-number-info').textContent = tossups[0].number; + questionNumber = 1; + } + + return true; +} + +/** + * Called when the users buzzes. + * The first "buzz" pauses the question, and the second "buzz" reveals the rest of the question + * and updates the score. + */ +function buzz () { + // Stop the question reading + clearTimeout(timeoutID); + currentlyBuzzing = true; + if (audio.soundEffects) audio.buzz.play(); + + buzzpointIndex = document.getElementById('question').textContent.length; + if (!tossupTextSplit.includes('(*)') && tossupText.includes('(*)')) { + buzzpointIndex += 3; + } + + // Include buzzpoint + document.getElementById('question').textContent += '(#) '; + + document.getElementById('buzz').textContent = 'Reveal'; + document.getElementById('next').disabled = true; + document.getElementById('start').disabled = true; + document.getElementById('pause').disabled = true; + + if (settings.timer) { + timer.stopTimer(); + timer.startTimer(ANSWER_TIME_LIMIT, () => document.getElementById('answer-submit').click()); + } +} + +/** + * Clears user stats. + */ +function clearStats () { + stats.powers = 0; + stats.tens = 0; + stats.negs = 0; + stats.dead = 0; + stats.points = 0; + stats.totalCorrectCelerity = 0; + + updateStatDisplay(); + window.sessionStorage.removeItem('tossup-stats'); +} + +async function giveAnswer (givenAnswer) { + currentlyBuzzing = false; + + const { directive, directedPrompt } = await api.checkAnswer(tossups[questionNumber - 1].answer, givenAnswer); + + switch (directive) { + case 'accept': { + const points = updateScore(true); + if (audio.soundEffects) { + if (points > 10) { + audio.power.play(); + } else { + audio.correct.play(); + } + } + revealQuestion(); + break; + } + case 'reject': + updateScore(false); + if (audio.soundEffects) audio.incorrect.play(); + if (settings.rebuzz) { + document.getElementById('buzz').disabled = false; + document.getElementById('buzz').textContent = 'Buzz'; + document.getElementById('next').disabled = false; + document.getElementById('pause').disabled = false; + document.getElementById('start').disabled = false; + readQuestion(Date.now()); + } else { + revealQuestion(); + } + break; + case 'prompt': + document.getElementById('answer-input-group').classList.remove('d-none'); + document.getElementById('answer-input').focus(); + document.getElementById('answer-input').placeholder = directedPrompt ? `Prompt: "${directedPrompt}"` : 'Prompt'; + break; + } +} + +function isPace (setName) { + if (!setName) { return false; } + + return setName.includes('PACE'); +} + +async function loadRandomTossups ({ alternateSubcategories, categories, difficulties, maxYear, minYear, number = 1, powermarkOnly, standardOnly, subcategories } = {}) { + randomTossups = []; + randomTossups = await api.getRandomTossup({ alternateSubcategories, categories, difficulties, maxYear, minYear, number, powermarkOnly, standardOnly, subcategories }); +} + +/** + * Get a random tossup. + * @returns + */ +async function getRandomTossup ({ alternateSubcategories, categories, difficulties, minYear, maxYear, powermarkOnly, subcategories, standardOnly } = {}, categoryManager = null) { + if (categoryManager?.percentView) { + categories = [categoryManager.getRandomCategory()]; + subcategories = []; + alternateSubcategories = []; + await loadRandomTossups({ alternateSubcategories, categories, difficulties, maxYear, minYear, powermarkOnly, subcategories, standardOnly }); + return randomTossups.pop(); + } + + if (randomTossups.length === 0) { + await loadRandomTossups({ alternateSubcategories, categories, difficulties, maxYear, minYear, number: 20, powermarkOnly, subcategories, standardOnly }); + } + + const randomQuestion = randomTossups.pop(); + + // Begin loading the next batch of questions (asynchronously) + if (randomTossups.length === 0) { + loadRandomTossups({ alternateSubcategories, categories, difficulties, maxYear, minYear, number: 20, powermarkOnly, subcategories, standardOnly }); + } + + return randomQuestion; +} + +async function next () { + // Stop reading the current question: + clearTimeout(timeoutID); + currentlyBuzzing = false; + if (settings.timer) { + timer.stopTimer(); + timer.tenthsRemaining = 0; + timer.updateDisplay(); + } + + if (await account.getUsername() && document.getElementById('answer').innerHTML) { + const pointValue = previous.isCorrect ? (previous.inPower ? previous.powerValue : 10) : (previous.endOfQuestion ? 0 : previous.negValue); + questionStats.recordTossup(tossups[questionNumber - 1], previous.isCorrect, pointValue, previous.celerity, false); + } + + document.getElementById('answer').textContent = ''; + document.getElementById('question').textContent = ''; + document.getElementById('toggle-correct').textContent = 'I was wrong'; + document.getElementById('toggle-correct').classList.add('d-none'); + + const hasNextQuestion = await advanceQuestion(); + + if (!hasNextQuestion) { + return; + } + + document.getElementById('buzz').textContent = 'Buzz'; + document.getElementById('buzz').disabled = false; + document.getElementById('next').textContent = 'Skip'; + document.getElementById('packet-number-info').textContent = query.packetNumbers[0]; + document.getElementById('packet-length-info').textContent = settings.selectBySetName ? tossups.length : '-'; + document.getElementById('pause').textContent = 'Pause'; + document.getElementById('pause').disabled = false; + document.getElementById('question').textContent = ''; + document.getElementById('set-name-info').textContent = query.setName; + + paused = false; + readQuestion(Date.now()); +} + +/** + * Toggles pausing or resuming the tossup. + */ +function pause () { + if (paused) { + document.getElementById('buzz').removeAttribute('disabled'); + document.getElementById('pause').textContent = 'Pause'; + readQuestion(Date.now()); + } else { + document.getElementById('buzz').setAttribute('disabled', 'disabled'); + document.getElementById('pause').textContent = 'Resume'; + clearTimeout(timeoutID); + } + paused = !paused; +} + +/** + * Recursively reads the question based on the reading speed. + */ +function readQuestion (expectedReadTime) { + if (!currentlyBuzzing && tossupTextSplit.length > 0) { + const word = tossupTextSplit.shift(); + if (word !== '(*)') { + document.getElementById('question').textContent += word + ' '; + } + + // calculate time needed before reading next word + let time = Math.log(word.length) + 1; + if ((word.endsWith('.') && word.charCodeAt(word.length - 2) > 96 && word.charCodeAt(word.length - 2) < 123) || + word.slice(-2) === '.\u201d' || word.slice(-2) === '!\u201d' || word.slice(-2) === '?\u201d') { time += 2; } else if (word.endsWith(',') || word.slice(-2) === ',\u201d') { time += 0.75; } else if (word === '(*)') { time = 0; } + + time = time * 0.9 * (125 - settings.readingSpeed); + const delay = time - Date.now() + expectedReadTime; + + timeoutID = window.setTimeout(() => { + readQuestion(time + expectedReadTime); + }, delay); + } else { + document.getElementById('pause').disabled = true; + if (settings.timer) { + timer.startTimer(DEAD_TIME_LIMIT, revealQuestion); + } + } +} + +function revealQuestion () { + document.getElementById('question').innerHTML = insertTokensIntoHTML(tossups[questionNumber - 1].question, tossups[questionNumber - 1].question_sanitized, [[buzzpointIndex]], [' (#) ']); + document.getElementById('answer').innerHTML = 'ANSWER: ' + tossups[questionNumber - 1].answer; + + document.getElementById('buzz').disabled = true; + document.getElementById('buzz').textContent = 'Buzz'; + document.getElementById('next').disabled = false; + document.getElementById('next').textContent = 'Next'; + document.getElementById('start').disabled = false; + + document.getElementById('toggle-correct').classList.remove('d-none'); + document.getElementById('toggle-correct').textContent = previous.isCorrect ? 'I was wrong' : 'I was right'; +} + +function toggleCorrect () { + const multiplier = previous.isCorrect ? -1 : 1; + + if (previous.inPower) { + stats.powers += multiplier * 1; + stats.points += multiplier * previous.powerValue; + } else { + stats.tens += multiplier * 1; + stats.points += multiplier * 10; + } + + if (previous.endOfQuestion) { + stats.dead += multiplier * -1; + } else { + stats.negs += multiplier * -1; + stats.points += multiplier * -previous.negValue; + } + + stats.totalCorrectCelerity += multiplier * previous.celerity; + + previous.isCorrect = !previous.isCorrect; + document.getElementById('toggle-correct').textContent = previous.isCorrect ? 'I was wrong' : 'I was right'; + + updateStatDisplay(); + window.sessionStorage.setItem('tossup-stats', JSON.stringify(stats)); +} + +function updateScore (isCorrect) { + const endOfQuestion = (tossupTextSplit.length === 0); + const inPower = tossupTextSplit.includes('(*)') && tossupText.includes('(*)'); + const powerValue = isPace(query.setName) ? 20 : 15; + const negValue = isPace(query.setName) ? 0 : -5; + const points = isCorrect ? (inPower ? powerValue : 10) : (endOfQuestion ? 0 : negValue); + + const characterCount = tossupTextSplit.join(' ').length; + const celerity = characterCount / tossupText.length; + + let result; + + if (isCorrect) { + result = inPower ? 'powers' : 'tens'; + stats.totalCorrectCelerity += celerity; + } else { + result = endOfQuestion ? 'dead' : 'negs'; + } + + stats[result] += 1; + stats.points += points; + + previous.celerity = celerity; + previous.endOfQuestion = endOfQuestion; + previous.inPower = inPower; + previous.negValue = negValue; + previous.powerValue = powerValue; + previous.isCorrect = isCorrect; + + updateStatDisplay(); + window.sessionStorage.setItem('tossup-stats', JSON.stringify(stats)); + + return points; +} + +/** + * Updates the displayed stat line. + */ +function updateStatDisplay () { + const { powers, tens, negs, dead, points, totalCorrectCelerity } = stats; + const numTossups = powers + tens + negs + dead; + const numCorrectTossups = powers + tens; + let celerity = numCorrectTossups === 0 ? 0 : parseFloat(totalCorrectCelerity) / numCorrectTossups; + celerity = Math.round(1000 * celerity) / 1000; + const includePlural = (numTossups === 1) ? '' : 's'; + document.getElementById('statline').innerHTML = + `${powers}/${tens}/${negs} with ${numTossups} tossup${includePlural} seen (${points} pts, celerity: ${celerity})`; + + // disable clear stats button if no stats + document.getElementById('clear-stats').disabled = (numTossups === 0); +} + +document.getElementById('answer-form').addEventListener('submit', function (event) { + event.preventDefault(); + event.stopPropagation(); + + if (settings.timer) { + timer.stopTimer(); + timer.tenthsRemaining = 0; + timer.updateDisplay(); + } + + const answer = document.getElementById('answer-input').value; + + document.getElementById('answer-input').value = ''; + document.getElementById('answer-input').blur(); + document.getElementById('answer-input').placeholder = 'Enter answer'; + document.getElementById('answer-input-group').classList.add('d-none'); + + giveAnswer(answer); +}); + +document.getElementById('buzz').addEventListener('click', function () { + this.blur(); + + // reveal answer on second click + // when NOT using type to answer + if (currentlyBuzzing) { + currentlyBuzzing = false; + updateScore(true); + revealQuestion(); + return; + } + + buzz(); + + if (settings.typeToAnswer) { + document.getElementById('answer-input-group').classList.remove('d-none'); + document.getElementById('answer-input').focus(); + this.disabled = true; + } +}); + +document.getElementById('clear-stats').addEventListener('click', function () { + this.blur(); + clearStats(); +}); + +document.getElementById('next').addEventListener('click', function () { + this.blur(); + createTossupCard(tossups[questionNumber - 1]); + next(); +}); + +document.getElementById('packet-number').addEventListener('change', function () { + // if field is blank, store blank result in `query` + query.packetNumbers = rangeToArray(this.value.trim(), 0); + window.localStorage.setItem('singleplayer-tossup-query', JSON.stringify(query)); + query.packetNumbers = rangeToArray(this.value.trim(), maxPacketNumber); +}); + +document.getElementById('pause').addEventListener('click', function () { + this.blur(); + pause(); +}); + +document.getElementById('reading-speed').addEventListener('input', function () { + settings.readingSpeed = this.value; + document.getElementById('reading-speed-display').textContent = this.value; + window.localStorage.setItem('singleplayer-tossup-settings', JSON.stringify(settings)); +}); + +document.getElementById('report-question-submit').addEventListener('click', function () { + api.reportQuestion( + document.getElementById('report-question-id').value, + document.getElementById('report-question-reason').value, + document.getElementById('report-question-description').value + ); +}); + +document.getElementById('set-name').addEventListener('change', async function () { + query.setName = this.value.trim(); + + // make border red if set name is not in set list + if (api.getSetList().includes(this.value) || this.value.length === 0) { + this.classList.remove('is-invalid'); + } else { + this.classList.add('is-invalid'); + } + + maxPacketNumber = await api.getNumPackets(this.value); + + if (this.value === '' || maxPacketNumber === 0) { + document.getElementById('packet-number').placeholder = 'Packet Numbers'; + } else { + document.getElementById('packet-number').placeholder = `Packet Numbers (1-${maxPacketNumber})`; + } + + window.localStorage.setItem('singleplayer-tossup-query', JSON.stringify(query)); +}); + +document.getElementById('start').addEventListener('click', async function () { + this.blur(); + if (query.setName.length === 0 && settings.selectBySetName) { + window.alert('Please enter a set name.'); + return false; + } + + if (query.packetNumbers.length === 0 && settings.selectBySetName) { + query.packetNumbers = rangeToArray(document.getElementById('packet-number').value.trim(), maxPacketNumber); + } + + document.getElementById('next').disabled = false; + document.getElementById('next').textContent = 'Skip'; + document.getElementById('settings').classList.add('d-none'); + + if (settings.selectBySetName) { + queryLock(); + questionNumber = 0; + try { + tossups = await api.getPacketTossups(query.setName, query.packetNumbers[0]); + } finally { + queryUnlock(); + } + } + + next(); +}); + +document.getElementById('toggle-correct').addEventListener('click', function () { + this.blur(); + toggleCorrect(); +}); + +document.getElementById('toggle-powermark-only').addEventListener('click', function () { + this.blur(); + query.powermarkOnly = this.checked; + loadRandomTossups(query); + window.localStorage.setItem('singleplayer-tossup-query', JSON.stringify(query)); +}); + +document.getElementById('toggle-select-by-set-name').addEventListener('click', function () { + this.blur(); + settings.selectBySetName = this.checked; + document.getElementById('toggle-powermark-only').disabled = this.checked; + document.getElementById('toggle-standard-only').disabled = this.checked; + + if (this.checked) { + document.getElementById('difficulty-settings').classList.add('d-none'); + document.getElementById('set-settings').classList.remove('d-none'); + } else { + document.getElementById('difficulty-settings').classList.remove('d-none'); + document.getElementById('set-settings').classList.add('d-none'); + } + + window.localStorage.setItem('singleplayer-tossup-settings', JSON.stringify(settings)); +}); + +document.getElementById('toggle-show-history').addEventListener('click', function () { + this.blur(); + settings.showHistory = this.checked; + + if (this.checked) { + document.getElementById('room-history').classList.remove('d-none'); + } else { + document.getElementById('room-history').classList.add('d-none'); + } + + window.localStorage.setItem('singleplayer-tossup-settings', JSON.stringify(settings)); +}); + +document.getElementById('toggle-standard-only').addEventListener('click', function () { + this.blur(); + query.standardOnly = this.checked; + loadRandomTossups(query); + window.localStorage.setItem('singleplayer-tossup-query', JSON.stringify(query)); +}); + +document.getElementById('toggle-timer').addEventListener('click', function () { + this.blur(); + settings.timer = this.checked; + document.getElementById('timer').classList.toggle('d-none'); + window.localStorage.setItem('singleplayer-tossup-settings', JSON.stringify(settings)); +}); + +document.getElementById('type-to-answer').addEventListener('click', function () { + this.blur(); + settings.typeToAnswer = this.checked; + document.getElementById('toggle-rebuzz').disabled = !this.checked; + window.localStorage.setItem('singleplayer-tossup-settings', JSON.stringify(settings)); +}); + +document.getElementById('toggle-settings').addEventListener('click', function () { + this.blur(); + document.getElementById('buttons').classList.toggle('col-lg-9'); + document.getElementById('buttons').classList.toggle('col-lg-12'); + document.getElementById('content').classList.toggle('col-lg-9'); + document.getElementById('content').classList.toggle('col-lg-12'); + document.getElementById('settings').classList.toggle('d-none'); + document.getElementById('settings').classList.toggle('d-lg-none'); +}); + +document.getElementById('toggle-rebuzz').addEventListener('click', function () { + this.blur(); + settings.rebuzz = this.checked; + window.localStorage.setItem('singleplayer-tossup-settings', JSON.stringify(settings)); +}); + +document.getElementById('year-range-a').onchange = function () { + query.minYear = $('#slider').slider('values', 0); + query.maxYear = $('#slider').slider('values', 1); + loadRandomTossups(query); + window.localStorage.setItem('singleplayer-tossup-query', JSON.stringify(query)); +}; + +document.getElementById('year-range-b').onchange = function () { + query.minYear = $('#slider').slider('values', 0); + query.maxYear = $('#slider').slider('values', 1); + loadRandomTossups(query); + window.localStorage.setItem('singleplayer-tossup-query', JSON.stringify(query)); +}; + +document.addEventListener('keydown', (event) => { + if (['INPUT', 'TEXTAREA', 'SELECT'].includes(document.activeElement.tagName)) return; + + switch (event.key) { + case ' ': + document.getElementById('buzz').click(); + // Prevent spacebar from scrolling the page: + if (event.target === document.body) event.preventDefault(); + break; + case 'e': + document.getElementById('toggle-settings').click(); + break; + case 'k': + document.getElementsByClassName('card-header-clickable')[0].click(); + break; + case 't': + document.getElementsByClassName('star-tossup')[0].click(); + break; + case 'y': + navigator.clipboard.writeText(tossups[0]?._id ?? ''); + break; + case 'n': + document.getElementById('next').click(); + break; + case 'p': + document.getElementById('pause').click(); + break; + case 's': + document.getElementById('start').click(); + break; + } +}); + +$(document).ready(function () { + $('#slider').slider('values', 0, query.minYear); + $('#slider').slider('values', 1, query.maxYear); +}); +document.getElementById('year-range-a').textContent = query.minYear; +document.getElementById('year-range-b').textContent = query.maxYear; + +ReactDOM.createRoot(document.getElementById('category-modal-root')).render( + { + ({ categories: query.categories, subcategories: query.subcategories, alternateSubcategories: query.alternateSubcategories } = categoryManager.export()); + loadRandomTossups(query); + window.localStorage.setItem('singleplayer-tossup-query', JSON.stringify(query)); + }} + /> +); + +ReactDOM.createRoot(document.getElementById('difficulty-dropdown-root')).render( + { + query.difficulties = getDropdownValues('difficulties'); + loadRandomTossups(query); + window.localStorage.setItem('singleplayer-tossup-query', JSON.stringify(query)); + }} + /> +); diff --git a/client/singleplayer/tossups/old-index.min.js b/client/singleplayer/tossups/old-index.min.js new file mode 100644 index 00000000..09471d56 --- /dev/null +++ b/client/singleplayer/tossups/old-index.min.js @@ -0,0 +1,36 @@ +import account from"../../scripts/accounts.js";import questionStats from"../../scripts/auth/question-stats.js";import api from"../../scripts/api/index.js";import audio from"../../audio/index.js";import Timer from"../../scripts/Timer.js";import{arrayToRange,createTossupCard,rangeToArray}from"../../scripts/utilities/index.js";import CategoryManager from"../../scripts/utilities/category-manager.js";import{getDropdownValues}from"../../scripts/utilities/dropdown-checklist.js";import{insertTokensIntoHTML}from"../../insert-tokens-into-html.js";import CategoryModal from"../../scripts/components/CategoryModal.min.js";import DifficultyDropdown from"../../scripts/components/DifficultyDropdown.min.js";// Functions and variables specific to the tossups page. +const ANSWER_TIME_LIMIT=10,DEAD_TIME_LIMIT=5;// Status variables +let buzzpointIndex=-1,currentlyBuzzing=!1,maxPacketNumber=24,paused=!1,questionNumber=0;// WARNING: 1-indexed +const timer=new Timer;/** + * An array of random questions. + * We get 20 random questions at a time so we don't have to make an HTTP request between every question. + */let randomTossups=[],timeoutID=-1,tossups=[{}],tossupText="",tossupTextSplit=[];const previous={isCorrect:!0,inPower:!1,negValue:-5,powerValue:15,endOfQuestion:!1,celerity:0},stats=window.sessionStorage.getItem("tossup-stats")?JSON.parse(window.sessionStorage.getItem("tossup-stats")):{powers:0,tens:0,negs:0,dead:0,points:0,totalCorrectCelerity:0},defaults={alternateSubcategories:[],categories:[],difficulties:[],minYear:2010,maxYear:2024,packetNumbers:[],powermarkOnly:!1,setName:"",standardOnly:!1,subcategories:[],version:"01-06-2024"};let query;window.localStorage.getItem("singleplayer-tossup-query")?(query=JSON.parse(window.localStorage.getItem("singleplayer-tossup-query")),query.version!==defaults.version&&(query=defaults,window.localStorage.setItem("singleplayer-tossup-query",JSON.stringify(query)))):query=defaults;const categoryManager=new CategoryManager(query.categories,query.subcategories,query.alternateSubcategories),settings=window.localStorage.getItem("singleplayer-tossup-settings")?JSON.parse(window.localStorage.getItem("singleplayer-tossup-settings")):{readingSpeed:50,rebuzz:!1,selectBySetName:!1,showHistory:!0,timer:!0,typeToAnswer:!0};// Load query and settings first so user doesn't see the default settings +settings.readingSpeed&&(document.getElementById("reading-speed-display").textContent=settings.readingSpeed,document.getElementById("reading-speed").value=settings.readingSpeed),settings.rebuzz&&(document.getElementById("toggle-rebuzz").checked=!0),settings.selectBySetName&&(document.getElementById("difficulty-settings").classList.add("d-none"),document.getElementById("set-settings").classList.remove("d-none"),document.getElementById("toggle-select-by-set-name").checked=!0,document.getElementById("toggle-powermark-only").disabled=!0,document.getElementById("toggle-standard-only").disabled=!0),settings.showHistory||(document.getElementById("toggle-show-history").checked=!1,document.getElementById("room-history").classList.add("d-none")),settings.timer||(document.getElementById("toggle-timer").checked=!1,document.getElementById("timer").classList.add("d-none")),settings.typeToAnswer||(document.getElementById("type-to-answer").checked=!1,document.getElementById("toggle-rebuzz").disabled=!0),query.packetNumbers&&(document.getElementById("packet-number").value=arrayToRange(query.packetNumbers)),query.powermarkOnly&&(document.getElementById("toggle-powermark-only").checked=!0),query.setName&&(document.getElementById("set-name").value=query.setName,api.getNumPackets(query.setName).then(a=>{maxPacketNumber=a,0===maxPacketNumber?document.getElementById("set-name").classList.add("is-invalid"):document.getElementById("packet-number").placeholder=`Packet Numbers (1-${maxPacketNumber})`})),updateStatDisplay();function queryLock(){document.getElementById("question").textContent="Fetching questions...",document.getElementById("start").disabled=!0,document.getElementById("next").disabled=!0,document.getElementById("pause").disabled=!0,document.getElementById("buzz").disabled=!0}function queryUnlock(){document.getElementById("question").textContent="",document.getElementById("start").disabled=!1,document.getElementById("next").disabled=!1,document.getElementById("pause").disabled=!1,document.getElementById("buzz").disabled=!1}/** + * @returns {Promise} Whether or not there is a next question + */async function advanceQuestion(){if(settings.selectBySetName){// Get the next question if the current one is in the wrong category and subcategory +do// Go to the next packet if you reach the end of this packet +if(questionNumber++,questionNumber>tossups.length){if(query.packetNumbers.shift(),0===query.packetNumbers.length)return window.alert("No more questions left"),document.getElementById("buzz").disabled=!0,document.getElementById("pause").disabled=!0,document.getElementById("next").disabled=!0,!1;// alert the user if there are no more packets +queryLock();try{tossups=await api.getPacketTossups(query.setName,query.packetNumbers[0])}finally{queryUnlock()}questionNumber=1}while(!categoryManager.isValidCategory(tossups[questionNumber-1]));0""!==a),document.getElementById("question-number-info").textContent=questionNumber)}else{queryLock();try{tossups=await getRandomTossup(query,categoryManager),tossups=[tossups]}finally{queryUnlock()}if(!tossups[0])return window.alert("No questions found"),!1;query.setName=tossups[0].set.name,query.packetNumbers=[tossups[0].packet.number],tossupText=tossups[0].question_sanitized,tossupTextSplit=tossupText.split(" ").filter(a=>""!==a),document.getElementById("question-number-info").textContent=tossups[0].number,questionNumber=1}return!0}/** + * Called when the users buzzes. + * The first "buzz" pauses the question, and the second "buzz" reveals the rest of the question + * and updates the score. + */function buzz(){// Stop the question reading +// Include buzzpoint +clearTimeout(timeoutID),currentlyBuzzing=!0,audio.soundEffects&&audio.buzz.play(),buzzpointIndex=document.getElementById("question").textContent.length,!tossupTextSplit.includes("(*)")&&tossupText.includes("(*)")&&(buzzpointIndex+=3),document.getElementById("question").textContent+="(#) ",document.getElementById("buzz").textContent="Reveal",document.getElementById("next").disabled=!0,document.getElementById("start").disabled=!0,document.getElementById("pause").disabled=!0,settings.timer&&(timer.stopTimer(),timer.startTimer(ANSWER_TIME_LIMIT,()=>document.getElementById("answer-submit").click()))}/** + * Clears user stats. + */function clearStats(){stats.powers=0,stats.tens=0,stats.negs=0,stats.dead=0,stats.points=0,stats.totalCorrectCelerity=0,updateStatDisplay(),window.sessionStorage.removeItem("tossup-stats")}async function giveAnswer(a){currentlyBuzzing=!1;const{directive:b,directedPrompt:c}=await api.checkAnswer(tossups[questionNumber-1].answer,a);switch(b){case"accept":{const a=updateScore(!0);audio.soundEffects&&(10b.charCodeAt(b.length-2)||".\u201D"===b.slice(-2)||"!\u201D"===b.slice(-2)||"?\u201D"===b.slice(-2)?c+=2:b.endsWith(",")||",\u201D"===b.slice(-2)?c+=.75:"(*)"===b&&(c=0),c=.9*c*(125-settings.readingSpeed);const d=c-Date.now()+a;timeoutID=window.setTimeout(()=>{readQuestion(c+a)},d)}else document.getElementById("pause").disabled=!0,settings.timer&&timer.startTimer(DEAD_TIME_LIMIT,revealQuestion)}function revealQuestion(){document.getElementById("question").innerHTML=insertTokensIntoHTML(tossups[questionNumber-1].question,tossups[questionNumber-1].question_sanitized,[[buzzpointIndex]],[" (#) "]),document.getElementById("answer").innerHTML="ANSWER: "+tossups[questionNumber-1].answer,document.getElementById("buzz").disabled=!0,document.getElementById("buzz").textContent="Buzz",document.getElementById("next").disabled=!1,document.getElementById("next").textContent="Next",document.getElementById("start").disabled=!1,document.getElementById("toggle-correct").classList.remove("d-none"),document.getElementById("toggle-correct").textContent=previous.isCorrect?"I was wrong":"I was right"}function toggleCorrect(){const a=previous.isCorrect?-1:1;previous.inPower?(stats.powers+=1*a,stats.points+=a*previous.powerValue):(stats.tens+=1*a,stats.points+=10*a),previous.endOfQuestion?stats.dead+=-1*a:(stats.negs+=-1*a,stats.points+=a*-previous.negValue),stats.totalCorrectCelerity+=a*previous.celerity,previous.isCorrect=!previous.isCorrect,document.getElementById("toggle-correct").textContent=previous.isCorrect?"I was wrong":"I was right",updateStatDisplay(),window.sessionStorage.setItem("tossup-stats",JSON.stringify(stats))}function updateScore(a){const b=0===tossupTextSplit.length,c=tossupTextSplit.includes("(*)")&&tossupText.includes("(*)"),d=isPace(query.setName)?20:15,e=isPace(query.setName)?0:-5,f=a?c?d:10:b?0:e,g=tossupTextSplit.join(" ").length,h=g/tossupText.length;let i;return a?(i=c?"powers":"tens",stats.totalCorrectCelerity+=h):i=b?"dead":"negs",stats[i]+=1,stats.points+=f,previous.celerity=h,previous.endOfQuestion=b,previous.inPower=c,previous.negValue=e,previous.powerValue=d,previous.isCorrect=a,updateStatDisplay(),window.sessionStorage.setItem("tossup-stats",JSON.stringify(stats)),f}/** + * Updates the displayed stat line. + */function updateStatDisplay(){const{powers:a,tens:b,negs:c,dead:d,points:e,totalCorrectCelerity:f}=stats,g=a+b+c+d,h=a+b;let i=0===h?0:parseFloat(f)/h;i=Math.round(1e3*i)/1e3;const j=1===g?"":"s";// disable clear stats button if no stats +document.getElementById("statline").innerHTML=`${a}/${b}/${c} with ${g} tossup${j} seen (${e} pts, celerity: ${i})`,document.getElementById("clear-stats").disabled=0===g}document.getElementById("answer-form").addEventListener("submit",function(a){a.preventDefault(),a.stopPropagation(),settings.timer&&(timer.stopTimer(),timer.tenthsRemaining=0,timer.updateDisplay());const b=document.getElementById("answer-input").value;document.getElementById("answer-input").value="",document.getElementById("answer-input").blur(),document.getElementById("answer-input").placeholder="Enter answer",document.getElementById("answer-input-group").classList.add("d-none"),giveAnswer(b)}),document.getElementById("buzz").addEventListener("click",function(){// reveal answer on second click +// when NOT using type to answer +return this.blur(),currentlyBuzzing?(currentlyBuzzing=!1,updateScore(!0),void revealQuestion()):void(buzz(),settings.typeToAnswer&&(document.getElementById("answer-input-group").classList.remove("d-none"),document.getElementById("answer-input").focus(),this.disabled=!0))}),document.getElementById("clear-stats").addEventListener("click",function(){this.blur(),clearStats()}),document.getElementById("next").addEventListener("click",function(){this.blur(),createTossupCard(tossups[questionNumber-1]),next()}),document.getElementById("packet-number").addEventListener("change",function(){// if field is blank, store blank result in `query` +query.packetNumbers=rangeToArray(this.value.trim(),0),window.localStorage.setItem("singleplayer-tossup-query",JSON.stringify(query)),query.packetNumbers=rangeToArray(this.value.trim(),maxPacketNumber)}),document.getElementById("pause").addEventListener("click",function(){this.blur(),pause()}),document.getElementById("reading-speed").addEventListener("input",function(){settings.readingSpeed=this.value,document.getElementById("reading-speed-display").textContent=this.value,window.localStorage.setItem("singleplayer-tossup-settings",JSON.stringify(settings))}),document.getElementById("report-question-submit").addEventListener("click",function(){api.reportQuestion(document.getElementById("report-question-id").value,document.getElementById("report-question-reason").value,document.getElementById("report-question-description").value)}),document.getElementById("set-name").addEventListener("change",async function(){query.setName=this.value.trim(),api.getSetList().includes(this.value)||0===this.value.length?this.classList.remove("is-invalid"):this.classList.add("is-invalid"),maxPacketNumber=await api.getNumPackets(this.value),document.getElementById("packet-number").placeholder=""===this.value||0===maxPacketNumber?"Packet Numbers":`Packet Numbers (1-${maxPacketNumber})`,window.localStorage.setItem("singleplayer-tossup-query",JSON.stringify(query))}),document.getElementById("start").addEventListener("click",async function(){if(this.blur(),0===query.setName.length&&settings.selectBySetName)return window.alert("Please enter a set name."),!1;if(0===query.packetNumbers.length&&settings.selectBySetName&&(query.packetNumbers=rangeToArray(document.getElementById("packet-number").value.trim(),maxPacketNumber)),document.getElementById("next").disabled=!1,document.getElementById("next").textContent="Skip",document.getElementById("settings").classList.add("d-none"),settings.selectBySetName){queryLock(),questionNumber=0;try{tossups=await api.getPacketTossups(query.setName,query.packetNumbers[0])}finally{queryUnlock()}}next()}),document.getElementById("toggle-correct").addEventListener("click",function(){this.blur(),toggleCorrect()}),document.getElementById("toggle-powermark-only").addEventListener("click",function(){this.blur(),query.powermarkOnly=this.checked,loadRandomTossups(query),window.localStorage.setItem("singleplayer-tossup-query",JSON.stringify(query))}),document.getElementById("toggle-select-by-set-name").addEventListener("click",function(){this.blur(),settings.selectBySetName=this.checked,document.getElementById("toggle-powermark-only").disabled=this.checked,document.getElementById("toggle-standard-only").disabled=this.checked,this.checked?(document.getElementById("difficulty-settings").classList.add("d-none"),document.getElementById("set-settings").classList.remove("d-none")):(document.getElementById("difficulty-settings").classList.remove("d-none"),document.getElementById("set-settings").classList.add("d-none")),window.localStorage.setItem("singleplayer-tossup-settings",JSON.stringify(settings))}),document.getElementById("toggle-show-history").addEventListener("click",function(){this.blur(),settings.showHistory=this.checked,this.checked?document.getElementById("room-history").classList.remove("d-none"):document.getElementById("room-history").classList.add("d-none"),window.localStorage.setItem("singleplayer-tossup-settings",JSON.stringify(settings))}),document.getElementById("toggle-standard-only").addEventListener("click",function(){this.blur(),query.standardOnly=this.checked,loadRandomTossups(query),window.localStorage.setItem("singleplayer-tossup-query",JSON.stringify(query))}),document.getElementById("toggle-timer").addEventListener("click",function(){this.blur(),settings.timer=this.checked,document.getElementById("timer").classList.toggle("d-none"),window.localStorage.setItem("singleplayer-tossup-settings",JSON.stringify(settings))}),document.getElementById("type-to-answer").addEventListener("click",function(){this.blur(),settings.typeToAnswer=this.checked,document.getElementById("toggle-rebuzz").disabled=!this.checked,window.localStorage.setItem("singleplayer-tossup-settings",JSON.stringify(settings))}),document.getElementById("toggle-settings").addEventListener("click",function(){this.blur(),document.getElementById("buttons").classList.toggle("col-lg-9"),document.getElementById("buttons").classList.toggle("col-lg-12"),document.getElementById("content").classList.toggle("col-lg-9"),document.getElementById("content").classList.toggle("col-lg-12"),document.getElementById("settings").classList.toggle("d-none"),document.getElementById("settings").classList.toggle("d-lg-none")}),document.getElementById("toggle-rebuzz").addEventListener("click",function(){this.blur(),settings.rebuzz=this.checked,window.localStorage.setItem("singleplayer-tossup-settings",JSON.stringify(settings))}),document.getElementById("year-range-a").onchange=function(){query.minYear=$("#slider").slider("values",0),query.maxYear=$("#slider").slider("values",1),loadRandomTossups(query),window.localStorage.setItem("singleplayer-tossup-query",JSON.stringify(query))},document.getElementById("year-range-b").onchange=function(){query.minYear=$("#slider").slider("values",0),query.maxYear=$("#slider").slider("values",1),loadRandomTossups(query),window.localStorage.setItem("singleplayer-tossup-query",JSON.stringify(query))},document.addEventListener("keydown",a=>{if(!["INPUT","TEXTAREA","SELECT"].includes(document.activeElement.tagName))switch(a.key){case" ":document.getElementById("buzz").click(),a.target===document.body&&a.preventDefault();break;case"e":document.getElementById("toggle-settings").click();break;case"k":document.getElementsByClassName("card-header-clickable")[0].click();break;case"t":document.getElementsByClassName("star-tossup")[0].click();break;case"y":navigator.clipboard.writeText(tossups[0]?._id??"");break;case"n":document.getElementById("next").click();break;case"p":document.getElementById("pause").click();break;case"s":document.getElementById("start").click()}}),$(document).ready(function(){$("#slider").slider("values",0,query.minYear),$("#slider").slider("values",1,query.maxYear)}),document.getElementById("year-range-a").textContent=query.minYear,document.getElementById("year-range-b").textContent=query.maxYear,ReactDOM.createRoot(document.getElementById("category-modal-root")).render(/*#__PURE__*/React.createElement(CategoryModal,{categoryManager:categoryManager,onClose:()=>{({categories:query.categories,subcategories:query.subcategories,alternateSubcategories:query.alternateSubcategories}=categoryManager.export()),loadRandomTossups(query),window.localStorage.setItem("singleplayer-tossup-query",JSON.stringify(query))}})),ReactDOM.createRoot(document.getElementById("difficulty-dropdown-root")).render(/*#__PURE__*/React.createElement(DifficultyDropdown,{startingDifficulties:query.difficulties,onChange:()=>{query.difficulties=getDropdownValues("difficulties"),loadRandomTossups(query),window.localStorage.setItem("singleplayer-tossup-query",JSON.stringify(query))}})); \ No newline at end of file diff --git a/quizbowl/Player.js b/quizbowl/Player.js index 9eadf04a..f4647ddf 100644 --- a/quizbowl/Player.js +++ b/quizbowl/Player.js @@ -1,8 +1,7 @@ -import { USERNAME_MAX_LENGTH } from '../server/multiplayer/constants.js'; - class Player { - constructor (userId) { + constructor (userId, MAX_USERNAME_LENGTH = undefined) { this.userId = userId; + this.MAX_USERNAME_LENGTH = MAX_USERNAME_LENGTH; this.username = ''; @@ -67,12 +66,11 @@ class Player { /** * Safely update the player's username, and return the new username. * @param {string} username - * @param {number} [maxLength=USERNAME_MAX_LENGTH] * @returns {string} newUsername */ - safelySetUsername (username, maxLength = USERNAME_MAX_LENGTH) { + safelySetUsername (username) { if (!username) { username = ''; } - username = username.substring(0, maxLength); + username = username.substring(0, this.MAX_USERNAME_LENGTH); this.username = username; return this.username; } diff --git a/quizbowl/TossupRoom.js b/quizbowl/TossupRoom.js index 71812dd7..14d94752 100644 --- a/quizbowl/TossupRoom.js +++ b/quizbowl/TossupRoom.js @@ -15,7 +15,7 @@ export default class TossupRoom extends Room { constructor (name, categories = [], subcategories = [], alternateSubcategories = []) { super(name); - this.checkAnswer = async function checkAnswer (answerline, givenAnswer) { console.log('hm'); throw new Error('Not implemented'); }; + this.checkAnswer = async function checkAnswer (answerline, givenAnswer) { throw new Error('Not implemented'); }; this.getRandomTossups = async function getRandomTossups (args) { throw new Error('Not implemented'); }; this.getSet = async function getSet (args) { throw new Error('Not implemented'); }; this.getSetList = async function getSetList (args) { throw new Error('Not implemented'); }; @@ -194,7 +194,7 @@ export default class TossupRoom extends Room { this.adjustQuery(['difficulties'], [value]); } - giveAnswer (userId, { givenAnswer }) { + async giveAnswer (userId, { givenAnswer }) { if (typeof givenAnswer !== 'string') { return false; } if (this.buzzedIn !== userId) { return false; } @@ -204,11 +204,7 @@ export default class TossupRoom extends Room { if (Object.keys(this.tossup).length === 0) { return; } - const celerity = this.questionSplit.slice(this.wordIndex).join(' ').length / this.tossup.question.length; - const endOfQuestion = (this.wordIndex === this.questionSplit.length); - const inPower = this.questionSplit.indexOf('(*)') >= this.wordIndex; - const { directive, directedPrompt } = this.checkAnswer(this.tossup.answer, givenAnswer); - const points = scoreTossup({ isCorrect: directive === 'accept', inPower, endOfQuestion }); + const { celerity, directive, directedPrompt, points } = await this.scoreTossup({ givenAnswer }); switch (directive) { case 'accept': @@ -277,12 +273,13 @@ export default class TossupRoom extends Room { if (this.questionProgress !== this.QuestionProgressEnum.ANSWER_REVEALED) { this.revealQuestion(); } + const oldTossup = this.tossup; const hasNextQuestion = await this.advanceQuestion(); this.queryingQuestion = false; if (!hasNextQuestion) { return; } const username = this.players[userId].username; - this.emitMessage({ type, userId, username, tossup: this.tossup }); + this.emitMessage({ type, userId, username, oldTossup, tossup: this.tossup }); this.wordIndex = 0; this.questionProgress = this.QuestionProgressEnum.READING; @@ -309,6 +306,15 @@ export default class TossupRoom extends Room { this.emitMessage({ type: 'pause', paused: this.paused, username }); } + async scoreTossup ({ givenAnswer }) { + const celerity = this.questionSplit.slice(this.wordIndex).join(' ').length / this.tossup.question.length; + const endOfQuestion = (this.wordIndex === this.questionSplit.length); + const inPower = this.questionSplit.indexOf('(*)') >= this.wordIndex; + const { directive, directedPrompt } = await this.checkAnswer(this.tossup.answer, givenAnswer); + const points = scoreTossup({ isCorrect: directive === 'accept', inPower, endOfQuestion }); + return { celerity, directive, directedPrompt, endOfQuestion, inPower, points }; + } + async setPacketNumbers (userId, { value }) { const allowedPacketNumbers = await this.getNumPackets(this.query.setName); if (value.some((value) => typeof value !== 'number' || value < 1 || value > allowedPacketNumbers)) { return false; } @@ -347,6 +353,11 @@ export default class TossupRoom extends Room { this.emitMessage({ type: 'set-username', userId, oldUsername, newUsername: username }); } + startServerTimer (time, ontick, callback) { + if (!this.settings.timer) { return; } + super.startServerTimer(time, ontick, callback); + } + async readQuestion (expectedReadTime) { if (Object.keys(this.tossup).length === 0) { return; } if (this.wordIndex >= this.questionSplit.length) { diff --git a/server/multiplayer/ServerPlayer.js b/server/multiplayer/ServerPlayer.js index 5c70f11f..54ab67d0 100644 --- a/server/multiplayer/ServerPlayer.js +++ b/server/multiplayer/ServerPlayer.js @@ -1,8 +1,9 @@ import Player from '../../quizbowl/Player.js'; +import { USERNAME_MAX_LENGTH } from './constants.js'; export default class ServerPlayer extends Player { constructor (userId) { - super(userId); + super(userId, USERNAME_MAX_LENGTH); this.online = true; } } diff --git a/server/multiplayer/ServerTossupRoom.js b/server/multiplayer/ServerTossupRoom.js index 51bd7f9d..091966fc 100644 --- a/server/multiplayer/ServerTossupRoom.js +++ b/server/multiplayer/ServerTossupRoom.js @@ -190,9 +190,9 @@ export default class ServerTossupRoom extends TossupRoom { this.emitMessage({ type: 'toggle-public', public: isPublic, username }); } - toggleSelectBySetName (userId, { selectBySetName }) { + toggleSelectBySetName (userId, { selectBySetName, setName }) { if (this.isPermanent) { return; } - super.toggleSelectBySetName(userId, { selectBySetName }); + super.toggleSelectBySetName(userId, { selectBySetName, setName }); } toggleTimer (userId, { timer }) {