This commit is contained in:
xbl
2025-10-22 11:51:33 +02:00
commit 94e1502aa2
14 changed files with 1684 additions and 0 deletions

57
app.py Executable file
View File

@ -0,0 +1,57 @@
from flask import Flask, render_template, request
import math
from profiles import profiles
app = Flask(__name__)
# --- Matching logic (server-side copy, used for initial render only) ---
def euclidean_distance(a, b):
return math.sqrt(sum((x - y) ** 2 for x, y in zip(a, b)))
@app.route("/")
def index():
return render_template("about.html")
@app.route("/inactive")
def inactive():
# Default user values (all 20). Client-side UI will live-update from there.
user_values = [20, 20, 20, 20, 20, 20]
sorted_matches = sorted(
profiles.items(), key=lambda p: euclidean_distance(user_values, p[1]['values'])
)
top_matches = sorted_matches[:3]
return render_template(
"inactive.html",
user=user_values,
top_matches=top_matches,
profiles=profiles,
)
@app.route("/profiles")
def profiles_page():
return render_template("profiles.html", profiles=profiles)
@app.route("/quiz", methods=["GET", "POST"])
def quiz():
if request.method == "POST":
user_values = [
int(request.form.get(f"point{i}", 20)) for i in range(1, 7)
]
sorted_matches = sorted(
profiles.items(),
key=lambda p: euclidean_distance(user_values, p[1]['values'])
)
top_matches = sorted_matches[:3]
return render_template("result.html",
user=user_values,
top_matches=top_matches,
profiles=profiles)
return render_template("quiz.html", profiles=profiles)
@app.route("/result")
def result():
return render_template("result.html", profiles=profiles)
if __name__ == "__main__":
app.run(host='localhost', port=1337, debug=False)

107
profiles.py Normal file
View File

@ -0,0 +1,107 @@
# Predefined radar profiles
profiles = {
"Alpenglow": {
"values": [25, 35, 45, 70, 50, 25],
"info_text0": "15%",
"info_text1": "<1%",
"info_text2": "Indica-dominant Hybrid",
"info_text3": "spicy-fruity, sour apple",
"info_text4": "relaxing, sleep promoting"
},
"Amnesia Haze": {
"values": [35, 65, 40, 35, 30, 35],
"info_text0": "20%",
"info_text1": "<1%",
"info_text2": "Sativa-dominant Hybrid",
"info_text3": "incense, citrus fruits",
"info_text4": "energizing, calming"
},
"Blue Dream": {
"values": [25, 35, 40, 41, 75, 35],
"info_text0": "15%",
"info_text1": "<1%",
"info_text2": "Sativa-dominant Hybrid",
"info_text3": "berries, sweet, slighty hazy",
"info_text4": "focusing, relaxing, slightly sleep promoting"
},
"Bubba Kush": {
"values": [30, 61, 40, 41, 31, 35],
"info_text0": "15%",
"info_text1": "<1%",
"info_text2": "Indica-dominant Hybrid",
"info_text3": "lavender, cocoa, roasted coffee",
"info_text4": "calming, relaxing"
},
"Cherry Valley Cake": {
"values": [30, 59, 65, 35, 30, 55],
"info_text0": "18%",
"info_text1": "<1%",
"info_text2": "Indica-dominant Hybrid",
"info_text3": "creamy, spicy, fermented cherry",
"info_text4": "inspring, relaxing, calming"
},
"Ciskei": {
"values": [70, 40, 35, 39, 41, 40],
"info_text0": "5%",
"info_text1": "7.5%",
"info_text2": "Sativa",
"info_text3": "herbaceous, cheesy, spicy",
"info_text4": "energizing, focusing"
},
"Cleopatra's Milk": {
"values": [30, 65, 45, 35, 35, 55],
"info_text0": "18%",
"info_text1": "<1%",
"info_text2": "Indica-dominant Hybrid",
"info_text3": "cake, oriental spices, fruit",
"info_text4": "inspring, calming"
},
"Grape Waves": {
"values": [25, 75, 45, 45, 35, 45],
"info_text0": "20%",
"info_text1": "<1%",
"info_text2": "Indica-dominant Hybrid",
"info_text3": "creamy, heavy, raisins",
"info_text4": "calming, inspiring"
},
"JBT Kush": {
"values": [30, 55, 50, 55, 45, 41],
"info_text0": "20%",
"info_text1": "<1%",
"info_text2": "Indica-dominant Hybrid",
"info_text3": "gassy (fuel), cheesy, pungent",
"info_text4": "calming, relaxing, sleep promoting"
},
"Mother's Choice": {
"values": [35, 35, 55, 35, 50, 61],
"info_text0": "15%",
"info_text1": "<1%",
"info_text2": "Indica-dominant Hybrid",
"info_text3": "spicy-sweet, candy",
"info_text4": "inspiring, focusing, relaxing"
},
"Royal Feast": {
"values": [25, 50, 55, 70, 55, 45],
"info_text0": "15%",
"info_text1": "<1%",
"info_text2": "Indica-dominant Hybrid",
"info_text3": "earthy, sour, heavy apple aroma",
"info_text4": "focusing, calming, sleep promoting"
},
"Strawberry Sundae": {
"values": [28, 32, 50, 42, 41, 47],
"info_text0": "15%",
"info_text1": "<1%",
"info_text2": "Indica-dominant Hybrid",
"info_text3": "fruity-creamy, strawberry yogurt",
"info_text4": "inspiring, relaxing"
},
"Wedding Cake": {
"values": [25, 70, 60, 50, 41, 50],
"info_text0": "20%",
"info_text1": "<1%",
"info_text2": "Indica-dominant Hybrid",
"info_text3": "creamy, lemon, vanilla, pastry",
"info_text4": "calming, relaxing, inspiring"
}
}

BIN
static/cic.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 71 KiB

BIN
static/cm.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 45 KiB

BIN
static/favicon.png Executable file

Binary file not shown.

After

Width:  |  Height:  |  Size: 21 KiB

168
static/js/inactive.js Executable file
View File

