mirror of
				https://github.com/monkeytypegame/monkeytype.git
				synced 2025-10-27 01:06:21 +08:00 
			
		
		
		
	* Add weekly seasons * Fix test * Add week timestamp tests * Add unit tests * Fix weeks before calculation * Update user.spec.ts * Remove minXp maxXp reward config * Record total time typed + last activity timestamp * Season -> Weekly XP Leaderboard * prettier * Add config hints * Update leaderboard.ts * monkeytype Co-authored-by: Miodec <jack@monkeytype.com>
		
			
				
	
	
		
			267 lines
		
	
	
	
		
			7 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			267 lines
		
	
	
	
		
			7 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| let state = {};
 | |
| let schema = {};
 | |
| 
 | |
| const buildLabel = (elementType, text, hintText) => {
 | |
|   const labelElement = document.createElement("label");
 | |
|   labelElement.innerHTML = text;
 | |
|   labelElement.style.fontWeight = elementType === "group" ? "bold" : "lighter";
 | |
| 
 | |
|   if (hintText) {
 | |
|     const hintElement = document.createElement("span");
 | |
|     hintElement.classList.add("tooltip");
 | |
|     hintElement.innerHTML = " ⓘ";
 | |
| 
 | |
|     const hintTextElement = document.createElement("span");
 | |
|     hintTextElement.classList.add("tooltip-text");
 | |
|     hintTextElement.innerHTML = hintText;
 | |
| 
 | |
|     hintElement.appendChild(hintTextElement);
 | |
| 
 | |
|     labelElement.appendChild(hintElement);
 | |
|   }
 | |
| 
 | |
|   return labelElement;
 | |
| };
 | |
| 
 | |
| const buildNumberInput = (schema, parentState, key) => {
 | |
|   const input = document.createElement("input");
 | |
|   input.classList.add("base-input");
 | |
|   input.type = "number";
 | |
|   input.value = parentState[key];
 | |
| 
 | |
|   const min = schema.min || 0;
 | |
|   input.min = min;
 | |
| 
 | |
|   input.addEventListener("change", () => {
 | |
|     const normalizedValue = parseFloat(input.value, 10);
 | |
|     parentState[key] = Math.max(normalizedValue, min);
 | |
|   });
 | |
| 
 | |
|   return input;
 | |
| };
 | |
| 
 | |
| const buildBooleanInput = (parentState, key) => {
 | |
|   const input = document.createElement("input");
 | |
|   input.classList.add("base-input");
 | |
|   input.type = "checkbox";
 | |
|   input.checked = parentState[key] || false;
 | |
| 
 | |
|   input.addEventListener("change", () => {
 | |
|     parentState[key] = input.checked;
 | |
|   });
 | |
| 
 | |
|   return input;
 | |
| };
 | |
| 
 | |
| const buildStringInput = (parentState, key) => {
 | |
|   const input = document.createElement("input");
 | |
|   input.classList.add("base-input");
 | |
|   input.type = "text";
 | |
|   input.value = parentState[key] || "";
 | |
| 
 | |
|   input.addEventListener("change", () => {
 | |
|     parentState[key] = input.value;
 | |
|   });
 | |
| 
 | |
|   return input;
 | |
| };
 | |
| 
 | |
| const defaultValueForType = (type) => {
 | |
|   switch (type) {
 | |
|     case "number":
 | |
|       return 0;
 | |
|     case "boolean":
 | |
|       return false;
 | |
|     case "string":
 | |
|       return "";
 | |
|     case "array":
 | |
|       return [];
 | |
|     case "object":
 | |
|       return {};
 | |
|   }
 | |
| 
 | |
|   return null;
 | |
| };
 | |
| 
 | |
| const arrayFormElementDecorator = (childElement, parentState, index) => {
 | |
|   const decoratedElement = document.createElement("div");
 | |
|   decoratedElement.classList.add("array-form-element-decorator");
 | |
| 
 | |
|   const removeButton = document.createElement("button");
 | |
|   removeButton.innerHTML = "X";
 | |
|   removeButton.classList.add("array-input", "array-input-delete", "button");
 | |
|   removeButton.addEventListener("click", () => {
 | |
|     parentState.splice(index, 1);
 | |
|     rerender();
 | |
|   });
 | |
| 
 | |
|   decoratedElement.appendChild(childElement);
 | |
|   decoratedElement.appendChild(removeButton);
 | |
| 
 | |
|   return decoratedElement;
 | |
| };
 | |
| 
 | |
| const buildArrayInput = (schema, parentState) => {
 | |
|   const itemType = schema.items.type;
 | |
|   const inputControlsDiv = document.createElement("div");
 | |
|   inputControlsDiv.classList.add("array-input-controls");
 | |
| 
 | |
|   const addButton = document.createElement("button");
 | |
|   addButton.innerHTML = "Add One";
 | |
|   addButton.classList.add("array-input", "button");
 | |
|   addButton.addEventListener("click", () => {
 | |
|     parentState.push(defaultValueForType(itemType));
 | |
|     rerender();
 | |
|   });
 | |
| 
 | |
|   const removeButton = document.createElement("button");
 | |
|   removeButton.innerHTML = "Delete All";
 | |
|   removeButton.classList.add("array-input", "array-input-delete", "button");
 | |
|   removeButton.addEventListener("click", () => {
 | |
|     parentState.splice(0, parentState.length);
 | |
|     rerender();
 | |
|   });
 | |
| 
 | |
|   inputControlsDiv.appendChild(addButton);
 | |
|   inputControlsDiv.appendChild(removeButton);
 | |
| 
 | |
|   return inputControlsDiv;
 | |
| };
 | |
