26 Commits

Author SHA1 Message Date
96cbac2702 Revert "0.1.2"
This reverts commit 1d788671c6.
2021-04-07 17:00:54 +02:00
1d788671c6 0.1.2 2021-04-07 16:59:42 +02:00
e7458c49f0 Now also triggering electron build
ref #14
2021-04-07 16:59:35 +02:00
2ec6e1c107 Merge branch 'feature/14-release_config' of git.odit.services:lfk/scanclient into feature/14-release_config 2021-04-07 16:51:47 +02:00
3c2300c09e Updated package version to the current version
ref #14
2021-04-07 16:51:46 +02:00
69ab9893d3 Updated version to the current electron version
ref #14
2021-04-07 16:51:25 +02:00
dbff163b6b Added release-it w/ config
ref #14
2021-04-07 16:50:38 +02:00
703168081c Merge pull request 'Symbols for valid/invalid scans feature/11-valid_icons' (#12) from feature/11-valid_icons into dev
Reviewed-on: #12
2021-04-06 19:01:07 +00:00
64398cdd74 Now also showing for errors
ref #11
2021-04-06 20:53:21 +02:00
a4d749cc3f Added svg titels
ref #11
2021-04-06 18:07:13 +02:00
838127cf72 Added missing translations
ref #11
2021-04-06 18:05:04 +02:00
379999e491 Changed test sizes to fit the page on smaller screens
ref #11
2021-04-06 18:03:13 +02:00
be6974af20 Added valid/invalid svg icons
ref #11
2021-04-06 18:00:31 +02:00
c0555c0662 Added valid/invalid logic
ref #11
2021-04-06 18:00:05 +02:00
db4d63da8b Merge pull request 'feature/9-move-url-config-from-env-to-local-config' (#10) from feature/9-move-url-config-from-env-to-local-config into dev
Reviewed-on: #10
close #9
2021-03-23 17:35:48 +00:00
b80a832256 Login - add trailing slash to api_endpoint url
ref #9
2021-03-23 18:10:58 +01:00
81b2db60ec 🧹 drop old env.js
ref #9
2021-03-22 19:28:42 +01:00
5563f1fca3 new api_endpoint store + Config UI
ref #9
2021-03-22 19:26:03 +01:00
517cfddc5f CI - mkdir out 2021-03-19 18:49:26 +01:00
ae0ec9d67c CI fix 2021-03-19 18:44:38 +01:00
cfd40d3f19 re-enable CI 2021-03-19 18:25:28 +01:00
5c117d6f48 Merge branch 'main' into dev 2021-03-19 16:40:32 +00:00
0296c26479 Merge branch 'feature/4-license-exporter' into dev
close #4 , close #5
2021-03-19 17:36:50 +01:00
d0facb2846 added husky formatting 🐶
ref #5
2021-03-19 17:33:23 +01:00
7deb0d26c4 added license:export script to pre-commit hook
ref #4
2021-03-19 17:32:54 +01:00
00b8a14bc3 husky 🐶 2021-03-19 17:24:39 +01:00
15 changed files with 5285 additions and 75 deletions

55
.drone.yml Normal file
View File

@@ -0,0 +1,55 @@
---
kind: secret
name: gitea_token
get:
path: odit-git-bot
name: apikey
---
kind: secret
name: git_ssh
get:
path: odit-git-bot
name: sshkey
---
kind: pipeline
type: kubernetes
name: build:tag
steps:
- name: run build
image: node:15.11.0-alpine3.13
commands:
- apk add git zip -f
- yarn
- yarn build
- mkdir out
- zip -r out/dist.zip dist
- name: gitea add packages to build
image: plugins/gitea-release
settings:
api_key:
from_secret: gitea_token
base_url: https://git.odit.services
files: out/*
checksum:
- md5
- sha1
- sha256
- sha512
- adler32
- crc32
- name: checkout electron
image: alpine/git
commands:
- git clone https://git.odit.services/lfk/scanclient-electron electron
- cd electron
- name: tag new version
image: node:alpine
commands:
- yarn
- yarn build
- npm --no-git-tag-version version ${DRONE_TAG}
trigger:
event:
- tag

1
.husky/.gitignore vendored Normal file
View File

@@ -0,0 +1 @@
_

5
.husky/pre-commit Normal file
View File

@@ -0,0 +1,5 @@
#!/bin/sh
. "$(dirname "$0")/_/husky.sh"
yarn format
yarn license:export

View File

@@ -9,7 +9,6 @@
</head> </head>
<body class="bg-white font-family-karla h-screen"> <body class="bg-white font-family-karla h-screen">
<script src="./env.js"></script>
<script type="module" src="./src/main.js"></script> <script type="module" src="./src/main.js"></script>
</body> </body>
</html> </html>

5033
licenses.md Normal file

File diff suppressed because it is too large Load Diff

View File

@@ -1,25 +1,47 @@
{ {
"name": "@lfk/scanclient", "name": "@lfk/scanclient",
"version": "0.0.0", "version": "0.1.1",
"scripts": { "scripts": {
"dev": "vite", "dev": "vite",
"build": "vite build", "build": "vite build",
"format": "prettier --write --plugin-search-dir=. ./**/*.html ./**/*.svelte" "format": "prettier --write --plugin-search-dir=. ./**/*.html ./**/*.svelte",
"prepare": "husky install",
"license:export": "license-exporter --markdown && git stage licenses.md",
"release": "release-it --only-version"
}, },
"devDependencies": { "devDependencies": {
"@odit/license-exporter": "^0.0.11",
"@svitejs/vite-plugin-svelte": "^0.11.1", "@svitejs/vite-plugin-svelte": "^0.11.1",
"@tsconfig/svelte": "^1.0.10", "@tsconfig/svelte": "^1.0.10",
"@types/html-minifier": "^4.0.0", "@types/html-minifier": "^4.0.0",
"axios": "^0.21.1", "axios": "^0.21.1",
"html-minifier": "^4.0.0", "html-minifier": "^4.0.0",
"husky": "^5.1.3",
"prettier": "^2.2.1", "prettier": "^2.2.1",
"prettier-plugin-svelte": "^2.2.0", "prettier-plugin-svelte": "^2.2.0",
"release-it": "^14.5.1",
"svelte": "^3.35.0", "svelte": "^3.35.0",
"svelte-i18n": "^3.3.7",
"svelte-preprocess": "^4.6.9", "svelte-preprocess": "^4.6.9",
"vite": "^2.1.2", "vite": "^2.1.2",
"vite-plugin-windicss": "^0.9.2" "vite-plugin-windicss": "^0.9.2"
}, },
"dependencies": { "release-it": {
"svelte-i18n": "^3.3.7" "git": {
"commit": true,
"requireCleanWorkingDir": false,
"commitMessage": "🚀Bumped version to v${version}",
"requireBranch": "dev",
"push": false,
"tag": true,
"tagName": null,
"tagAnnotation": "v${version}"
},
"npm": {
"publish": false
},
"hooks": {
"after:bump": "npx auto-changelog --commit-limit false -p -u --hide-credit && git add CHANGELOG.md && node order.js && git add src/locales"
}
} }
} }