@ -0,0 +1,168 @@
document.addEventListener("DOMContentLoaded", () => {
const labels = ['Energy','Calm','Relax','Sleep','Focus','Inspire'];
let userData = [...window.appData.initialUserData];
const profiles = window.appData.profiles;
const youLineColor = 'rgba(255,255,255,0.69)';
const pointBgColor = 'rgba(0,0,0,0.1)';
const youFillColor = 'rgba(255,255,255,0.1)';
const matchLineColor = 'rgb(191,181,49)';
const matchFillColor = 'rgba(191,181,49,0.420)';
const radialLineColor= 'rgba(255,255,255,0.420)';
function distance(a,b){ return Math.sqrt(a.reduce((s,v,i)=>s+(v-b[i])**2,0)); }
function getTopMatches() {
return Object.entries(profiles)
.sort((a,b)=>distance(userData,a[1].values)-distance(userData,b[1].values))
.slice(0,3);
}
function getFontSize(){ return window.innerWidth < 600 ? 10 : 39; }
function getBottomChartFontSize(){ return window.innerWidth < 600 ? 8 : 24; }
function getPointRadius(){ return window.innerWidth < 600 ? 11 : 24; }
function getPointHoverRadius(){ return window.innerWidth < 600 ? 12 : 39; }
function getPointBorderColor(){ return '#BFB531'; }
// === Initialize charts ===
const charts = [1,2,3].map(i => {
const ctx = document.getElementById(`chart-${i}`).getContext('2d');
const isBest = i === 1;
const datasets = isBest ? [
{
label:'You',
data:[...userData],
fill:true,
borderColor: youLineColor,
backgroundColor: youFillColor,
pointBackgroundColor: pointBgColor,
pointBorderColor: getPointBorderColor(),
pointRadius: getPointRadius(),
pointHoverRadius: getPointHoverRadius()
},
{
label:'Match',
data:[],
fill:true,
borderColor: matchLineColor,
backgroundColor: matchFillColor,
pointBackgroundColor: matchLineColor,
pointRadius:0
}
] : [
{
label:'Match',
data:[],
fill:true,
borderColor: matchLineColor,
backgroundColor: matchFillColor,
pointBackgroundColor: matchLineColor,
pointRadius:0
}
];
return new Chart(ctx,{
type:'radar',
data:{ labels, datasets },
options:{
responsive:true,
maintainAspectRatio:false,
scales:{
r:{
min:0, max:80,
ticks:{ display:false, stepSize:20 },
pointLabels:{ font:{ size:isBest?getFontSize():getBottomChartFontSize(), weight:'bold' }, color:'#FFF' },
angleLines:{ color:radialLineColor },
grid:{ drawticks:false, drawOnChartArea:false, circular:false }
}
},
plugins:{
tooltip:{ enabled:false },
legend:{ display:false }
}
}
});
});
// === Update charts & DOM ===
function updateCharts(){
const matches = getTopMatches();
matches.forEach(([name, prof], idx)=>{
const chart = charts[idx];
if(idx===0){
chart.data.datasets[0].data = [...userData];
chart.data.datasets[1].data = prof.values;
chart.data.datasets[1].label = name;
} else {
chart.data.datasets[0].data = prof.values;
chart.data.datasets[0].label = name;
}
chart.options.scales.r.pointLabels.font.size = idx===0 ? getFontSize() : getBottomChartFontSize();
chart.update('none');
// Update main card
document.getElementById(`name-${idx+1}`).textContent = name;
for (let i = 0; i <= 4; i++) {
const el = document.getElementById(`info${i}-${idx+1}`);
if (el) el.textContent = prof[`info_text${i}`] || '';
}
// Update popup
const popupName = document.getElementById(`popup-name-${idx+1}`);
if (popupName) popupName.textContent = name;
for (let i = 0; i <= 4; i++) {
const popupEl = document.getElementById(`popup-info${i}-${idx+1}`);
if (popupEl) popupEl.textContent = prof[`info_text${i}`] || '';
}
});
}
updateCharts();
// === Reset button ===
document.getElementById('resetBtn').addEventListener('click', ()=>{
userData = [...window.appData.initialUserData];
updateCharts();
});
// === Dragging top chart ===
const topChart = charts[0];
const topCanvas = document.getElementById('chart-1');
topCanvas.style.touchAction = 'none';
let draggingIndex = null;
topCanvas.addEventListener('pointerdown', e => {
const points = topChart.getElementsAtEventForMode(e, 'nearest', {intersect:true}, true);
if(points.length && points[0].datasetIndex===0) draggingIndex = points[0].index;
});
topCanvas.addEventListener('pointermove', e => {
if(draggingIndex===null) return;
const rect = topCanvas.getBoundingClientRect();
const x = e.clientX - rect.left;
const y = e.clientY - rect.top;
const cx = rect.width/2;
const cy = rect.height/2;
const dx = x-cx;
const dy = cy-y;
const dist = Math.sqrt(dx*dx + dy*dy);
const rMax = Math.min(rect.width, rect.height)/2 * 0.9;
let value = (dist/rMax)*80;
value = Math.min(Math.max(Math.round(value),0),80);
userData[draggingIndex] = value;
updateCharts();
});
topCanvas.addEventListener('pointerup', ()=> draggingIndex=null);
topCanvas.addEventListener('pointerleave', ()=> draggingIndex=null);
// bottom charts scroll normally
[2,3].forEach(i => {
document.getElementById(`chart-${i}`).style.touchAction='auto';
});
window.addEventListener('resize', updateCharts);
});

16
static/js/popup.js Executable file
View File

@ -0,0 +1,16 @@
function togglePopup(event, id) {
event.stopPropagation();
const popup = document.getElementById(id);
if (popup.classList.contains('active')) {
popup.classList.remove('active');
} else {
document.querySelectorAll('.popup').forEach(p => p.classList.remove('active'));
popup.classList.add('active');
}
}
// Close on outside click
document.addEventListener('click', () => {
document.querySelectorAll('.popup').forEach(p => p.classList.remove('active'));
});

BIN
static/scc.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 414 KiB

687
static/style.css Normal file
View File

