feat: initial commit
This commit is contained in:
commit
5c2b5fe673
21 changed files with 3723 additions and 0 deletions
24
.gitignore
vendored
Normal file
24
.gitignore
vendored
Normal file
|
@ -0,0 +1,24 @@
|
|||
# Logs
|
||||
logs
|
||||
*.log
|
||||
npm-debug.log*
|
||||
yarn-debug.log*
|
||||
yarn-error.log*
|
||||
pnpm-debug.log*
|
||||
lerna-debug.log*
|
||||
|
||||
node_modules
|
||||
dist
|
||||
dist-ssr
|
||||
*.local
|
||||
|
||||
# Editor directories and files
|
||||
.vscode/*
|
||||
!.vscode/extensions.json
|
||||
.idea
|
||||
.DS_Store
|
||||
*.suo
|
||||
*.ntvs*
|
||||
*.njsproj
|
||||
*.sln
|
||||
*.sw?
|
22
index.html
Normal file
22
index.html
Normal file
|
@ -0,0 +1,22 @@
|
|||
<!DOCTYPE html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="UTF-8" />
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||
<meta name="description" content="Secret santa name drawing" />
|
||||
<title>SecSan</title>
|
||||
<link rel="icon" href="/favicon.png" />
|
||||
<link
|
||||
rel="apple-touch-icon"
|
||||
href="/apple-touch-icon.png"
|
||||
sizes="180x180"
|
||||
/>
|
||||
<meta name="theme-color" content="#ffffff" />
|
||||
<link rel="preconnect" href="https://fonts.bunny.net">
|
||||
<link href="https://fonts.bunny.net/css?family=ubuntu:500" rel="stylesheet" />
|
||||
</head>
|
||||
<body>
|
||||
<div id="app"></div>
|
||||
<script type="module" src="/src/main.tsx"></script>
|
||||
</body>
|
||||
</html>
|
24
package.json
Normal file
24
package.json
Normal file
|
@ -0,0 +1,24 @@
|
|||
{
|
||||
"name": "secsan",
|
||||
"private": true,
|
||||
"version": "0.0.0",
|
||||
"type": "module",
|
||||
"scripts": {
|
||||
"dev": "vite",
|
||||
"build": "tsc && vite build",
|
||||
"preview": "vite preview",
|
||||
"test": "vitest"
|
||||
},
|
||||
"dependencies": {
|
||||
"@preact/signals": "^1.1.2",
|
||||
"preact": "^10.11.2"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@preact/preset-vite": "^2.4.0",
|
||||
"@types/node": "^18.11.9",
|
||||
"typescript": "^4.6.4",
|
||||
"vite": "^3.2.3",
|
||||
"vite-plugin-pwa": "^0.13.3",
|
||||
"vitest": "^0.25.1"
|
||||
}
|
||||
}
|
3188
pnpm-lock.yaml
Normal file
3188
pnpm-lock.yaml
Normal file
File diff suppressed because it is too large
Load diff
BIN
public/apple-touch-icon.png
Normal file
BIN
public/apple-touch-icon.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 13 KiB |
BIN
public/favicon-192.png
Normal file
BIN
public/favicon-192.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 14 KiB |
BIN
public/favicon.png
Normal file
BIN
public/favicon.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 14 KiB |
1
public/vite.svg
Normal file
1
public/vite.svg
Normal file
|
@ -0,0 +1 @@
|
|||
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" aria-hidden="true" role="img" class="iconify iconify--logos" width="31.88" height="32" preserveAspectRatio="xMidYMid meet" viewBox="0 0 256 257"><defs><linearGradient id="IconifyId1813088fe1fbc01fb466" x1="-.828%" x2="57.636%" y1="7.652%" y2="78.411%"><stop offset="0%" stop-color="#41D1FF"></stop><stop offset="100%" stop-color="#BD34FE"></stop></linearGradient><linearGradient id="IconifyId1813088fe1fbc01fb467" x1="43.376%" x2="50.316%" y1="2.242%" y2="89.03%"><stop offset="0%" stop-color="#FFEA83"></stop><stop offset="8.333%" stop-color="#FFDD35"></stop><stop offset="100%" stop-color="#FFA800"></stop></linearGradient></defs><path fill="url(#IconifyId1813088fe1fbc01fb466)" d="M255.153 37.938L134.897 252.976c-2.483 4.44-8.862 4.466-11.382.048L.875 37.958c-2.746-4.814 1.371-10.646 6.827-9.67l120.385 21.517a6.537 6.537 0 0 0 2.322-.004l117.867-21.483c5.438-.991 9.574 4.796 6.877 9.62Z"></path><path fill="url(#IconifyId1813088fe1fbc01fb467)" d="M185.432.063L96.44 17.501a3.268 3.268 0 0 0-2.634 3.014l-5.474 92.456a3.268 3.268 0 0 0 3.997 3.378l24.777-5.718c2.318-.535 4.413 1.507 3.936 3.838l-7.361 36.047c-.495 2.426 1.782 4.5 4.151 3.78l15.304-4.649c2.372-.72 4.652 1.36 4.15 3.788l-11.698 56.621c-.732 3.542 3.979 5.473 5.943 2.437l1.313-2.028l72.516-144.72c1.215-2.423-.88-5.186-3.54-4.672l-25.505 4.922c-2.396.462-4.435-1.77-3.759-4.114l16.646-57.705c.677-2.35-1.37-4.583-3.769-4.113Z"></path></svg>
|
After Width: | Height: | Size: 1.5 KiB |
188
src/app.css
Normal file
188
src/app.css
Normal file
|
@ -0,0 +1,188 @@
|
|||
:root {
|
||||
--background: #13092b;
|
||||
--background-transparent: #13092baa;
|
||||
--background-2: #090f2b;
|
||||
--foreground: #ddd;
|
||||
--foreground-bright: #fff;
|
||||
--foreground-dim: #aaa;
|
||||
--error: orange;
|
||||
--accent: #f9027a;
|
||||
--accent-dim: hsl(331, 50%, 49%);
|
||||
--primary: #8da1ee;
|
||||
--confirm: #82f1b1;
|
||||
--warning: #fb7185;
|
||||
}
|
||||
|
||||
body {
|
||||
font-family: 'Ubuntu', 'Noto Sans', sans-serif;
|
||||
font-size: 24px;
|
||||
background-color: var(--background);
|
||||
background-image: url('https://ashhhleyyy.dev/assets-gen/background.svg');
|
||||
color: var(--foreground);
|
||||
margin: 0;
|
||||
box-sizing: border-box;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
flex-direction: column;
|
||||
min-height: 100vh;
|
||||
max-width: 100vw;
|
||||
}
|
||||
|
||||
* {
|
||||
box-sizing: inherit;
|
||||
}
|
||||
|
||||
@media (prefers-reduced-motion: no-preference) {
|
||||
html {
|
||||
scroll-behavior: smooth;
|
||||
}
|
||||
}
|
||||
|
||||
main {
|
||||
margin: 16px;
|
||||
padding: 8px;
|
||||
width: calc(100vw - 32px);
|
||||
max-width: 720px;
|
||||
flex: 1;
|
||||
background-color: var(--background-transparent);
|
||||
}
|
||||
|
||||
a {
|
||||
color: inherit;
|
||||
font-family: inherit;
|
||||
}
|
||||
|
||||
h1 {
|
||||
margin: 0;
|
||||
display: flex;
|
||||
flex-direction: row;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
h1::before {
|
||||
content: '# ';
|
||||
margin-right: 16px;
|
||||
color: var(--foreground-dim);
|
||||
font-family: 'JetBrains Mono', 'Oxygen Mono', monospace;
|
||||
}
|
||||
|
||||
h2::before {
|
||||
content: '## ';
|
||||
color: var(--foreground-dim);
|
||||
font-family: 'JetBrains Mono', 'Oxygen Mono', monospace;
|
||||
}
|
||||
|
||||
h3::before {
|
||||
content: '### ';
|
||||
color: var(--foreground-dim);
|
||||
font-family: 'JetBrains Mono', 'Oxygen Mono', monospace;
|
||||
}
|
||||
|
||||
h4::before {
|
||||
content: '#### ';
|
||||
color: var(--foreground-dim);
|
||||
font-family: 'JetBrains Mono', 'Oxygen Mono', monospace;
|
||||
}
|
||||
|
||||
h5::before {
|
||||
content: '##### ';
|
||||
color: var(--foreground-dim);
|
||||
font-family: 'JetBrains Mono', 'Oxygen Mono', monospace;
|
||||
}
|
||||
|
||||
h6::before {
|
||||
content: '###### ';
|
||||
color: var(--foreground-dim);
|
||||
font-family: 'JetBrains Mono', 'Oxygen Mono', monospace;
|
||||
}
|
||||
|
||||
button {
|
||||
border: none;
|
||||
padding: 8px;
|
||||
border-radius: 8px;
|
||||
cursor: pointer;
|
||||
text-decoration: none;
|
||||
font-size: initial;
|
||||
color: black;
|
||||
transition: filter 200ms ease;
|
||||
vertical-align: middle;
|
||||
font-family: inherit;
|
||||
}
|
||||
|
||||
button:hover {
|
||||
filter: brightness(.8);
|
||||
}
|
||||
|
||||
button.primary {
|
||||
background-color: var(--primary);
|
||||
}
|
||||
|
||||
button.confirm {
|
||||
background-color: var(--confirm);
|
||||
}
|
||||
|
||||
button.warning {
|
||||
background-color: var(--warning);
|
||||
}
|
||||
|
||||
button:disabled {
|
||||
filter: saturate(25%) brightness(.85);
|
||||
cursor: not-allowed;
|
||||
}
|
||||
|
||||
a.primary {
|
||||
color: var(--primary);
|
||||
}
|
||||
|
||||
a.confirm {
|
||||
color: var(--confirm);
|
||||
}
|
||||
|
||||
a.warning {
|
||||
color: var(--warning);
|
||||
}
|
||||
|
||||
input {
|
||||
margin: 8px 0;
|
||||
background-color: rgba(128, 128, 128, .25);
|
||||
color: white;
|
||||
border: #888 2px solid;
|
||||
padding: 8px;
|
||||
border-radius: 8px;
|
||||
font-family: inherit;
|
||||
font-size: medium;
|
||||
}
|
||||
|
||||
input[type="text"]:focus-visible,
|
||||
input[type="password"]:focus-visible {
|
||||
outline: none;
|
||||
border-color: var(--accent);
|
||||
}
|
||||
|
||||
.button-row {
|
||||
display: flex;
|
||||
width: 100%;
|
||||
flex-direction: row;
|
||||
flex-wrap: wrap;
|
||||
gap: 8px;
|
||||
margin: 8px 0;
|
||||
}
|
||||
|
||||
.button-row button {
|
||||
flex: 1;
|
||||
}
|
||||
|
||||
.button-group {
|
||||
display: flex;
|
||||
width: 100%;
|
||||
flex-direction: column;
|
||||
gap: 8px;
|
||||
margin: 8px 0;
|
||||
}
|
||||
|
||||
.drawn-name {
|
||||
font-size: 72px;
|
||||
width: 100%;
|
||||
display: grid;
|
||||
place-content: center;
|
||||
}
|
24
src/app.tsx
Normal file
24
src/app.tsx
Normal file
|
@ -0,0 +1,24 @@
|
|||
import { signal } from '@preact/signals';
|
||||
import { useState } from 'preact/hooks';
|
||||
import './app.css';
|
||||
import DrawName from './pages/DrawName';
|
||||
import EnterNames from './pages/EnterNames';
|
||||
import { buildNameMap } from './shuffle';
|
||||
|
||||
const names = signal<string[]>([]);
|
||||
|
||||
export function App() {
|
||||
const [shuffled, setShuffled] = useState<[string, string][] | null>(null);
|
||||
|
||||
function start() {
|
||||
setShuffled(buildNameMap(names.value));
|
||||
}
|
||||
|
||||
return (
|
||||
<main>
|
||||
<h1>Secret Santa</h1>
|
||||
|
||||
{shuffled ? <DrawName names={shuffled} /> : <EnterNames start={start} names={names} />}
|
||||
</main>
|
||||
)
|
||||
}
|
0
src/index.css
Normal file
0
src/index.css
Normal file
5
src/main.tsx
Normal file
5
src/main.tsx
Normal file
|
@ -0,0 +1,5 @@
|
|||
import { render } from 'preact'
|
||||
import { App } from './app'
|
||||
import './index.css'
|
||||
|
||||
render(<App />, document.getElementById('app') as HTMLElement)
|
50
src/pages/DrawName.tsx
Normal file
50
src/pages/DrawName.tsx
Normal file
|
@ -0,0 +1,50 @@
|
|||
import { FunctionComponent } from "preact";
|
||||
import { useState } from "preact/hooks";
|
||||
import { useWindowEvent } from "../util";
|
||||
|
||||
interface Props {
|
||||
names: [string, string][];
|
||||
}
|
||||
|
||||
const DrawName: FunctionComponent<Props> = ({ names }) => {
|
||||
const [currentPerson, setCurrentPerson] = useState<number | null>(null);
|
||||
const [completed, setCompleted] = useState<number[]>([]);
|
||||
|
||||
useWindowEvent('click', (e) => {
|
||||
if (e.button !== 0) return;
|
||||
|
||||
if (currentPerson !== null) {
|
||||
setCompleted(completed => [...completed, currentPerson])
|
||||
setCurrentPerson(null);
|
||||
}
|
||||
});
|
||||
|
||||
if (currentPerson === null) {
|
||||
if (completed.length === names.length) {
|
||||
return <>
|
||||
<h2>
|
||||
All done!
|
||||
</h2>
|
||||
</>
|
||||
}
|
||||
return <>
|
||||
<div class="button-group">
|
||||
{names.map(([name, _], i) => {
|
||||
if (completed.includes(i)) {
|
||||
return null;
|
||||
} else {
|
||||
return <button key={i + "-" + name} class="primary" onClick={() => setCurrentPerson(i)}>{name}</button>;
|
||||
}
|
||||
})}
|
||||
</div>
|
||||
</>
|
||||
}
|
||||
|
||||
return <>
|
||||
<div class="drawn-name">
|
||||
{names[currentPerson][1]}
|
||||
</div>
|
||||
</>
|
||||
}
|
||||
|
||||
export default DrawName;
|
52
src/pages/EnterNames.tsx
Normal file
52
src/pages/EnterNames.tsx
Normal file
|
@ -0,0 +1,52 @@
|
|||
import { Signal } from "@preact/signals";
|
||||
import { FunctionComponent } from "preact";
|
||||
import { useState } from "preact/hooks";
|
||||
|
||||
interface Props {
|
||||
names: Signal<string[]>;
|
||||
start: () => void;
|
||||
}
|
||||
|
||||
const EnterNames: FunctionComponent<Props> = ({ names, start }) => {
|
||||
const [name, setName] = useState('');
|
||||
|
||||
function addName() {
|
||||
if (name === '') return;
|
||||
if (names.value.includes(name)) {
|
||||
alert(`${name} has already been added!`);
|
||||
return;
|
||||
}
|
||||
names.value = [...names.value, name];
|
||||
setName('');
|
||||
}
|
||||
|
||||
return <>
|
||||
<ul>
|
||||
{names.value.map((name, i) => <li key={i + "-" + name}>
|
||||
{name}
|
||||
</li>)}
|
||||
</ul>
|
||||
|
||||
<div>
|
||||
<input
|
||||
type="text"
|
||||
placeholder="Name..."
|
||||
value={name}
|
||||
onInput={(e) => setName((e.target! as HTMLInputElement).value)}
|
||||
onKeyPress={(e) => {
|
||||
if (e.key === 'Enter') {
|
||||
addName();
|
||||
e.preventDefault();
|
||||
}
|
||||
}}
|
||||
/>
|
||||
|
||||
<div class="button-row">
|
||||
<button class="primary" onClick={addName}>Add name</button>
|
||||
<button class='confirm' disabled={names.value.length <= 1} onClick={start}>Start drawing</button>
|
||||
</div>
|
||||
</div>
|
||||
</>
|
||||
}
|
||||
|
||||
export default EnterNames;
|
56
src/shuffle.ts
Normal file
56
src/shuffle.ts
Normal file
|
@ -0,0 +1,56 @@
|
|||
// https://stackoverflow.com/a/2450976
|
||||
function shuffle<T>(array: T[]) {
|
||||
let currentIndex = array.length, randomIndex;
|
||||
|
||||
// While there remain elements to shuffle.
|
||||
while (currentIndex != 0) {
|
||||
// Pick a remaining element.
|
||||
randomIndex = Math.floor(Math.random() * currentIndex);
|
||||
currentIndex--;
|
||||
|
||||
// And swap it with the current element.
|
||||
[array[currentIndex], array[randomIndex]] = [array[randomIndex], array[currentIndex]];
|
||||
}
|
||||
|
||||
return array;
|
||||
}
|
||||
|
||||
function checkShuffled<T>(old: T[], nw: T[]): boolean {
|
||||
if (old.length !== nw.length) {
|
||||
throw new Error('cannot compare arrays of non-equal length');
|
||||
}
|
||||
|
||||
if (old.length <= 1) {
|
||||
return true;
|
||||
}
|
||||
|
||||
for (let i = 0; i < old.length; i++) {
|
||||
if (old[i] === nw[i]) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
function nonEqualShuffle<T>(input: T[]): T[] {
|
||||
const nw = input.slice();
|
||||
|
||||
shuffle(nw);
|
||||
while (!checkShuffled(input, nw)) {
|
||||
shuffle(nw);
|
||||
}
|
||||
|
||||
return nw;
|
||||
}
|
||||
|
||||
export function buildNameMap(names: string[]): [person: string, target: string][] {
|
||||
const ret: [string, string][] = [];
|
||||
|
||||
const shuffled = nonEqualShuffle(names);
|
||||
for (let i = 0; i < shuffled.length; i++) {
|
||||
ret.push([names[i], shuffled[i]]);
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
8
src/util.ts
Normal file
8
src/util.ts
Normal file
|
@ -0,0 +1,8 @@
|
|||
import { useEffect } from "preact/hooks";
|
||||
|
||||
export function useWindowEvent<K extends keyof WindowEventMap>(type: K, listener: (this: Window, ev: WindowEventMap[K]) => any, options?: boolean | AddEventListenerOptions): void {
|
||||
useEffect(() => {
|
||||
window.addEventListener(type, listener, options);
|
||||
return () => window.removeEventListener(type, listener);
|
||||
});
|
||||
}
|
1
src/vite-env.d.ts
vendored
Normal file
1
src/vite-env.d.ts
vendored
Normal file
|
@ -0,0 +1 @@
|
|||
/// <reference types="vite/client" />
|
12
test/shuffle.test.ts
Normal file
12
test/shuffle.test.ts
Normal file
|
@ -0,0 +1,12 @@
|
|||
import { expect, test } from 'vitest';
|
||||
import { buildNameMap } from '../src/shuffle';
|
||||
|
||||
test('shuffle does not include matching pairs', () => {
|
||||
const names = Array(10).fill(null).map((_, i) => "Test " + i);
|
||||
for (let i = 0; i < 50000; i++) {
|
||||
const shuffled = buildNameMap(names);
|
||||
for (const pair of shuffled) {
|
||||
expect(pair[0]).not.toBe(pair[1]);
|
||||
}
|
||||
}
|
||||
});
|
22
tsconfig.json
Normal file
22
tsconfig.json
Normal file
|
@ -0,0 +1,22 @@
|
|||
{
|
||||
"compilerOptions": {
|
||||
"target": "ESNext",
|
||||
"useDefineForClassFields": true,
|
||||
"lib": ["DOM", "DOM.Iterable", "ESNext"],
|
||||
"allowJs": false,
|
||||
"skipLibCheck": true,
|
||||
"esModuleInterop": false,
|
||||
"allowSyntheticDefaultImports": true,
|
||||
"strict": true,
|
||||
"forceConsistentCasingInFileNames": true,
|
||||
"module": "ESNext",
|
||||
"moduleResolution": "Node",
|
||||
"resolveJsonModule": true,
|
||||
"isolatedModules": true,
|
||||
"noEmit": true,
|
||||
"jsx": "react-jsx",
|
||||
"jsxImportSource": "preact"
|
||||
},
|
||||
"include": ["src"],
|
||||
"references": [{ "path": "./tsconfig.node.json" }]
|
||||
}
|
9
tsconfig.node.json
Normal file
9
tsconfig.node.json
Normal file
|
@ -0,0 +1,9 @@
|
|||
{
|
||||
"compilerOptions": {
|
||||
"composite": true,
|
||||
"module": "ESNext",
|
||||
"moduleResolution": "Node",
|
||||
"allowSyntheticDefaultImports": true
|
||||
},
|
||||
"include": ["vite.config.ts"]
|
||||
}
|
37
vite.config.ts
Normal file
37
vite.config.ts
Normal file
|
@ -0,0 +1,37 @@
|
|||
import { defineConfig } from 'vite';
|
||||
import preact from '@preact/preset-vite';
|
||||
import { VitePWA } from 'vite-plugin-pwa';
|
||||
|
||||
// https://vitejs.dev/config/
|
||||
export default defineConfig({
|
||||
plugins: [
|
||||
preact(),
|
||||
VitePWA({
|
||||
registerType: 'autoUpdate',
|
||||
includeAssets: ['favicon.png', 'apple-touch-icon.png'],
|
||||
manifest: {
|
||||
name: 'SecSan',
|
||||
short_name: 'SecSan',
|
||||
description: 'Secret santa name drawing',
|
||||
theme_color: '#ffffff',
|
||||
icons: [
|
||||
{
|
||||
src: 'favicon.png',
|
||||
sizes: '256x256',
|
||||
type: 'image/png',
|
||||
},
|
||||
{
|
||||
src: 'favicon-192.png',
|
||||
sizes: '192x192',
|
||||
type: 'image/png',
|
||||
},
|
||||
],
|
||||
},
|
||||
}),
|
||||
],
|
||||
server: {
|
||||
hmr: {
|
||||
clientPort: parseInt(process.env.CLIENT_PORT || '5173'),
|
||||
},
|
||||
},
|
||||
});
|
Loading…
Reference in a new issue