View File

@@ -1,3 +0,0 @@
const config = {
endpoint: 'https://dev.lauf-fuer-kaya.de/'
};

View File

@@ -1,3 +0,0 @@
const config = {
endpoint: 'https://dev.lauf-fuer-kaya.de/'
};

View File

@@ -1,5 +1,5 @@
<script> <script>
console.log("app started with base url " + config.endpoint); import { apikey, api_endpoint, lang, page } from "./store.js";
import { addMessages, init } from "svelte-i18n"; import { addMessages, init } from "svelte-i18n";
import en from "./locales/en.json"; import en from "./locales/en.json";
import de from "./locales/de.json"; import de from "./locales/de.json";
@@ -11,13 +11,13 @@
import Scanner from "./Scanner.svelte"; import Scanner from "./Scanner.svelte";
import Login from "./Login.svelte"; import Login from "./Login.svelte";
import Settings from "./Settings.svelte"; import Settings from "./Settings.svelte";
import { apikey, lang, page } from "./store.js";
$: is_configured = $apikey && $apikey !== "null" && $apikey !== ""; $: is_configured = $apikey && $apikey !== "null" && $apikey !== "";
$: settings_open = $page === "settings"; $: settings_open = $page === "settings";
init({ init({
fallbackLocale: "en-US", fallbackLocale: "en-US",
initialLocale: $lang, initialLocale: $lang,
}); });
console.log("app started with base url " + $api_endpoint);
</script> </script>
{#if settings_open && is_configured} {#if settings_open && is_configured}

View File

@@ -1,16 +1,27 @@
<script> <script>
import { apikey, lang, stationinfo } from "./store.js"; import { apikey, lang, stationinfo, api_endpoint } from "./store.js";
import axios from "axios"; import axios from "axios";
import { _, locale } from "svelte-i18n"; import { _, locale } from "svelte-i18n";
let token; let token;
let api_endpoint_input;
$: error = false; $: error = false;
$: errormessage = ""; $: errormessage = "";
$: isTokenValid = $: isTokenValid =
token?.length === 44 && token?.length === 44 &&
token.split(".")[0].length === 7 && token.split(".")[0].length === 7 &&
isUUID(token.split(".")[1]); isUUID(token.split(".")[1]);
function isLocale(l) { $: isEndpointValid = validURL(api_endpoint_input);
return $locale == l; function validURL(str) {
var pattern = new RegExp(
"^(https?:\\/\\/)?" + // protocol
"((([a-z\\d]([a-z\\d-]*[a-z\\d])*)\\.)+[a-z]{2,}|" + // domain name
"((\\d{1,3}\\.){3}\\d{1,3}))" + // OR ip (v4) address
"(\\:\\d+)?(\\/[-a-z\\d%_.~+]*)*" + // port and path
"(\\?[;&a-z\\d%_.~+=-]*)?" + // query string
"(\\#[-a-z\\d_]*)?$",
"i"
); // fragment locator
return !!pattern.test(str);
} }
function isUUID(uuid) { function isUUID(uuid) {
let s = "" + uuid; let s = "" + uuid;
@@ -71,57 +82,99 @@
</div> </div>
{/if} {/if}
{/if} {/if}
{#if $api_endpoint}
<form <form
class="flex flex-col pt-3 md:pt-8" class="flex flex-col pt-3 md:pt-8"
onsubmit="event.preventDefault();" onsubmit="event.preventDefault();"
on:submit={() => { on:submit={() => {
axios axios
.request({ .request({
method: "GET", method: "GET",
url: config.endpoint + "api/stations/me", url: $api_endpoint + "api/stations/me",
headers: { Authorization: "Bearer " + token }, headers: { Authorization: "Bearer " + token },
}) })
.then(function (response) { .then(function (response) {
error = false; error = false;
errormessage = ""; errormessage = "";
apikey.set(token); apikey.set(token);
stationinfo.set(JSON.stringify(response.data)); stationinfo.set(JSON.stringify(response.data));
}) })
.catch(function (e) { .catch(function (e) {
error = true; error = true;
errormessage = e.response.data.short; errormessage = e.response.data.short;
}); });
}} }}
>
<div class="flex flex-col pt-4">
<label for="token" class="text-lg">{$_("client_token")}</label>
<input
type="text"
id="token"
placeholder={$_("client_token")}
bind:value={token}
class:border-red-500={!isTokenValid}
class:border-solid={!isTokenValid}
class:border-3={!isTokenValid}
class="shadow appearance-none border rounded w-full py-2 px-3 text-gray-700 mt-1 leading-tight focus:outline-none focus:shadow-outline"
/>
</div>
{#if !isTokenValid}
<span class="text-sm"
>{$_("please_provide_a_valid_client_token")}</span
>
{/if}
<button
disabled={!isTokenValid}
class:cursor-pointer={isTokenValid}
class:opacity-50={!isTokenValid}
id="configure"
type="submit"
class="bg-black text-white font-bold text-lg hover:bg-gray-700 p-2 mt-8 hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-black"
>{$_("configure")}</button
> >
</form> <div class="flex flex-col pt-4">
<label for="token" class="text-lg">{$_("client_token")}</label>
<input
type="text"
id="token"
placeholder={$_("client_token")}
bind:value={token}
class:border-red-500={!isTokenValid}
class:border-solid={!isTokenValid}
class:border-3={!isTokenValid}
class="shadow appearance-none border rounded w-full py-2 px-3 text-gray-700 mt-1 leading-tight focus:outline-none focus:shadow-outline"
/>
</div>
{#if !isTokenValid}
<span class="text-sm"
>{$_("please_provide_a_valid_client_token")}</span
>
{/if}
<button
disabled={!isTokenValid}
class:cursor-pointer={isTokenValid}
class:opacity-50={!isTokenValid}
id="configure"
type="submit"
class="bg-black text-white font-bold text-lg hover:bg-gray-700 p-2 mt-8 hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-black"
>{$_("configure")}</button
>
</form>
{:else}
<form
class="flex flex-col pt-3 md:pt-8"
onsubmit="event.preventDefault();"
on:submit={() => {
if (api_endpoint_input.substr(-1) !== "/") {
api_endpoint_input = api_endpoint_input + "/";
}
api_endpoint.set(api_endpoint_input);
}}
>
<div class="flex flex-col pt-4">
<label for="api_endpoint" class="text-lg"
>{$_("api_endpoint")}</label
>
<input
type="text"
id="api_endpoint"
placeholder={$_("api_endpoint")}
bind:value={api_endpoint_input}
class:border-red-500={!isEndpointValid}
class:border-solid={!isEndpointValid}
class:border-3={!isEndpointValid}
class="shadow appearance-none border rounded w-full py-2 px-3 text-gray-700 mt-1 leading-tight focus:outline-none focus:shadow-outline"
/>
</div>
{#if !isEndpointValid}
<span class="text-sm"
>{$_("please_provide_a_valid_client_api_endpoint")}</span
>
{/if}
<button
disabled={!isEndpointValid}
class:cursor-pointer={isEndpointValid}
class:opacity-50={!isEndpointValid}
id="configure"
type="submit"
class="bg-black text-white font-bold text-lg hover:bg-gray-700 p-2 mt-8 hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-black"
>{$_("configure")}</button
>
</form>
{/if}
<div class="text-center pt-12 pb-12"> <div class="text-center pt-12 pb-12">
<p> <p>
<svg <svg

View File

@@ -1,7 +1,7 @@
<script> <script>
import axios from "axios"; import axios from "axios";
import { _ } from "svelte-i18n"; import { _ } from "svelte-i18n";
import { apikey, page, stationinfo } from "./store.js"; import { apikey, api_endpoint, page, stationinfo } from "./store.js";
function init(el) { function init(el) {
el.focus(); el.focus();
} }
@@ -9,6 +9,7 @@
let lastscan_time = ""; let lastscan_time = "";
let lastscan_laptime = ""; let lastscan_laptime = "";
let lastscan_totaldistance = ""; let lastscan_totaldistance = "";
let lastscan_valid = true;
let card = ""; let card = "";
// live clock at the top // live clock at the top
let time = new Date(); let time = new Date();
@@ -64,7 +65,7 @@
axios axios
.request({ .request({
method: "POST", method: "POST",
url: config.endpoint + "api/scans/trackscans", url: $api_endpoint + "api/scans/trackscans",
headers: { Authorization: "Bearer " + $apikey }, headers: { Authorization: "Bearer " + $apikey },
data: { card }, data: { card },
}) })
@@ -83,6 +84,7 @@
) + ) +
"s"; "s";
lastscan_laptime = response.data.lapTime; lastscan_laptime = response.data.lapTime;
lastscan_valid = response.data.valid;
lastscan_totaldistance = lastscan_totaldistance =
Math.floor(response.data.runner.distance / 1000) + Math.floor(response.data.runner.distance / 1000) +
"km " + "km " +
@@ -112,14 +114,44 @@
/> />
</label> </label>
{#if lastscan_totaldistance} {#if lastscan_totaldistance}
<h1 class="text-3xl font-bold text-center">last scan</h1> {#if !lastscan_valid || lastscan_error}
<h1 class="text-5xl font-bold text-center">{lastscan_time}</h1> <svg
<h1 class="text-3xl font-bold text-center">total distance</h1> xmlns="http://www.w3.org/2000/svg"
<h1 class="text-8xl font-bold text-center"> width="20rem"
height="20rem"
class="ml-auto mr-auto"
version="1.0"
title="Invalid"
viewBox="0 0 100 100"
><g fill="none" stroke="red"
><path d="M6 6l88 88" stroke-width="18.1" /><path
d="M6 94L94 6"
stroke-width="17.8"
/></g
></svg
>
{:else}
<svg
xmlns="http://www.w3.org/2000/svg"
width="20rem"
height="20rem"
class="ml-auto mr-auto"
title="Valid"
viewBox="0 0 600 600"
><path
d="M8 405s115 129 138 182h99c41-126 203-429 341-535 28-37-43-52-102-27-87 36-252 317-283 384-44 12-90-74-90-74z"
fill="#181"
/></svg
>
{/if}
<h1 class="text-2xl font-bold text-center">{$_("total-distance")}</h1>
<h1 class="text-6xl font-bold text-center">
{lastscan_totaldistance} {lastscan_totaldistance}
</h1> </h1>
<h1 class="text-3xl font-bold text-center">lap time</h1> <h1 class="text-2xl font-bold text-center">{$_("lap-time")}</h1>
<h1 class="text-8xl font-bold text-center">{lastscan_laptime}</h1> <h1 class="text-6xl font-bold text-center">{lastscan_laptime}</h1>
<h1 class="text-2xl font-bold text-center">{$_("last-scan")}</h1>
<h1 class="text-5xl font-bold text-center">{lastscan_time}</h1>
{:else} {:else}
<h1 class="text-3xl font-bold text-center"> <h1 class="text-3xl font-bold text-center">
{$_("please_scan_a_card")} {$_("please_scan_a_card")}

View File

@@ -1,7 +1,7 @@
<script> <script>
import { _ } from "svelte-i18n"; import { _ } from "svelte-i18n";
import { apikey, lang, page, stationinfo } from "./store.js"; import { apikey, api_endpoint, lang, page, stationinfo } from "./store.js";
</script> </script>
<div class="p-5 min-h-screen"> <div class="p-5 min-h-screen">
@@ -103,6 +103,7 @@
<button <button
on:click={() => { on:click={() => {
apikey.set(""); apikey.set("");
api_endpoint.set("");
page.set(""); page.set("");
}} }}
class="w-full py-3 bg-black text-white focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-black" class="w-full py-3 bg-black text-white focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-black"

View File

@@ -1,4 +1,5 @@
{ {
"api_endpoint": "API-Endpunkt",
"api_key": "API Key", "api_key": "API Key",
"back_to_scanner": "Zurück zum Scanner", "back_to_scanner": "Zurück zum Scanner",
"client_token": "Client Token", "client_token": "Client Token",
@@ -6,9 +7,12 @@
"configure": "Konfigurieren", "configure": "Konfigurieren",
"error": "Error!", "error": "Error!",
"language": "Sprache", "language": "Sprache",
"lap-time": "Rundenzeit",
"last-scan": "Letzter Scan um",
"log_out_from_this_client": "Von diesem Scanner abmelden", "log_out_from_this_client": "Von diesem Scanner abmelden",
"minimum_lap_time": "minimale Rundenzeit", "minimum_lap_time": "minimale Rundenzeit",
"please_check_your_token_and_try_again": "Bitte überprüfe den Token und versuche es erneut...", "please_check_your_token_and_try_again": "Bitte überprüfe den Token und versuche es erneut...",
"please_provide_a_valid_client_api_endpoint": "Bitte gebe einen gültigen API-Endpunkt an ...",
"please_provide_a_valid_client_token": "Bitte gebe einen gültigen Client-Token an ...", "please_provide_a_valid_client_token": "Bitte gebe einen gültigen Client-Token an ...",
"please_provide_the_scan_client_token": "Bitte gebe den Scan-Client-Token an.", "please_provide_the_scan_client_token": "Bitte gebe den Scan-Client-Token an.",
"please_scan_a_card": "Bitte scanne eine Karte ...", "please_scan_a_card": "Bitte scanne eine Karte ...",
@@ -20,6 +24,7 @@
"station_id": "Scanstations-ID", "station_id": "Scanstations-ID",
"the_provided_scan_station_is_disabled": "Die angegebene Scanstation ist deaktiviert.", "the_provided_scan_station_is_disabled": "Die angegebene Scanstation ist deaktiviert.",
"the_provided_scan_station_token_is_invalid": "Der angegebene Scanstation-Token ist ungültig.", "the_provided_scan_station_token_is_invalid": "Der angegebene Scanstation-Token ist ungültig.",
"total-distance": "Gesamtdistanz",
"track_distance": "Länge des Tracks", "track_distance": "Länge des Tracks",
"track_id": "Track ID", "track_id": "Track ID",
"track_name": "Track Name" "track_name": "Track Name"

View File

@@ -1,4 +1,5 @@
{ {
"api_endpoint": "API Endpoint",
"api_key": "API Key", "api_key": "API Key",
"back_to_scanner": "Back to Scanner", "back_to_scanner": "Back to Scanner",
"client_token": "Client Token", "client_token": "Client Token",
@@ -6,9 +7,12 @@
"configure": "Configure", "configure": "Configure",
"error": "Error!", "error": "Error!",
"language": "Language", "language": "Language",
"lap-time": "lap time",
"last-scan": "last scan",
"log_out_from_this_client": "Log Out from this Client", "log_out_from_this_client": "Log Out from this Client",
"minimum_lap_time": "minimum lap time", "minimum_lap_time": "minimum lap time",
"please_check_your_token_and_try_again": "Please check your token and try again...", "please_check_your_token_and_try_again": "Please check your token and try again...",
"please_provide_a_valid_client_api_endpoint": "Please provide a valid api endpoint...",
"please_provide_a_valid_client_token": "Please provide a valid client token...", "please_provide_a_valid_client_token": "Please provide a valid client token...",
"please_provide_the_scan_client_token": "Please provide the scan client token.", "please_provide_the_scan_client_token": "Please provide the scan client token.",
"please_scan_a_card": "please scan a card...", "please_scan_a_card": "please scan a card...",
@@ -20,6 +24,7 @@
"station_id": "Scanstation ID", "station_id": "Scanstation ID",
"the_provided_scan_station_is_disabled": "The provided scan station is disabled.", "the_provided_scan_station_is_disabled": "The provided scan station is disabled.",
"the_provided_scan_station_token_is_invalid": "The provided scan station token is invalid.", "the_provided_scan_station_token_is_invalid": "The provided scan station token is invalid.",
"total-distance": "total distance",
"track_distance": "Track Distance", "track_distance": "Track Distance",
"track_id": "Track ID", "track_id": "Track ID",
"track_name": "Track Name" "track_name": "Track Name"

View File

@@ -1,5 +1,10 @@
import { writable } from 'svelte/store'; import { writable } from 'svelte/store';
const stored_api_endpoint = localStorage.getItem('api_endpoint')||"";
export const api_endpoint = writable(stored_api_endpoint);
api_endpoint.subscribe((value) => {
localStorage.setItem('api_endpoint', value);
});
const stored_apikey = localStorage.getItem('apikey'); const stored_apikey = localStorage.getItem('apikey');
export const apikey = writable(stored_apikey); export const apikey = writable(stored_apikey);
apikey.subscribe((value) => { apikey.subscribe((value) => {