@ -0,0 +1,687 @@
/* === Reset & Base Styles === */
* { box-sizing: border-box; }
html, body {
margin: 0;
padding: 0;
height: 100%;
width: 100%;
font-family: Arial, sans-serif;
color: #FFFFFF;
overflow-x: hidden;
}
h1, h2, h3 {
line-height: 1.3;
font-weight: bold;
text-align: center;
margin: 0 0 0 0;
}
h1 {
font-size: 2.25rem;
}
.top-match h2 {
font-size: 4rem;
}
.top-match h3 {
margin-top: 0.25em;
font-size: 2rem;
}
.bottom-row h3 {
margin-top: 0.25em;
font-size: 1.5rem;
}
.top-match p {
font-size: 1.5rem;
line-height: 1;
margin: 0.5em 0 0.5em;
color: #FFFFFF;
}
.bottom-row h2 {
font-size: 3rem;
}
.bottom-row p {
font-size: 1.25rem;
margin: 0 0 0;
}
.info-block {
#border: 3px solid #BFB531;
border-radius: 25px;
padding: 0.5em 0.5em;
margin: 0.5em 4em;
background-color: rgba(191,181,49,0.1);
text-align: center;
box-sizing: border-box;
}
.profile-info-block {
width: 100%;
}
/* ABOUT PAGE*/
.aboutbody {
display: flex;
flex-direction: column;
}
.abouttext a, p {
font-size: 1.25rem;
text-align: center;
text-decoration: none;
}
.abouttextcontainer {
margin-top: 3vh;
margin-bottom: 3vh;
}
.aboutlink {
color: #BFB531;
}
.made {
font-weight: bold;
color: #FF6600;
}
.aboutlogocontainer {
display: flex;
justify-content: center;
align-items: center;
}
.aboutlogo {
display: flex;
justify-content: center;
align-items: center;
overflow: hidden;
flex: 1;
max-height: 150px;
max-width: 95%;
width: auto;
margin-top: 1em;
margin-bottom: 3em;
}
.aboutlogo img {
max-height: 100%;
max-width: 100%;
object-fit: contain;
}
input[type="hidden"] { display: none; }
/* === Body & Sticky Footer === */
body {
display: flex;
flex-direction: column;
min-height: 95vh; /* ensures footer sticks at bottom */
background: #1A202C;
}
/* === Page container fills remaining space === */
.page-container {
flex: 1;
display: flex;
flex-direction: column;
width: 100%;
padding: 20px;
}
/* === Logo === */
.logo-container {
display: flex;
justify-content: center;
align-items: center;
}
.logo {
width: 100%;
height: auto;
max-width: 480px;
}
/* === Grid Layout (profiles page) === */
.profile-grid {
display: grid;
padding: 20px;
grid-template-columns: repeat(auto-fit, minmax(420px, 420px));
gap: 25px;
width: 100%;
margin: 0 auto;
justify-content: center;
}
.profile-grid canvas {
width: 100%;
max-width: 420px;
height: auto;
max-height: 420px;
display: block;
object-fit: contain;
}
/* === Footer === */
.footer {
padding: 15px;
flex-shrink: 0; /* ensure footer doesn't shrink */
text-align: center;
}
.footer p {
font-size: 1rem;
color: #FFFFFF;
}
/* === Cards === */
.profile-card {
background: rgba(0,0,0,0.39);
border-radius: 8px;
box-shadow: 0 2px 6px rgba(0,0,0,0.12);
display: flex;
flex-direction: column;
width: 100%;
}
.profile-profile-card {
padding: 1em;
flex-direction: column;
align-items: center;
background: rgba(0,0,0,0.39);
border-radius: 8px;
box-shadow: 0 2px 6px rgba(0,0,0,0.12);
display: flex;
max-width: 420px;
max-height: 690px;
}
.profile-profile-card canvas {
display: block;
width: 100%;
height: auto;
max-width: 420px;
max-height: 420px;
margin: 1em auto;
}
/* Canvas sizing */
.profile-card canvas {
margin: 3em 0;
touch-action: pan-y;
#height: auto;
max-height: 840px;
margin: 0.5em;
display: block;
object-fit: contain;
z-index: 0;
}
/* === Matches container === */
.matches-container {
display: flex;
flex-direction: column;
gap: 18px;
align-items: stretch;
width: 100%;
}
/* Top row: full width */
.best-row {
width: 100%;
}
/* Bottom row: use CSS grid for auto-equal height columns */
.bottom-row {
display: grid;
grid-template-columns: 1fr; /* mobile default: one column */
gap: 15px;
width: 100%;
justify-content: center;
#align-items: stretch;
}
/* Profile-card inside grid cells fill the cell */
.bottom-row .profile-card {
width: 100%;
height: auto;
align-items: stretch;
display: flex;
flex-direction: column;
text-align: center;
}
/* Top match card styling */
.best-card {
border: 2px solid #BFB531;
box-shadow: 0 6px 24px rgba(0,0,0,0.18);
width: 100%;
display: flex;
flex-direction: column;
}
/* Side match default layout (stacked) */
.side-match {
width: 100%;
display: flex;
flex-direction: column;
}
/* === Buttons === */
button[type="submit"],
button.magenta-button {
display: block;
margin: 16px 0;
padding: 11px 22px;
font-size: 1em;
font-weight: 700;
color: rgba(0,0,0,1);
border: none;
border-radius: 8px;
cursor: pointer;
box-shadow: 0 4px 10px rgba(0,0,0,0.18);
transition: background-color 0.18s ease, transform 0.12s ease;
}
button[type="submit"] { background-color: #808080; }
button[type="submit"]:hover { background-color: #FFFFFF; transform: translateY(1px); }
button[type="submit"]:active { background-color: #000000; transform: translateY(0); }
button.magenta-button { background-color: #BFB531; }
button.magenta-button:hover { background-color: #9E9928; transform: translateY(1px); }
button.magenta-button:active { background-color: #E6E66B; transform: translateY(0); }
.button-row {
display: flex;
justify-content: center;
gap: 12px;
flex-wrap: wrap;
margin: 0.5em 0;
width: 100%;
}
/* Popup hidden initially */
.popup {
display: none;
position: fixed;
inset: 0;
background: rgba(0,0,0,0.6);
z-index: 9999;
justify-content: center;
align-items: center;
}
/* Show when active */
.popup.active {
display: flex;
}
.popup-content {
background: #262626;
padding: 1.5em 0;
border-radius: 12px;
border: 2px solid #BFB531;
max-width: 500px;
width: 90%;
color: #FFF;
text-align: center;
box-shadow: 0 8px 28px rgba(0,0,0,0.5);
font-size: 2rem;
}
.glow-text {
text-shadow:
0 0 5px #BFB531,
0 0 10px #BFB531,
0 0 20px #BFB531,
0 0 40px #BFB531;
}
.glow-info {
padding-bottom: 0.25em;
text-shadow:
0 0 5px #000000,
0 0 10px #000000,
0 0 20px #000000,
0 0 40px #000000;
}
.info-text {
padding: 0.1em 0;
margin: 0.1em 0;
}
.popup-content h3 {
font-size: 3rem;
margin: 0.5em;
}
.popup-content h4 {
font-size: 2rem;
margin: 0.5em 0 0 0;
color: rgba(181,191,49,0.42);
}
.popup-content p {
font-size: 1.5rem;
margin: 0 0 1em 0;
}
/* Circle "i" info icon */
.info-icon {
display: inline-flex;
justify-content: center;
align-items: center;
width: 69px;
height: 69px;
border-radius: 50%;
background-color: rgba(191,181,49,0.25); /* golden circle */
color: #BFB531;
font-weight: bold;
cursor: pointer;
position: relative;
font-size: 42px;
}
.profile-info-icon {
width: 42px;
height: 42px;
font-size: 2rem;
}
.bottom-row .info-icon {
width: 32px; /* smaller circle */
height: 32px;
margin-right: 0;
}
.bottom-row .info-icon::before {
font-size: 24px;
}
.info-icon::before {
content: "i"; /* the actual "i" */
font-family: Arial, sans-serif;
line-height: 1;
font-weight: 900;
}
/* Hover effect */
.info-icon:hover {
background-color: #FFF;
}
.title-with-icon {
display: block;
#align-items: center;
margin: 0.25em 1em 0.25em 1em;
}
/* === Animations === */
@keyframes gradientMove {
0% { background-position: 0% 50%; }
50% { background-position: 100% 50%; }
100% { background-position: 0% 50%; }
}
/* === Small mobile tweaks (≤600px) === */
@media (max-width: 600px) {
.profile-card {
width: 100%;
max-width: 100%;
padding: 0.5em;
margin: 0;
}
.profile-card canvas {
max-height: 360px;
}
.top-match h2, .bottom-row h2 {
font-size: 1.5rem;
}
.top-match h3, .bottom-row h3 {
font-size: 1rem;
margin: 0;
padding: 0;
}
.top-match p, .bottom-row p {
font-size: 1rem;
}
.info-block {
padding: 0.25em 2em;
margin: 0.25em;
}
.button-row {
gap: 5px;
}
button.magenta-button {
padding: 5px 10px;
}
input[type="hidden"] { display: none; }
.info-icon {
width: 32px;
height: 32px;
}
.info-icon::before {
font-size: 17px;
}
.bottom-row .info-icon {
width: 24px;
height: 24px;
}
.bottom-row .info-icon::before {
font-size: 13px;
}
.popup-content h3 {
font-size: 2rem;
margin: 0.5em;
}
.popup-content h4 {
font-size: 1.5rem;
margin: 0.5em 0 0 0;
color: rgba(181,191,49,0.42);
}
.popup-content p {
font-size: 1rem;
margin: 0 0 1em 0;
}
.cards-container {
grid-template-columns: 1fr;
align-items: center;
}
.footer p {
font-size: 0.5rem;
}
}
/* === Large screen width (≥1000px) === */
@media (min-width: 1000px) {
/* Bottom row: two equal columns */
.bottom-row {
grid-template-columns: repeat(2, 1fr);
justify-content: center;
}
/* Top card canvas larger */
.best-card canvas {
max-height: 50vh;
}
/* Side match canvas */
.bottom-row .profile-card canvas {
}
.cards-container {
grid-template-columns: repeat(2, 1fr);
align-items: center;
}
}
/* === Landscape Center Hero + Sidecards layout === */
@media only screen and (orientation: landscape) {
.aboutbody {
flex-direction: row;
align-items: flex-start;
justify-content: center;
}
.abouttextcontainer {
min-width: 500px;
}
.matches-container {
display: grid;
grid-template-columns: 1fr 2fr 1fr; /* left, center, right */
gap: 20px;
align-items: start;
}
.footer {
display: flex;
justify-content: center;
left: 0;
width: 100%;
bottom: 0;
position: fixed;
}
.best-row {
grid-column: 2;
grid-row: 1;
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
}
.best-card {
max-width: 50vw;
width: 100%;
}
.info-icon {
height: 39px;
width: 39px;
font-size: 1.5rem;
}
.title-with-icon h2 {
font-size: 2rem;
}
.top-match p {
font-size: 1.25rem;
}
.title-with-icon p {
font-size: 1.25rem;
}
.profile-card h3 {
font-size: 1.5rem;
}
/* Place children explicitly */
.best-row {
grid-column: 2;
grid-row: 1;
display: flex;
flex-direction: column;
justify-content: center;
margin: 0;
padding: 0;
}
.bottom-row {
display: contents; /* flatten so its children can go left/right */
}
#card-2 {
grid-column: 1;
grid-row: 1;
align-self: center;
}
#card-3 {
grid-column: 3;
grid-row: 1;
align-self: center;
}
/* Scale down side cards relative to the hero */
#card-2, #card-3 {
transform: scale(0.85);
}
.best-card canvas {
max-height: 39vh;
}
.profile-card canvas {
max-height: 45vh;
}
}
/* === Category Cards === */
.cards-container {
display: grid;
gap: 15px;
padding: 20px;
width: 90%;
max-width: 1000px;
margin: 0 auto;
}
.category-card {
background: rgba(0, 0, 0, 0.24);
border-radius: 10px;
padding: 20px;
text-align: center;
cursor: pointer;
border: 3px solid transparent;
transition: all 0.3s ease;
user-select: none;
position: relative;
}
.category-card.active {
border-color: #;
background: rgba(0, 0, 0, 0.69);
}
.category-card.emphasis {
border-color: #BFB531;
background: rgba(0, 0, 0, 0.69);
}
.category-card .checkmark {
position: absolute;
top: 8px;
right: 16px;
font-size: 25px;
font-weight: bold;
color: #BFB531;
opacity: 1;
}
.category-card .checkmark span {
opacity: 0.39;
}
.category-card.active .checkmark span.first {
opacity: 1;
}
.category-card.emphasis .checkmark span.first,
.category-card.emphasis .checkmark span.second {
opacity: 1;
}
/* === Confetti canvas fix === */
canvas.confetti-canvas {
display: block;
position: fixed;
top: 0;
left: 0;
width: 100% !important;
height: 100% !important;
pointer-events: none; /* dont block buttons/links */
z-index: 9999; /* stays above your charts/cards */
}

42
templates/about.html Executable file
View File

@ -0,0 +1,42 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1" />
<title>cannamatch about</title>
<link rel="stylesheet" href="{{ url_for('static', filename='style.css') }}">
<link rel="icon" type="image/png" href="{{ url_for('static', filename='favicon.png') }}">
</head>
<body class="profiles-page">
<div class="page-container">
<p style="display:none">HUEEEEEEEEEEEEED 42069</p>
<div class="logo-container">
<a href="{{ url_for('index') }}">
<img src="{{ url_for('static', filename='cm.png') }}" alt="Logo" class="logo">
</a>
</div>
<div class="button-row">
<button type="button" class="magenta-button" onclick="location.href='{{ url_for('index') }}'">About</button>
<button type="button" class="magenta-button" onclick="location.href='{{ url_for('profiles_page') }}'">All Profiles</button>
<button type="button" class="magenta-button" onclick="location.href='{{ url_for('inactive') }}'">Interactive</button>
<button type="button" class="magenta-button" onclick="location.href='{{ url_for('quiz') }}'">Quiz</button>
</div>
<div class="aboutbody">
<div class="abouttextcontainer">
<p class="abouttext">Made for:</p>
<div class="aboutlogocontainer">
<a target="_blank" href="https://cicada.je"><img class="aboutlogo" src="static/scc.png" alt="scc"></img></a>
</div>
<p class="abouttext">Learn about Swiss Cannabis Research <a class="aboutlink" target="_blank" href="https://swisscannabis-research.ch">here</a>.</p>
</div>
<div class="abouttextcontainer">
<p class="abouttext">Inspired by:</p>
<div class="aboutlogocontainer">
<a target="blank" href="https://cicada.je"><img class="aboutlogo" src="static/cic.png" alt="cic"></img></a>
</div>
<p class="abouttext">Learn about the Terpene Wheel <a class="aboutlink" target="blank" href="https://cicada.je/terpene-wheel/">here</a>.</p>
</div>
</div>
</div>
</body>
</html>

139
templates/inactive.html Executable file
View File

@ -0,0 +1,139 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1" />
<title>cannamatch interactive</title>
<script src="https://cdn.jsdelivr.net/npm/chart.js"></script>
<script src="https://cdn.jsdelivr.net/npm/chartjs-plugin-dragdata@2.0.2/dist/chartjs-plugin-dragdata.min.js"></script>
<script src="{{ url_for('static', filename='js/popup.js') }}"></script>
<link rel="stylesheet" href="{{ url_for('static', filename='style.css') }}">
<link rel="icon" type="image/png" href="{{ url_for('static', filename='favicon.png') }}">
</head>
<body class="profiles-page">
<div class="page-container">
<div class="logo-container">
<a href="{{ url_for('index') }}">
<img src="{{ url_for('static', filename='cm.png') }}" alt="Logo" class="logo">
</a>
</div>
<div class="button-row">
<button type="button" class="magenta-button" onclick="location.href='{{ url_for('index') }}'">About</button>
<button type="button" class="magenta-button" onclick="location.href='{{ url_for('profiles_page') }}'">All Profiles</button>
<button type="button" class="magenta-button" onclick="location.href='{{ url_for('inactive') }}'">Interactive</button>
<button type="button" class="magenta-button" onclick="location.href='{{ url_for('quiz') }}'">Quiz</button>
<button type="button" class="magenta-button" id="resetBtn" style="display:none">Reset</button>
</div>
<div class="matches-container">
<!-- Top Match -->
<div class="matches-row best-row">
<div class="profile-card best-card top-match" id="card-1">
<h3>Best</h3>
<div class="info-block">
<div class="title-with-icon">
<h2 class="glow-info" id="name-1"></h2>
<span class="info-icon glow-text" onclick="togglePopup(event, 'popup-1')"></span>
</div>
<p class="info-text" id="info0-1"></p>
<p class="info-text" id="info2-1"></p>
</div>
<canvas id="chart-1"></canvas>
</div>
<!-- Popup for Top Match -->
<div class="popup" id="popup-1">
<div class="popup-content">
<h3 class="info-text glow-info" id="popup-name-1"></h3>
<h4 class="info-text glow-info">THC</h4>
<p class="info-text" id="popup-info0-1"></p>
<h4 class="info-text glow-info">CBD</h4>
<p class="info-text" id="popup-info1-1"></p>
<h4 class="info-text glow-info">GENETICS</h4>
<p class="info-text" id="popup-info2-1"></p>
<h4 class="info-text glow-info">TASTE</h4>
<p class="info-text" id="popup-info3-1"></p>
<h4 class="info-text glow-info">EFFECT</h4>
<p class="info-text" id="popup-info4-1"></p>
</div>
</div>
</div>
<!-- Bottom Row Matches -->
<div class="matches-row bottom-row">
<!-- Runner-up -->
<div class="profile-card side-match" id="card-2">
<h3>Runner-up</h3>
<div class="info-block">
<div class="title-with-icon">
<h2 class="glow-info" id="name-2"></h2>
<span class="info-icon glow-text" onclick="togglePopup(event, 'popup-2')"></span>
</div>
<p class="info-text" id="info0-2"></p>
<p class="info-text" id="info2-2"></p>
</div>
<canvas id="chart-2"></canvas>
</div>
<!-- Popup for Runner-up -->
<div class="popup" id="popup-2">
<div class="popup-content">
<h3 class="info-text glow-info" id="popup-name-2"></h3>
<h4 class="info-text glow-info">THC</h4>
<p class="info-text" id="popup-info0-2"></p>
<h4 class="info-text glow-info">CBD</h4>
<p class="info-text" id="popup-info1-2"></p>
<h4 class="info-text glow-info">GENETICS</h4>
<p class="info-text" id="popup-info2-2"></p>
<h4 class="info-text glow-info">TASTE</h4>
<p class="info-text" id="popup-info3-2"></p>
<h4 class="info-text glow-info">EFFECT</h4>
<p class="info-text" id="popup-info4-2"></p>
</div>
</div>
<!-- Maybe -->
<div class="profile-card side-match" id="card-3">
<h3>Maybe...</h3>
<div class="info-block">
<div class="title-with-icon">
<h2 class="glow-info" id="name-3"></h2>
<span class="info-icon glow-text" onclick="togglePopup(event, 'popup-3')"></span>
</div>
<p class="info-text" id="info0-3"></p>
<p class="info-text" id="info2-3"></p>
</div>
<canvas id="chart-3"></canvas>
</div>
<!-- Popup for Maybe -->
<div class="popup" id="popup-3">
<div class="popup-content">
<h3 class="info-text glow-info" id="popup-name-3"></h3>
<h4 class="info-text glow-info">THC</h4>
<p class="info-text" id="popup-info0-3"></p>
<h4 class="info-text glow-info">CBD</h4>
<p class="info-text" id="popup-info1-3"></p>
<h4 class="info-text glow-info">GENETICS</h4>
<p class="info-text" id="popup-info2-3"></p>
<h4 class="info-text glow-info">TASTE</h4>
<p class="info-text" id="popup-info3-3"></p>
<h4 class="info-text glow-info">EFFECT</h4>
<p class="info-text" id="popup-info4-3"></p>
</div>
</div>
</div>
</div>
</div>
<script>
window.appData = {
profiles: {{ profiles | tojson }},
initialUserData: [39,39,39,39,39,39]
};
</script>
<script src="{{ url_for('static', filename='js/inactive.js') }}"></script>
</body>
</html>

102
templates/profiles.html Executable file
View File

@ -0,0 +1,102 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1" />
<title>cannamatch profiles</title>
<script src="https://cdn.jsdelivr.net/npm/chart.js"></script>
<!-- Popup helper (same as inactive) -->
<script src="{{ url_for('static', filename='js/popup.js') }}"></script>
<link rel="stylesheet" href="{{ url_for('static', filename='style.css') }}">
<link rel="icon" type="image/png" href="{{ url_for('static', filename='favicon.png') }}">
</head>
<body class="profiles-page">
<div class="page-container">
<div class="logo-container">
<a href="{{ url_for('index') }}">
<img src="{{ url_for('static', filename='cm.png') }}" alt="Logo" class="logo">
</a>
</div>
<div class="button-row">
<button type="button" class="magenta-button" onclick="location.href='{{ url_for('index') }}'">About</button>
<button type="button" class="magenta-button" onclick="location.href='{{ url_for('profiles_page') }}'">All Profiles</button>
<button type="button" class="magenta-button" onclick="location.href='{{ url_for('inactive') }}'">Interactive</button>
<button type="button" class="magenta-button" onclick="location.href='{{ url_for('quiz') }}'">Quiz</button>
</div>
<div class="profile-grid">
{% for name, values in profiles.items() %}
<div class="profile-profile-card">
<div class="info-block profile-info-block">
<div class="title-with-icon">
<div class="profile-name"><h2 class="glow-info">{{ name }}</h2></div>
<span class="info-icon profile-info-icon glow-text" onclick="togglePopup(event, 'popup-{{ loop.index }}')"></span>
<p class="info-text">{{ values.info_text0 }}</p>
<p class="info-text">{{ values.info_text2 }}</p>
</div>
</div>
<canvas id="chart-{{ loop.index }}"></canvas>
<div class="popup" id="popup-{{ loop.index }}">
<div class="popup-content profile-popup-content">
<h3 class="info-text" id="popup-name-{{ loop.index }}">{{ name }}</h3>
<h4 class="info-text glow-info">THC</h4>
<p class="info-text" id="popup-info0-1">{{ values.info_text0 }}</p>
<h4 class="info-text glow-info">CBD</h4>
<p class="info-text" id="popup-info1-1">{{ values.info_text1 }}</p>
<h4 class="info-text glow-info">GENETICS</h4>
<p class="info-text" id="popup-info2-1">{{ values.info_text2 }}</p>
<h4 class="info-text glow-info">TASTE</h4>
<p class="info-text" id="popup-info3-1">{{ values.info_text3 }}</p>
<h4 class="info-text glow-info">EFFECT</h4>
<p class="info-text" id="popup-info4-1">{{ values.info_text4 }}</p>
</div>
</div>
</div>
{% endfor %}
</div>
</div>
<script>
const labels = ['Energy', 'Calm', 'Relax', 'Sleep', 'Focus', 'Inspire'];
const profiles = {{ profiles | tojson }};
Object.entries(profiles).forEach(([name, data], index) => {
const ctx = document.getElementById(`chart-${index + 1}`).getContext('2d');
new Chart(ctx, {
type: 'radar',
data: {
labels: labels,
datasets: [{
label: name,
data: data.values,
fill: true,
borderColor: 'rgba(191,181,49,1)',
pointBackgroundColor: 'rgba(0,0,0,0)',
pointBorderColor: 'rgba(191,181,49,1)',
backgroundColor: 'rgba(191,181,49,0.420)',
pointRadius: 0,
pointHoverRadius: 0
}]
},
options: {
responsive: true,
maintainAspectRatio: false,
scales: {
r: {
min: 0,
max: 80,
ticks: { stepSize: 20, display: false },
pointLabels: { font: { size: 15, weight: 'bold' }, color: '#FFF' },
grid: { color: 'rgba(255,255,255,0.420)', circular: false },
angleLines: { color: 'rgba(255,255,255,0.69)' }
}
},
plugins: { tooltip: { enabled: false}, legend: { display: false } }
}
});
});
</script>
</body>
</html>

67
templates/quiz.html Executable file
View File

@ -0,0 +1,67 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1" />
<title>cannamatch quiz</title>
<link rel="stylesheet" href="{{ url_for('static', filename='style.css') }}">
<link rel="icon" type="image/png" href="{{ url_for('static', filename='favicon.png') }}">
</head>
<body class="profiles-page">
<div class="page-container">
<div class="logo-container">
<a href="{{ url_for('index') }}">
<img src="{{ url_for('static', filename='cm.png') }}" alt="Logo" class="logo">
</a>
</div>
<div class="button-row">
<button type="button" class="magenta-button" onclick="location.href='{{ url_for('index') }}'">About</button>
<button type="button" class="magenta-button" onclick="location.href='{{ url_for('profiles_page') }}'">All Profiles</button>
<button type="button" class="magenta-button" onclick="location.href='{{ url_for('inactive') }}'">Interactive</button>
<button type="button" class="magenta-button" onclick="location.href='{{ url_for('quiz') }}'">Quiz</button>
</div>
<h2>What are you looking for?</h2>
<form method="POST">
<div class="cards-container">
{% set categories = [
{'name': 'Energy', 'desc': 'Boost endurance and alertness.'},
{'name': 'Calm', 'desc': 'Stay grounded and centered.'},
{'name': 'Relax', 'desc': 'Ease tension in body and mind.'},
{'name': 'Sleep', 'desc': 'Support restfulness and recovery.'},
{'name': 'Focus', 'desc': 'Enhance concentration and clarity.'},
{'name': 'Inspire', 'desc': 'Spark creativity and new ideas.'}
] %}
{% for cat in categories %}
<div class="category-card" onclick="toggleCard(this, {{ loop.index }})">
<h3>{{ cat.name }}</h3>
<p>{{ cat.desc }}</p>
<input type="hidden" id="point{{ loop.index }}" name="point{{ loop.index }}" value="20">
<div class="checkmark">
<span class="first">+</span><span class="second">+</span>
</div>
</div>
{% endfor %}
</div>
<div class="button-row">
<button type="submit">Submit</button>
</div>
</form>
</div>
<script>
function toggleCard(card, index) {
const input = document.getElementById(`point${index}`);
const values = [20, 40, 80];
let value = parseInt(input.value, 10);
if (isNaN(value)) value = 20;
// Find next value in the cycle
const nextValue = values[(values.indexOf(value) + 1) % values.length];
input.value = nextValue;
card.classList.remove("active", "emphasis");
if (nextValue === 40) card.classList.add("active");
else if (nextValue === 80) card.classList.add("emphasis");
}
</script>
</body>
</html>

299
templates/result.html Executable file
View File

@ -0,0 +1,299 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1" />
<title>cannamatch result</title>
<!-- Chart.js & confetti -->
<script src="https://cdn.jsdelivr.net/npm/chart.js"></script>
<script src="https://cdn.jsdelivr.net/npm/canvas-confetti@1.5.1/dist/confetti.browser.min.js"></script>
<!-- Popup helper (same as inactive) -->
<script src="{{ url_for('static', filename='js/popup.js') }}"></script>
<!-- Shared styles / favicon (same as inactive) -->
<link rel="stylesheet" href="{{ url_for('static', filename='style.css') }}" />
<link rel="icon" type="image/png" href="{{ url_for('static', filename='favicon.png') }}" />
</head>
<body class="profiles-page">
<div class="page-container">
<!-- Logo (same block as inactive) -->
<div class="logo-container">
<a href="{{ url_for('index') }}">
<img src="{{ url_for('static', filename='cm.png') }}" alt="Logo" class="logo">
</a>
</div>
<!-- Buttons row (kept minimal for results page) -->
<div class="button-row">
<button type="button" class="magenta-button" onclick="location.href='{{ url_for('index') }}'">About</button>
<button type="button" class="magenta-button" onclick="location.href='{{ url_for('profiles_page') }}'">All Profiles</button>
<button type="button" class="magenta-button" onclick="location.href='{{ url_for('inactive') }}'">Interactive</button>
<button type="button" class="magenta-button" onclick="location.href='{{ url_for('quiz') }}'">Quiz</button>
</div>
<div class="matches-container">
<!-- Top Match -->
<div class="matches-row best-row">
<div class="profile-card best-card top-match" id="card-1">
<h3>Best</h3>
<div class="info-block">
<div class="title-with-icon">
<h2 class="glow-info" id="name-1">{{ top_matches[0][0] if top_matches|length > 0 else '' }}</h2>
<span class="info-icon glow-text" onclick="togglePopup(event, 'popup-1')"></span>
</div>
{% if top_matches|length > 0 %}
<p class="info-text" id="info0-1">{{ top_matches[0][1].get('info_text0','') }}</p>
<p class="info-text" id="info2-1">{{ top_matches[0][1].get('info_text2','') }}</p>
{% endif %}
</div>
<canvas id="chart-1"></canvas>
</div>
<!-- Popup for Top Match -->
<div class="popup" id="popup-1">
<div class="popup-content">
<h3 class="info-text" id="popup-name-1">{{ top_matches[0][0] if top_matches|length > 0 else '' }}</h3>
{% if top_matches|length > 0 %}
<h4 class="info-text glow-info">THC</h4>
<p class="info-text" id="popup-info0-1">{{ top_matches[0][1].get('info_text0','') }}</p>
<h4 class="info-text glow-info">CBD</h4>
<p class="info-text" id="popup-info1-1">{{ top_matches[0][1].get('info_text1','') }}</p>
<h4 class="info-text glow-info">GENETICS</h4>
<p class="info-text" id="popup-info2-1">{{ top_matches[0][1].get('info_text2','') }}</p>
<h4 class="info-text glow-info">TASTE</h4>
<p class="info-text" id="popup-info3-1">{{ top_matches[0][1].get('info_text3','') }}</p>
<h4 class="info-text glow-info">EFFECT</h4>
<p class="info-text" id="popup-info4-1">{{ top_matches[0][1].get('info_text4','') }}</p>
{% endif %}
</div>
</div>
</div>
<!-- Bottom Row Matches -->
<div class="matches-row bottom-row">
<!-- Runner-up -->
<div class="profile-card side-match" id="card-2">
<h3>Runner-up</h3>
<div class="info-block">
<div class="title-with-icon">
<h2 class="glow-info" id="name-2">{{ top_matches[1][0] if top_matches|length > 1 else '' }}</h2>
<span class="info-icon glow-text" onclick="togglePopup(event, 'popup-2')"></span>
</div>
{% if top_matches|length > 1 %}
<p class="info-text" id="info0-2">{{ top_matches[1][1].get('info_text0','') }}</p>
<p class="info-text" id="info2-2">{{ top_matches[1][1].get('info_text2','') }}</p>
{% endif %}
</div>
<canvas id="chart-2"></canvas>
</div>
<!-- Popup for Runner-up -->
<div class="popup" id="popup-2">
<div class="popup-content">
<h3 class="info-text" id="popup-name-2">{{ top_matches[1][0] if top_matches|length > 1 else '' }}</h3>
{% if top_matches|length > 1 %}
<h4 class="info-text glow-info">THC</h4>
<p class="info-text" id="popup-info0-2">{{ top_matches[1][1].get('info_text0','') }}</p>
<h4 class="info-text glow-info">CBD</h4>
<p class="info-text" id="popup-info1-2">{{ top_matches[1][1].get('info_text1','') }}</p>
<h4 class="info-text glow-info">GENETICS</h4>
<p class="info-text" id="popup-info2-2">{{ top_matches[1][1].get('info_text2','') }}</p>
<h4 class="info-text glow-info">TASTE</h4>
<p class="info-text" id="popup-info3-2">{{ top_matches[1][1].get('info_text3','') }}</p>
<h4 class="info-text glow-info">EFFECT</h4>
<p class="info-text" id="popup-info4-2">{{ top_matches[1][1].get('info_text4','') }}</p>
{% endif %}
</div>
</div>
<!-- Maybe -->
<div class="profile-card side-match" id="card-3">
<h3>Maybe...</h3>
<div class="info-block">
<div class="title-with-icon">
<h2 class="glow-info" id="name-3">{{ top_matches[2][0] if top_matches|length > 2 else '' }}</h2>
<span class="info-icon glow-text" onclick="togglePopup(event, 'popup-3')"></span>
</div>
{% if top_matches|length > 2 %}
<p class="info-text" id="info0-3">{{ top_matches[2][1].get('info_text0','') }}</p>
<p class="info-text" id="info2-3">{{ top_matches[2][1].get('info_text2','') }}</p>
{% endif %}
</div>
<canvas id="chart-3"></canvas>
</div>
<!-- Popup for Maybe -->
<div class="popup" id="popup-3">
<div class="popup-content">
<h3 class="info-text" id="popup-name-3">{{ top_matches[2][0] if top_matches|length > 2 else '' }}</h3>
{% if top_matches|length > 2 %}
<h4 class="info-text glow-info">THC</h4>
<p class="info-text" id="popup-info0-3">{{ top_matches[2][1].get('info_text0','') }}</p>
<h4 class="info-text glow-info">CBD</h4>
<p class="info-text" id="popup-info1-3">{{ top_matches[2][1].get('info_text1','') }}</p>
<h4 class="info-text glow-info">GENETICS</h4>
<p class="info-text" id="popup-info2-3">{{ top_matches[2][1].get('info_text2','') }}</p>
<h4 class="info-text glow-info">TASTE</h4>
<p class="info-text" id="popup-info3-3">{{ top_matches[2][1].get('info_text3','') }}</p>
<h4 class="info-text glow-info">EFFECT</h4>
<p class="info-text" id="popup-info4-3">{{ top_matches[2][1].get('info_text4','') }}</p>
{% endif %}
</div>
</div>
</div>
</div>
</div>
<script>
// Same visual language as inactive.js
const labels = ['Energy','Calm','Relax','Sleep','Focus','Inspire'];
const userData = {{ user | tojson }};
const matches = {{ top_matches | tojson }};
const youLineColor = 'rgba(255,255,255,0.5)';
const youFillColor = 'rgba(255,255,255,0.1)';
const matchLineColor = 'rgb(191,181,49)';
const matchFillColor = 'rgba(191,181,49,0.420)';
const radialLineColor= 'rgba(255,255,255,0.420)';
function getFontSize(){ return window.innerWidth < 600 ? 10 : 42; }
function getBottomChartFontSize(){ return window.innerWidth < 600 ? 8 : 24; }
function getPointRadius(){ return window.innerWidth < 600 ? 11 : 24; }
function getPointHoverRadius(){ return window.innerWidth < 600 ? 12 : 39; }
const charts = [];
// Build top chart (You + best match), if present
if (matches.length > 0) {
const [bestName, bestProf] = matches[0];
const ctx1 = document.getElementById('chart-1').getContext('2d');
charts[0] = new Chart(ctx1, {
type: 'radar',
data: {
labels,
datasets: [
{
label: bestName,
data: bestProf.values,
fill: true,
borderColor: matchLineColor,
backgroundColor: matchFillColor,
pointBackgroundColor: matchLineColor,
pointRadius: 0
},
{
label: 'You',
data: userData,
fill: true,
borderColor: youLineColor,
backgroundColor: youFillColor,
pointBackgroundColor: youLineColor,
pointBorderColor: matchLineColor,
pointRadius: 0,
pointHoverRadius: 0
}
]
},
options: {
responsive: true,
maintainAspectRatio: false,
scales: {
r: {
min: 0,
max: 80,
ticks: { display: false, stepSize: 20 },
pointLabels: { font: { size: getFontSize(), weight: 'bold' }, color: '#FFF' },
angleLines: { color: radialLineColor },
grid: { drawticks: false, drawOnChartArea: false, circular: false }
}
},
plugins: { tooltip: { enabled: false }, legend: { display: false } }
}
});
}
// Build bottom charts (matches only)
for (let i = 1; i < Math.min(matches.length, 3); i++) {
const [name, prof] = matches[i];
const ctx = document.getElementById(`chart-${i+1}`).getContext('2d');
charts[i] = new Chart(ctx, {
type: 'radar',
data: {
labels,
datasets: [{
label: name,
data: prof.values,
fill: true,
borderColor: matchLineColor,
backgroundColor: matchFillColor,
pointBackgroundColor: matchLineColor,
pointRadius: 0
}]
},
options: {
responsive: true,
maintainAspectRatio: false,
scales: {
r: {
min: 0,
max: 80,
ticks: { display: false, stepSize: 20 },
pointLabels: { font: { size: getBottomChartFontSize(), weight: 'bold' }, color: '#FFF' },
angleLines: { color: radialLineColor },
grid: { drawticks: false, drawOnChartArea: false, circular: false }
}
},
plugins: { tooltip: { enabled: false }, legend: { display: false } }
}
});
}
// Adjust label font sizes on resize to mirror inactive page
window.addEventListener('resize', () => {
if (charts[0]) {
charts[0].options.scales.r.pointLabels.font.size = getFontSize();
charts[0].update('none');
}
if (charts[1]) {
charts[1].options.scales.r.pointLabels.font.size = getBottomChartFontSize();
charts[1].update('none');
}
if (charts[2]) {
charts[2].options.scales.r.pointLabels.font.size = getBottomChartFontSize();
charts[2].update('none');
}
});
// Confetti like your previous result page: from left & right of best card
window.addEventListener('load', () => {
const bestCard = document.querySelector('.best-card');
if (!bestCard) return;
const rect = bestCard.getBoundingClientRect();
const colors = ['#BFB531', '#FFFFFF'];
function fire(xOrigin) {
confetti({
particleCount: 69,
angle: xOrigin < 0.5 ? 60 : 120,
spread: 69,
origin: {
x: xOrigin,
y: (rect.top + rect.height / 2) / window.innerHeight
},
colors
});
}
const leftOrigin = (rect.left) / window.innerWidth;
const rightOrigin = (rect.right) / window.innerWidth;
fire(leftOrigin);
fire(rightOrigin);
});
</script>
</body>
</html>