diff options
37 files changed, 1017 insertions, 375 deletions
@@ -3,21 +3,36 @@ TO GET IT RUNNING: -Install postgresql for local user & get it running +Install postgresql for local user & get it running (as in course material) - $ wget https://github.com/hy-tsoha/local-pg/raw/master/pg-install.sh - $ bash pg-install.sh - $ postrgres & - + +Install poetry if nessesary. (refer your distro) +- $ pip install --user poetry +- $ pipx install poetry + Clone the source, get tables ready & install poetry dependencies - $ git clone https://github.com/triionhe/kyselma.git - $ cd kyselma -- $ psql < SCHEMA.sql -- $ poetry install +- $ psql < SCHEMA.sql (BE CAREFUL! This drops some tables.) +- $ poetry install --no-root Start the app in poetry virtual environment -- $ SQLALCHEMY_DATABASE_URI="postgresql:///$USER" SECRET_KEY=923987295 poetry run flask run +- $ SQLALCHEMY_DATABASE_URI="postgresql:///$USER" SECRET_KEY=29347884 poetry run flask run + +Surf to the webpage and start two sessions for better testing +- $ firefox http://localhost:5000/ http://127.0.0.1:5000/ + + DONE: +- Vastausten anto +- Vastausten tarkistelu +- Kyselyn luomisen näyttö +- Kyselyn lukitseminen vastattavaksi +- Kysymyksen näyttö +- Ulkoasu - Voi lisätä kyselyn - Kysymyksen mukana lisätään vastaus - Kysymyksen lisäys @@ -31,13 +46,11 @@ DONE: - Nettisivurunko TODO: -- Kyselyn luomisen näyttö -- Kyselyn lukitseminen vastattavaksi -- Kysymyksen näyttö -- Ulkoasu - Moderointi -- Vastausten anto -- Vastausten tarkistelu +- Parempi ulkoasu +- Vastauksen ja luomisen aloittamisen yksinkertaistaminen +- Tietoturvaseikat? +- Eniten ja vähiten yhdenmukaiset vastaajat ... Tarkoitus on luoda sivu jossa voi luoda kysymyksiä ja kyselyitä, joita @@ -92,3 +105,6 @@ Kysymykset, kyselyt, vastaukset tallennetaan asianmukaisiin tauluihin. Ominaisuuksia joiden tekemistä voi harkita: - Jokaisella kyselyllä on elinikä jonka jälkeen ne siivotaan. - Kysymyksiin voi liittää kuvia + +Ikonit (CC) pixellove: +https://www.svgrepo.com/collection/pixellove-bordered-vectors/ @@ -1,3 +1,4 @@ +DROP TABLE users, questions, answers, questionaires, quiz_links; CREATE TABLE users ( id SERIAL PRIMARY KEY, nick TEXT, @@ -21,6 +22,50 @@ CREATE TABLE questionaires ( id SERIAL PRIMARY KEY, questionset INT[] DEFAULT '{}', creator_id INT, - ready BOOLEAN DEFAULT FALSE, created INT ); +CREATE TABLE quiz_links ( + id SERIAL PRIMARY KEY, + quiz_id INT, + link TEXT, + created INT +); +INSERT INTO users (nick, created) VALUES ('Raili Niemi', 57 ); +INSERT INTO users (nick, created) VALUES ('Tea Jalava', 36 ); +INSERT INTO users (nick, created) VALUES ('Eero Metsäranta', 47 ); +INSERT INTO questions (question, neg_answer, pos_answer, created) + VALUES ('Mikä ikä?', '0', '100', 99 ); +INSERT INTO questions (question, neg_answer, pos_answer, created) + VALUES ('Mikä vointi?', 'huono', 'hyvä', 99 ); +INSERT INTO questions (question, neg_answer, pos_answer, created) + VALUES ('Ulkoiletko?', 'viime vuonna', 'joka päivä', 99 ); +INSERT INTO questions (question, neg_answer, pos_answer, created) + VALUES ('kys?', 'ei vielä', 'ei koskaan', 99 ); +INSERT INTO answers (user_id, question_id, answer, created) + VALUES (1, 1, 570, 99 ); +INSERT INTO answers (user_id, question_id, answer, created) + VALUES (1, 2, 670, 99 ); +INSERT INTO answers (user_id, question_id, answer, created) + VALUES (1, 3, 888, 99 ); +INSERT INTO answers (user_id, question_id, answer, created) + VALUES (1, 4, 999, 99 ); +INSERT INTO answers (user_id, question_id, answer, created) + VALUES (2, 1, 360, 99 ); +INSERT INTO answers (user_id, question_id, answer, created) + VALUES (2, 2, 230, 99 ); +INSERT INTO answers (user_id, question_id, answer, created) + VALUES (2, 3, 120, 99 ); +INSERT INTO answers (user_id, question_id, answer, created) + VALUES (2, 4, 123, 99 ); +INSERT INTO answers (user_id, question_id, answer, created) + VALUES (3, 1, 470, 99 ); +INSERT INTO answers (user_id, question_id, answer, created) + VALUES (3, 2, 570, 99 ); +INSERT INTO answers (user_id, question_id, answer, created) + VALUES (3, 3, 321, 99 ); +INSERT INTO answers (user_id, question_id, answer, created) + VALUES (3, 4, 785, 99 ); +INSERT INTO questionaires (questionset, creator_id, created) + VALUES ('{1,2,3,4}', 1, 666); +INSERT INTO quiz_links (quiz_id, link, created) + VALUES (1, 'kyselama', 666); @@ -8,8 +8,8 @@ app.config["SQLALCHEMY_DATABASE_URI"] = getenv("SQLALCHEMY_DATABASE_URI") db.init_app(app) import routes.base -import routes.set.nick -import routes.set.quiz -import routes.set.question -import routes.set.answers -import routes.get.quiz +import routes.answer +import routes.create +import routes.analyse +import routes.question + diff --git a/db_actions.py b/db_actions.py index 5a76b00..5f793fc 100644 --- a/db_actions.py +++ b/db_actions.py @@ -65,6 +65,32 @@ def quiz_add( quiz_id, question_id ): }) db.session.commit() + +def set_quiz_link( quiz_id, link ): + sql = "INSERT \ + INTO quiz_links (quiz_id, link, created) \ + VALUES (:quiz_id, :link, :created)" + result = db.session.execute( text(sql), + { "quiz_id":quiz_id, "link":link, "created":int(time()) } ) + db.session.commit() + + +def find_quiz_by_link( link ): + sql = "SELECT quiz_id \ + FROM quiz_links \ + WHERE link=:link;" + result = db.session.execute(text(sql), { "link":link }).fetchone() + return result[0] if result else result + + +def get_quiz_link( quiz_id ): + sql = "SELECT link \ + FROM quiz_links \ + WHERE quiz_id=:quiz_id;" + result = db.session.execute(text(sql), { "quiz_id":quiz_id }).fetchone() + return result[0] if result else result + + def answer_new(user_id, question_id, answer): sql = "INSERT \ INTO answers (user_id, question_id, answer, created) \ @@ -77,6 +103,7 @@ def answer_new(user_id, question_id, answer): } ) db.session.commit() + def get_questions(quiz_id): sql = "SELECT q.id, q.question, q.neg_answer, q.pos_answer, a.answer \ FROM questionaires quiz \ @@ -85,15 +112,62 @@ def get_questions(quiz_id): WHERE a.question_id = q.id AND quiz.id = (:quiz_id);" return db.session.execute( text(sql), { "quiz_id":quiz_id } ).fetchall() +def get_comparable(quiz_id,user1,user2): + sql = "SELECT q.question, q.neg_answer, q.pos_answer, \ + a1.answer, a2.answer, \ + 100 - ABS( a1.answer - a2.answer ) / 10 \ + FROM questionaires quiz \ + JOIN questions q ON q.id = ANY(quiz.questionset) \ + JOIN answers a1 ON a1.user_id = (:user1) \ + JOIN answers a2 ON a2.user_id = (:user2) \ + WHERE a1.question_id = q.id \ + AND a2.question_id = q.id \ + AND quiz.id = (:quiz_id);" + return db.session.execute( text(sql), { + "quiz_id":quiz_id, + "user1":user1, + "user2":user2 + } ).fetchall() + def get_user_answer(user_id, question_id): sql = "SELECT answer \ - FROM answers \ - WHERE question_id = (:question_id) AND user_id = (:user_id);" + FROM answers \ + WHERE question_id = (:question_id) AND user_id = (:user_id);" result = db.session.execute( text(sql), { 'question_id': question_id, 'user_id': user_id } ).fetchone() return result[0] if result else result -
\ No newline at end of file + +def get_user_answers_for_quiz(quiz_id, user_id): + sql = "SELECT a.question_id, a.answer \ + FROM questionaires quiz \ + JOIN answers a ON a.question_id = ANY(quiz.questionset) \ + WHERE a.user_id = (:user_id) AND quiz.id = (:quiz_id);" + return db.session.execute( text(sql), { + 'user_id': user_id, + 'quiz_id': quiz_id + } ).fetchall() + +def get_users_answered(quiz_id): + sql = "SELECT DISTINCT a.user_id, u.nick \ + FROM questionaires quiz \ + JOIN answers a ON a.question_id = quiz.questionset[1] \ + JOIN users u ON u.id = a.user_id \ + WHERE quiz.id = (:quiz_id);" + return db.session.execute( text(sql), { + 'quiz_id': quiz_id + } ).fetchall() + +def is_user_answered(quiz_id, user_id): + sql = "SELECT a.answer \ + FROM questionaires quiz \ + JOIN answers a ON a.question_id = quiz.questionset[1] \ + WHERE quiz.id = (:quiz_id) AND a.user_id = (:user_id);" + results = db.session.execute( text(sql), { + 'quiz_id': quiz_id, + 'user_id': user_id + } ).fetchone() + return results[0] if results else results diff --git a/routes/analyse.py b/routes/analyse.py new file mode 100644 index 0000000..b333e70 --- /dev/null +++ b/routes/analyse.py @@ -0,0 +1,50 @@ +from app import app +from flask import render_template,session,request,redirect +import db_actions as D +from routes.tools import rows2dicts, get_alert, get_nick, red + +@app.route("/pages/analyse.html") +def analyse(): + if "id" in session: + sid = session["id"] + else: + return red["nick"] + + if "answer_id" in session and D.is_user_answered(session["answer_id"],sid): + aid = session["answer_id"] + else: + return render_template("analyse.html", + alert=get_alert(), + nick=get_nick() + ) + + uid1 = session["anal_user1"] if "anal_user1" in session else sid + uid1 = sid if uid1 != sid and not D.is_user_answered(aid,uid1) else uid1 + + uid2 = session["anal_user2"] if "anal_user2" in session else sid + uid2 = sid if uid2 != sid and not D.is_user_answered(aid,uid2) else uid2 + + comparable = D.get_comparable( aid, uid1, uid2 ) + avg=0 + for i in range(len(comparable)): + avg += comparable[i][5] + avg//=len(comparable) + + + return render_template("analyse.html", + alert=get_alert(), + nick=get_nick(), + code=D.get_quiz_link(aid), + questions = rows2dicts( comparable, ['q','n','p','a1','a2','c'] ), + users = rows2dicts( D.get_users_answered(aid), ['id','nick'] ), + user1=int(uid1), + user2=int(uid2), + avg = avg + ) + +@app.route("/set/compare",methods=["POST"]) +def set_compare(): + session["anal_user1"] = request.form["user1"] + session["anal_user2"] = request.form["user2"] + return redirect("/#analyse") + diff --git a/routes/answer.py b/routes/answer.py new file mode 100644 index 0000000..fa3d138 --- /dev/null +++ b/routes/answer.py @@ -0,0 +1,100 @@ +from app import app +from flask import render_template, session, request, redirect +import db_actions as D +from routes.tools import rows2dicts, get_alert, get_nick, red + + +@app.route("/pages/new_answer.html") +def new_answer(): + if "id" not in session: + return red["nick"] + return render_template("new_answer.html", + alert=get_alert(), + nick=get_nick() + ) + +@app.route("/kys/<link>") +def kys_link(link): + if aid := D.find_quiz_by_link( link ): + session["answer_id"] = aid + return redirect("/#answer") + return redirect("/") + +@app.route("/set/answer_id",methods=["POST"]) +def answer_id(): + if "id" not in session: + session["alert"] = "Nimimerkkiä ei ole asetettu." + return redirect("/#nick") + else: + sid = session["id"] + + if "next" not in request.form: + next = "/#answer" + else: + next = "/#"+request.form["next"] + + if "link" not in request.form or request.form["link"]=="": + session["alert"] = "Kyselmän nimeä ei ole annettu." + return redirect(next) + + if aid := D.find_quiz_by_link( request.form["link"] ): + session["answer_id"] = aid + else: + session["alert"] = "Koodilla ei löytynyt kyselmää" + return redirect(next) + + if next == "/#analyse" and not D.is_user_answered( aid, sid ): + session["alert"] = "Et ole vielä vastannut tähän kyselmään. \ + Voit tutkia vastaksia vastattuasi." + return redirect("/#answer") + + return redirect( next ) + +@app.route("/pages/answer.html") +def answer(): + if "id" in session: + sid = session["id"] + else: + return red["nick"] + + if "answer_id" in session: + aid = session["answer_id"] + else: + return red["new_answer"] + + if D.is_user_answered(aid, sid): + return red["new_answer"] + + return render_template("answer.html", + alert = get_alert(), + nick = get_nick(), + questions = rows2dicts( D.get_questions(aid), ['i','q','n','p'] ), + link = D.get_quiz_link( aid ) + ) + +@app.route("/set/answers",methods=["POST"]) +def set_answers(): + if "id" not in session: + session["alert"]="Nimimerkkiä ei ole vielä valittu!" + return redirect( "/#nick" ) + if "answer_id" not in session: + session["alert"]="Kyselyä ei ole valittu vastaamista varten!" + return redirect( "/#answer" ) + + sid = session["id"] + for qid, answer in request.form.items(): + try: + if int(answer) < 0 or int(answer) > 999: + session["alert"]="Luvattoman pieniä tai suuria lukuja!" + return redirect( "/#answer" ) + elif D.get_user_answer(sid, int(qid) ): + session["alert"]="Kyselyyn olikin jo saatu vastauksia." + return redirect( "/#answer" ) + except ValueError: + session["alert"] = "Vastaukset ei ole lukuja!" + return redirect( "/#answer" ) + + for qid, answer in request.form.items(): + D.answer_new(sid, int(qid), int(answer)) + + return redirect("/#analyse") diff --git a/routes/base.py b/routes/base.py index 144eeef..695388b 100644 --- a/routes/base.py +++ b/routes/base.py @@ -1,24 +1,7 @@ from app import app -from flask import render_template,session +from flask import render_template,session,request,redirect import db_actions as D - -def get_alert(): - if "alert" in session: - alert = session["alert"] - del session["alert"] - return f"{alert}" - return "" - -def get_nick(): - while "id" in session.keys(): - nick = D.user_get_nick(session["id"]) - if not nick: - del session['id'] - if "quiz_id" in session.keys(): - del session['quiz_id'] - break - return nick - return "(ei nimimerkkiä)" +from routes.tools import rows2dicts, get_alert, get_nick @app.route("/") def index(): @@ -29,50 +12,40 @@ def info(): return render_template("info.html", alert=get_alert() ) - -@app.route("/pages/create.html") -def create(): - if "id" not in session.keys(): - return "redirect = #nick" - if "quiz_id" not in session.keys(): - return "redirect = #quiz" - return render_template("create.html", - alert=get_alert(), - nick=get_nick() - ) - -@app.route("/pages/answer.html") -def answer(): - if "id" not in session.keys(): - return "redirect = #nick" - return render_template("answer.html", - alert=get_alert(), - nick=get_nick() - ) - -@app.route("/pages/analyse.html") -def analyse(): - if "id" not in session.keys(): - return "redirect = #nick" - return render_template("analyse.html", - alert=get_alert(), - nick=get_nick() - ) - -@app.route("/pages/moderate.html") -def moderate(): - return render_template("moderate.html", - alert=get_alert() - ) - + @app.route("/pages/nick.html") def nick(): - return render_template("nick.html", alert=get_alert() ) + return render_template("nick.html", + alert=get_alert() + ) -@app.route("/pages/question.html") -def question(): - return render_template("question.html", alert=get_alert() ) +@app.route("/set/nick",methods=["POST"]) +def new_nick(): + if "id" in session.keys(): + session["alert"]="Sinulla on jo nimimerkki. Käytä sitä." + return redirect("/") + if "nick" not in request.form or request.form["nick"]=="": + session["alert"]="Nimimerkkiä ei voi asettaa ilman nimimerkkiä." + return redirect("/#nick") + else: + nick = request.form["nick"] + if len(nick) < 4: + session["alert"]="Nimimerkki on liian lyhyt" + return redirect("/#nick") + if not nick.isalnum(): + session["alert"]="Nimimerkissä saa olla vain kirjaimia ja numeroita." + return redirect("/#nick") + if D.user_exists(nick): + session["alert"]="Nimimerkki jonka olet ottamassa on jo varattu." + return redirect("/#nick") + session["id"] = D.user_new(nick) + return redirect("/") + + + +#@app.route("/pages/moderate.html") +#def moderate(): +# return render_template("moderate.html", +# alert=get_alert() +# ) -@app.route("/pages/quiz.html") -def build(): - return render_template("quiz.html", alert=get_alert() ) diff --git a/routes/create.py b/routes/create.py new file mode 100644 index 0000000..0a2a343 --- /dev/null +++ b/routes/create.py @@ -0,0 +1,55 @@ +from app import app +from flask import render_template,session,request,redirect +import db_actions as D +from routes.tools import rows2dicts, get_alert, get_nick, generate_link, red + + +@app.route("/pages/create.html") +def create(): + if "id" not in session: + return red["nick"] + if "quiz_id" not in session: + return red["quiz"] + if D.get_quiz_link(session["quiz_id"]): + return red["quiz"] + + return render_template("create.html", + alert=get_alert(), + nick=get_nick(), + questions=rows2dicts( + D.get_questions(session["quiz_id"]), + ['i','q','n','p','a'] + ) + ) + +@app.route("/pages/quiz.html") +def build(): + if "id" not in session: + return red["nick"] + return render_template("quiz.html", + alert=get_alert(), + nick=get_nick() + ) + +@app.route("/set/quiz",methods=["POST"]) +def new_quiz(): + if not "id" in session.keys(): + session["alert"]="Tarvitset nimimerkin loudaksesi" + return redirect("/#nick") + user_id = session["id"] + session["quiz_id"] = D.quiz_new( user_id ) + return redirect("/#create") + + +@app.route("/set/quiz_ready",methods=["POST"]) +def quiz_ready(): + if "quiz_id" not in session.keys(): + return "KUOLETTAVA: kyselyä ei ole" + if not D.is_user_answered(session["quiz_id"], session["id"]): + session["alert"] = "Tyhjän kyselmän luominen ei käy päinsä!" + return redirect("/#create") + quiz_id = session["quiz_id"] + session["answer_id"] = session["quiz_id"] + D.set_quiz_link(session["quiz_id"], generate_link()) + return redirect("/#analyse") + diff --git a/routes/get/quiz.py b/routes/get/quiz.py deleted file mode 100644 index 69e2613..0000000 --- a/routes/get/quiz.py +++ /dev/null @@ -1,18 +0,0 @@ -from app import app -from flask import render_template, session, request, redirect, jsonify -import db_actions as D - - -@app.route("/get/quiz_creator",methods=["GET"]) -def get_questions_by_id(): - if "quiz_id" not in session.keys(): - return "KUOLETTAVA: Sessiota / kyselmä id:tä ei ole" - - results = D.get_questions(session['quiz_id']) - r={} - names=['i','q','n','p','a'] - for i in range(len(results)): - r[i]={} - for j in range(len(results[i])): - r[i][names[j]]=results[i][j] - return (jsonify(r)) diff --git a/routes/question.py b/routes/question.py new file mode 100644 index 0000000..02261ca --- /dev/null +++ b/routes/question.py @@ -0,0 +1,49 @@ +from app import app +from flask import render_template,session,request,redirect +import db_actions as D +from routes.tools import rows2dicts, get_alert, get_nick + +@app.route("/pages/question.html") +def question(): + return render_template("question.html", + alert=get_alert(), + nick=get_nick() + ) + +@app.route("/set/question",methods=["POST"]) +def new_question(): + try: + question = request.form["question"] + neg_ans = request.form["neg_ans"] + pos_ans = request.form["pos_ans"] + answer = int(request.form["answer"]) + except (KeyError, ValueError): + session["alert"] = "Nyt kaikkea ei tullut perille tai jotain outoa." + return redirect("/#question") + + try: + sid = session["id"] + except (KeyError): + session["alert"] = "Nimimerkki puuttuukin." + return redirect("/#nick") + + try: + qid = session["quiz_id"] + except (KeyError): + session["alert"] = "Kyselmän luonti ei ollutkaan kesken." + return redirect("/#quiz") + + for entry in [question, neg_ans, pos_ans]: + if len(entry) < 2 or len(entry) > 80: + session["alert"] = "Syötteiden tulee olla 2-80 merkkiä pitkiä" + return redirect("/#question") + + if answer < 0 or answer > 999: + session["alert"] = "Vastauksessasi on nyt jotain häikkää." + return redirect("/#question") + + question_id = D.question_new( question, neg_ans, pos_ans ) + D.quiz_add(qid, question_id) + D.answer_new(sid, question_id, answer) + return redirect("/#create") + diff --git a/routes/set/answers.py b/routes/set/answers.py deleted file mode 100644 index 859d65c..0000000 --- a/routes/set/answers.py +++ /dev/null @@ -1,32 +0,0 @@ -from app import app -from flask import render_template, session, request, redirect -import db_actions as D - -def validate_answer(ans): - if len(ans)<1: - return False - try: - value=int(ans) - if value<0 or value>1000: - return False - except ValueError: - return False - return True - -@app.route("/set/answers",methods=["POST"]) -def set_answers(): - if "id" not in session.keys(): - return "KUOLETTAVA: Nimimerkkiä ei ole vielä valittu!" - if "quiz_id" not in session.keys(): - return "KUOLETTAVA: Yrität vastata kyselyyn ilman sen valintaa!" - - user_id = session["id"] - for id, answer in request.form.items(): - question_id = int(id) - if not validate_answer(answer): - return "KUOLETTAVA: Epäkelpo vastaus!" - if D.get_user_answer(user_id,question_id): - return "KUOLETTAVA: On jo vastattu!" - D.answer_new(user_id, question_id, answer) - - return redirect("/#analyse") diff --git a/routes/set/nick.py b/routes/set/nick.py deleted file mode 100644 index 67ddeea..0000000 --- a/routes/set/nick.py +++ /dev/null @@ -1,28 +0,0 @@ -from app import app -from flask import render_template, session, request, redirect -import db_actions as D - - -@app.route("/set/nick",methods=["POST"]) -def new_nick(): - nick = request.form["nick"] - if "id" in session.keys(): - msg = "You already have a nick." - elif D.user_exists(nick): - msg = "Nick is already reserved." - elif msg := invalid_nick(nick): - pass - else: - session["id"] = D.user_new(nick) - return redirect("/") - session["alert"]="Nick in not created: "+msg - return redirect("/#nick") - - -def invalid_nick(nick): - if len(nick)<4: - return "Nick is too short" - if not nick.isalnum(): - return "Only letters and numbers are allowed" - return 0 -
\ No newline at end of file diff --git a/routes/set/question.py b/routes/set/question.py deleted file mode 100644 index deaf9be..0000000 --- a/routes/set/question.py +++ /dev/null @@ -1,40 +0,0 @@ -from app import app -from flask import render_template, session, request, redirect -import db_actions as D - - -def validate_answer(ans): - if len(ans)<1: - return False - return True - -def validate_question(question): - if len(question)<2: - return False - return True - -@app.route("/set/question",methods=["POST"]) -def new_question(): - question = request.form["question"] - neg_ans = request.form["neg_ans"] - pos_ans = request.form["pos_ans"] - answer = request.form["answer"] - if not validate_question(question): - msg = "Kysymys on virheellinen" - elif not validate_answer(neg_ans): - msg = "Vasen selite on virheellinen" - elif not validate_answer(pos_ans): - msg = "Oikea selite on virheellinen" - elif "id" not in session.keys(): - msg = "Tarvitaan nimimerkki" - elif "quiz_id" not in session.keys(): - msg = "Ei voi lisätä kysymystä ilman kyselmää" - else: - quiz_id = session["quiz_id"] - user_id = session["id"] - question_id = D.question_new( question, neg_ans, pos_ans ) - D.quiz_add(quiz_id, question_id) - D.answer_new(user_id, question_id, answer) - return redirect("/#create") - session["alert"]="Kysymystä ei luotu: "+msg - return redirect("/#create") diff --git a/routes/set/quiz.py b/routes/set/quiz.py deleted file mode 100644 index 9ad13da..0000000 --- a/routes/set/quiz.py +++ /dev/null @@ -1,13 +0,0 @@ -from app import app -from flask import render_template, session, request, redirect -import db_actions as D - - -@app.route("/set/quiz",methods=["POST"]) -def new_quiz(): - if not "id" in session.keys(): - session["alert"]="Tarvitset nimimerkin loudaksesi" - return redirect("/#nick") - user_id = session["id"] - session["quiz_id"] = D.quiz_new( user_id ) - return redirect("/#create") diff --git a/routes/tools.py b/routes/tools.py new file mode 100644 index 0000000..69e6fef --- /dev/null +++ b/routes/tools.py @@ -0,0 +1,46 @@ +from random import randint +from flask import session +import db_actions as D + +red = { + "nick": "<script>window.location.hash=\"nick\"</script>", + "new_answer": "<script>window.location.hash=\"new_answer\"</script>", + "quiz": "<script>window.location.hash=\"quiz\"</script>" +} + + +def rows2dicts( rows, names ): + dlist=[] + for i in range(len(rows)): + row={} + for j in range(len(names)): + row[names[j]]=rows[i][j] + dlist.append(row) + return dlist + +def get_alert(): + if "alert" in session: + alert = session["alert"] + del session["alert"] + return f"{alert}" + return "" + +def get_nick(): + while "id" in session.keys(): + nick = D.user_get_nick(session["id"]) + if not nick: + del session['id'] + if "quiz_id" in session.keys(): + del session['quiz_id'] + break + return nick + return "(ei nimimerkkiä)" + +def generate_link(): + konso="rtpshjklvnm" + vocal="eyuioa" + str="" + for i in range(4): + str+=konso[randint(0,len(konso)-1)] + str+=vocal[randint(0,len(vocal)-1)] + return str diff --git a/static/alert-timeout.js b/static/alert-timeout.js new file mode 100644 index 0000000..01bd260 --- /dev/null +++ b/static/alert-timeout.js @@ -0,0 +1,5 @@ +Array.from(document.getElementsByClassName('kysAlert')).forEach( (a) => { + a.addEventListener('click', (event) => { + event.target.remove(); + }) + }) diff --git a/static/answer.js b/static/answer.js deleted file mode 100644 index 61882a8..0000000 --- a/static/answer.js +++ /dev/null @@ -1,68 +0,0 @@ -var questions = {} - -createQuestionDiv = ( question ) => { - const questionDiv = document.createElement('div') - questionDiv.className = 'kysQuestion' - - const qDiv = document.createElement('div') - qDiv.appendChild( document.createTextNode( question.q ) ) - qDiv.className = 'kysText' - questionDiv.appendChild( qDiv ) - - const npDiv = document.createElement('div') - npDiv.className = 'kysScale' - - const nDiv = document.createElement('div') - nDiv.appendChild( document.createTextNode( question.n ) ) - nDiv.className = 'kysNegative' - npDiv.appendChild( nDiv ) - - const sDiv = document.createElement('div') - sDiv.className = 'kysScaleSpacer' - npDiv.appendChild( sDiv ) - - const pDiv = document.createElement('div') - pDiv.appendChild( document.createTextNode( question.p ) ) - pDiv.className = 'kysPositive' - npDiv.appendChild( pDiv ) - - questionDiv.appendChild( npDiv ) - - const aInput = document.createElement('input') - aInput.className = 'kysAnswer' - aInput.type = 'range' - aInput.min = 0 - aInput.max = 999 - aInput.value = 500 - aInput.name = question.i - questionDiv.appendChild( aInput ) - - return questionDiv -} - -createQuestions = () => { - const kysForm = document.getElementById('questionForm') - const questionsDiv = document.createElement('div') - Object.keys(questions).forEach(k => { - questionsDiv.appendChild( createQuestionDiv( questions[k] ) ) - }) - kysForm.appendChild( questionsDiv ) - const submitInput = document.createElement('input') - submitInput.type='submit' - submitInput.value='Vastaa kyselyyn' - submitInput.className = 'kysSubmitAnswers' - kysForm.appendChild( submitInput ) -} - -loadQuestions = async() => { - await fetch( 'get/quiz_creator' ) - .then( response => response.json() ) - .then( json => questions = json ) - .catch( error => { - alert("dkd") - } ) - - createQuestions() -} - -loadQuestions() diff --git a/static/create.js b/static/create.js deleted file mode 100644 index d34a2da..0000000 --- a/static/create.js +++ /dev/null @@ -1,62 +0,0 @@ -var questions = {} - -createQuestionDiv = ( question ) => { - const questionDiv = document.createElement('div') - questionDiv.className = 'kysQuestion' - - const qDiv = document.createElement('div') - qDiv.appendChild( document.createTextNode( question.q ) ) - qDiv.className = 'kysText' - questionDiv.appendChild( qDiv ) - - const npDiv = document.createElement('div') - npDiv.className = 'kysScale' - - const nDiv = document.createElement('div') - nDiv.appendChild( document.createTextNode( question.n ) ) - nDiv.className = 'kysNegative' - npDiv.appendChild( nDiv ) - - const sDiv = document.createElement('div') - sDiv.className = 'kysScaleSpacer' - npDiv.appendChild( sDiv ) - - const pDiv = document.createElement('div') - pDiv.appendChild( document.createTextNode( question.p ) ) - pDiv.className = 'kysPositive' - npDiv.appendChild( pDiv ) - - questionDiv.appendChild( npDiv ) - - const aDiv = document.createElement('input') - aDiv.className = 'kysAnswer' - aDiv.type = 'range' - aDiv.min = 0 - aDiv.max = 999 - aDiv.disabled = true - aDiv.value = question.a - questionDiv.appendChild( aDiv ) - - return questionDiv -} - -createQuestions = () => { - const questionsDiv = document.getElementById('questions') - questionsDiv.className = 'kysQuestions' - Object.keys(questions).forEach(k => { - questionsDiv.appendChild( createQuestionDiv( questions[k] ) ) - }) -} - -loadQuestions = async() => { - await fetch( 'get/quiz_creator' ) - .then( response => response.json() ) - .then( json => questions = json ) - .catch( error => { - alert("dkd") - } ) - - createQuestions() -} - -loadQuestions() diff --git a/static/icons/cel-rings-love-svgrepo-com.svg b/static/icons/cel-rings-love-svgrepo-com.svg new file mode 100644 index 0000000..2e49668 --- /dev/null +++ b/static/icons/cel-rings-love-svgrepo-com.svg @@ -0,0 +1,31 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<!-- Uploaded to: SVG Repo, www.svgrepo.com, Generator: SVG Repo Mixer Tools -->
+<svg width="800px" height="800px" viewBox="0 0 64 64" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
+
+ <title>cel-rings-love</title>
+ <desc>Created with Sketch.</desc>
+ <defs>
+
+</defs>
+ <g id="General" stroke="none" stroke-width="1" fill="none" fill-rule="evenodd">
+ <g id="SLICES-64px" transform="translate(-630.000000, 0.000000)">
+
+</g>
+ <g id="ICONS" transform="translate(-625.000000, 5.000000)">
+ <g id="cel-rings-love" transform="translate(630.000000, 2.000000)">
+ <path d="M34.4229,9.3516 C36.5259,7.2126 36.5259,3.7446 34.4229,1.6046 C32.3199,-0.5354 28.9109,-0.5344 26.8079,1.6046 L25.9999,2.3476 L25.1929,1.6046 C23.0899,-0.5344 19.6799,-0.5344 17.5769,1.6046 C15.4739,3.7446 15.4749,7.2126 17.5769,9.3516 L25.9999,17.9996 L34.4229,9.3516 Z" id="Fill-747" fill="#F16963">
+
+</path>
+ <path d="M34.4229,9.3516 C36.5259,7.2126 36.5259,3.7446 34.4229,1.6046 C32.3199,-0.5354 28.9109,-0.5344 26.8079,1.6046 L25.9999,2.3476 L25.1929,1.6046 C23.0899,-0.5344 19.6799,-0.5344 17.5769,1.6046 C15.4739,3.7446 15.4749,7.2126 17.5769,9.3516 L25.9999,17.9996 L34.4229,9.3516 Z" id="Stroke-748" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M28.8071,26.4072 C30.8121,29.0802 32.0001,32.4012 32.0001,36.0002 C32.0001,44.8372 24.8361,52.0002 16.0001,52.0002 C7.1631,52.0002 0.0001,44.8372 0.0001,36.0002 C0.0001,27.1632 7.1631,20.0002 16.0001,20.0002 C18.2091,20.0002 20.3131,20.4472 22.2281,21.2582" id="Stroke-749" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M23.1929,45.5928 C21.1879,42.9198 19.9999,39.5988 19.9999,35.9998 C19.9999,27.1628 27.1639,19.9998 35.9999,19.9998 C44.8369,19.9998 51.9999,27.1628 51.9999,35.9998 C51.9999,44.8368 44.8369,51.9998 35.9999,51.9998 C33.5979,51.9998 31.3199,51.4708 29.2759,50.5228" id="Stroke-750" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ </g>
+ </g>
+ </g>
+</svg>
\ No newline at end of file diff --git a/static/icons/cld-server-svgrepo-com.svg b/static/icons/cld-server-svgrepo-com.svg new file mode 100644 index 0000000..b484198 --- /dev/null +++ b/static/icons/cld-server-svgrepo-com.svg @@ -0,0 +1,64 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<!-- Uploaded to: SVG Repo, www.svgrepo.com, Generator: SVG Repo Mixer Tools -->
+<svg width="800px" height="800px" viewBox="0 0 64 64" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
+
+ <title>cld-server</title>
+ <desc>Created with Sketch.</desc>
+ <defs>
+
+</defs>
+ <g id="General" stroke="none" stroke-width="1" fill="none" fill-rule="evenodd">
+ <g id="SLICES-64px" transform="translate(-810.000000, -200.000000)">
+
+</g>
+ <g id="ICONS" transform="translate(-805.000000, -195.000000)">
+ <g id="cld-server" transform="translate(810.000000, 204.000000)">
+ <path d="M48,12 C51.313,12 54,9.313 54,6 C54,2.687 51.313,0 48,0 L6,0 C2.687,0 0,2.687 0,6 C0,9.313 2.687,12 6,12 L48,12 Z" id="Fill-424" fill="#969CE3">
+
+</path>
+ <path d="M10,6 C10,7.104 9.104,8 8,8 C6.896,8 6,7.104 6,6 C6,4.896 6.896,4 8,4 C9.104,4 10,4.896 10,6" id="Fill-425" fill="#7BBDEC">
+
+</path>
+ <path d="M48,30 C51.313,30 54,27.313 54,24 C54,20.687 51.313,18 48,18 L6,18 C2.687,18 0,20.687 0,24 C0,27.313 2.687,30 6,30 L48,30 Z" id="Fill-426" fill="#969CE3">
+
+</path>
+ <path d="M10,24 C10,25.104 9.104,26 8,26 C6.896,26 6,25.104 6,24 C6,22.896 6.896,22 8,22 C9.104,22 10,22.896 10,24" id="Fill-427" fill="#7BBDEC">
+
+</path>
+ <path d="M48,48 C51.313,48 54,45.313 54,42 C54,38.687 51.313,36 48,36 L6,36 C2.687,36 0,38.687 0,42 C0,45.313 2.687,48 6,48 L48,48 Z" id="Fill-428" fill="#969CE3">
+
+</path>
+ <path d="M10,42 C10,43.104 9.104,44 8,44 C6.896,44 6,43.104 6,42 C6,40.896 6.896,40 8,40 C9.104,40 10,40.896 10,42" id="Fill-429" fill="#7BBDEC">
+
+</path>
+ <path d="M48,12 C51.313,12 54,9.313 54,6 C54,2.687 51.313,0 48,0 L6,0 C2.687,0 0,2.687 0,6 C0,9.313 2.687,12 6,12 L48,12 Z" id="Stroke-430" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M10,6 C10,7.104 9.104,8 8,8 C6.896,8 6,7.104 6,6 C6,4.896 6.896,4 8,4 C9.104,4 10,4.896 10,6 Z" id="Stroke-431" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M48,6 L36,6" id="Stroke-432" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M48,30 C51.313,30 54,27.313 54,24 C54,20.687 51.313,18 48,18 L6,18 C2.687,18 0,20.687 0,24 C0,27.313 2.687,30 6,30 L48,30 Z" id="Stroke-433" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M10,24 C10,25.104 9.104,26 8,26 C6.896,26 6,25.104 6,24 C6,22.896 6.896,22 8,22 C9.104,22 10,22.896 10,24 Z" id="Stroke-434" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M48,24 L36,24" id="Stroke-435" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M48,48 C51.313,48 54,45.313 54,42 C54,38.687 51.313,36 48,36 L6,36 C2.687,36 0,38.687 0,42 C0,45.313 2.687,48 6,48 L48,48 Z" id="Stroke-436" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M10,42 C10,43.104 9.104,44 8,44 C6.896,44 6,43.104 6,42 C6,40.896 6.896,40 8,40 C9.104,40 10,40.896 10,42 Z" id="Stroke-437" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M48,42 L36,42" id="Stroke-438" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ </g>
+ </g>
+ </g>
+</svg>
\ No newline at end of file diff --git a/static/icons/cle-spraycan-svgrepo-com.svg b/static/icons/cle-spraycan-svgrepo-com.svg new file mode 100644 index 0000000..86e3c64 --- /dev/null +++ b/static/icons/cle-spraycan-svgrepo-com.svg @@ -0,0 +1,70 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<!-- Uploaded to: SVG Repo, www.svgrepo.com, Generator: SVG Repo Mixer Tools -->
+<svg width="800px" height="800px" viewBox="0 0 64 64" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
+
+ <title>cle-spraycan</title>
+ <desc>Created with Sketch.</desc>
+ <defs>
+
+</defs>
+ <g id="General" stroke="none" stroke-width="1" fill="none" fill-rule="evenodd">
+ <g id="SLICES-64px" transform="translate(-720.000000, -100.000000)">
+
+</g>
+ <g id="ICONS" transform="translate(-715.000000, -95.000000)">
+ <g id="cle-spraycan" transform="translate(728.000000, 100.000000)">
+ <path d="M3,54 L25,54 C26.657,54 28,52.657 28,51 L28,20 L0,20 L0,51 C0,52.657 1.343,54 3,54" id="Fill-295" fill="#F16963">
+
+</path>
+ <polygon id="Fill-296" fill="#E9EFFA" points="18 12 10 12 0 20 28 20">
+
+</polygon>
+ <polygon id="Fill-297" fill="#E9EFFA" points="10 12 18 12 18 4 10 4">
+
+</polygon>
+ <polygon id="Fill-298" fill="#F1F0E2" points="16 46 28 46 28 26 16 26">
+
+</polygon>
+ <polyline id="Stroke-299" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" points="0 20 10 12 18 12 28 20">
+
+</polyline>
+ <polyline id="Stroke-300" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" points="10 12 10 4 18 4 18 12">
+
+</polyline>
+ <polyline id="Stroke-301" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" points="28 46 16 46 16 26 28 26">
+
+</polyline>
+ <path d="M3,54 L25,54 C26.657,54 28,52.657 28,51 L28,20 L0,20 L0,51 C0,52.657 1.343,54 3,54 Z" id="Stroke-302" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M22,8 L24.5,8" id="Stroke-303" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M28.5527,8 L33.4737,8" id="Stroke-304" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" stroke-dasharray="2.895,4.053">
+
+</path>
+ <path d="M35.5,8 L38,8" id="Stroke-305" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M22,5 L24.354,4.159" id="Stroke-306" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M27.7773,2.9365 L31.9333,1.4525" id="Stroke-307" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" stroke-dasharray="2.596,3.635">
+
+</path>
+ <path d="M33.6455,0.8408 L35.9995,-0.0002" id="Stroke-308" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M22,11 L24.354,11.841" id="Stroke-309" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M27.7773,13.0635 L31.9333,14.5475" id="Stroke-310" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" stroke-dasharray="2.596,3.635">
+
+</path>
+ <path d="M33.6455,15.1592 L35.9995,16.0002" id="Stroke-311" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ </g>
+ </g>
+ </g>
+</svg>
\ No newline at end of file diff --git a/static/icons/con-ruler-pencil-svgrepo-com.svg b/static/icons/con-ruler-pencil-svgrepo-com.svg new file mode 100644 index 0000000..2d3d484 --- /dev/null +++ b/static/icons/con-ruler-pencil-svgrepo-com.svg @@ -0,0 +1,91 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<!-- Uploaded to: SVG Repo, www.svgrepo.com, Generator: SVG Repo Mixer Tools -->
+<svg width="800px" height="800px" viewBox="0 0 64 64" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
+
+ <title>con-ruler-pencil</title>
+ <desc>Created with Sketch.</desc>
+ <defs>
+
+</defs>
+ <g id="General" stroke="none" stroke-width="1" fill="none" fill-rule="evenodd">
+ <g id="SLICES-64px" transform="translate(-540.000000, -300.000000)">
+
+</g>
+ <g id="ICONS" transform="translate(-535.000000, -295.000000)">
+ <g id="con-ruler-pencil" transform="translate(548.000000, 298.000000)">
+ <polygon id="Fill-501" fill="#EEC261" points="0 58 14 58 14 0 0 0">
+
+</polygon>
+ <path d="M33,0 L27,0 C25.344,0 24,1.344 24,3 L24,8 L36,8 L36,3 C36,1.344 34.656,0 33,0" id="Fill-502" fill="#F16963">
+
+</path>
+ <polygon id="Fill-503" fill="#99A5B7" points="32.0371 52 27.9631 52 30.0001 58">
+
+</polygon>
+ <polygon id="Fill-504" fill="#F3E777" points="24.375 41.3662 28 38.0002 32 42.0002 36 38.0002 36 8.0002 24 8.0002 24 40.3262 24.355 41.3732">
+
+</polygon>
+ <polygon id="Fill-505" fill="#F1F0E2" points="36 38 32 42 28 38 24.375 41.366 24.355 41.373 27.963 52 32.037 52 36 40.326">
+
+</polygon>
+ <path d="M27.9629,52 L24.3559,41.372" id="Stroke-506" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <polyline id="Stroke-507" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" points="36 37.9995 36 40.3265 32.037 52.0005">
+
+</polyline>
+ <polyline id="Stroke-508" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" points="24.375 41.3662 28 38.0002 32 42.0002 36 38.0002">
+
+</polyline>
+ <path d="M27.9629,52 L32.0369,52" id="Stroke-509" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M28,34 L28,12" id="Stroke-510" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <polyline id="Stroke-511" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" points="32.0371 52.0005 30.0001 58.0005 27.9631 52.0005">
+
+</polyline>
+ <polyline id="Stroke-512" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" points="24.3555 41.3721 24.0005 40.3261 24.0005 8.0001">
+
+</polyline>
+ <path d="M36,8 L36,38" id="Stroke-513" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <polygon id="Stroke-514" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" points="0 58 14 58 14 0 0 0">
+
+</polygon>
+ <path d="M6,32 L14,32" id="Stroke-515" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M10,26 L14,26" id="Stroke-516" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M6,20 L14,20" id="Stroke-517" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M6,8 L14,8" id="Stroke-518" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M10,14 L14,14" id="Stroke-519" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M8,50 L14,50" id="Stroke-520" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M6,44 L14,44" id="Stroke-521" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M10,38 L14,38" id="Stroke-522" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M24,8 L36,8" id="Stroke-523" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <path d="M24,8 L24,3 C24,1.344 25.344,0 27,0 L33,0 C34.656,0 36,1.344 36,3 L36,8" id="Stroke-524" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ </g>
+ </g>
+ </g>
+</svg>
\ No newline at end of file diff --git a/static/icons/gen-heart-rate-svgrepo-com.svg b/static/icons/gen-heart-rate-svgrepo-com.svg new file mode 100644 index 0000000..8880970 --- /dev/null +++ b/static/icons/gen-heart-rate-svgrepo-com.svg @@ -0,0 +1,28 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<!-- Uploaded to: SVG Repo, www.svgrepo.com, Generator: SVG Repo Mixer Tools -->
+<svg width="800px" height="800px" viewBox="0 0 64 64" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
+
+ <title>gen-heart-rate</title>
+ <desc>Created with Sketch.</desc>
+ <defs>
+
+</defs>
+ <g id="General" stroke="none" stroke-width="1" fill="none" fill-rule="evenodd">
+ <g id="SLICES-64px" transform="translate(-180.000000, 0.000000)">
+
+</g>
+ <g id="ICONS" transform="translate(-175.000000, 5.000000)">
+ <g id="gen-heart-rate" transform="translate(180.000000, 6.000000)">
+ <path d="M26.0001,5.9995 L28.1011,4.1005 C33.5671,-1.3665 42.4331,-1.3665 47.8991,4.1005 C53.3671,9.5685 53.3661,18.4325 47.8991,23.8985 L26.0001,45.9995 L4.1011,23.8985 C-1.3659,18.4325 -1.3669,9.5685 4.1011,4.1005 C9.5671,-1.3665 18.4331,-1.3665 23.8991,4.1005 L26.0001,5.9995 Z" id="Fill-249" fill="#F16963">
+
+</path>
+ <path d="M26.0001,5.9995 L28.1011,4.1005 C33.5671,-1.3665 42.4331,-1.3665 47.8991,4.1005 C53.3671,9.5685 53.3661,18.4325 47.8991,23.8985 L26.0001,45.9995 L4.1011,23.8985 C-1.3659,18.4325 -1.3669,9.5685 4.1011,4.1005 C9.5671,-1.3665 18.4331,-1.3665 23.8991,4.1005 L26.0001,5.9995 Z" id="Stroke-250" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
+
+</path>
+ <polyline id="Stroke-251" stroke="#414547" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" points="16.0001 23.9995 23.0001 23.9995 26.0001 11.9995 31.0001 33.9995 35.0001 19.9995 37.0001 23.9995 52.0001 23.9995">
+
+</polyline>
+ </g>
+ </g>
+ </g>
+</svg>
\ No newline at end of file diff --git a/static/kyselma.css b/static/kyselma.css index a7a60db..14c7653 100644 --- a/static/kyselma.css +++ b/static/kyselma.css @@ -2,13 +2,17 @@ .kysQuestion { display: flex; flex-direction: column; - max-width: 25em; + max-width: 25em; + padding: 0.5em; + border: 0.15em #fed solid; } .kysText { align-self: center; } .kysScale { display: flex; + width: 100%; + resize: horizontal; } .kysScaleSpacer { flex-grow: 1; @@ -26,3 +30,63 @@ .kysCreateText { width: 100%; } + +.kysNick::before { + content: "Olet:"; + padding-right: 0.4em; +} + +.kysNick { + border: 0.1em solid; + padding: 0.4em; + position: fixed; + top: 100vh; + left: 100vw; + transform: translate(-100%, -100%); + display: block; + background-color: #DAF; +} + +.kysAlert { + border: 0.4em solid; + padding: 0.8em; + display: block; + background-color: #F55; + border-top-color: #F97; + border-left-color: #F96; + border-right-color: #F13; + border-bottom-color: #F14; + position: fixed; + top: 50vh; + left: 50vw; + transform: translate(-50%, -50%); +} + +.kysTotal { + align-self: center; + font-size: xxx-large; +} + +.kysSelectU1 { + background: #faa; +} + +.kysSelectU2 { + background: #aaf; +} + +.kysUser1::-webkit-slider-runnable-track, .kysUser1::-moz-range-track { + background: #faa; +} + +.kysUser2::-webkit-slider-runnable-track, .kysUser2::-moz-range-track { + background: #aaf; +} + +.kysUser1::-webkit-slider-thumb, .kysUser1::-moz-range-thumb { + background: #c00; +} + +.kysUser2::-webkit-slider-thumb, .kysUser2::-moz-range-thumb { + background: #00c; +} diff --git a/static/menu.css b/static/menu.css index 04127ae..4cf7c5f 100644 --- a/static/menu.css +++ b/static/menu.css @@ -93,7 +93,7 @@ .menuHidden, .menuSpacer, .menuTitle { display: flex !important } .menuSpacer { flex-grow: 2; } .menuText { - padding-left: calc(var(--menuHeight)/2); + padding-left: calc(var(--menuHeight)/8); padding-right: calc(var(--menuHeight)/2); } } diff --git a/static/menu.js b/static/menu.js index 5256ef2..8d0baee 100644 --- a/static/menu.js +++ b/static/menu.js @@ -132,7 +132,7 @@ hashToPage = async () => { else document.getElementById(`${currentPage}_page`).className = 'page' pageElement.className = 'page pageActive' - if ( !pageElement.loaded ) await fetch( pages[p].URL ) + if ( pages[p].dynamic || !pageElement.loaded ) await fetch( pages[p].URL ) .then( response => { if (!response.ok) return `ERROR loading "${pages[p].URL}"!` return response.text() @@ -141,10 +141,10 @@ hashToPage = async () => { pageElement.innerHTML = text; pageElement.loaded = true; } ) - if ( pageElement.innerHTML.startsWith("redirect = ") ) { +/* if ( pageElement.innerHTML.startsWith("redirect = ") ) { pageElement.loaded = false; window.location.assign( pageElement.innerHTML.slice( 11 ) ); - } + }*/ // https://plnkr.co/edit/MMegiu by Allen Kim Array.from(pageElement.querySelectorAll("script")).forEach(oldScript => { diff --git a/static/pages.json b/static/pages.json index 618bc7a..37e59e4 100644 --- a/static/pages.json +++ b/static/pages.json @@ -3,7 +3,7 @@ "pages": { "info": { "URL": "pages/info.html", - "menuIcon": "kyselma.svg", + "menuIcon": "icons/gen-heart-rate-svgrepo-com.svg", "menuName": "Kyselmä" }, "spacer-1": { @@ -11,27 +11,36 @@ "create": { "pageTitle": "Luo kysely", "URL": "pages/create.html", - "menuName": "Luo" + "menuIcon": "icons/con-ruler-pencil-svgrepo-com.svg", + "menuName": "Luo", + "dynamic": "yes" }, "answer": { "pageTitle": "Vastaa kyselyyn", "URL": "pages/answer.html", - "menuName": "Vastaa" + "menuIcon": "icons/cld-server-svgrepo-com.svg", + "menuName": "Vastaa", + "dynamic": "yes" }, "analyse": { "pageTitle": "Tutki vastauksia", "URL": "pages/analyse.html", - "menuName": "Tutki" + "menuIcon": "icons/cel-rings-love-svgrepo-com.svg", + "menuName": "Tutki", + "dynamic": "yes" }, "moderate": { "pageTitle": "Moderoi sivustoa", "URL": "pages/moderate.html", + "menuIcon": "icons/cle-spraycan-svgrepo-com.svg", + "hidden": "yes", "menuName": "Moderoi" }, "nick": { "pageTitle": "Anna nimimerkki", "URL": "pages/nick.html", - "hidden": "yes" + "hidden": "yes", + "dynamic": "yes" }, "question": { "pageTitle": "Lisää kysymys", @@ -39,9 +48,14 @@ "hidden": "yes" }, "quiz": { - "pageTitle": "Rakenna kysely", + "pageTitle": "Luodaanko uusi kyselmä?", "URL": "pages/quiz.html", "hidden": "yes" + }, + "new_answer": { + "pageTitle": "Anna kyselyn koodi", + "URL": "pages/new_answer.html", + "hidden": "yes" } } } diff --git a/templates/analyse.html b/templates/analyse.html index 6970764..70fb98e 100644 --- a/templates/analyse.html +++ b/templates/analyse.html @@ -1,4 +1,73 @@ -<div id="nickbox">{{ nick }}</div> -<div id="alertbox">{{ alert }}</div> -<h1>analyse</h1> +{% include 'base.html' %} + +{% if users %} + +Tutkit kyselyä: {{ code }} + + +<div class="kysQuestion"><div class="kysScale"> +<form action="/set/compare" method="POST"> +<input type="submit" value="Vertaa" class="kysButton"> +<select class="kysSelectU1 kysSelect" name="user1"> +{% for user in users %} + {% if user1==user.id %} + <option value="{{ user.id }}" selected>{{ user.nick }}</option> + {% else %} + <option value="{{ user.id }}">{{ user.nick }}</option> + {% endif %} +{% endfor %} +</select> +<select class="kysSelectU2 kysSelect" name="user2"> +{% for user in users %} + {% if user2==user.id %} + <option value="{{ user.id }}" selected>{{ user.nick }}</option> + {% else %} + <option value="{{ user.id }}">{{ user.nick }}</option> + {% endif %} +{% endfor %} +</select> +<div class="kysScaleSpacer"></div> +</form> +</div></div> + +{% endif %} + +{% if code %} + +<div id="questions" class="kysQuestions"> + <div class="kysQuestion"> + <div class="kysTotal">{{ avg }}%</div> + </div> +{% for q in questions %} +<div class="kysQuestion"> + <div class="kysText">{{ q.q }} ({{ q.c }}%)</div> + <input class="kysAnswer kysUser1" type="range" min="0" max="999" + value="{{ q.a1 }}" disabled=""> + <input class="kysAnswer kysUser2" type="range" min="0" max="999" + value="{{ q.a2 }}" disabled=""> + <div class="kysScale"> + <div class="kysNegative">{{ q.n }}</div> + <div class="kysScaleSpacer"></div> + <div class="kysPositive">{{ q.p }}</div> + </div> +</div> +{% endfor %} + +</div> + +{% endif %} + +<form action="/set/answer_id" method="POST"> +Vaihda kyselyn koodia: +<input type="text" name="link"> +<input type="text" name="next" value="analyse" hidden="true"> +<input type="submit" value="Vaihda"> +</form> + +{% if code %} +Linkki tähän kyselyyn: +<a href="http://anal.fi:5000/kys/{{ code }}"> + http://anal.fi:5000/kys/{{ code }} +</a> +{% endif %} diff --git a/templates/answer.html b/templates/answer.html index 0ed0a05..eb49ff8 100644 --- a/templates/answer.html +++ b/templates/answer.html @@ -1,5 +1,21 @@ -<script src="answer.js"></script> -<div id="nickbox">{{ nick }}</div> -<div id="alertbox">{{ alert }}</div> -<h1>answer NOW!</h1> -<form id=questionForm action="/set/answers" method="POST"></form> +{% include 'base.html' %} + +Vastaa kyselmään "{{ link }}": + +<form class="kysQuestionaire" action="/set/answers" method="POST"> +<div id="questions" class="kysQuestions"> +{% for q in questions %} +<div class="kysQuestion"> +<div class="kysText">{{ q.q }}</div> +<div class="kysScale"> + <div class="kysNegative">{{ q.n }}</div> + <div class="kysScaleSpacer"></div> + <div class="kysPositive">{{ q.p }}</div> +</div> +<input class="kysAnswer" type="range" min="0" max="999" name="{{ q.i }}"> +</div> +{% endfor %} +<input class="kysSubmitAnswers" type="submit" value="Vastaa kyselyyn"> +</div> +</form> + diff --git a/templates/base.html b/templates/base.html new file mode 100644 index 0000000..839cc88 --- /dev/null +++ b/templates/base.html @@ -0,0 +1,7 @@ +{% if nick %} + <div class="kysNick">{{ nick }}</div> +{% endif %} +{% if alert %} + <div class="kysAlert">{{ alert }}</div> + <script src="alert-timeout.js"></script> +{% endif %} diff --git a/templates/create.html b/templates/create.html index fe2aa24..b369ae8 100644 --- a/templates/create.html +++ b/templates/create.html @@ -1,9 +1,29 @@ -<script src="create.js"></script> -<div id="nickbox">{{ nick }}</div> -<div id="alertbox">{{ alert }}</div> -<h1>create</h1> +{% include 'base.html' %} -<div id=questions></div> -<a href="#question">Lisää kysymys</a> +<div id="questions" class="kysQuestions"> +{% for q in questions %} +<div class="kysQuestion"> +<div class="kysText">{{ q.q }}</div> +<div class="kysScale"> + <div class="kysNegative">{{ q.n }}</div> + <div class="kysScaleSpacer"></div> + <div class="kysPositive">{{ q.p }}</div> +</div> +<input class="kysAnswer" type="range" min="0" max="999" + value="{{ q.a }}" disabled=""></div> +{% endfor %} + +<div class="kysQuestion"> +<div class="kysScale"> +<a href="#question"> + <input type="button" value="Lisää kysymys" class="kysButton"> +</a> +<div class="kysScaleSpacer"></div> +<form action="/set/quiz_ready" method="POST"> +<input type="text" name="ok" hidden=true> +<input type="submit" value="Valmis" class="kysButton"> </form> +<div class="kysScale"> +</div> +</div>
\ No newline at end of file diff --git a/templates/info.html b/templates/info.html index 3f984a6..30e84e2 100644 --- a/templates/info.html +++ b/templates/info.html @@ -1,2 +1,11 @@ -<div id="alertbox">{{ alert }}</div> -<h1>info</h1> +{% include 'base.html' %} +<h1>Kyselmä</h1> + +Tervetuloa kyselmien ihmeelliseen maailmaan. Täällä laadit ikioman kyselmän +ja vastailet kavereidesi luomiin kyselmiin. Vastattuasi pääset vertailemaan +vastauksiasi muiden vatausten kassa. Alkuun pääset painamalla ylhäätä "Luo" +ja tekemällä uuden kyselmän. Tämän jälkeen vain lisäät niin monta kysymystä +vastausvaihtoehtoineen kuin haluat ja lopuksi painat valmis, jonka jälkeen +muutkin pääsee vastaamaan kyselmään koodilla. Tarkastelussa voit valita +mitkä tahansa kaksi vastaaja ja tutkia vastausten samankaltaisuutta ja +kaiken huipuksi vastauksista lasketaan yhtäläisyys prosentti. diff --git a/templates/moderate.html b/templates/moderate.html index 116f2ff..118c8d4 100644 --- a/templates/moderate.html +++ b/templates/moderate.html @@ -1,2 +1,2 @@ -<div id="alertbox">{{ alert }}</div> +{% include 'base.html' %} <h1>moderate</h1> diff --git a/templates/new_answer.html b/templates/new_answer.html new file mode 100644 index 0000000..0d690d4 --- /dev/null +++ b/templates/new_answer.html @@ -0,0 +1,7 @@ +{% include 'base.html' %} +<form action="/set/answer_id" method="POST"> +Vastaa kyselyyn koodilla: +<input type="text" name="link"> +<input type="submit" value="Kyselmään"> +</form> +Saatoit myös päätyä tänne mikäli olet jo vastannut. diff --git a/templates/nick.html b/templates/nick.html index fe4a50a..92e166f 100644 --- a/templates/nick.html +++ b/templates/nick.html @@ -1,6 +1,6 @@ -<div id="alertbox">{{ alert }}</div> +{% include 'base.html' %} <form action="/set/nick" method="POST"> Anna itsellesi nimimerkki ensin: -<input type="text" name="nick" rows="1" cols="40"></textarea> +<input type="text" name="nick"> <input type="submit" value="Lähetä"> </form> diff --git a/templates/question.html b/templates/question.html index 016a1ac..62afaaf 100644 --- a/templates/question.html +++ b/templates/question.html @@ -1,4 +1,4 @@ -<div id="kysAlert">{{ alert }}</div> +{% include 'base.html' %} <form action="/set/question" method="POST"> <div class="kysQuestion"> <div class="kysCreateText">Kysymys:</div> diff --git a/templates/quiz.html b/templates/quiz.html index b2083ae..9e356c3 100644 --- a/templates/quiz.html +++ b/templates/quiz.html @@ -1,4 +1,4 @@ -<div id="alertbox">{{ alert }}</div> +{% include 'base.html' %} <form action="/set/quiz" method="POST"> <input type="submit" value="Aloita uusi kyselmä"> </form> |