| 
 | |
| const buildUnknownInput = () => {
 | |
|   const disclaimer = document.createElement("div");
 | |
|   disclaimer.innerHTML = `<i class="unknown-input">This configuration is not yet supported</i>`;
 | |
| 
 | |
|   return disclaimer;
 | |
| };
 | |
| 
 | |
| const render = (state, schema) => {
 | |
|   const build = (
 | |
|     schema,
 | |
|     state,
 | |
|     parentState,
 | |
|     currentKey = "",
 | |
|     path = "configuration"
 | |
|   ) => {
 | |
|     const parent = document.createElement("div");
 | |
|     parent.classList.add("form-element");
 | |
| 
 | |
|     const { type, label, hint, fields, items } = schema;
 | |
| 
 | |
|     if (label) {
 | |
|       parent.appendChild(buildLabel(type, label, hint));
 | |
|     }
 | |
| 
 | |
|     parent.id = path;
 | |
| 
 | |
|     if (type === "object") {
 | |
|       const entries = Object.entries(fields);
 | |
|       entries.forEach(([key, value]) => {
 | |
|         if (!state[key]) {
 | |
|           state[key] = defaultValueForType(value.type);
 | |
|         }
 | |
| 
 | |
|         const childElement = build(
 | |
|           value,
 | |
|           state[key],
 | |
|           state,
 | |
|           key,
 | |
|           `${path}.${key}`
 | |
|         );
 | |
|         parent.appendChild(childElement);
 | |
|       });
 | |
|     } else if (type === "array") {
 | |
|       const arrayInputControls = buildArrayInput(schema, state);
 | |
|       parent.appendChild(arrayInputControls);
 | |
| 
 | |
|       if (state && state.length > 0) {
 | |
|         state.forEach((element, index) => {
 | |
|           const childElement = build(
 | |
|             items,
 | |
|             element,
 | |
|             state,
 | |
|             index,
 | |
|             `${path}[${index}]`
 | |
|           );
 | |
| 
 | |
|           const decoratedChildElement = arrayFormElementDecorator(
 | |
|             childElement,
 | |
|             state,
 | |
|             index
 | |
|           );
 | |
|           parent.appendChild(decoratedChildElement);
 | |
|         });
 | |
|       }
 | |
|     } else if (type === "number") {
 | |
|       parent.appendChild(buildNumberInput(schema, parentState, currentKey));
 | |
|       parent.classList.add("input-label");
 | |
|     } else if (type === "string") {
 | |
|       parent.appendChild(buildStringInput(parentState, currentKey));
 | |
|       parent.classList.add("input-label");
 | |
|     } else if (type === "boolean") {
 | |
|       parent.appendChild(buildBooleanInput(parentState, currentKey));
 | |
|       parent.classList.add("input-label");
 | |
|     } else {
 | |
|       parent.appendChild(buildUnknownInput());
 | |
|     }
 | |
| 
 | |
|     return parent;
 | |
|   };
 | |
| 
 | |
|   return build(schema, state, state);
 | |
| };
 | |
| 
 | |
| function rerender() {
 | |
|   const root = document.querySelector("#root");
 | |
|   root.innerHTML = "";
 | |
|   root?.append(render(state, schema));
 | |
| }
 | |
| 
 | |
| window.onload = async () => {
 | |
|   const schemaResponse = await fetch("/configuration/schema");
 | |
|   const dataResponse = await fetch("/configuration");
 | |
| 
 | |
|   const schemaResponseJson = await schemaResponse.json();
 | |
|   const dataResponseJson = await dataResponse.json();
 | |
| 
 | |
|   const { data: formSchema } = schemaResponseJson;
 | |
|   const { data: initialData } = dataResponseJson;
 | |
| 
 | |
|   state = initialData;
 | |
|   schema = formSchema;
 | |
| 
 | |
|   rerender();
 | |
| 
 | |
|   const saveButton = document.querySelector("#save");
 | |
| 
 | |
|   saveButton?.addEventListener("click", async () => {
 | |
|     if (saveButton.disabled) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     saveButton.innerHTML = "Saving...";
 | |
|     saveButton.disabled = true;
 | |
|     const response = await fetch("/configuration", {
 | |
|       method: "PATCH",
 | |
|       headers: {
 | |
|         "Content-Type": "application/json",
 | |
|       },
 | |
|       body: JSON.stringify({
 | |
|         configuration: state,
 | |
|       }),
 | |
|     });
 | |
|     if (response.status === 200) {
 | |
|       saveButton.innerHTML = "Saved!";
 | |
|       saveButton.classList.add("good");
 | |
|     } else {
 | |
|       saveButton.innerHTML = "Failed!";
 | |
|       saveButton.classList.add("bad");
 | |
|     }
 | |
|     setTimeout(() => {
 | |
|       saveButton.innerHTML = "Save Changes";
 | |
|       saveButton.classList.remove("good");
 | |
|       saveButton.classList.remove("bad");
 | |
|       saveButton.disabled = false;
 | |
|     }, 3000);
 | |
|   });
 | |
| };
 |