Compare commits

..

26 Commits

Author SHA1 Message Date
51d058bf96 added 'yarn postbuild' script for fixing env.js in dist/index.html
ref #31
2021-04-13 17:58:43 +02:00
ad13bae068 dependency bump 2021-04-13 17:57:41 +02:00
152e74190d 🚀Bumped version to v0.7.0
All checks were successful
continuous-integration/drone/push Build is passing
2021-04-06 21:28:02 +02:00
4c83e2e738 Merge pull request 'Donation list feature/39-donation_list' (#40) from feature/39-donation_list into dev
All checks were successful
continuous-integration/drone/push Build is passing
Reviewed-on: #40
2021-04-06 19:27:30 +00:00
cb1b9d330b Sorted translations 🌍
ref #39
2021-04-06 21:25:36 +02:00
3b4c11e4c0 Merge branch 'feature/39-donation_list' of git.odit.services:lfk/selfservice into feature/39-donation_list 2021-04-06 21:25:12 +02:00
89820d4450 Added total to bottom of page
ref #39
2021-04-06 21:25:10 +02:00
121022c843 added distance formatting
ref #39
2021-04-06 21:23:24 +02:00
d713fbef94 Added translations 🌍
ref #39
2021-04-06 17:27:18 +02:00
815a36f202 Now w/ formatted currency amount
ref #39
2021-04-06 17:20:41 +02:00
baa6da3dd0 Fixed spaces in name
ref #39
2021-04-06 17:18:09 +02:00
f141130db5 Added basic sponsoring table
ref #39
2021-04-06 17:14:22 +02:00
5a123b0cf8 Updated env description
All checks were successful
continuous-integration/drone/push Build is passing
2021-04-06 17:03:05 +02:00
f5ae2145df 🚀Bumped version to v0.6.1
All checks were successful
continuous-integration/drone/push Build is passing
2021-04-06 17:01:20 +02:00
96c0e56986 Fixed imprint/privacy default links
All checks were successful
continuous-integration/drone/push Build is passing
2021-04-06 17:00:55 +02:00
bc1de2acdc 🚀Bumped version to v0.6.0
All checks were successful
continuous-integration/drone/push Build is passing
2021-04-03 18:53:32 +02:00
5c558aa9ee Merge branch 'dev' of git.odit.services:lfk/selfservice into dev 2021-04-03 18:53:14 +02:00
145b49906b Revert "🚀Bumped version to v0.6.0"
This reverts commit 0a62e8f5d1.
2021-04-03 18:53:12 +02:00
0a62e8f5d1 🚀Bumped version to v0.6.0 2021-04-03 18:52:54 +02:00
53eeb6be3d Merge pull request 'Certificate download feature/19-runner_certficates' (#38) from feature/19-runner_certficates into dev
All checks were successful
continuous-integration/drone/push Build is passing
Reviewed-on: #38
2021-04-03 16:51:22 +00:00
a554a90e74 Merge branch 'dev' into feature/19-runner_certficates
# Conflicts:
#	index.html
2021-04-03 18:50:16 +02:00
c3b2b93d90 Reverted relative linking fix
ref #19
2021-04-03 18:48:53 +02:00
7064a5bd82 Merge branch 'dev' into feature/19-runner_certficates 2021-04-03 16:28:18 +00:00
f0a7f35dec Renoved fixed data
ref #19
2021-04-03 18:11:57 +02:00
b938cfc49e Fixed blob handling
ref #19
2021-04-03 18:11:22 +02:00
f6334397dc first part of certificate generation with manual data
ref #19
2021-04-03 17:57:24 +02:00
10 changed files with 529 additions and 357 deletions

View File

@@ -2,9 +2,44 @@
All notable changes to this project will be documented in this file. Dates are displayed in UTC. All notable changes to this project will be documented in this file. Dates are displayed in UTC.
#### [0.7.0](https://git.odit.services/lfk/selfservice/compare/0.6.1...0.7.0)
- Merge pull request 'Donation list feature/39-donation_list' (#40) from feature/39-donation_list into dev [`4c83e2e`](https://git.odit.services/lfk/selfservice/commit/4c83e2e738a075354383dca4d500808f761247d2)
- Added total to bottom of page [`89820d4`](https://git.odit.services/lfk/selfservice/commit/89820d44501793365248b8e778522cdc188afa70)
- Sorted translations 🌍 [`cb1b9d3`](https://git.odit.services/lfk/selfservice/commit/cb1b9d330b28f11cceed9691aee4ccee5246d346)
- Added basic sponsoring table [`f141130`](https://git.odit.services/lfk/selfservice/commit/f141130db5ede60a623747f4e324f66259e5bc75)
- Added translations 🌍 [`d713fbe`](https://git.odit.services/lfk/selfservice/commit/d713fbef94e75fbbc62254ea8f0ca50dc4e93d79)
- Now w/ formatted currency amount [`815a36f`](https://git.odit.services/lfk/selfservice/commit/815a36f20271f47f49c3814b001c3404d43113cd)
- Fixed spaces in name [`baa6da3`](https://git.odit.services/lfk/selfservice/commit/baa6da3dd06c493fccc61945b9bcbd9b2e79d910)
- Updated env description [`5a123b0`](https://git.odit.services/lfk/selfservice/commit/5a123b0cf89f49d450becbbc03b28c5bb6416b7c)
- added distance formatting [`121022c`](https://git.odit.services/lfk/selfservice/commit/121022c8434484a363a3f2ea68aba4d8ef7cad2d)
#### [0.6.1](https://git.odit.services/lfk/selfservice/compare/0.6.0...0.6.1)
> 6 April 2021
- 🚀Bumped version to v0.6.1 [`f5ae214`](https://git.odit.services/lfk/selfservice/commit/f5ae2145df07413329f60c229d9571fd1de2ca79)
- Fixed imprint/privacy default links [`96c0e56`](https://git.odit.services/lfk/selfservice/commit/96c0e5698697361e65c5aa80c9a0aada5c3d5f30)
#### [0.6.0](https://git.odit.services/lfk/selfservice/compare/0.5.1...0.6.0)
> 3 April 2021
- 🚀Bumped version to v0.6.0 [`bc1de2a`](https://git.odit.services/lfk/selfservice/commit/bc1de2acdcbf8f3319026887cb6b3d594fcd63dc)
- first part of certificate generation with manual data [`f633439`](https://git.odit.services/lfk/selfservice/commit/f6334397dc99cd38d31ef524bc26ba7d386b33dd)
- Renoved fixed data [`f0a7f35`](https://git.odit.services/lfk/selfservice/commit/f0a7f35dec07508a07d4c11abda297ad0bf91187)
- Revert "🚀Bumped version to v0.6.0" [`145b499`](https://git.odit.services/lfk/selfservice/commit/145b49906bdf2d75abf93e1779f188a691a3a909)
- 🚀Bumped version to v0.6.0 [`0a62e8f`](https://git.odit.services/lfk/selfservice/commit/0a62e8f5d1929bfe9a53a9709a3031f06a2487c3)
- Merge pull request 'Certificate download feature/19-runner_certficates' (#38) from feature/19-runner_certficates into dev [`53eeb6b`](https://git.odit.services/lfk/selfservice/commit/53eeb6be3ddf4735a11b6c4636c28b4d16aa58ee)
- Fixed blob handling [`b938cfc`](https://git.odit.services/lfk/selfservice/commit/b938cfc49e269e76761a752f8257ebbab88f6959)
- Reverted relative linking fix [`c3b2b93`](https://git.odit.services/lfk/selfservice/commit/c3b2b93d90102b4e31cfce15220acfc8fe48a7c5)
#### [0.5.1](https://git.odit.services/lfk/selfservice/compare/0.5.0...0.5.1) #### [0.5.1](https://git.odit.services/lfk/selfservice/compare/0.5.0...0.5.1)
> 3 April 2021
- Merge pull request 'feature/30-profile-forgot-link' (#35) from feature/30-profile-forgot-link into dev [`#30`](https://git.odit.services/lfk/selfservice/issues/30) - Merge pull request 'feature/30-profile-forgot-link' (#35) from feature/30-profile-forgot-link into dev [`#30`](https://git.odit.services/lfk/selfservice/issues/30)
- 🚀Bumped version to v0.5.1 [`1a3c9ed`](https://git.odit.services/lfk/selfservice/commit/1a3c9edeb3987907ffe2223da8f3be079a1c80d2)
- Merge pull request 'Env linking bugfix bugfix/31-env_linking' (#37) from bugfix/31-env_linking into dev [`fd6bd88`](https://git.odit.services/lfk/selfservice/commit/fd6bd88d4238cca2a2755ab1bc51ff7870bef947) - Merge pull request 'Env linking bugfix bugfix/31-env_linking' (#37) from bugfix/31-env_linking into dev [`fd6bd88`](https://git.odit.services/lfk/selfservice/commit/fd6bd88d4238cca2a2755ab1bc51ff7870bef947)
- basic ProfileNone layout [`863568d`](https://git.odit.services/lfk/selfservice/commit/863568d7d0e132dd1b7d13a2cb1afe8db02c472c) - basic ProfileNone layout [`863568d`](https://git.odit.services/lfk/selfservice/commit/863568d7d0e132dd1b7d13a2cb1afe8db02c472c)
- add forgot view + action [`30f3a51`](https://git.odit.services/lfk/selfservice/commit/30f3a51ef82ad966225abc6fd95508af56106924) - add forgot view + action [`30f3a51`](https://git.odit.services/lfk/selfservice/commit/30f3a51ef82ad966225abc6fd95508af56106924)

View File

@@ -3,6 +3,7 @@ WORKDIR /app
COPY . . COPY . .
RUN yarn RUN yarn
RUN yarn build RUN yarn build
RUN yarn postbuild
# final image # final image
FROM fholzer/nginx-brotli:v1.19.1 FROM fholzer/nginx-brotli:v1.19.1
COPY --from=0 /app/dist /usr/share/nginx/html COPY --from=0 /app/dist /usr/share/nginx/html

6
env_fix.js Normal file
View File

@@ -0,0 +1,6 @@
const fs = require('fs');
if (fs.existsSync('./dist/index.html')) {
const content = fs.readFileSync('./dist/index.html', { encoding: 'utf8' });
const newcontent = content.replace(`"/env.js"`, `"./env.js"`);
fs.writeFileSync('./dist/index.html', newcontent);
}

View File

@@ -10,8 +10,8 @@
<body class="dark:bg-gray-900 text-black dark:text-white p-0"> <body class="dark:bg-gray-900 text-black dark:text-white p-0">
<div id="app"></div> <div id="app"></div>
<script src="./env.js"></script> <script src="/env.js"></script>
<script defer type="module" src="./src/main.js"></script> <script defer type="module" src="/src/main.js"></script>
</body> </body>
</html> </html>

View File

@@ -1,50 +1,51 @@
{ {
"name": "@odit/lfk-selfservice", "name": "@odit/lfk-selfservice",
"version": "0.5.1", "version": "0.7.0",
"scripts": { "scripts": {
"dev": "vite", "dev": "vite",
"build": "vite build", "build": "vite build",
"release": "release-it --only-version" "release": "release-it --only-version",
}, "postbuild": "node env_fix.js"
"dependencies": { },
"marked": "2.0.1", "dependencies": {
"redaxios": "0.4.1", "marked": "2.0.3",
"toastify-js": "1.10.0", "redaxios": "0.4.1",
"validator": "13.5.2", "toastify-js": "1.10.0",
"vue-i18n": "9.0.0", "validator": "13.5.2",
"vue-toastification": "2.0.0-rc.1", "vue-i18n": "9.1.4",
"vue": "3.0.9", "vue-toastification": "2.0.0-rc.1",
"vue-router": "4.0.5" "vue": "3.0.9",
}, "vue-router": "4.0.5"
"devDependencies": { },
"@tailwindcss/jit": "0.1.18", "devDependencies": {
"@tailwindcss/aspect-ratio": "0.2.0", "@tailwindcss/jit": "0.1.18",
"@tailwindcss/forms": "0.3.2", "@tailwindcss/aspect-ratio": "0.2.0",
"@tailwindcss/line-clamp": "0.2.0", "@tailwindcss/forms": "0.3.2",
"@tailwindcss/typography": "0.4.0", "@tailwindcss/line-clamp": "0.2.0",
"@vitejs/plugin-vue": "1.2.1", "@tailwindcss/typography": "0.4.0",
"@vue/compiler-sfc": "3.0.10", "@vitejs/plugin-vue": "1.2.1",
"autoprefixer": "10.2.5", "@vue/compiler-sfc": "3.0.11",
"tailwindcss": "2.0.4", "autoprefixer": "10.2.5",
"release-it": "14.5.0", "tailwindcss": "2.1.1",
"vite": "2.1.5" "release-it": "14.6.1",
}, "vite": "2.1.5"
"release-it": { },
"git": { "release-it": {
"commit": true, "git": {
"requireCleanWorkingDir": false, "commit": true,
"commitMessage": "🚀Bumped version to v${version}", "requireCleanWorkingDir": false,
"requireBranch": "dev", "commitMessage": "🚀Bumped version to v${version}",
"push": false, "requireBranch": "dev",
"tag": true, "push": false,
"tagName": null, "tag": true,
"tagAnnotation": "v${version}" "tagName": null,
}, "tagAnnotation": "v${version}"
"npm": { },
"publish": false "npm": {
}, "publish": false
"hooks": { },
"after:bump": "npx auto-changelog --commit-limit false -p -u --hide-credit && git add CHANGELOG.md && node order_i18n.js && git add src/locales" "hooks": {
} "after:bump": "npx auto-changelog --commit-limit false -p -u --hide-credit && git add CHANGELOG.md && node order_i18n.js && git add src/locales"
} }
} }
}

View File

@@ -5,8 +5,8 @@ const config = {
baseurl: '', baseurl: '',
// optional, will fallback to /selfservice/ // optional, will fallback to /selfservice/
baseurl_selfservice: '/selfservice/', baseurl_selfservice: '/selfservice/',
// optional, will fallback to /imprint // optional, will fallback to baseurl_selfservice/imprint
url_imprint: '', url_imprint: '',
// optional, will fallback to /privacy // optional, will fallback to baseurl_selfservice/privacy
url_privacy: '' url_privacy: ''
}; };

View File

@@ -31,8 +31,8 @@
export default { export default {
data() { data() {
return { return {
imprint_url: config.url_imprint || "/imprint/" imprint_url: config.url_imprint || config.baseurl_selfservice + "imprint"
, privacy_url: config.url_privacy || "/privacy/" , privacy_url: config.url_privacy || config.baseurl_selfservice + "privacy"
} }
}, },
} }

View File

@@ -1,12 +1,14 @@
{ {
"access_is_only_provided_via_your_email_link": "Der Zugang erfolgt nur über den Link, den Sie bei der Registrierung erhalten haben.", "access_is_only_provided_via_your_email_link": "Der Zugang erfolgt nur über den Link, den Sie bei der Registrierung erhalten haben.",
"already_have_an_account": "Sie haben bereits einen Account?", "already_have_an_account": "Sie haben bereits einen Account?",
"amount_per_kilometer_in_eur": "Betrag pro Kilometer (in €)",
"apartment_suite_etc": "Addresszeile 2", "apartment_suite_etc": "Addresszeile 2",
"cancel_keep_my_data": "Abbrechen, meine Daten behalten", "cancel_keep_my_data": "Abbrechen, meine Daten behalten",
"configuration_error": "Konfigurationsfehler", "configuration_error": "Konfigurationsfehler",
"confirm_delete_all_of_my_data": "Bestätigung, meine gesamten Daten löschen", "confirm_delete_all_of_my_data": "Bestätigung, meine gesamten Daten löschen",
"confirm_personal_data": "Hiermit bestätige ich die Vollständigkeit und Richtigkeit der oben genannten Angaben", "confirm_personal_data": "Hiermit bestätige ich die Vollständigkeit und Richtigkeit der oben genannten Angaben",
"delete-all-of-my-data": "Meine gesamten Daten löschen", "current_total_amount_in_eur": "Aktueller Gesamtbetrag (in €)",
"delete_all_of_my_data": "Meine gesamten Daten löschen",
"delete_my_data": "Meine Daten löschen", "delete_my_data": "Meine Daten löschen",
"deletion_in_progress": "Daten werden gelöscht...", "deletion_in_progress": "Daten werden gelöscht...",
"distance": "Distanz", "distance": "Distanz",
@@ -24,6 +26,7 @@
"mittelname": "Mittelname", "mittelname": "Mittelname",
"nachname": "Nachname", "nachname": "Nachname",
"no_laps_scans_were_recorded_yet": "Es wurden noch keine Runden / Scans aufgezeichnet ...", "no_laps_scans_were_recorded_yet": "Es wurden noch keine Runden / Scans aufgezeichnet ...",
"no_sponsorings_for_you_were_recorded_yet": "Es gibt noch keine Sponsorings für dich",
"not_registered_yet": "Noch nicht registriert?", "not_registered_yet": "Noch nicht registriert?",
"organization": "Organisation", "organization": "Organisation",
"ort": "Ort", "ort": "Ort",
@@ -47,6 +50,7 @@
"the_system_is_not_properly_configured_please_contact_the_system_administrator_for_help": "Das System ist nicht richtig konfiguriert. Bitte wenden Sie sich an den Systemadministrator, um Hilfe zu erhalten.", "the_system_is_not_properly_configured_please_contact_the_system_administrator_for_help": "Das System ist nicht richtig konfiguriert. Bitte wenden Sie sich an den Systemadministrator, um Hilfe zu erhalten.",
"this_is_not_a_valid_international_phone_number": "Dies ist keine gültige internationale Telefonnummer", "this_is_not_a_valid_international_phone_number": "Dies ist keine gültige internationale Telefonnummer",
"tos": "AGBs", "tos": "AGBs",
"total": "Gesamt",
"view_my_data": "Meine Läuferdaten einsehen", "view_my_data": "Meine Läuferdaten einsehen",
"vorname": "Vorname", "vorname": "Vorname",
"you_have_been_registered": "Sie wurden registriert!", "you_have_been_registered": "Sie wurden registriert!",

View File

@@ -2,11 +2,13 @@
"access_is_only_provided_via_your_email_link": "Access is only provided via the link you received upon registration.", "access_is_only_provided_via_your_email_link": "Access is only provided via the link you received upon registration.",
"all_data_deleted": "All Data deleted!", "all_data_deleted": "All Data deleted!",
"already_have_an_account": "Already have an account?", "already_have_an_account": "Already have an account?",
"amount_per_kilometer_in_eur": "Amount per kilometer (in €)",
"apartment_suite_etc": "Apartment, suite, etc.", "apartment_suite_etc": "Apartment, suite, etc.",
"cancel_keep_my_data": "Cancel, keep my data", "cancel_keep_my_data": "Cancel, keep my data",
"configuration_error": "Configuration error", "configuration_error": "Configuration error",
"confirm_delete_all_of_my_data": "Confirm, delete all of my data", "confirm_delete_all_of_my_data": "Confirm, delete all of my data",
"confirm_personal_data": "I hereby confirm that the above information is complete and correct", "confirm_personal_data": "I hereby confirm that the above information is complete and correct",
"current_total_amount_in_eur": "Current total amount (in €)",
"delete_my_data": "Delete my data", "delete_my_data": "Delete my data",
"deletion_in_progress": "Deletion in progress...", "deletion_in_progress": "Deletion in progress...",
"distance": "Distance", "distance": "Distance",
@@ -24,6 +26,7 @@
"mittelname": "Middlename", "mittelname": "Middlename",
"nachname": "Lastname", "nachname": "Lastname",
"no_laps_scans_were_recorded_yet": "No laps/ scans were recorded yet...", "no_laps_scans_were_recorded_yet": "No laps/ scans were recorded yet...",
"no_sponsorings_for_you_were_recorded_yet": "No sponsorings for you were recorded yet...",
"not_registered_yet": "Not registered yet?", "not_registered_yet": "Not registered yet?",
"organization": "Organization", "organization": "Organization",
"ort": "City", "ort": "City",
@@ -47,6 +50,7 @@
"the_system_is_not_properly_configured_please_contact_the_system_administrator_for_help": "The system is not properly configured. Please contact the system administrator for help.", "the_system_is_not_properly_configured_please_contact_the_system_administrator_for_help": "The system is not properly configured. Please contact the system administrator for help.",
"this_is_not_a_valid_international_phone_number": "This is not a valid international phone number", "this_is_not_a_valid_international_phone_number": "This is not a valid international phone number",
"tos": "Terms of Service", "tos": "Terms of Service",
"total": "Total",
"view_my_data": "View my data", "view_my_data": "View my data",
"vorname": "Firstname", "vorname": "Firstname",
"you_have_been_registered": "You have been registered!", "you_have_been_registered": "You have been registered!",

View File

@@ -1,301 +1,422 @@
<template> <template>
<div class="min-h-screen w-full p-4"> <div class="min-h-screen w-full p-4">
<section class="text-white body-font"> <section class="text-white body-font">
<div class="container mx-auto flex items-center md:flex-row flex-col"> <div class="container mx-auto flex items-center md:flex-row flex-col">
<div <div
class="flex flex-col md:pr-10 md:mb-0 mb-6 pr-0 w-full md:w-auto md:text-left text-center text-black dark:text-gray-200" class="flex flex-col md:pr-10 md:mb-0 mb-6 pr-0 w-full md:w-auto md:text-left text-center text-black dark:text-gray-200"
> >
<p <p
class="text-3xl font-bold whitespace-nowrap" class="text-3xl font-bold whitespace-nowrap"
v-text="(state.firstname || '') + ' ' + (state.middlename || '') + ' ' + (state.lastname || '')" v-text="(state.firstname || '') + ' ' + (state.middlename || '') + ' ' + (state.lastname || '')"
></p> ></p>
<p class="text-md whitespace-nowrap">{{ state.group }}</p> <p class="text-md whitespace-nowrap">{{ state.group }}</p>
</div> </div>
<div class="inline-flex md:ml-auto md:mr-0 mx-auto items-center"> <div class="inline-flex md:ml-auto md:mr-0 mx-auto items-center">
<div v-if="(state.delete_active === false)"> <div v-if="(state.delete_active === false)">
<button <button
type="button" type="button"
class="focus:border-black focus:ring-2 focus:ring-black text-white text-sm py-2.5 px-5 rounded-md bg-blue-500 hover:bg-blue-600 hover:shadow-lg" class="focus:border-black focus:ring-2 focus:ring-black text-white text-sm py-2.5 px-5 rounded-md bg-blue-500 hover:bg-blue-600 hover:shadow-lg"
> @click="get_certificate"
<svg >
xmlns="http://www.w3.org/2000/svg" <svg
width="24" xmlns="http://www.w3.org/2000/svg"
height="24" width="24"
viewBox="0 0 24 24" height="24"
fill="none" viewBox="0 0 24 24"
stroke="currentColor" fill="none"
stroke-width="2" stroke="currentColor"
stroke-linecap="round" stroke-width="2"
stroke-linejoin="round" stroke-linecap="round"
class="feather feather-download" stroke-linejoin="round"
style="display: inline;height: 1rem;vertical-align: sub;" class="feather feather-download"
> style="display: inline;height: 1rem;vertical-align: sub;"
<path d="M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4" /> >
<polyline points="7 10 12 15 17 10" /> <path d="M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4" />
<line x1="12" y1="15" x2="12" y2="3" /> <polyline points="7 10 12 15 17 10" />
</svg> <line x1="12" y1="15" x2="12" y2="3" />
{{ $t('download_certificate') }} </svg>
</button> {{ $t('download_certificate') }}
<button </button>
type="button" <button
class="focus:border-black focus:ring-2 focus:ring-black text-white text-sm py-2.5 px-5 rounded-md bg-red-600 hover:bg-red-700 hover:shadow-lg ml-1" type="button"
@click="() => { state.delete_active = true }" class="focus:border-black focus:ring-2 focus:ring-black text-white text-sm py-2.5 px-5 rounded-md bg-red-600 hover:bg-red-700 hover:shadow-lg ml-1"
> @click="() => { state.delete_active = true }"
<svg >
xmlns="http://www.w3.org/2000/svg" <svg
width="24" xmlns="http://www.w3.org/2000/svg"
height="24" width="24"
viewBox="0 0 24 24" height="24"
fill="none" viewBox="0 0 24 24"
stroke="none" fill="none"
stroke-width="2" stroke="none"
stroke-linecap="round" stroke-width="2"
stroke-linejoin="round" stroke-linecap="round"
class="feather feather-download" stroke-linejoin="round"
style="display: inline;height: 1rem;vertical-align: sub;" class="feather feather-download"
> style="display: inline;height: 1rem;vertical-align: sub;"
<path d="M0 0h24v24H0z" /> >
<path <path d="M0 0h24v24H0z" />
fill="currentColor" <path
d="M17 6h5v2h-2v13a1 1 0 01-1 1H5a1 1 0 01-1-1V8H2V6h5V3a1 1 0 011-1h8a1 1 0 011 1v3zm1 2H6v12h12V8zm-5 6l2 2-1 1-2-2-2 2-1-1 2-2-2-2 1-1 2 2 2-2 1 1-2 2zM9 4v2h6V4H9z" fill="currentColor"
/> d="M17 6h5v2h-2v13a1 1 0 01-1 1H5a1 1 0 01-1-1V8H2V6h5V3a1 1 0 011-1h8a1 1 0 011 1v3zm1 2H6v12h12V8zm-5 6l2 2-1 1-2-2-2 2-1-1 2-2-2-2 1-1 2 2 2-2 1 1-2 2zM9 4v2h6V4H9z"
</svg> />
{{ $t('delete_my_data') }} </svg>
</button> {{ $t('delete_my_data') }}
</div> </button>
<div v-if="(state.delete_active === true)"> </div>
<button <div v-if="(state.delete_active === true)">
type="button" <button
class="focus:border-black focus:ring-2 focus:ring-black text-white text-sm py-2.5 px-5 rounded-md bg-blue-500 hover:bg-blue-600 hover:shadow-lg" type="button"
@click="() => { state.delete_active = false }" class="focus:border-black focus:ring-2 focus:ring-black text-white text-sm py-2.5 px-5 rounded-md bg-blue-500 hover:bg-blue-600 hover:shadow-lg"
> @click="() => { state.delete_active = false }"
<svg >
xmlns="http://www.w3.org/2000/svg" <svg
width="24" xmlns="http://www.w3.org/2000/svg"
height="24" width="24"
viewBox="0 0 24 24" height="24"
fill="none" viewBox="0 0 24 24"
stroke="none" fill="none"
stroke-width="2" stroke="none"
stroke-linecap="round" stroke-width="2"
stroke-linejoin="round" stroke-linecap="round"
class="feather feather-download" stroke-linejoin="round"
style="display: inline;height: 1rem;vertical-align: sub;" class="feather feather-download"
> style="display: inline;height: 1rem;vertical-align: sub;"
<path fill="none" d="M0 0h24v24H0z" /> >
<path fill="currentColor" d="M12 11l5-5 1 1-5 5 5 5-1 1-5-5-5 5-1-1 5-5-5-5 1-1z" /> <path fill="none" d="M0 0h24v24H0z" />
</svg> <path fill="currentColor" d="M12 11l5-5 1 1-5 5 5 5-1 1-5-5-5 5-1-1 5-5-5-5 1-1z" />
{{ $t('cancel_keep_my_data') }} </svg>
</button> {{ $t('cancel_keep_my_data') }}
<button </button>
type="button" <button
class="focus:border-black focus:ring-2 focus:ring-black text-white text-sm py-2.5 px-5 rounded-md bg-red-600 hover:bg-red-700 hover:shadow-lg ml-1" type="button"
@click="delete_me" class="focus:border-black focus:ring-2 focus:ring-black text-white text-sm py-2.5 px-5 rounded-md bg-red-600 hover:bg-red-700 hover:shadow-lg ml-1"
> @click="delete_me"
<svg >
xmlns="http://www.w3.org/2000/svg" <svg
width="24" xmlns="http://www.w3.org/2000/svg"
height="24" width="24"
viewBox="0 0 24 24" height="24"
fill="none" viewBox="0 0 24 24"
stroke="none" fill="none"
stroke-width="2" stroke="none"
stroke-linecap="round" stroke-width="2"
stroke-linejoin="round" stroke-linecap="round"
class="feather feather-download" stroke-linejoin="round"
style="display: inline;height: 1rem;vertical-align: sub;" class="feather feather-download"
> style="display: inline;height: 1rem;vertical-align: sub;"
<path d="M0 0h24v24H0z" /> >
<path <path d="M0 0h24v24H0z" />
fill="currentColor" <path
d="M17 6h5v2h-2v13a1 1 0 01-1 1H5a1 1 0 01-1-1V8H2V6h5V3a1 1 0 011-1h8a1 1 0 011 1v3zm1 2H6v12h12V8zm-5 6l2 2-1 1-2-2-2 2-1-1 2-2-2-2 1-1 2 2 2-2 1 1-2 2zM9 4v2h6V4H9z" fill="currentColor"
/> d="M17 6h5v2h-2v13a1 1 0 01-1 1H5a1 1 0 01-1-1V8H2V6h5V3a1 1 0 011-1h8a1 1 0 011 1v3zm1 2H6v12h12V8zm-5 6l2 2-1 1-2-2-2 2-1-1 2-2-2-2 1-1 2 2 2-2 1 1-2 2zM9 4v2h6V4H9z"
</svg> />
{{ $t('confirm_delete_all_of_my_data') }} </svg>
</button> {{ $t('confirm_delete_all_of_my_data') }}
</div> </button>
</div> </div>
</div> </div>
</section> </div>
<div class="flex flex-wrap"> </section>
<div class="w-full p-4"> <div class="flex flex-wrap">
<div class="flex flex-wrap flex-col w-full tabs"> <div class="w-full p-4">
<div class="flex lg:flex-wrap flex-row lg:space-x-2"> <div class="flex flex-wrap flex-col w-full tabs">
<div class="flex-none"> <div class="flex lg:flex-wrap flex-row lg:space-x-2">
<button <div class="flex-none">
@click="() => { state.activetab = 'profile' }" <button
:class="{ 'tab-active border-b-2 font-medium border-blue-500': (state.activetab === 'profile') }" @click="() => { state.activetab = 'profile' }"
class="tab tab-underline py-4 px-6 block" :class="{ 'tab-active border-b-2 font-medium border-blue-500': (state.activetab === 'profile') }"
type="button" class="tab tab-underline py-4 px-6 block"
>{{ $t('profile') }}</button> type="button"
</div> >{{ $t('profile') }}</button>
<div class="flex-none"> </div>
<button <div class="flex-none">
@click="() => { state.activetab = 'laptimes' }" <button
:class="{ 'tab-active border-b-2 font-medium border-blue-500': (state.activetab === 'laptimes') }" @click="() => { state.activetab = 'laptimes' }"
class="tab tab-underline py-4 px-6 block" :class="{ 'tab-active border-b-2 font-medium border-blue-500': (state.activetab === 'laptimes') }"
type="button" class="tab tab-underline py-4 px-6 block"
>{{ $t('lap_times') }}</button> type="button"
</div> >{{ $t('lap_times') }}</button>
<div class="flex-none"> </div>
<button <div class="flex-none">
@click="() => { state.activetab = 'sponsorings' }" <button
:class="{ 'tab-active border-b-2 font-medium border-blue-500': (state.activetab === 'sponsorings') }" @click="() => { state.activetab = 'sponsorings' }"
class="tab tab-underline py-4 px-6 block" :class="{ 'tab-active border-b-2 font-medium border-blue-500': (state.activetab === 'sponsorings') }"
type="button" class="tab tab-underline py-4 px-6 block"
>{{ $t('sponsoring') }}</button> type="button"
</div> >{{ $t('sponsoring') }}</button>
</div> </div>
<div v-if="(state.activetab === 'profile')" class="tab-content block"> </div>
<div class="py-4 w-full"> <div v-if="(state.activetab === 'profile')" class="tab-content block">
<div class="flex flex-col"> <div class="py-4 w-full">
<form class="form flex flex-wrap w-full"> <div class="flex flex-col">
<div class="w-full"> <form class="form flex flex-wrap w-full">
<div class="form-element"> <div class="w-full">
<div class="text-lg">{{ $t('vorname') }}</div> <div class="form-element">
<p <div class="text-lg">{{ $t('vorname') }}</div>
class="h-10 w-full dark:bg-gray-800 rounded text-base outline-none dark:text-gray-100 text-gray-600 py-1 px-3 leading-8 transition-colors duration-200 ease-in-out" <p
v-text="state.firstname" class="h-10 w-full dark:bg-gray-800 rounded text-base outline-none dark:text-gray-100 text-gray-600 py-1 px-3 leading-8 transition-colors duration-200 ease-in-out"
/> v-text="state.firstname"
</div> />
<div class="form-element"> </div>
<div class="text-lg">{{ $t('mittelname') }}</div> <div class="form-element">
<p <div class="text-lg">{{ $t('mittelname') }}</div>
class="h-10 w-full dark:bg-gray-800 rounded text-base outline-none dark:text-gray-100 text-gray-600 py-1 px-3 leading-8 transition-colors duration-200 ease-in-out" <p
v-text="state.middlename" class="h-10 w-full dark:bg-gray-800 rounded text-base outline-none dark:text-gray-100 text-gray-600 py-1 px-3 leading-8 transition-colors duration-200 ease-in-out"
/> v-text="state.middlename"
</div> />
<div class="form-element"> </div>
<div class="text-lg">{{ $t('nachname') }}</div> <div class="form-element">
<p <div class="text-lg">{{ $t('nachname') }}</div>
class="h-10 w-full dark:bg-gray-800 rounded text-base outline-none dark:text-gray-100 text-gray-600 py-1 px-3 leading-8 transition-colors duration-200 ease-in-out" <p
v-text="state.lastname" class="h-10 w-full dark:bg-gray-800 rounded text-base outline-none dark:text-gray-100 text-gray-600 py-1 px-3 leading-8 transition-colors duration-200 ease-in-out"
/> v-text="state.lastname"
</div> />
<div class="form-element"> </div>
<div class="text-lg">{{ $t('e_mail_adress') }}</div> <div class="form-element">
<p <div class="text-lg">{{ $t('e_mail_adress') }}</div>
class="h-10 w-full dark:bg-gray-800 rounded text-base outline-none dark:text-gray-100 text-gray-600 py-1 px-3 leading-8 transition-colors duration-200 ease-in-out" <p
v-text="state.email" class="h-10 w-full dark:bg-gray-800 rounded text-base outline-none dark:text-gray-100 text-gray-600 py-1 px-3 leading-8 transition-colors duration-200 ease-in-out"
/> v-text="state.email"
</div> />
<div class="form-element"> </div>
<div class="text-lg">{{ $t('phone_number') }}</div> <div class="form-element">
<p <div class="text-lg">{{ $t('phone_number') }}</div>
class="h-10 w-full dark:bg-gray-800 rounded text-base outline-none dark:text-gray-100 text-gray-600 py-1 px-3 leading-8 transition-colors duration-200 ease-in-out" <p
v-text="state.phone" class="h-10 w-full dark:bg-gray-800 rounded text-base outline-none dark:text-gray-100 text-gray-600 py-1 px-3 leading-8 transition-colors duration-200 ease-in-out"
/> v-text="state.phone"
</div> />
</div> </div>
</form> </div>
</div> </form>
</div> </div>
</div> </div>
<div v-if="(state.activetab === 'laptimes')" class="tab-content block"> </div>
<div class="py-4 w-full"> <div v-if="(state.activetab === 'laptimes')" class="tab-content block">
<section class="text-gray-400 dark:bg-gray-900 body-font"> <div class="py-4 w-full">
<div class="container mx-auto"> <section class="text-gray-400 dark:bg-gray-900 body-font">
<div class="lg:w-2/3 w-full mx-auto overflow-auto"> <div class="container mx-auto">
<table <div class="lg:w-2/3 w-full mx-auto overflow-auto">
v-if="state.scans.length > 0" <table
class="table-auto w-full text-left whitespace-no-wrap" v-if="state.scans.length > 0"
> class="table-auto w-full text-left whitespace-no-wrap"
<thead >
class="text-black bg-gray-300 dark:text-white text-sm dark:bg-gray-800" <thead
> class="text-black bg-gray-300 dark:text-white text-sm dark:bg-gray-800"
<tr> >
<th <tr>
class="px-4 py-3 title-font tracking-wider font-medium" <th
>{{ $t('distance') }}</th> class="px-4 py-3 title-font tracking-wider font-medium"
<th >{{ $t('distance') }}</th>
class="px-4 py-3 title-font tracking-wider font-medium" <th
>{{ $t('lap_time') }}</th> class="px-4 py-3 title-font tracking-wider font-medium"
</tr> >{{ $t('lap_time') }}</th>
</thead> </tr>
<tbody class="text-gray-900 dark:text-gray-50"> </thead>
<tr v-for="s in state.scans" :key="s.id"> <tbody class="text-gray-900 dark:text-gray-50">
<td class="px-4 py-3"> <tr v-for="s in state.scans" :key="s.id">
<span v-text="s.distance"></span>m <td class="px-4 py-3">
</td> <span v-text="s.distance"></span>
<td class="px-4 py-3" v-text="s.lapTime"></td> </td>
</tr> <td class="px-4 py-3" v-text="s.lapTime"></td>
</tbody> </tr>
</table> </tbody>
<div v-else class="text-center font-bold text-black dark:text-white text-2xl"> </table>
<img <div v-else class="text-center font-bold text-black dark:text-white text-2xl">
src="../assets/empty_laps.svg" <img
style="height:25rem; margin:0 auto;" src="../assets/empty_laps.svg"
:alt="[[$t('no_laps_scans_were_recorded_yet')]]" style="height:25rem; margin:0 auto;"
/> :alt="[[$t('no_laps_scans_were_recorded_yet')]]"
{{ $t('no_laps_scans_were_recorded_yet') }} />
</div> {{ $t('no_laps_scans_were_recorded_yet') }}
</div> </div>
</div> </div>
</section> </div>
</div> </section>
</div> </div>
<div v-if="(state.activetab === 'sponsorings')" class="tab-content block"> </div>
<div class="py-4 w-full">coming soon...</div> <div v-if="(state.activetab === 'sponsorings')" class="tab-content block">
</div> <div class="py-4 w-full">
</div> <section class="text-gray-400 dark:bg-gray-900 body-font">
</div> <div class="container mx-auto">
</div> <div class="lg:w-2/3 w-full mx-auto overflow-auto">
</div> <table
</template> v-if="state.sponsorings.length > 0"
class="table-auto w-full text-left whitespace-no-wrap"
<script setup> >
import { reactive } from "vue"; <thead
import { useToast } from "vue-toastification"; class="text-black bg-gray-300 dark:text-white text-sm dark:bg-gray-800"
import axios from "redaxios"; >
<tr>
const state = reactive({ <th class="px-4 py-3 title-font tracking-wider font-medium">Name</th>
phone: "", <th
email: "", class="px-4 py-3 title-font tracking-wider font-medium"
firstname: "", >{{ $t('amount_per_kilometer_in_eur') }}</th>
middlename: "", <th
lastname: "", class="px-4 py-3 title-font tracking-wider font-medium"
scans: [], >{{ $t('current_total_amount_in_eur') }}</th>
group: "", </tr>
activetab: "profile", </thead>
delete_active: false, <tbody class="text-gray-900 dark:text-gray-50">
}) <tr v-for="s in state.sponsorings" :key="s.id">
const toast = useToast(); <td class="px-4 py-3">
const props = defineProps({ <span v-text="s.donor.firstname"></span>
token: String <span v-if="s.donor.middlename">
}) <span v-text="s.donor.middlename"></span>
const accesstoken = atob(props.token); </span>
axios.get(`${config.baseurl}api/runners/me/${accesstoken}`) <span v-text="s.donor.lastname"></span>
.then(({ data }) => { </td>
state.phone = data.phone; <td class="px-4 py-3">
state.email = data.email; <span
state.firstname = data.firstname; v-text="(s.amountPerDistance / 100)
state.middlename = data.middlename; .toFixed(2)
state.lastname = data.lastname; .toLocaleString('de-DE', { valute: 'EUR' })"
state.group = data.group; ></span>€
}).catch((error) => { </td>
toast.error("An error occured while loading your profile data"); <td class="px-4 py-3">
}) <span
axios.get(`${config.baseurl}api/runners/me/${accesstoken}/scans`) v-text="(s.amount / 100)
.then(({ data }) => { .toFixed(2)
data.map(function(s) { .toLocaleString('de-DE', { valute: 'EUR' })"
s.lapTime = Math.floor(s.lapTime / 60) + 'min ' + (Math.floor(s.lapTime % 60) + "").padStart(2, "0") + "s" ></span>€
return s; </td>
}) </tr>
data.filter(s => s.valid === true); </tbody>
state.scans = data; <tfoot class="text-gray-900 dark:text-gray-50">
}).catch((error) => { <tr>
toast.error("An error occured while loading your profile data"); <td class="px-4 py-3">{{ $t('total') }}</td>
}) <td class="px-4 py-3">
function delete_me() { <span
toast("Deletion in progress..."); v-text="(state.sponsorings.reduce(function(sum, current) {
let url = `${config.baseurl}api/runners/me/${accesstoken}?force=true` return sum + current.amountPerDistance;
axios.delete(url) }, 0) / 100)
.then(() => { .toFixed(2)
toast("All Data deleted!"); .toLocaleString('de-DE', { valute: 'EUR' })"
location.replace(`${config.baseurl_selfservice}`); ></span>€
}) </td>
.catch((error) => { <td class="px-4 py-3">
toast.error("An error occured while deleting your profile data"); <span
}); v-text="(state.sponsorings.reduce(function(sum, current) {
} return sum + current.amount;
}, 0) / 100)
.toFixed(2)
.toLocaleString('de-DE', { valute: 'EUR' })"
></span>€
</td>
</tr>
</tfoot>
</table>
<div v-else class="text-center font-bold text-black dark:text-white text-2xl">
<img
src="../assets/empty_laps.svg"
style="height:25rem; margin:0 auto;"
:alt="[[$t('no_sponsorings_for_you_were_recorded_yet')]]"
/>
{{ $t('no_sponsorings_for_you_were_recorded_yet') }}
</div>
</div>
</div>
</section>
</div>
</div>
</div>
</div>
</div>
</div>
</template>
<script setup>
import { reactive } from "vue";
import { useToast } from "vue-toastification";
import axios from "redaxios";
const state = reactive({
phone: "",
email: "",
firstname: "",
middlename: "",
lastname: "",
scans: [],
sponsorings: [],
group: "",
activetab: "profile",
delete_active: false,
fullobject: {}
})
const toast = useToast();
const props = defineProps({
token: String
})
const accesstoken = atob(props.token);
axios.get(`${config.baseurl}api/runners/me/${accesstoken}`)
.then(({ data }) => {
state.phone = data.phone;
state.email = data.email;
state.firstname = data.firstname;
state.middlename = data.middlename;
state.lastname = data.lastname;
state.group = data.group;
state.sponsorings = data.distanceDonations;
state.fullobject = data;
}).catch((error) => {
toast.error("An error occured while loading your profile data");
})
axios.get(`${config.baseurl}api/runners/me/${accesstoken}/scans`)
.then(({ data }) => {
data.map(function(s) {
s.lapTime = Math.floor(s.lapTime / 60) + 'min ' + (Math.floor(s.lapTime % 60) + "").padStart(2, "0") + "s"
s.distance = Math.floor(s.distance / 1000) + 'km ' + (Math.floor(s.distance % 1000) + "").padStart(3, "0") + "m"
return s;
})
data.filter(s => s.valid === true);
state.scans = data;
}).catch((error) => {
toast.error("An error occured while loading your profile data");
})
function delete_me() {
toast("Deletion in progress...");
let url = `${config.baseurl}api/runners/me/${accesstoken}?force=true`
axios.delete(url)
.then(() => {
toast("All Data deleted!");
location.replace(`${config.baseurl_selfservice}`);
})
.catch((error) => {
toast.error("An error occured while deleting your profile data");
});
}
function get_certificate() {
toast("Generation in progress...");
const browserlocale = ((navigator.languages && navigator.languages[0]) || '').substr(0, 2);
let url = `${config.baseurl}documents/certificates?locale=${browserlocale}&download=true&key=${config.documentserver_key}`;
let postdata = Object.assign({}, state.fullobject);
postdata.group = {
name: postdata.group
}
postdata = [postdata]
axios.post(url, postdata, {
responseType: "blob"
})
.then((response) => {
console.log(response)
if (response.status != "200") {
toast.error("An error occured while generateing your certificate!");
} else {
var fileURL = window.URL.createObjectURL(new Blob([response.data], { type: 'application/pdf' }));
var fileLink = document.createElement('a');
fileLink.href = fileURL;
fileLink.setAttribute('download', 'Certificate.pdf');
document.body.appendChild(fileLink);
fileLink.click();
fileLink.remove();
toast("Document generated!");
}
})
.catch((err) => {
console.error(err);
toast.error("An error occured while deleting your profile data");
});
}
</script> </script>