Compare commits
	
		
			21 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| c184eb5bde | |||
| 29912e84b0 | |||
| 789d580e62 | |||
| 5a61c78fa0 | |||
| 77b941de7e | |||
| 15c2bc2aa3 | |||
| b7c1ce947c | |||
| 22a8953332 | |||
| fdfb7be739 | |||
| a426e57b6f | |||
| 91c257f3ba | |||
| 0ca99e7a16 | |||
| 1e882a37f5 | |||
| 387002b261 | |||
| 42d6fa1bb8 | |||
| 318408add8 | |||
| fc147d0f58 | |||
| 9181ac5443 | |||
| cb922e9ce9 | |||
| 36ca85ebf7 | |||
| d96f32cc84 | 
							
								
								
									
										62
									
								
								.drone.yml
									
									
									
									
									
								
							
							
						
						
									
										62
									
								
								.drone.yml
									
									
									
									
									
								
							| @@ -1,62 +0,0 @@ | ||||
| --- | ||||
| kind: secret | ||||
| name: gitea_token | ||||
| get: | ||||
|   path: odit-git-bot | ||||
|   name: apikey | ||||
|  | ||||
| --- | ||||
| kind: secret | ||||
| name: ci_token | ||||
| get: | ||||
|   path: odit-ci-bot | ||||
|   name: apikey | ||||
|  | ||||
| --- | ||||
| kind: secret | ||||
| name: npm_url | ||||
| get: | ||||
|   path: odit-npm-cache | ||||
|   name: url | ||||
|  | ||||
| --- | ||||
| kind: pipeline | ||||
| type: kubernetes | ||||
| name: build:tag | ||||
| steps: | ||||
|   - name: run build | ||||
|     image: registry.odit.services/hub/library/node:19.5.0-alpine3.16 | ||||
|     commands: | ||||
|       - apk add git zip -f | ||||
|       - npm config set registry $NPM_REGISTRY_URL && npm i -g pnpm@8 | ||||
|       - pnpm i | ||||
|       - pnpm build | ||||
|       - mkdir out | ||||
|       - zip -r out/dist.zip dist | ||||
|     environment: | ||||
|       NPM_REGISTRY_URL: | ||||
|         from_secret: npm_url | ||||
|   - 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/* | ||||
|       title: Release ${DRONE_TAG} | ||||
|       checksum: | ||||
|         - md5 | ||||
|         - sha1 | ||||
|         - sha256 | ||||
|         - sha512 | ||||
|         - adler32 | ||||
|         - crc32 | ||||
|   - name: trigger electron build | ||||
|     image: idcooldi/drone-webhook | ||||
|     settings: | ||||
|       urls: https://ci.odit.services/api/repos/lfk/scanclient-electron/builds?SOURCE_TAG=${DRONE_TAG} | ||||
|       bearer: | ||||
|         from_secret: ci_token | ||||
| trigger: | ||||
|   event: | ||||
|     - tag | ||||
							
								
								
									
										27
									
								
								.gitea/workflows/release.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								.gitea/workflows/release.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| name: Build release images | ||||
| on: | ||||
|   push: | ||||
|     tags: | ||||
|       - "*.*.*" | ||||
|  | ||||
| jobs: | ||||
|   build-container: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v4 | ||||
|       - name: Login to registry | ||||
|         uses: docker/login-action@v3 | ||||
|         with: | ||||
|           registry: registry.odit.services | ||||
|           username: ${{ vars.REGISTRY_USERNAME }} | ||||
|           password: ${{ secrets.REGISTRY_PASSWORD }} | ||||
|       - name: Set up Docker Buildx | ||||
|         uses: docker/setup-buildx-action@v3 | ||||
|       - name: Build and push | ||||
|         uses: docker/build-push-action@v6 | ||||
|         with: | ||||
|           push: true | ||||
|           tags: | | ||||
|             ${{ vars.REGISTRY }}/lfk/scanclient:${{ github.ref_name }} | ||||
|           platforms: linux/amd64,linux/arm64 | ||||
							
								
								
									
										31
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										31
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,9 +1,24 @@ | ||||
| /node_modules/ | ||||
| /dist/ | ||||
| # Logs | ||||
| logs | ||||
| *.log | ||||
| npm-debug.log* | ||||
| yarn-debug.log* | ||||
| yarn-error.log* | ||||
| pnpm-debug.log* | ||||
| lerna-debug.log* | ||||
|  | ||||
| node_modules | ||||
| dist | ||||
| dist-ssr | ||||
| *.local | ||||
|  | ||||
| # Editor directories and files | ||||
| .vscode/* | ||||
| !.vscode/extensions.json | ||||
| .idea | ||||
| .DS_Store | ||||
| package-lock.json | ||||
| yarn.lock | ||||
| /out | ||||
| /yarn.lock | ||||
| /app/node_modules | ||||
| /app/dist | ||||
| *.suo | ||||
| *.ntvs* | ||||
| *.njsproj | ||||
| *.sln | ||||
| *.sw? | ||||
|   | ||||
							
								
								
									
										3
									
								
								.vscode/extensions.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								.vscode/extensions.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| { | ||||
|   "recommendations": ["svelte.svelte-vscode"] | ||||
| } | ||||
							
								
								
									
										48
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										48
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -2,13 +2,59 @@ | ||||
|  | ||||
| All notable changes to this project will be documented in this file. Dates are displayed in UTC. | ||||
|  | ||||
| #### [1.0.0](https://git.odit.services/lfk/scanclient/compare/0.2.0...1.0.0) | ||||
| #### [1.2.1](https://git.odit.services/lfk/scanclient/compare/1.2.0...1.2.1) | ||||
|  | ||||
| - feat: add imprint & privacy [`789d580`](https://git.odit.services/lfk/scanclient/commit/789d580e62d2d2fed0e72f3321f299ae05977035) | ||||
| - refactor: move to nginx:1.27-alpine [`29912e8`](https://git.odit.services/lfk/scanclient/commit/29912e84b0e9e1b950eeb22ffe75c1533c0c6913) | ||||
|  | ||||
| #### [1.2.0](https://git.odit.services/lfk/scanclient/compare/1.0.3...1.2.0) | ||||
|  | ||||
| > 6 April 2025 | ||||
|  | ||||
| - chore: update to new svelte,vite,tailwindcss [`22a8953`](https://git.odit.services/lfk/scanclient/commit/22a8953332855b9797bb2f2d1e9ab1eee7726cce) | ||||
| - chore: pnpm@9 + node@23 [`0ca99e7`](https://git.odit.services/lfk/scanclient/commit/0ca99e7a1693b7f3d3bf25425a4c6b0cf626500d) | ||||
| - chore: tmp release it swap [`a426e57`](https://git.odit.services/lfk/scanclient/commit/a426e57b6f8a190fb5545a8464c830bde1de528b) | ||||
| - chore(release): 1.2.0 [`5a61c78`](https://git.odit.services/lfk/scanclient/commit/5a61c78fa08da7cb8739917df6a14e4ae74eb001) | ||||
| - cleanups, brand font, etc. [`b7c1ce9`](https://git.odit.services/lfk/scanclient/commit/b7c1ce947c8959f3b06231892c1c425eeaaba45f) | ||||
| - chore(deps): bump some [`fdfb7be`](https://git.odit.services/lfk/scanclient/commit/fdfb7be739ba98534c57ca23f04733138e4c24b9) | ||||
| - feat(ci)!: Switch to woodpecker [`387002b`](https://git.odit.services/lfk/scanclient/commit/387002b261ef2c0681ad6318984581c0e67f4389) | ||||
| - ci: fix order.js [`77b941d`](https://git.odit.services/lfk/scanclient/commit/77b941de7efc2a2b49364f958275529cc9a44c45) | ||||
| - chore: README [`91c257f`](https://git.odit.services/lfk/scanclient/commit/91c257f3ba3dcd5796e643d930f50fd8dbf0610a) | ||||
| - fix(ci): Switched to non-drone env vars [`1e882a3`](https://git.odit.services/lfk/scanclient/commit/1e882a37f506fa456053a6ce5032962553bb5584) | ||||
| - ci: add release script [`15c2bc2`](https://git.odit.services/lfk/scanclient/commit/15c2bc2aa39b8fe94101f979f1dfc3e8e69b40e1) | ||||
|  | ||||
| #### [1.0.3](https://git.odit.services/lfk/scanclient/compare/1.0.2...1.0.3) | ||||
|  | ||||
| > 15 April 2023 | ||||
|  | ||||
| - 🚀Bumped version to 1.0.3 [`42d6fa1`](https://git.odit.services/lfk/scanclient/commit/42d6fa1bb8f7c057b8feade8d3a33be93d084c88) | ||||
| - Moved padding to div [`318408a`](https://git.odit.services/lfk/scanclient/commit/318408add82bcc71804d43f44e437bd38a553c0a) | ||||
| - Added X top padding [`fc147d0`](https://git.odit.services/lfk/scanclient/commit/fc147d0f5813241a868abeeafc83ce0167a9513a) | ||||
| - Enable push on release [`9181ac5`](https://git.odit.services/lfk/scanclient/commit/9181ac54434e4650099631e32a6351b9c5da0d13) | ||||
|  | ||||
| #### [1.0.2](https://git.odit.services/lfk/scanclient/compare/1.0.1...1.0.2) | ||||
|  | ||||
| > 15 April 2023 | ||||
|  | ||||
| - 🚀Bumped version to 1.0.2 [`cb922e9`](https://git.odit.services/lfk/scanclient/commit/cb922e9ce9da5dd17b4ff9416053a34daf9e4edf) | ||||
|  | ||||
| #### [1.0.1](https://git.odit.services/lfk/scanclient/compare/1.0.0...1.0.1) | ||||
|  | ||||
| > 15 April 2023 | ||||
|  | ||||
| - 🚀Bumped version to 1.0.1 [`36ca85e`](https://git.odit.services/lfk/scanclient/commit/36ca85ebf76ee5696875922eec633a864d323fde) | ||||
| - reload on language change [`d96f32c`](https://git.odit.services/lfk/scanclient/commit/d96f32cc8499f7a4b1e6458bb68132394c41f782) | ||||
|  | ||||
| ### [1.0.0](https://git.odit.services/lfk/scanclient/compare/0.2.0...1.0.0) | ||||
|  | ||||
| > 15 April 2023 | ||||
|  | ||||
| - feat(Scanner): move clock to monospace font [`0768492`](https://git.odit.services/lfk/scanclient/commit/076849221a9c6353ec752d4f2213ae2e10b6480b) | ||||
| - Lockfile [`5f86508`](https://git.odit.services/lfk/scanclient/commit/5f865081a64e2f59a1541a6166e85da3d7cfdbfa) | ||||
| - Pinned deependencies [`0edf31a`](https://git.odit.services/lfk/scanclient/commit/0edf31a9ec228d3a027c4ef0d61d31318ded8d9e) | ||||
| - deps: node@19.9.0 [`4ce0eef`](https://git.odit.services/lfk/scanclient/commit/4ce0eeffea6eee4d8af3ba130d2244669e905ed9) | ||||
| - New checkmark and lower padding [`a8a0d00`](https://git.odit.services/lfk/scanclient/commit/a8a0d0018ee27588eede9bb0456d1b41060f35cd) | ||||
| - 🚀Bumped version to 1.0.0 [`bd4fc7a`](https://git.odit.services/lfk/scanclient/commit/bd4fc7a2f46b2c6f84c0881242e099d64ca4d695) | ||||
| - drop html-minifier [`c6700a9`](https://git.odit.services/lfk/scanclient/commit/c6700a9153f96d330f0a6636adaa616b03d232ee) | ||||
| - Moved ci to pnpm w/ cache [`8610e0b`](https://git.odit.services/lfk/scanclient/commit/8610e0b285939f118fab952ca00e76b1ff659a16) | ||||
| - Removed svg padding [`38a91f7`](https://git.odit.services/lfk/scanclient/commit/38a91f730b46206eac714cae69be26212c01af6f) | ||||
|   | ||||
							
								
								
									
										10
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								Dockerfile
									
									
									
									
									
								
							| @@ -1,13 +1,11 @@ | ||||
| FROM registry.odit.services/hub/library/node:19.5.0-alpine3.16 as build | ||||
| ARG NPM_REGISTRY_URL=https://registry.npmjs.org | ||||
| FROM registry.odit.services/hub/library/node:23.11.0-alpine3.21 AS build | ||||
| WORKDIR /app | ||||
|  | ||||
| COPY package.json ./ | ||||
| RUN npm config set registry $NPM_REGISTRY_URL && npm i -g pnpm@8 | ||||
| RUN mkdir /pnpm && pnpm config set store-dir /pnpm && pnpm i | ||||
| COPY ./package.json ./pnpm-lock.yaml ./pnpm-workspace.yaml ./ | ||||
| RUN npm i -g pnpm@10.7 && pnpm i | ||||
|  | ||||
| COPY . . | ||||
| RUN pnpm build | ||||
|  | ||||
| FROM registry.odit.services/library/nginx-brotli:3.15 as final | ||||
| FROM registry.odit.services/hub/library/nginx:1.27-alpine AS final | ||||
| COPY --from=build /app/dist /usr/share/nginx/html | ||||
							
								
								
									
										362
									
								
								LICENSE
									
									
									
									
									
								
							
							
						
						
									
										362
									
								
								LICENSE
									
									
									
									
									
								
							| @@ -1,362 +0,0 @@ | ||||
| Creative Commons Attribution-NonCommercial-ShareAlike 4.0 International Creative | ||||
| Commons Corporation ("Creative Commons") is not a law firm and does not provide | ||||
| legal services or legal advice. Distribution of Creative Commons public licenses | ||||
| does not create a lawyer-client or other relationship. Creative Commons makes | ||||
| its licenses and related information available on an "as-is" basis. Creative | ||||
| Commons gives no warranties regarding its licenses, any material licensed | ||||
| under their terms and conditions, or any related information. Creative Commons | ||||
| disclaims all liability for damages resulting from their use to the fullest | ||||
| extent possible. | ||||
|  | ||||
| Using Creative Commons Public Licenses | ||||
|  | ||||
| Creative Commons public licenses provide a standard set of terms and conditions | ||||
| that creators and other rights holders may use to share original works of | ||||
| authorship and other material subject to copyright and certain other rights | ||||
| specified in the public license below. The following considerations are for | ||||
| informational purposes only, are not exhaustive, and do not form part of our | ||||
| licenses. | ||||
|  | ||||
| Considerations for licensors: Our public licenses are intended for use by | ||||
| those authorized to give the public permission to use material in ways otherwise | ||||
| restricted by copyright and certain other rights. Our licenses are irrevocable. | ||||
| Licensors should read and understand the terms and conditions of the license | ||||
| they choose before applying it. Licensors should also secure all rights necessary | ||||
| before applying our licenses so that the public can reuse the material as | ||||
| expected. Licensors should clearly mark any material not subject to the license. | ||||
| This includes other CC-licensed material, or material used under an exception | ||||
| or limitation to copyright. More considerations for licensors : wiki.creativecommons.org/Considerations_for_licensors | ||||
|  | ||||
| Considerations for the public: By using one of our public licenses, a licensor | ||||
| grants the public permission to use the licensed material under specified | ||||
| terms and conditions. If the licensor's permission is not necessary for any | ||||
| reason–for example, because of any applicable exception or limitation to copyright–then | ||||
| that use is not regulated by the license. Our licenses grant only permissions | ||||
| under copyright and certain other rights that a licensor has authority to | ||||
| grant. Use of the licensed material may still be restricted for other reasons, | ||||
| including because others have copyright or other rights in the material. A | ||||
| licensor may make special requests, such as asking that all changes be marked | ||||
| or described. Although not required by our licenses, you are encouraged to | ||||
| respect those requests where reasonable. More considerations for the public | ||||
| : wiki.creativecommons.org/Considerations_for_licensees | ||||
|  | ||||
| Creative Commons Attribution-NonCommercial-ShareAlike 4.0 International Public | ||||
| License | ||||
|  | ||||
| By exercising the Licensed Rights (defined below), You accept and agree to | ||||
| be bound by the terms and conditions of this Creative Commons Attribution-NonCommercial-ShareAlike | ||||
| 4.0 International Public License ("Public License"). To the extent this Public | ||||
| License may be interpreted as a contract, You are granted the Licensed Rights | ||||
| in consideration of Your acceptance of these terms and conditions, and the | ||||
| Licensor grants You such rights in consideration of benefits the Licensor | ||||
| receives from making the Licensed Material available under these terms and | ||||
| conditions. | ||||
|  | ||||
| Section 1 – Definitions. | ||||
|  | ||||
| a. Adapted Material means material subject to Copyright and Similar Rights | ||||
| that is derived from or based upon the Licensed Material and in which the | ||||
| Licensed Material is translated, altered, arranged, transformed, or otherwise | ||||
| modified in a manner requiring permission under the Copyright and Similar | ||||
| Rights held by the Licensor. For purposes of this Public License, where the | ||||
| Licensed Material is a musical work, performance, or sound recording, Adapted | ||||
| Material is always produced where the Licensed Material is synched in timed | ||||
| relation with a moving image. | ||||
|  | ||||
| b. Adapter's License means the license You apply to Your Copyright and Similar | ||||
| Rights in Your contributions to Adapted Material in accordance with the terms | ||||
| and conditions of this Public License. | ||||
|  | ||||
| c. BY-NC-SA Compatible License means a license listed at creativecommons.org/compatiblelicenses, | ||||
| approved by Creative Commons as essentially the equivalent of this Public | ||||
| License. | ||||
|  | ||||
| d. Copyright and Similar Rights means copyright and/or similar rights closely | ||||
| related to copyright including, without limitation, performance, broadcast, | ||||
| sound recording, and Sui Generis Database Rights, without regard to how the | ||||
| rights are labeled or categorized. For purposes of this Public License, the | ||||
| rights specified in Section 2(b)(1)-(2) are not Copyright and Similar Rights. | ||||
|  | ||||
| e. Effective Technological Measures means those measures that, in the absence | ||||
| of proper authority, may not be circumvented under laws fulfilling obligations | ||||
| under Article 11 of the WIPO Copyright Treaty adopted on December 20, 1996, | ||||
| and/or similar international agreements. | ||||
|  | ||||
| f. Exceptions and Limitations means fair use, fair dealing, and/or any other | ||||
| exception or limitation to Copyright and Similar Rights that applies to Your | ||||
| use of the Licensed Material. | ||||
|  | ||||
| g. License Elements means the license attributes listed in the name of a Creative | ||||
| Commons Public License. The License Elements of this Public License are Attribution, | ||||
| NonCommercial, and ShareAlike. | ||||
|  | ||||
| h. Licensed Material means the artistic or literary work, database, or other | ||||
| material to which the Licensor applied this Public License. | ||||
|  | ||||
| i. Licensed Rights means the rights granted to You subject to the terms and | ||||
| conditions of this Public License, which are limited to all Copyright and | ||||
| Similar Rights that apply to Your use of the Licensed Material and that the | ||||
| Licensor has authority to license. | ||||
|  | ||||
| j. Licensor means the individual(s) or entity(ies) granting rights under this | ||||
| Public License. | ||||
|  | ||||
| k. NonCommercial means not primarily intended for or directed towards commercial | ||||
| advantage or monetary compensation. For purposes of this Public License, the | ||||
| exchange of the Licensed Material for other material subject to Copyright | ||||
| and Similar Rights by digital file-sharing or similar means is NonCommercial | ||||
| provided there is no payment of monetary compensation in connection with the | ||||
| exchange. | ||||
|  | ||||
| l. Share means to provide material to the public by any means or process that | ||||
| requires permission under the Licensed Rights, such as reproduction, public | ||||
| display, public performance, distribution, dissemination, communication, or | ||||
| importation, and to make material available to the public including in ways | ||||
| that members of the public may access the material from a place and at a time | ||||
| individually chosen by them. | ||||
|  | ||||
| m. Sui Generis Database Rights means rights other than copyright resulting | ||||
| from Directive 96/9/EC of the European Parliament and of the Council of 11 | ||||
| March 1996 on the legal protection of databases, as amended and/or succeeded, | ||||
| as well as other essentially equivalent rights anywhere in the world. | ||||
|  | ||||
| n. You means the individual or entity exercising the Licensed Rights under | ||||
| this Public License. Your has a corresponding meaning. | ||||
|  | ||||
| Section 2 – Scope. | ||||
|  | ||||
|    a. License grant. | ||||
|  | ||||
| 1. Subject to the terms and conditions of this Public License, the Licensor | ||||
| hereby grants You a worldwide, royalty-free, non-sublicensable, non-exclusive, | ||||
| irrevocable license to exercise the Licensed Rights in the Licensed Material | ||||
| to: | ||||
|  | ||||
| A. reproduce and Share the Licensed Material, in whole or in part, for NonCommercial | ||||
| purposes only; and | ||||
|  | ||||
| B. produce, reproduce, and Share Adapted Material for NonCommercial purposes | ||||
| only. | ||||
|  | ||||
| 2. Exceptions and Limitations. For the avoidance of doubt, where Exceptions | ||||
| and Limitations apply to Your use, this Public License does not apply, and | ||||
| You do not need to comply with its terms and conditions. | ||||
|  | ||||
|       3. Term. The term of this Public License is specified in Section 6(a). | ||||
|  | ||||
| 4. Media and formats; technical modifications allowed. The Licensor authorizes | ||||
| You to exercise the Licensed Rights in all media and formats whether now known | ||||
| or hereafter created, and to make technical modifications necessary to do | ||||
| so. The Licensor waives and/or agrees not to assert any right or authority | ||||
| to forbid You from making technical modifications necessary to exercise the | ||||
| Licensed Rights, including technical modifications necessary to circumvent | ||||
| Effective Technological Measures. For purposes of this Public License, simply | ||||
| making modifications authorized by this Section 2(a)(4) never produces Adapted | ||||
| Material. | ||||
|  | ||||
|       5. Downstream recipients. | ||||
|  | ||||
| A. Offer from the Licensor – Licensed Material. Every recipient of the Licensed | ||||
| Material automatically receives an offer from the Licensor to exercise the | ||||
| Licensed Rights under the terms and conditions of this Public License. | ||||
|  | ||||
| B. Additional offer from the Licensor – Adapted Material. Every recipient | ||||
| of Adapted Material from You automatically receives an offer from the Licensor | ||||
| to exercise the Licensed Rights in the Adapted Material under the conditions | ||||
| of the Adapter's License You apply. | ||||
|  | ||||
| C. No downstream restrictions. You may not offer or impose any additional | ||||
| or different terms or conditions on, or apply any Effective Technological | ||||
| Measures to, the Licensed Material if doing so restricts exercise of the Licensed | ||||
| Rights by any recipient of the Licensed Material. | ||||
|  | ||||
| 6. No endorsement. Nothing in this Public License constitutes or may be construed | ||||
| as permission to assert or imply that You are, or that Your use of the Licensed | ||||
| Material is, connected with, or sponsored, endorsed, or granted official status | ||||
| by, the Licensor or others designated to receive attribution as provided in | ||||
| Section 3(a)(1)(A)(i). | ||||
|  | ||||
|    b. Other rights. | ||||
|  | ||||
| 1. Moral rights, such as the right of integrity, are not licensed under this | ||||
| Public License, nor are publicity, privacy, and/or other similar personality | ||||
| rights; however, to the extent possible, the Licensor waives and/or agrees | ||||
| not to assert any such rights held by the Licensor to the limited extent necessary | ||||
| to allow You to exercise the Licensed Rights, but not otherwise. | ||||
|  | ||||
| 2. Patent and trademark rights are not licensed under this Public License. | ||||
|  | ||||
| 3. To the extent possible, the Licensor waives any right to collect royalties | ||||
| from You for the exercise of the Licensed Rights, whether directly or through | ||||
| a collecting society under any voluntary or waivable statutory or compulsory | ||||
| licensing scheme. In all other cases the Licensor expressly reserves any right | ||||
| to collect such royalties, including when the Licensed Material is used other | ||||
| than for NonCommercial purposes. | ||||
|  | ||||
| Section 3 – License Conditions. | ||||
|  | ||||
| Your exercise of the Licensed Rights is expressly made subject to the following | ||||
| conditions. | ||||
|  | ||||
|    a. Attribution. | ||||
|  | ||||
| 1. If You Share the Licensed Material (including in modified form), You must: | ||||
|  | ||||
| A. retain the following if it is supplied by the Licensor with the Licensed | ||||
| Material: | ||||
|  | ||||
| i. identification of the creator(s) of the Licensed Material and any others | ||||
| designated to receive attribution, in any reasonable manner requested by the | ||||
| Licensor (including by pseudonym if designated); | ||||
|  | ||||
|             ii. a copyright notice; | ||||
|  | ||||
|             iii. a notice that refers to this Public License; | ||||
|  | ||||
|             iv. a notice that refers to the disclaimer of warranties; | ||||
|  | ||||
|              | ||||
|  | ||||
| v. a URI or hyperlink to the Licensed Material to the extent reasonably practicable; | ||||
|  | ||||
| B. indicate if You modified the Licensed Material and retain an indication | ||||
| of any previous modifications; and | ||||
|  | ||||
| C. indicate the Licensed Material is licensed under this Public License, and | ||||
| include the text of, or the URI or hyperlink to, this Public License. | ||||
|  | ||||
| 2. You may satisfy the conditions in Section 3(a)(1) in any reasonable manner | ||||
| based on the medium, means, and context in which You Share the Licensed Material. | ||||
| For example, it may be reasonable to satisfy the conditions by providing a | ||||
| URI or hyperlink to a resource that includes the required information. | ||||
|  | ||||
| 3. If requested by the Licensor, You must remove any of the information required | ||||
| by Section 3(a)(1)(A) to the extent reasonably practicable. | ||||
|  | ||||
| b. ShareAlike.In addition to the conditions in Section 3(a), if You Share | ||||
| Adapted Material You produce, the following conditions also apply. | ||||
|  | ||||
| 1. The Adapter's License You apply must be a Creative Commons license with | ||||
| the same License Elements, this version or later, or a BY-NC-SA Compatible | ||||
| License. | ||||
|  | ||||
| 2. You must include the text of, or the URI or hyperlink to, the Adapter's | ||||
| License You apply. You may satisfy this condition in any reasonable manner | ||||
| based on the medium, means, and context in which You Share Adapted Material. | ||||
|  | ||||
| 3. You may not offer or impose any additional or different terms or conditions | ||||
| on, or apply any Effective Technological Measures to, Adapted Material that | ||||
| restrict exercise of the rights granted under the Adapter's License You apply. | ||||
|  | ||||
| Section 4 – Sui Generis Database Rights. | ||||
|  | ||||
| Where the Licensed Rights include Sui Generis Database Rights that apply to | ||||
| Your use of the Licensed Material: | ||||
|  | ||||
| a. for the avoidance of doubt, Section 2(a)(1) grants You the right to extract, | ||||
| reuse, reproduce, and Share all or a substantial portion of the contents of | ||||
| the database for NonCommercial purposes only; | ||||
|  | ||||
| b. if You include all or a substantial portion of the database contents in | ||||
| a database in which You have Sui Generis Database Rights, then the database | ||||
| in which You have Sui Generis Database Rights (but not its individual contents) | ||||
| is Adapted Material, including for purposes of Section 3(b); and | ||||
|  | ||||
| c. You must comply with the conditions in Section 3(a) if You Share all or | ||||
| a substantial portion of the contents of the database. | ||||
|  | ||||
| For the avoidance of doubt, this Section 4 supplements and does not replace | ||||
| Your obligations under this Public License where the Licensed Rights include | ||||
| other Copyright and Similar Rights. | ||||
|  | ||||
| Section 5 – Disclaimer of Warranties and Limitation of Liability. | ||||
|  | ||||
| a. Unless otherwise separately undertaken by the Licensor, to the extent possible, | ||||
| the Licensor offers the Licensed Material as-is and as-available, and makes | ||||
| no representations or warranties of any kind concerning the Licensed Material, | ||||
| whether express, implied, statutory, or other. This includes, without limitation, | ||||
| warranties of title, merchantability, fitness for a particular purpose, non-infringement, | ||||
| absence of latent or other defects, accuracy, or the presence or absence of | ||||
| errors, whether or not known or discoverable. Where disclaimers of warranties | ||||
| are not allowed in full or in part, this disclaimer may not apply to You. | ||||
|  | ||||
| b. To the extent possible, in no event will the Licensor be liable to You | ||||
| on any legal theory (including, without limitation, negligence) or otherwise | ||||
| for any direct, special, indirect, incidental, consequential, punitive, exemplary, | ||||
| or other losses, costs, expenses, or damages arising out of this Public License | ||||
| or use of the Licensed Material, even if the Licensor has been advised of | ||||
| the possibility of such losses, costs, expenses, or damages. Where a limitation | ||||
| of liability is not allowed in full or in part, this limitation may not apply | ||||
| to You. | ||||
|  | ||||
| c. The disclaimer of warranties and limitation of liability provided above | ||||
| shall be interpreted in a manner that, to the extent possible, most closely | ||||
| approximates an absolute disclaimer and waiver of all liability. | ||||
|  | ||||
| Section 6 – Term and Termination. | ||||
|  | ||||
| a. This Public License applies for the term of the Copyright and Similar Rights | ||||
| licensed here. However, if You fail to comply with this Public License, then | ||||
| Your rights under this Public License terminate automatically. | ||||
|  | ||||
| b. Where Your right to use the Licensed Material has terminated under Section | ||||
| 6(a), it reinstates: | ||||
|  | ||||
| 1. automatically as of the date the violation is cured, provided it is cured | ||||
| within 30 days of Your discovery of the violation; or | ||||
|  | ||||
|       2. upon express reinstatement by the Licensor. | ||||
|  | ||||
| For the avoidance of doubt, this Section 6(b) does not affect any right the | ||||
| Licensor may have to seek remedies for Your violations of this Public License. | ||||
|  | ||||
| c. For the avoidance of doubt, the Licensor may also offer the Licensed Material | ||||
| under separate terms or conditions or stop distributing the Licensed Material | ||||
| at any time; however, doing so will not terminate this Public License. | ||||
|  | ||||
|    d. Sections 1, 5, 6, 7, and 8 survive termination of this Public License. | ||||
|  | ||||
| Section 7 – Other Terms and Conditions. | ||||
|  | ||||
| a. The Licensor shall not be bound by any additional or different terms or | ||||
| conditions communicated by You unless expressly agreed. | ||||
|  | ||||
| b. Any arrangements, understandings, or agreements regarding the Licensed | ||||
| Material not stated herein are separate from and independent of the terms | ||||
| and conditions of this Public License. | ||||
|  | ||||
| Section 8 – Interpretation. | ||||
|  | ||||
| a. For the avoidance of doubt, this Public License does not, and shall not | ||||
| be interpreted to, reduce, limit, restrict, or impose conditions on any use | ||||
| of the Licensed Material that could lawfully be made without permission under | ||||
| this Public License. | ||||
|  | ||||
| b. To the extent possible, if any provision of this Public License is deemed | ||||
| unenforceable, it shall be automatically reformed to the minimum extent necessary | ||||
| to make it enforceable. If the provision cannot be reformed, it shall be severed | ||||
| from this Public License without affecting the enforceability of the remaining | ||||
| terms and conditions. | ||||
|  | ||||
| c. No term or condition of this Public License will be waived and no failure | ||||
| to comply consented to unless expressly agreed to by the Licensor. | ||||
|  | ||||
| d. Nothing in this Public License constitutes or may be interpreted as a limitation | ||||
| upon, or waiver of, any privileges and immunities that apply to the Licensor | ||||
| or You, including from the legal processes of any jurisdiction or authority. | ||||
|  | ||||
| Creative Commons is not a party to its public licenses. Notwithstanding, Creative | ||||
| Commons may elect to apply one of its public licenses to material it publishes | ||||
| and in those instances will be considered the "Licensor." The text of the | ||||
| Creative Commons public licenses is dedicated to the public domain under the | ||||
| CC0 Public Domain Dedication. Except for the limited purpose of indicating | ||||
| that material is shared under a Creative Commons public license or as otherwise | ||||
| permitted by the Creative Commons policies published at creativecommons.org/policies, | ||||
| Creative Commons does not authorize the use of the trademark "Creative Commons" | ||||
| or any other trademark or logo of Creative Commons without its prior written | ||||
| consent including, without limitation, in connection with any unauthorized | ||||
| modifications to any of its public licenses or any other arrangements, understandings, | ||||
| or agreements concerning use of licensed material. For the avoidance of doubt, | ||||
| this paragraph does not form part of the public licenses. | ||||
|  | ||||
| Creative Commons may be contacted at creativecommons.org. | ||||
							
								
								
									
										11
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								README.md
									
									
									
									
									
								
							| @@ -2,20 +2,19 @@ | ||||
|  | ||||
| ## ✒️ Overview | ||||
| This is an API client for [https://git.odit.services/lfk/backend](@lfk/backend) | ||||
| - WebApp built with [Svelte](https://svelte.dev), [WindiCSS](https://windicss.org/) (to compile [TailwindCSS](https://tailwindcss.com/)) and [Vite](https://vitejs.dev). | ||||
| - Packaged with electron here: [https://git.odit.services/lfk/selfservice-electron](@lfk/selfservice-electron) | ||||
| - WebApp built with [Svelte](https://svelte.dev), [TailwindCSS](https://tailwindcss.com/) and [Vite](https://vitejs.dev). | ||||
|  | ||||
| ## 🚀 Getting Started | ||||
| ``` | ||||
| yarn | ||||
| pnpm i | ||||
| ``` | ||||
| ## Development | ||||
| ``` | ||||
| yarn dev | ||||
| pnpm dev | ||||
| / | ||||
| yarn dev --open | ||||
| pnpm dev --open | ||||
| ``` | ||||
| ## Build | ||||
| ``` | ||||
| yarn build | ||||
| pnpm build | ||||
| ``` | ||||
							
								
								
									
										11
									
								
								index.html
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								index.html
									
									
									
									
									
								
							| @@ -1,14 +1,13 @@ | ||||
| <!DOCTYPE html> | ||||
| <!doctype html> | ||||
| <html lang="en"> | ||||
|   <head> | ||||
|     <meta charset="UTF-8" /> | ||||
|     <link rel="icon" type="image/svg+xml" href="/favicon.png" /> | ||||
|     <meta name="viewport" content="width=device-width, initial-scale=1.0" /> | ||||
|     <title>LfK!Scan</title> | ||||
|     <base href="./" /> | ||||
|     <link rel="icon" type="image/png" href="./favicon.png" /> | ||||
|   </head> | ||||
|  | ||||
|   <body class="bg-white font-family-karla h-screen"> | ||||
|     <script type="module" src="./src/main.js"></script> | ||||
|   <body> | ||||
|     <div id="app"></div> | ||||
|     <script type="module" src="/src/main.ts"></script> | ||||
|   </body> | ||||
| </html> | ||||
|   | ||||
							
								
								
									
										10145
									
								
								licenses.md
									
									
									
									
									
								
							
							
						
						
									
										10145
									
								
								licenses.md
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										16
									
								
								order.js
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								order.js
									
									
									
									
									
								
							| @@ -1,16 +0,0 @@ | ||||
| const fs = require('fs'); | ||||
| // get all language files | ||||
| const files = fs.readdirSync('./src/locales/'); | ||||
| files.forEach((f) => { | ||||
| 	// read file as object | ||||
| 	const unordered = JSON.parse(fs.readFileSync(`src/locales/${f}`)); | ||||
| 	// order object by keys alpabetically A-Z | ||||
| 	const ordered = Object.keys(unordered).sort().reduce((obj, key) => { | ||||
| 		obj[key] = unordered[key]; | ||||
| 		return obj; | ||||
| 	}, {}); | ||||
| 	// format output as json for commit diff compatibility | ||||
| 	const out = JSON.stringify(ordered, 0, 4); | ||||
| 	// write output file | ||||
| 	fs.writeFileSync(`src/locales/${f}`, out); | ||||
| }); | ||||
							
								
								
									
										18
									
								
								order.mjs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								order.mjs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| import { readFileSync, readdirSync, writeFileSync } from "node:fs"; | ||||
| // get all language files | ||||
| const files = readdirSync("./src/locales/"); | ||||
| files.forEach((f) => { | ||||
| 	// read file as object | ||||
| 	const unordered = JSON.parse(readFileSync(`src/locales/${f}`)); | ||||
| 	// order object by keys alpabetically A-Z | ||||
| 	const ordered = Object.keys(unordered) | ||||
| 		.sort() | ||||
| 		.reduce((obj, key) => { | ||||
| 			obj[key] = unordered[key]; | ||||
| 			return obj; | ||||
| 		}, {}); | ||||
| 	// format output as json for commit diff compatibility | ||||
| 	const out = JSON.stringify(ordered, 0, 4); | ||||
| 	// write output file | ||||
| 	writeFileSync(`src/locales/${f}`, out); | ||||
| }); | ||||
							
								
								
									
										48
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										48
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,49 +1,53 @@ | ||||
| { | ||||
| 	"name": "@lfk/scanclient", | ||||
| 	"version": "1.0.0", | ||||
| 	"private": true, | ||||
| 	"version": "1.2.1", | ||||
| 	"type": "module", | ||||
| 	"scripts": { | ||||
| 		"dev": "vite", | ||||
| 		"release": "release-it", | ||||
| 		"build": "vite build", | ||||
| 		"preview": "vite preview", | ||||
| 		"format": "prettier --write --plugin-search-dir=. ./**/*.html ./**/*.svelte", | ||||
| 		"license:export": "license-exporter --markdown && git stage licenses.md", | ||||
| 		"release": "release-it --only-version" | ||||
| 		"check": "svelte-check --tsconfig ./tsconfig.app.json && tsc -p tsconfig.node.json" | ||||
| 	}, | ||||
| 	"devDependencies": { | ||||
| 		"@odit/license-exporter": "0.0.12", | ||||
| 		"@svitejs/vite-plugin-svelte": "0.11.1", | ||||
| 		"@tsconfig/svelte": "1.0.10", | ||||
| 		"axios": "0.21.1", | ||||
| 		"prettier": "2.2.1", | ||||
| 		"prettier-plugin-svelte": "2.2.0", | ||||
| 		"release-it": "14.5.1", | ||||
| 		"svelte": "3.35.0", | ||||
| 		"svelte-i18n": "3.3.7", | ||||
| 		"svelte-preprocess": "4.6.9", | ||||
| 		"vite": "2.1.2", | ||||
| 		"vite-plugin-windicss": "0.9.2" | ||||
| 		"@odit/license-exporter": "0.2.0", | ||||
| 		"@sveltejs/vite-plugin-svelte": "^5.0.3", | ||||
| 		"@tsconfig/svelte": "^5.0.4", | ||||
| 		"axios": "1.8.4", | ||||
| 		"prettier": "3.5.3", | ||||
| 		"prettier-plugin-svelte": "3.3.3", | ||||
| 		"release-it": "^18.1.2", | ||||
| 		"svelte": "^5.20.2", | ||||
| 		"svelte-check": "^4.1.4", | ||||
| 		"svelte-i18n": "4.0.1", | ||||
| 		"typescript": "~5.8.3", | ||||
| 		"vite": "^6.2.0" | ||||
| 	}, | ||||
| 	"dependencies": { | ||||
| 		"validator": "13.5.2" | ||||
| 		"@fontsource/athiti": "^5.2.5", | ||||
| 		"@tailwindcss/vite": "^4.1.3", | ||||
| 		"tailwindcss": "^4.1.3", | ||||
| 		"validator": "13.15.0" | ||||
| 	}, | ||||
| 	"release-it": { | ||||
| 		"git": { | ||||
| 			"commit": true, | ||||
| 			"requireCleanWorkingDir": false, | ||||
| 			"commitMessage": "🚀Bumped version to ${version}", | ||||
| 			"commitMessage": "chore(release): ${version}", | ||||
| 			"requireBranch": "dev", | ||||
| 			"push": false, | ||||
| 			"push": true, | ||||
| 			"tag": true, | ||||
| 			"tagName": null, | ||||
| 			"tagName": "${version}", | ||||
| 			"tagAnnotation": "${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" | ||||
| 			"after:bump": "npx auto-changelog --commit-limit false -p -u --hide-credit && git add CHANGELOG.md && node order.mjs  && git add src/locales" | ||||
| 		} | ||||
| 	}, | ||||
| 	"volta": { | ||||
| 		"node": "19.9.0" | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										5769
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										5769
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										3
									
								
								pnpm-workspace.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								pnpm-workspace.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| onlyBuiltDependencies: | ||||
|   - es5-ext | ||||
|   - esbuild | ||||
							
								
								
									
										511
									
								
								src/Login.svelte
									
									
									
									
									
								
							
							
						
						
									
										511
									
								
								src/Login.svelte
									
									
									
									
									
								
							| @@ -1,263 +1,262 @@ | ||||
| <script> | ||||
|   import isURL from "validator/lib/isURL"; | ||||
|   import isUUID from "validator/lib/isUUID"; | ||||
|   import { apikey, lang, stationinfo, api_endpoint, page } from "./store.js"; | ||||
|   import axios from "axios"; | ||||
|   import background from "./kaya_kids_background.jpg"; | ||||
|   import { _, locale } from "svelte-i18n"; | ||||
|   let token; | ||||
|   let api_endpoint_input = ""; | ||||
|   $: error = false; | ||||
|   $: errormessage = ""; | ||||
|   $: isTokenValid = | ||||
|     token === "rst" || | ||||
|     (token?.length === 44 && | ||||
|       token.split(".")[0].length === 7 && | ||||
|       isUUID(token.split(".")[1])); | ||||
|   $: isEndpointValid = isURL(api_endpoint_input); | ||||
| 	import isURL from "validator/lib/isURL"; | ||||
| 	import isUUID from "validator/lib/isUUID"; | ||||
| 	import { apikey, lang, stationinfo, api_endpoint, page } from "./store.js"; | ||||
| 	import axios from "axios"; | ||||
| 	import background from "./kaya_kids_background.jpg"; | ||||
| 	import { _, locale } from "svelte-i18n"; | ||||
| 	let token; | ||||
| 	let api_endpoint_input = ""; | ||||
| 	$: error = false; | ||||
| 	$: errormessage = ""; | ||||
| 	$: isTokenValid = | ||||
| 		token === "rst" || | ||||
| 		(token?.length === 44 && | ||||
| 			token.split(".")[0].length === 7 && | ||||
| 			isUUID(token.split(".")[1])); | ||||
| 	$: isEndpointValid = isURL(api_endpoint_input); | ||||
| </script> | ||||
|  | ||||
| <div class="w-full flex flex-wrap"> | ||||
|   <!-- Login Section --> | ||||
|   <div class="w-full md:w-1/2 flex flex-col"> | ||||
|     <div class="flex justify-center md:justify-start pt-12 md:pl-12 md:-mb-24"> | ||||
|       <div class="bg-black text-white font-bold text-xl p-4"> | ||||
|         <img src="./favicon.png" alt="" style="height: 3rem;display: inline;" /> | ||||
|         LfK!Scan | ||||
|       </div> | ||||
|     </div> | ||||
| 	<!-- Login Section --> | ||||
| 	<div class="w-full md:w-1/2 flex flex-col"> | ||||
| 		<div class="flex justify-center md:justify-start pt-12 md:pl-12 md:-mb-24"> | ||||
| 			<div class="bg-black text-white font-bold text-xl p-4"> | ||||
| 				<img src="./favicon.png" alt="" style="height: 3rem;display: inline;" /> | ||||
| 				LfK!Scan | ||||
| 			</div> | ||||
| 		</div> | ||||
|  | ||||
|     <div | ||||
|       class="flex flex-col justify-center md:justify-start my-auto pt-8 md:pt-0 px-8 md:px-24 lg:px-32" | ||||
|     > | ||||
|       <p class="text-center text-3xl">{$_("configuration")}</p> | ||||
|       <p class="text-center"> | ||||
|         {$_("please_provide_the_scan_client_token")}<br /><a | ||||
|           target="_blank" | ||||
|           class="underline" | ||||
|           href="https://docs.lauf-fuer-kaya.de/" | ||||
|           >{$_("see_our_configuration_guide")}</a | ||||
|         > | ||||
|       </p> | ||||
|       {#if error} | ||||
|         {#if errormessage === "invalid_token"} | ||||
|           <div | ||||
|             class="text-white px-6 py-4 border-0 rounded relative bg-red-500 mt-2" | ||||
|           > | ||||
|             <span class="inline-block align-middle"> | ||||
|               <b class="capitalize">{$_("error")}</b><br />{$_( | ||||
|                 "the_provided_scan_station_token_is_invalid" | ||||
|               )}<br />{$_("please_check_your_token_and_try_again")} | ||||
|             </span> | ||||
|           </div> | ||||
|         {/if} | ||||
|         {#if errormessage === "station_disabled"} | ||||
|           <div | ||||
|             class="text-white px-6 py-4 border-0 rounded relative bg-red-500 mt-2" | ||||
|           > | ||||
|             <span class="inline-block align-middle"> | ||||
|               <b class="capitalize">{$_("error")}</b><br />{$_( | ||||
|                 "the_provided_scan_station_is_disabled" | ||||
|               )} | ||||
|             </span> | ||||
|           </div> | ||||
|         {/if} | ||||
|       {/if} | ||||
|       {#if $api_endpoint} | ||||
|         <form | ||||
|           class="flex flex-col pt-3 md:pt-8" | ||||
|           onsubmit="event.preventDefault();" | ||||
|           on:submit={() => { | ||||
|             if (token === "rst") { | ||||
|               apikey.set(""); | ||||
|               api_endpoint.set(""); | ||||
|               page.set(""); | ||||
|               token = ""; | ||||
|               api_endpoint_input = ""; | ||||
|             } else { | ||||
|               axios | ||||
|                 .request({ | ||||
|                   method: "GET", | ||||
|                   url: $api_endpoint + "api/stations/me", | ||||
|                   headers: { Authorization: "Bearer " + token }, | ||||
|                 }) | ||||
|                 .then(function (response) { | ||||
|                   error = false; | ||||
|                   errormessage = ""; | ||||
|                   apikey.set(token); | ||||
|                   stationinfo.set(JSON.stringify(response.data)); | ||||
|                 }) | ||||
|                 .catch(function (e) { | ||||
|                   error = true; | ||||
|                   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> | ||||
|       {:else} | ||||
|         <form | ||||
|           class="flex flex-col pt-3 md:pt-8" | ||||
|           onsubmit="event.preventDefault();" | ||||
|           on:submit={() => { | ||||
|             if (api_endpoint_input === "rst") { | ||||
|               apikey.set(""); | ||||
|               api_endpoint.set(""); | ||||
|               page.set(""); | ||||
|               token = ""; | ||||
|               api_endpoint_input = ""; | ||||
|             } else { | ||||
|               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"> | ||||
|         <p> | ||||
|           <svg | ||||
|             style="height: 1rem;display: inline;" | ||||
|             xmlns="http://www.w3.org/2000/svg" | ||||
|             fill="none" | ||||
|             stroke="currentColor" | ||||
|             stroke-width="2" | ||||
|             stroke-linecap="round" | ||||
|             stroke-linejoin="round" | ||||
|             class="feather feather-zap" | ||||
|             viewBox="0 0 24 24" | ||||
|           > | ||||
|             <path d="M13 2L3 14h9l-1 8 10-12h-9l1-8z" /> | ||||
|           </svg><span | ||||
|             >powered by <a | ||||
|               href="https://odit.services" | ||||
|               target="_blank" | ||||
|               class="underline">ODIT.Services</a | ||||
|             >.</span | ||||
|           > | ||||
|         </p> | ||||
|       </div> | ||||
|     </div> | ||||
|     <div class="w-full p-3"> | ||||
|       <div class="inline-block mr-2 mt-2"> | ||||
|         <button | ||||
|           on:click={() => { | ||||
|             lang.set("de-DE"); | ||||
|             locale.set("de-DE"); | ||||
|           }} | ||||
|           type="button" | ||||
|           class="bg-black focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-black text-white text-sm py-2.5 px-5 rounded-md hover:bg-blue-700" | ||||
|           >Deutsch | ||||
|           <svg | ||||
|             class="h-4 inline" | ||||
|             xmlns="http://www.w3.org/2000/svg" | ||||
|             viewBox="0 0 512 512" | ||||
|             ><path | ||||
|               d="M15.923 345.043C52.094 442.527 145.929 512 256 512s203.906-69.473 240.077-166.957L256 322.783l-240.077 22.26z" | ||||
|               fill="#ffda44" | ||||
|             /><path | ||||
|               d="M256 0C145.929 0 52.094 69.472 15.923 166.957L256 189.217l240.077-22.261C459.906 69.472 366.071 0 256 0z" | ||||
|             /><path | ||||
|               d="M15.923 166.957C5.633 194.69 0 224.686 0 256s5.633 61.31 15.923 89.043h480.155C506.368 317.31 512 287.314 512 256s-5.632-61.31-15.923-89.043H15.923z" | ||||
|               fill="#d80027" | ||||
|             /></svg | ||||
|           ></button | ||||
|         > | ||||
|       </div> | ||||
|       <div class="inline-block mr-2 mt-2"> | ||||
|         <button | ||||
|           on:click={() => { | ||||
|             lang.set("en-US"); | ||||
|             locale.set("en-US"); | ||||
|           }} | ||||
|           type="button" | ||||
|           class="bg-black focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-black text-white text-sm py-2.5 px-5 rounded-md hover:bg-blue-700" | ||||
|           >English | ||||
|           <svg | ||||
|             class="h-4 inline" | ||||
|             xmlns="http://www.w3.org/2000/svg" | ||||
|             viewBox="0 0 512 512" | ||||
|           > | ||||
|             <circle cx="256" cy="256" r="256" fill="#f0f0f0" /> | ||||
|             <g fill="#d80027"> | ||||
|               <path | ||||
|                 d="M244.87 256H512c0-23.106-3.08-45.49-8.819-66.783H244.87V256zM244.87 122.435h229.556a257.35 257.35 0 00-59.07-66.783H244.87v66.783zM256 512c60.249 0 115.626-20.824 159.356-55.652H96.644C140.374 491.176 195.751 512 256 512zM37.574 389.565h436.852a254.474 254.474 0 0028.755-66.783H8.819a254.474 254.474 0 0028.755 66.783z" | ||||
|               /> | ||||
|             </g> | ||||
|             <path | ||||
|               d="M118.584 39.978h23.329l-21.7 15.765 8.289 25.509-21.699-15.765-21.699 15.765 7.16-22.037a257.407 257.407 0 00-49.652 55.337h7.475l-13.813 10.035a255.58 255.58 0 00-6.194 10.938l6.596 20.301-12.306-8.941a253.567 253.567 0 00-8.372 19.873l7.267 22.368h26.822l-21.7 15.765 8.289 25.509-21.699-15.765-12.998 9.444A258.468 258.468 0 000 256h256V0c-50.572 0-97.715 14.67-137.416 39.978zm9.918 190.422l-21.699-15.765L85.104 230.4l8.289-25.509-21.7-15.765h26.822l8.288-25.509 8.288 25.509h26.822l-21.7 15.765 8.289 25.509zm-8.289-100.083l8.289 25.509-21.699-15.765-21.699 15.765 8.289-25.509-21.7-15.765h26.822l8.288-25.509 8.288 25.509h26.822l-21.7 15.765zM220.328 230.4l-21.699-15.765L176.93 230.4l8.289-25.509-21.7-15.765h26.822l8.288-25.509 8.288 25.509h26.822l-21.7 15.765 8.289 25.509zm-8.289-100.083l8.289 25.509-21.699-15.765-21.699 15.765 8.289-25.509-21.7-15.765h26.822l8.288-25.509 8.288 25.509h26.822l-21.7 15.765zm0-74.574l8.289 25.509-21.699-15.765-21.699 15.765 8.289-25.509-21.7-15.765h26.822l8.288-25.509 8.288 25.509h26.822l-21.7 15.765z" | ||||
|               fill="#0052b4" | ||||
|             /> | ||||
|           </svg></button | ||||
|         > | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
| 		<div | ||||
| 			class="flex flex-col justify-center md:justify-start my-auto pt-8 md:pt-0 px-8 md:px-24 lg:px-32" | ||||
| 		> | ||||
| 			<p class="text-center text-3xl">{$_("configuration")}</p> | ||||
| 			<p class="text-center"> | ||||
| 				{$_("please_provide_the_scan_client_token")}<br /><a | ||||
| 					target="_blank" | ||||
| 					class="underline" | ||||
| 					href="https://docs.lauf-fuer-kaya.de/" | ||||
| 					>{$_("see_our_configuration_guide")}</a | ||||
| 				> | ||||
| 			</p> | ||||
| 			{#if error} | ||||
| 				{#if errormessage === "invalid_token"} | ||||
| 					<div | ||||
| 						class="text-white px-6 py-4 border-0 rounded relative bg-red-500 mt-2" | ||||
| 					> | ||||
| 						<span class="inline-block align-middle"> | ||||
| 							<b class="capitalize">{$_("error")}</b><br />{$_( | ||||
| 								"the_provided_scan_station_token_is_invalid" | ||||
| 							)}<br />{$_("please_check_your_token_and_try_again")} | ||||
| 						</span> | ||||
| 					</div> | ||||
| 				{/if} | ||||
| 				{#if errormessage === "station_disabled"} | ||||
| 					<div | ||||
| 						class="text-white px-6 py-4 border-0 rounded relative bg-red-500 mt-2" | ||||
| 					> | ||||
| 						<span class="inline-block align-middle"> | ||||
| 							<b class="capitalize">{$_("error")}</b><br />{$_( | ||||
| 								"the_provided_scan_station_is_disabled" | ||||
| 							)} | ||||
| 						</span> | ||||
| 					</div> | ||||
| 				{/if} | ||||
| 			{/if} | ||||
| 			{#if $api_endpoint} | ||||
| 				<form | ||||
| 					class="flex flex-col pt-3 md:pt-8" | ||||
| 					onsubmit={(e) => { | ||||
| 						e.preventDefault(); | ||||
| 						if (token === "rst") { | ||||
| 							apikey.set(""); | ||||
| 							api_endpoint.set(""); | ||||
| 							page.set(""); | ||||
| 							token = ""; | ||||
| 							api_endpoint_input = ""; | ||||
| 						} else { | ||||
| 							axios | ||||
| 								.request({ | ||||
| 									method: "GET", | ||||
| 									url: $api_endpoint + "api/stations/me", | ||||
| 									headers: { Authorization: "Bearer " + token }, | ||||
| 								}) | ||||
| 								.then(function (response) { | ||||
| 									error = false; | ||||
| 									errormessage = ""; | ||||
| 									apikey.set(token); | ||||
| 									stationinfo.set(JSON.stringify(response.data)); | ||||
| 								}) | ||||
| 								.catch(function (e) { | ||||
| 									error = true; | ||||
| 									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 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={(e) => { | ||||
| 						e.preventDefault(); | ||||
| 						if (api_endpoint_input === "rst") { | ||||
| 							apikey.set(""); | ||||
| 							api_endpoint.set(""); | ||||
| 							page.set(""); | ||||
| 							token = ""; | ||||
| 							api_endpoint_input = ""; | ||||
| 						} else { | ||||
| 							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 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 mt-2"> | ||||
| 				<p> | ||||
| 					powered by <a | ||||
| 						href="https://odit.services?ref=lfk" | ||||
| 						target="_blank" | ||||
| 						class="underline">ODIT.Services</a | ||||
| 					> | ||||
| 				</p> | ||||
| 				<p> | ||||
| 					<a | ||||
| 						href="https://lauf-fuer-kaya.de/datenschutz/" | ||||
| 						target="_blank" | ||||
| 						class="underline">Datenschutz</a | ||||
| 					> | ||||
| 					- | ||||
| 					<a | ||||
| 						href="https://lauf-fuer-kaya.de/impressum/" | ||||
| 						target="_blank" | ||||
| 						class="underline">Impressum</a | ||||
| 					> | ||||
| 				</p> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 		<div class="w-full p-3"> | ||||
| 			<div class="inline-block mr-2 mt-2"> | ||||
| 				<button | ||||
| 					onclick={() => { | ||||
| 						lang.set("de-DE"); | ||||
| 						locale.set("de-DE"); | ||||
| 					}} | ||||
| 					type="button" | ||||
| 					class="bg-black focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-black text-white text-sm py-2.5 px-5 rounded-md hover:bg-blue-700 cursor-pointer" | ||||
| 					>Deutsch | ||||
| 					<svg | ||||
| 						class="h-4 inline" | ||||
| 						xmlns="http://www.w3.org/2000/svg" | ||||
| 						viewBox="0 0 512 512" | ||||
| 						><path | ||||
| 							d="M15.923 345.043C52.094 442.527 145.929 512 256 512s203.906-69.473 240.077-166.957L256 322.783l-240.077 22.26z" | ||||
| 							fill="#ffda44" | ||||
| 						/><path | ||||
| 							d="M256 0C145.929 0 52.094 69.472 15.923 166.957L256 189.217l240.077-22.261C459.906 69.472 366.071 0 256 0z" | ||||
| 						/><path | ||||
| 							d="M15.923 166.957C5.633 194.69 0 224.686 0 256s5.633 61.31 15.923 89.043h480.155C506.368 317.31 512 287.314 512 256s-5.632-61.31-15.923-89.043H15.923z" | ||||
| 							fill="#d80027" | ||||
| 						/></svg | ||||
| 					></button | ||||
| 				> | ||||
| 			</div> | ||||
| 			<div class="inline-block mr-2 mt-2"> | ||||
| 				<button | ||||
| 					onclick={() => { | ||||
| 						lang.set("en-US"); | ||||
| 						locale.set("en-US"); | ||||
| 					}} | ||||
| 					type="button" | ||||
| 					class="bg-black focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-black text-white text-sm py-2.5 px-5 rounded-md hover:bg-blue-700 cursor-pointer" | ||||
| 					>English | ||||
| 					<svg | ||||
| 						class="h-4 inline" | ||||
| 						xmlns="http://www.w3.org/2000/svg" | ||||
| 						viewBox="0 0 512 512" | ||||
| 					> | ||||
| 						<circle cx="256" cy="256" r="256" fill="#f0f0f0" /> | ||||
| 						<g fill="#d80027"> | ||||
| 							<path | ||||
| 								d="M244.87 256H512c0-23.106-3.08-45.49-8.819-66.783H244.87V256zM244.87 122.435h229.556a257.35 257.35 0 00-59.07-66.783H244.87v66.783zM256 512c60.249 0 115.626-20.824 159.356-55.652H96.644C140.374 491.176 195.751 512 256 512zM37.574 389.565h436.852a254.474 254.474 0 0028.755-66.783H8.819a254.474 254.474 0 0028.755 66.783z" | ||||
| 							/> | ||||
| 						</g> | ||||
| 						<path | ||||
| 							d="M118.584 39.978h23.329l-21.7 15.765 8.289 25.509-21.699-15.765-21.699 15.765 7.16-22.037a257.407 257.407 0 00-49.652 55.337h7.475l-13.813 10.035a255.58 255.58 0 00-6.194 10.938l6.596 20.301-12.306-8.941a253.567 253.567 0 00-8.372 19.873l7.267 22.368h26.822l-21.7 15.765 8.289 25.509-21.699-15.765-12.998 9.444A258.468 258.468 0 000 256h256V0c-50.572 0-97.715 14.67-137.416 39.978zm9.918 190.422l-21.699-15.765L85.104 230.4l8.289-25.509-21.7-15.765h26.822l8.288-25.509 8.288 25.509h26.822l-21.7 15.765 8.289 25.509zm-8.289-100.083l8.289 25.509-21.699-15.765-21.699 15.765 8.289-25.509-21.7-15.765h26.822l8.288-25.509 8.288 25.509h26.822l-21.7 15.765zM220.328 230.4l-21.699-15.765L176.93 230.4l8.289-25.509-21.7-15.765h26.822l8.288-25.509 8.288 25.509h26.822l-21.7 15.765 8.289 25.509zm-8.289-100.083l8.289 25.509-21.699-15.765-21.699 15.765 8.289-25.509-21.7-15.765h26.822l8.288-25.509 8.288 25.509h26.822l-21.7 15.765zm0-74.574l8.289 25.509-21.699-15.765-21.699 15.765 8.289-25.509-21.7-15.765h26.822l8.288-25.509 8.288 25.509h26.822l-21.7 15.765z" | ||||
| 							fill="#0052b4" | ||||
| 						/> | ||||
| 					</svg></button | ||||
| 				> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 	</div> | ||||
|  | ||||
|   <!-- Image Section --> | ||||
|   <div class="w-1/2 shadow-2xl"> | ||||
|     <img | ||||
|       alt="" | ||||
|       class="object-cover w-full h-screen hidden md:block" | ||||
|       src={background} | ||||
|     /> | ||||
|   </div> | ||||
| 	<!-- Image Section --> | ||||
| 	<div class="w-1/2 shadow-2xl"> | ||||
| 		<img | ||||
| 			alt="" | ||||
| 			class="object-cover w-full h-screen hidden md:block" | ||||
| 			src={background} | ||||
| 		/> | ||||
| 	</div> | ||||
| </div> | ||||
|   | ||||
| @@ -1,166 +1,166 @@ | ||||
| <script> | ||||
|   import axios from "axios"; | ||||
|   import { _ } from "svelte-i18n"; | ||||
|   import { apikey, api_endpoint, page, stationinfo } from "./store.js"; | ||||
|   function init(el) { | ||||
|     el.focus(); | ||||
|   } | ||||
|   let lastscan_error = ""; | ||||
|   let lastscan_time = ""; | ||||
|   let lastscan_laptime = ""; | ||||
|   let lastscan_totaldistance = ""; | ||||
|   let lastscan_valid = true; | ||||
|   let card = ""; | ||||
|   // live clock at the top | ||||
|   let time = new Date(); | ||||
|   $: hours = (time.getHours() + "").padStart(2, "0"); | ||||
|   $: minutes = (time.getMinutes() + "").padStart(2, "0"); | ||||
|   $: seconds = (time.getSeconds() + "").padStart(2, "0"); | ||||
|   setInterval(() => { | ||||
|     time = new Date(); | ||||
|   }, 1000); | ||||
| <script lang="ts"> | ||||
| 	import axios from "axios"; | ||||
| 	import { _ } from "svelte-i18n"; | ||||
| 	import { apikey, api_endpoint, page, stationinfo } from "./store.js"; | ||||
| 	function init(el: HTMLInputElement) { | ||||
| 		el.focus(); | ||||
| 	} | ||||
| 	let lastscan_error = ""; | ||||
| 	let lastscan_time = ""; | ||||
| 	let lastscan_laptime = ""; | ||||
| 	let lastscan_totaldistance = ""; | ||||
| 	let lastscan_valid = true; | ||||
| 	let card = ""; | ||||
| 	// live clock at the top | ||||
| 	let time = new Date(); | ||||
| 	$: hours = (time.getHours() + "").padStart(2, "0"); | ||||
| 	$: minutes = (time.getMinutes() + "").padStart(2, "0"); | ||||
| 	$: seconds = (time.getSeconds() + "").padStart(2, "0"); | ||||
| 	setInterval(() => { | ||||
| 		time = new Date(); | ||||
| 	}, 1000); | ||||
| </script> | ||||
|  | ||||
| <div class="min-h-screen"> | ||||
|   <div class="bg-white shadow p-2"> | ||||
|     <div class="flex flex-wrap -mx-1 overflow-hidden"> | ||||
|       <div class="my-1 px-1 w-1/3 overflow-hidden text-center self-center"> | ||||
|         <img src="/favicon.png" alt="" class="h-14 mx-auto" /> | ||||
|       </div> | ||||
| 	<div class="bg-white shadow p-2"> | ||||
| 		<div class="flex flex-wrap -mx-1 overflow-hidden"> | ||||
| 			<div class="my-1 px-1 w-1/3 overflow-hidden text-center self-center"> | ||||
| 				<img src="/favicon.png" alt="" class="h-14 mx-auto" /> | ||||
| 			</div> | ||||
|  | ||||
|       <div class="my-1 px-1 w-1/3 overflow-hidden text-center self-center"> | ||||
|         Lauf Für Kaya! Scan | ||||
|       </div> | ||||
| 			<div class="my-1 px-1 w-1/3 overflow-hidden text-center self-center"> | ||||
| 				LfK!Scan | ||||
| 			</div> | ||||
|  | ||||
|       <div class="my-1 px-1 w-1/3 overflow-hidden text-center self-center"> | ||||
|         {JSON.parse($stationinfo).track.name} - #{JSON.parse($stationinfo).track | ||||
|           .id} - {JSON.parse($stationinfo).track.distance}m | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
| 			<div class="my-1 px-1 w-1/3 overflow-hidden text-center self-center"> | ||||
| 				{JSON.parse($stationinfo).track.name} - #{JSON.parse($stationinfo).track | ||||
| 					.id} - {JSON.parse($stationinfo).track.distance}m | ||||
| 			</div> | ||||
| 		</div> | ||||
| 	</div> | ||||
|  | ||||
|   <h1 class="mr-6 text-7xl font-semibold text-center text-gray-900 font-mono"> | ||||
|     {hours}:{minutes}:{seconds} | ||||
|   </h1> | ||||
|   <section class="px-4 py-2 mx-auto max-w-7xl"> | ||||
|     <div class="mx-auto space-y-5 w-full md:w-1/2"> | ||||
|       {#if lastscan_error} | ||||
|         <div | ||||
|           class="text-white px-6 py-4 border-0 rounded relative bg-red-500 mt-2" | ||||
|         > | ||||
|           <span class="inline-block align-middle"> | ||||
|             <b class="capitalize">Error!</b><br />{lastscan_error} | ||||
|           </span> | ||||
|         </div> | ||||
|       {/if} | ||||
|       <form | ||||
|         onsubmit="event.preventDefault();" | ||||
|         on:submit={() => { | ||||
|           if (card === "cnf") { | ||||
|             page.set("settings"); | ||||
|           } else { | ||||
|             card = parseInt(card); | ||||
|             lastscan_error = ""; | ||||
|             axios | ||||
|               .request({ | ||||
|                 method: "POST", | ||||
|                 url: $api_endpoint + "api/scans/trackscans", | ||||
|                 headers: { Authorization: "Bearer " + $apikey }, | ||||
|                 data: { card }, | ||||
|               }) | ||||
|               .then((response) => { | ||||
|                 const time = new Date(); | ||||
|                 const hours = (time.getHours() + "").padStart(2, "0"); | ||||
|                 const minutes = (time.getMinutes() + "").padStart(2, "0"); | ||||
|                 const seconds = (time.getSeconds() + "").padStart(2, "0"); | ||||
|                 lastscan_time = hours + ":" + minutes + ":" + seconds; | ||||
|                 response.data.lapTime = | ||||
|                   Math.floor(response.data.lapTime / 60) + | ||||
|                   "min " + | ||||
|                   (Math.floor(response.data.lapTime % 60) + "").padStart( | ||||
|                     2, | ||||
|                     "0" | ||||
|                   ) + | ||||
|                   "s"; | ||||
|                 lastscan_laptime = response.data.lapTime; | ||||
|                 lastscan_valid = response.data.valid; | ||||
|                 lastscan_totaldistance = | ||||
|                   Math.floor(response.data.runner.distance / 1000) + | ||||
|                   "km " + | ||||
|                   ( | ||||
|                     Math.floor(response.data.runner.distance % 1000) + "" | ||||
|                   ).padStart(3, "0") + | ||||
|                   "m"; | ||||
|               }) | ||||
|               .catch((e) => { | ||||
|                 lastscan_error = e.response.data.message; | ||||
|               }); | ||||
|           } | ||||
|           card = ""; | ||||
|         }} | ||||
|       > | ||||
|         <label class="block"> | ||||
|           <span class="block mb-1 text-xs font-medium text-gray-700" | ||||
|             >{$_("runner_card")}</span | ||||
|           > | ||||
|           <input | ||||
|             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" | ||||
|             type="text" | ||||
|             placeholder="123456789" | ||||
|             required | ||||
|             use:init | ||||
|             bind:value={card} | ||||
|           /> | ||||
|         </label> | ||||
|         {#if lastscan_totaldistance} | ||||
|           <div class="w-full text-center items-center"> | ||||
|             {#if !lastscan_valid || lastscan_error} | ||||
|               <svg | ||||
|                 xmlns="http://www.w3.org/2000/svg" | ||||
|                 fill="none" | ||||
|                 stroke-width="1.5" | ||||
|                 stroke="currentColor" | ||||
|                 class="w-30 h-30 text-center mx-auto text-red-600" | ||||
|                 viewBox="5.25 5.25 13.5 13.5" | ||||
|               > | ||||
|                 <path | ||||
|                   stroke-linecap="round" | ||||
|                   stroke-linejoin="round" | ||||
|                   d="M6 18L18 6M6 6l12 12" | ||||
|                 /> | ||||
|               </svg> | ||||
|             {:else} | ||||
|               <svg | ||||
|                 xmlns="http://www.w3.org/2000/svg" | ||||
|                 fill="none" | ||||
|                 stroke-width="1.5" | ||||
|                 stroke="currentColor" | ||||
|                 class="w-30 h-30 text-center mx-auto text-green-600" | ||||
|                 viewBox="3.75 4.5 16.5 15" | ||||
|               > | ||||
|                 <path | ||||
|                   stroke-linecap="round" | ||||
|                   stroke-linejoin="round" | ||||
|                   d="m4.5 12.75 6 6 9-13.5" | ||||
|                 /> | ||||
|               </svg> | ||||
|             {/if} | ||||
|           </div> | ||||
|           <h1 class="text-2xl font-bold text-center">{$_("total-distance")}</h1> | ||||
|           <h1 class="text-6xl font-bold text-center"> | ||||
|             {lastscan_totaldistance} | ||||
|           </h1> | ||||
|           <h1 class="text-2xl font-bold text-center">{$_("lap-time")}</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} | ||||
|           <h1 class="text-3xl font-bold text-center"> | ||||
|             {$_("please_scan_a_card")} | ||||
|           </h1> | ||||
|         {/if} | ||||
|         <button type="submit" class="hidden">{$_("scan")}</button> | ||||
|       </form> | ||||
|     </div> | ||||
|   </section> | ||||
| 	<h1 class="mr-6 text-7xl font-semibold text-center text-gray-900 font-mono"> | ||||
| 		{hours}:{minutes}:{seconds} | ||||
| 	</h1> | ||||
| 	<section class="px-4 py-2 mx-auto max-w-7xl"> | ||||
| 		<div class="mx-auto space-y-5 w-full md:w-1/2"> | ||||
| 			{#if lastscan_error} | ||||
| 				<div | ||||
| 					class="text-white px-6 py-4 border-0 rounded relative bg-red-500 mt-2" | ||||
| 				> | ||||
| 					<span class="inline-block align-middle"> | ||||
| 						<b class="capitalize">Error!</b><br />{lastscan_error} | ||||
| 					</span> | ||||
| 				</div> | ||||
| 			{/if} | ||||
| 			<form | ||||
| 				onsubmit={(event) => { | ||||
| 					event.preventDefault(); | ||||
| 					if (card === "cnf") { | ||||
| 						page.set("settings"); | ||||
| 					} else { | ||||
| 						card = parseInt(card); | ||||
| 						lastscan_error = ""; | ||||
| 						axios | ||||
| 							.request({ | ||||
| 								method: "POST", | ||||
| 								url: $api_endpoint + "api/scans/trackscans", | ||||
| 								headers: { Authorization: "Bearer " + $apikey }, | ||||
| 								data: { card }, | ||||
| 							}) | ||||
| 							.then((response) => { | ||||
| 								const time = new Date(); | ||||
| 								const hours = (time.getHours() + "").padStart(2, "0"); | ||||
| 								const minutes = (time.getMinutes() + "").padStart(2, "0"); | ||||
| 								const seconds = (time.getSeconds() + "").padStart(2, "0"); | ||||
| 								lastscan_time = hours + ":" + minutes + ":" + seconds; | ||||
| 								response.data.lapTime = | ||||
| 									Math.floor(response.data.lapTime / 60) + | ||||
| 									"min " + | ||||
| 									(Math.floor(response.data.lapTime % 60) + "").padStart( | ||||
| 										2, | ||||
| 										"0" | ||||
| 									) + | ||||
| 									"s"; | ||||
| 								lastscan_laptime = response.data.lapTime; | ||||
| 								lastscan_valid = response.data.valid; | ||||
| 								lastscan_totaldistance = | ||||
| 									Math.floor(response.data.runner.distance / 1000) + | ||||
| 									"km " + | ||||
| 									( | ||||
| 										Math.floor(response.data.runner.distance % 1000) + "" | ||||
| 									).padStart(3, "0") + | ||||
| 									"m"; | ||||
| 							}) | ||||
| 							.catch((e) => { | ||||
| 								lastscan_error = e.response.data.message; | ||||
| 							}); | ||||
| 					} | ||||
| 					card = ""; | ||||
| 				}} | ||||
| 			> | ||||
| 				<label class="block"> | ||||
| 					<span class="block text-base font-semibold text-gray-700" | ||||
| 						>{$_("runner_card")}</span | ||||
| 					> | ||||
| 					<input | ||||
| 						class="shadow appearance-none border rounded w-full py-2 px-3 text-gray-700 leading-tight focus:outline-none focus:shadow-outline" | ||||
| 						type="text" | ||||
| 						placeholder="123456789" | ||||
| 						required | ||||
| 						use:init | ||||
| 						bind:value={card} | ||||
| 					/> | ||||
| 				</label> | ||||
| 				{#if lastscan_totaldistance} | ||||
| 					<div class="w-full text-center items-center pt-2"> | ||||
| 						{#if !lastscan_valid || lastscan_error} | ||||
| 							<svg | ||||
| 								xmlns="http://www.w3.org/2000/svg" | ||||
| 								fill="none" | ||||
| 								stroke-width="1.5" | ||||
| 								stroke="currentColor" | ||||
| 								class="w-30 h-30 text-center mx-auto text-red-600" | ||||
| 								viewBox="5.25 5.25 13.5 13.5" | ||||
| 							> | ||||
| 								<path | ||||
| 									stroke-linecap="round" | ||||
| 									stroke-linejoin="round" | ||||
| 									d="M6 18L18 6M6 6l12 12" | ||||
| 								/> | ||||
| 							</svg> | ||||
| 						{:else} | ||||
| 							<svg | ||||
| 								xmlns="http://www.w3.org/2000/svg" | ||||
| 								fill="none" | ||||
| 								stroke-width="1.5" | ||||
| 								stroke="currentColor" | ||||
| 								class="w-30 h-30 text-center mx-auto text-green-600" | ||||
| 								viewBox="3.75 4.5 16.5 15" | ||||
| 							> | ||||
| 								<path | ||||
| 									stroke-linecap="round" | ||||
| 									stroke-linejoin="round" | ||||
| 									d="m4.5 12.75 6 6 9-13.5" | ||||
| 								/> | ||||
| 							</svg> | ||||
| 						{/if} | ||||
| 					</div> | ||||
| 					<h1 class="text-2xl font-bold text-center">{$_("total-distance")}</h1> | ||||
| 					<h1 class="text-6xl font-bold text-center"> | ||||
| 						{lastscan_totaldistance} | ||||
| 					</h1> | ||||
| 					<h1 class="text-2xl font-bold text-center">{$_("lap-time")}</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} | ||||
| 					<h1 class="text-3xl font-bold text-center"> | ||||
| 						{$_("please_scan_a_card")} | ||||
| 					</h1> | ||||
| 				{/if} | ||||
| 				<button type="submit" class="hidden">{$_("scan")}</button> | ||||
| 			</form> | ||||
| 		</div> | ||||
| 	</section> | ||||
| </div> | ||||
|   | ||||
| @@ -1,112 +1,114 @@ | ||||
| <script> | ||||
|   import { _ } from "svelte-i18n"; | ||||
| 	import { _ } from "svelte-i18n"; | ||||
|  | ||||
|   import { apikey, api_endpoint, lang, page, stationinfo } from "./store.js"; | ||||
| 	import { apikey, api_endpoint, lang, page, stationinfo } from "./store.js"; | ||||
| </script> | ||||
|  | ||||
| <div class="p-5 min-h-screen"> | ||||
|   <h1 class="font-bold text-3xl w-full text-center text-gray-900"> | ||||
|     Lauf Für Kaya! Scan | ||||
|   </h1> | ||||
|   <h1 class="text-3xl w-full text-center text-gray-900">{$_("settings")}</h1> | ||||
|   <p class="block text-sm font-bold text-gray-700 mt-2">{$_("api_key")}</p> | ||||
|   <p class="block text-sm text-gray-700">{$apikey}</p> | ||||
|   <p class="block text-sm font-bold text-gray-700 mt-2"> | ||||
|     {$_("station_description")} | ||||
|   </p> | ||||
|   <p class="block text-sm text-gray-700"> | ||||
|     {JSON.parse($stationinfo).description} | ||||
|   </p> | ||||
|   <p class="block text-sm font-bold text-gray-700 mt-2">{$_("station_id")}</p> | ||||
|   <p class="block text-sm text-gray-700">{JSON.parse($stationinfo).id}</p> | ||||
|   <p class="block text-sm font-bold text-gray-700 mt-2">{$_("track_id")}</p> | ||||
|   <p class="block text-sm text-gray-700">{JSON.parse($stationinfo).track.id}</p> | ||||
|   <p class="block text-sm font-bold text-gray-700 mt-2">{$_("track_name")}</p> | ||||
|   <p class="block text-sm text-gray-700"> | ||||
|     {JSON.parse($stationinfo).track.name} | ||||
|   </p> | ||||
|   <p class="block text-sm font-bold text-gray-700 mt-2"> | ||||
|     {$_("track_distance")} | ||||
|   </p> | ||||
|   <p class="block text-sm text-gray-700"> | ||||
|     {JSON.parse($stationinfo).track.distance} | ||||
|   </p> | ||||
|   <p class="block text-sm font-bold text-gray-700 mt-2"> | ||||
|     {$_("minimum_lap_time")} | ||||
|   </p> | ||||
|   <p class="block text-sm text-gray-700"> | ||||
|     {JSON.parse($stationinfo).track.minimumLapTime}s | ||||
|   </p> | ||||
|   <p class="block text-sm font-bold text-gray-700 mt-2">{$_("language")}</p> | ||||
|   <div class="w-full"> | ||||
|     <div class="inline-block mr-2 mt-2"> | ||||
|       <button | ||||
|         on:click={() => { | ||||
|           lang.set("de-DE"); | ||||
|         }} | ||||
|         type="button" | ||||
|         class:bg-blue-700={$lang === "de-DE"} | ||||
|         class="bg-black focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-black text-white text-sm py-2.5 px-5 rounded-md hover:bg-blue-700" | ||||
|         >Deutsch | ||||
|         <svg | ||||
|           class="h-4 inline" | ||||
|           xmlns="http://www.w3.org/2000/svg" | ||||
|           viewBox="0 0 512 512" | ||||
|           ><path | ||||
|             d="M15.923 345.043C52.094 442.527 145.929 512 256 512s203.906-69.473 240.077-166.957L256 322.783l-240.077 22.26z" | ||||
|             fill="#ffda44" | ||||
|           /><path | ||||
|             d="M256 0C145.929 0 52.094 69.472 15.923 166.957L256 189.217l240.077-22.261C459.906 69.472 366.071 0 256 0z" | ||||
|           /><path | ||||
|             d="M15.923 166.957C5.633 194.69 0 224.686 0 256s5.633 61.31 15.923 89.043h480.155C506.368 317.31 512 287.314 512 256s-5.632-61.31-15.923-89.043H15.923z" | ||||
|             fill="#d80027" | ||||
|           /></svg | ||||
|         ></button | ||||
|       > | ||||
|     </div> | ||||
|     <div class="inline-block mr-2 mt-2"> | ||||
|       <button | ||||
|         on:click={() => { | ||||
|           lang.set("en-EN"); | ||||
|         }} | ||||
|         type="button" | ||||
|         class:bg-blue-700={$lang === "en-EN"} | ||||
|         class="bg-black focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-black text-white text-sm py-2.5 px-5 rounded-md hover:bg-blue-700" | ||||
|         >English | ||||
|         <svg | ||||
|           class="h-4 inline" | ||||
|           xmlns="http://www.w3.org/2000/svg" | ||||
|           viewBox="0 0 512 512" | ||||
|         > | ||||
|           <circle cx="256" cy="256" r="256" fill="#f0f0f0" /> | ||||
|           <g fill="#d80027"> | ||||
|             <path | ||||
|               d="M244.87 256H512c0-23.106-3.08-45.49-8.819-66.783H244.87V256zM244.87 122.435h229.556a257.35 257.35 0 00-59.07-66.783H244.87v66.783zM256 512c60.249 0 115.626-20.824 159.356-55.652H96.644C140.374 491.176 195.751 512 256 512zM37.574 389.565h436.852a254.474 254.474 0 0028.755-66.783H8.819a254.474 254.474 0 0028.755 66.783z" | ||||
|             /> | ||||
|           </g> | ||||
|           <path | ||||
|             d="M118.584 39.978h23.329l-21.7 15.765 8.289 25.509-21.699-15.765-21.699 15.765 7.16-22.037a257.407 257.407 0 00-49.652 55.337h7.475l-13.813 10.035a255.58 255.58 0 00-6.194 10.938l6.596 20.301-12.306-8.941a253.567 253.567 0 00-8.372 19.873l7.267 22.368h26.822l-21.7 15.765 8.289 25.509-21.699-15.765-12.998 9.444A258.468 258.468 0 000 256h256V0c-50.572 0-97.715 14.67-137.416 39.978zm9.918 190.422l-21.699-15.765L85.104 230.4l8.289-25.509-21.7-15.765h26.822l8.288-25.509 8.288 25.509h26.822l-21.7 15.765 8.289 25.509zm-8.289-100.083l8.289 25.509-21.699-15.765-21.699 15.765 8.289-25.509-21.7-15.765h26.822l8.288-25.509 8.288 25.509h26.822l-21.7 15.765zM220.328 230.4l-21.699-15.765L176.93 230.4l8.289-25.509-21.7-15.765h26.822l8.288-25.509 8.288 25.509h26.822l-21.7 15.765 8.289 25.509zm-8.289-100.083l8.289 25.509-21.699-15.765-21.699 15.765 8.289-25.509-21.7-15.765h26.822l8.288-25.509 8.288 25.509h26.822l-21.7 15.765zm0-74.574l8.289 25.509-21.699-15.765-21.699 15.765 8.289-25.509-21.7-15.765h26.822l8.288-25.509 8.288 25.509h26.822l-21.7 15.765z" | ||||
|             fill="#0052b4" | ||||
|           /> | ||||
|         </svg></button | ||||
|       > | ||||
|     </div> | ||||
|   </div> | ||||
|   <br /> | ||||
|   <button | ||||
|     on:click={() => { | ||||
|       page.set(""); | ||||
|     }} | ||||
|     class="mb-3 w-full py-3 border-black border-3 text-black focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-black" | ||||
|     >{$_("back_to_scanner")}</button | ||||
|   > | ||||
|   <button | ||||
|     on:click={() => { | ||||
|       apikey.set(""); | ||||
|       api_endpoint.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" | ||||
|     >{$_("log_out_from_this_client")}</button | ||||
|   > | ||||
| 	<h1 class="font-bold text-3xl w-full text-center text-gray-900"> | ||||
| 		Lauf Für Kaya! Scan | ||||
| 	</h1> | ||||
| 	<h1 class="text-3xl w-full text-center text-gray-900">{$_("settings")}</h1> | ||||
| 	<p class="block text-sm font-bold text-gray-700 mt-2">{$_("api_key")}</p> | ||||
| 	<p class="block text-sm text-gray-700">{$apikey}</p> | ||||
| 	<p class="block text-sm font-bold text-gray-700 mt-2"> | ||||
| 		{$_("station_description")} | ||||
| 	</p> | ||||
| 	<p class="block text-sm text-gray-700"> | ||||
| 		{JSON.parse($stationinfo).description} | ||||
| 	</p> | ||||
| 	<p class="block text-sm font-bold text-gray-700 mt-2">{$_("station_id")}</p> | ||||
| 	<p class="block text-sm text-gray-700">{JSON.parse($stationinfo).id}</p> | ||||
| 	<p class="block text-sm font-bold text-gray-700 mt-2">{$_("track_id")}</p> | ||||
| 	<p class="block text-sm text-gray-700">{JSON.parse($stationinfo).track.id}</p> | ||||
| 	<p class="block text-sm font-bold text-gray-700 mt-2">{$_("track_name")}</p> | ||||
| 	<p class="block text-sm text-gray-700"> | ||||
| 		{JSON.parse($stationinfo).track.name} | ||||
| 	</p> | ||||
| 	<p class="block text-sm font-bold text-gray-700 mt-2"> | ||||
| 		{$_("track_distance")} | ||||
| 	</p> | ||||
| 	<p class="block text-sm text-gray-700"> | ||||
| 		{JSON.parse($stationinfo).track.distance} | ||||
| 	</p> | ||||
| 	<p class="block text-sm font-bold text-gray-700 mt-2"> | ||||
| 		{$_("minimum_lap_time")} | ||||
| 	</p> | ||||
| 	<p class="block text-sm text-gray-700"> | ||||
| 		{JSON.parse($stationinfo).track.minimumLapTime}s | ||||
| 	</p> | ||||
| 	<p class="block text-sm font-bold text-gray-700 mt-2">{$_("language")}</p> | ||||
| 	<div class="w-full"> | ||||
| 		<div class="inline-block mr-2 mt-2"> | ||||
| 			<button | ||||
| 				on:click={() => { | ||||
| 					lang.set("de-DE"); | ||||
| 					location.reload(); | ||||
| 				}} | ||||
| 				type="button" | ||||
| 				class:bg-blue-700={$lang === "de-DE"} | ||||
| 				class="bg-black focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-black text-white text-sm py-2.5 px-5 rounded-md hover:bg-blue-700 cursor-pointer" | ||||
| 				>Deutsch | ||||
| 				<svg | ||||
| 					class="h-4 inline" | ||||
| 					xmlns="http://www.w3.org/2000/svg" | ||||
| 					viewBox="0 0 512 512" | ||||
| 					><path | ||||
| 						d="M15.923 345.043C52.094 442.527 145.929 512 256 512s203.906-69.473 240.077-166.957L256 322.783l-240.077 22.26z" | ||||
| 						fill="#ffda44" | ||||
| 					/><path | ||||
| 						d="M256 0C145.929 0 52.094 69.472 15.923 166.957L256 189.217l240.077-22.261C459.906 69.472 366.071 0 256 0z" | ||||
| 					/><path | ||||
| 						d="M15.923 166.957C5.633 194.69 0 224.686 0 256s5.633 61.31 15.923 89.043h480.155C506.368 317.31 512 287.314 512 256s-5.632-61.31-15.923-89.043H15.923z" | ||||
| 						fill="#d80027" | ||||
| 					/></svg | ||||
| 				></button | ||||
| 			> | ||||
| 		</div> | ||||
| 		<div class="inline-block mr-2 mt-2"> | ||||
| 			<button | ||||
| 				on:click={() => { | ||||
| 					lang.set("en-EN"); | ||||
| 					location.reload(); | ||||
| 				}} | ||||
| 				type="button" | ||||
| 				class:bg-blue-700={$lang === "en-EN"} | ||||
| 				class="bg-black focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-black text-white text-sm py-2.5 px-5 rounded-md hover:bg-blue-700 cursor-pointer" | ||||
| 				>English | ||||
| 				<svg | ||||
| 					class="h-4 inline" | ||||
| 					xmlns="http://www.w3.org/2000/svg" | ||||
| 					viewBox="0 0 512 512" | ||||
| 				> | ||||
| 					<circle cx="256" cy="256" r="256" fill="#f0f0f0" /> | ||||
| 					<g fill="#d80027"> | ||||
| 						<path | ||||
| 							d="M244.87 256H512c0-23.106-3.08-45.49-8.819-66.783H244.87V256zM244.87 122.435h229.556a257.35 257.35 0 00-59.07-66.783H244.87v66.783zM256 512c60.249 0 115.626-20.824 159.356-55.652H96.644C140.374 491.176 195.751 512 256 512zM37.574 389.565h436.852a254.474 254.474 0 0028.755-66.783H8.819a254.474 254.474 0 0028.755 66.783z" | ||||
| 						/> | ||||
| 					</g> | ||||
| 					<path | ||||
| 						d="M118.584 39.978h23.329l-21.7 15.765 8.289 25.509-21.699-15.765-21.699 15.765 7.16-22.037a257.407 257.407 0 00-49.652 55.337h7.475l-13.813 10.035a255.58 255.58 0 00-6.194 10.938l6.596 20.301-12.306-8.941a253.567 253.567 0 00-8.372 19.873l7.267 22.368h26.822l-21.7 15.765 8.289 25.509-21.699-15.765-12.998 9.444A258.468 258.468 0 000 256h256V0c-50.572 0-97.715 14.67-137.416 39.978zm9.918 190.422l-21.699-15.765L85.104 230.4l8.289-25.509-21.7-15.765h26.822l8.288-25.509 8.288 25.509h26.822l-21.7 15.765 8.289 25.509zm-8.289-100.083l8.289 25.509-21.699-15.765-21.699 15.765 8.289-25.509-21.7-15.765h26.822l8.288-25.509 8.288 25.509h26.822l-21.7 15.765zM220.328 230.4l-21.699-15.765L176.93 230.4l8.289-25.509-21.7-15.765h26.822l8.288-25.509 8.288 25.509h26.822l-21.7 15.765 8.289 25.509zm-8.289-100.083l8.289 25.509-21.699-15.765-21.699 15.765 8.289-25.509-21.7-15.765h26.822l8.288-25.509 8.288 25.509h26.822l-21.7 15.765zm0-74.574l8.289 25.509-21.699-15.765-21.699 15.765 8.289-25.509-21.7-15.765h26.822l8.288-25.509 8.288 25.509h26.822l-21.7 15.765z" | ||||
| 						fill="#0052b4" | ||||
| 					/> | ||||
| 				</svg></button | ||||
| 			> | ||||
| 		</div> | ||||
| 	</div> | ||||
| 	<br /> | ||||
| 	<button | ||||
| 		on:click={() => { | ||||
| 			page.set(""); | ||||
| 		}} | ||||
| 		class="mb-3 w-full py-3 border-black border-3 text-black focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-black cursor-pointer" | ||||
| 		>{$_("back_to_scanner")}</button | ||||
| 	> | ||||
| 	<button | ||||
| 		on:click={() => { | ||||
| 			apikey.set(""); | ||||
| 			api_endpoint.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 cursor-pointer" | ||||
| 		>{$_("log_out_from_this_client")}</button | ||||
| 	> | ||||
| </div> | ||||
|   | ||||
							
								
								
									
										4
									
								
								src/app.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								src/app.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| @import "tailwindcss"; | ||||
| body { | ||||
| 	font-family: "Athiti", sans-serif; | ||||
| } | ||||
| @@ -1,8 +0,0 @@ | ||||
| import App from './App.svelte'; | ||||
| import 'windi.css'; | ||||
|  | ||||
| const app = new App({ | ||||
| 	target: document.body | ||||
| }); | ||||
|  | ||||
| export default app; | ||||
							
								
								
									
										10
									
								
								src/main.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								src/main.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
| import { mount } from 'svelte' | ||||
| import '@fontsource/athiti'; | ||||
| import './app.css' | ||||
| import App from './App.svelte' | ||||
|  | ||||
| const app = mount(App, { | ||||
|   target: document.getElementById('app')!, | ||||
| }) | ||||
|  | ||||
| export default app | ||||
							
								
								
									
										2
									
								
								src/vite-env.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								src/vite-env.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| /// <reference types="svelte" /> | ||||
| /// <reference types="vite/client" /> | ||||
							
								
								
									
										7
									
								
								svelte.config.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								svelte.config.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| import { vitePreprocess } from '@sveltejs/vite-plugin-svelte' | ||||
|  | ||||
| export default { | ||||
|   // Consult https://svelte.dev/docs#compile-time-svelte-preprocess | ||||
|   // for more information about preprocessors | ||||
|   preprocess: vitePreprocess(), | ||||
| } | ||||
| @@ -1,13 +0,0 @@ | ||||
| module.exports = { | ||||
|   theme: { | ||||
|     extend: { | ||||
|       colors: { | ||||
|         reepolee: { | ||||
|           500: "#b40000", | ||||
|           600: "#9c0000", | ||||
|           700: "#750000", | ||||
|         }, | ||||
|       }, | ||||
|     }, | ||||
|   }, | ||||
| }; | ||||
							
								
								
									
										20
									
								
								tsconfig.app.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								tsconfig.app.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | ||||
| { | ||||
|   "extends": "@tsconfig/svelte/tsconfig.json", | ||||
|   "compilerOptions": { | ||||
|     "target": "ESNext", | ||||
|     "useDefineForClassFields": true, | ||||
|     "module": "ESNext", | ||||
|     "resolveJsonModule": true, | ||||
|     /** | ||||
|      * Typecheck JS in `.svelte` and `.js` files by default. | ||||
|      * Disable checkJs if you'd like to use dynamic types in JS. | ||||
|      * Note that setting allowJs false does not prevent the use | ||||
|      * of JS in `.svelte` files. | ||||
|      */ | ||||
|     "allowJs": true, | ||||
|     "checkJs": true, | ||||
|     "isolatedModules": true, | ||||
|     "moduleDetection": "force" | ||||
|   }, | ||||
|   "include": ["src/**/*.ts", "src/**/*.js", "src/**/*.svelte"] | ||||
| } | ||||
							
								
								
									
										7
									
								
								tsconfig.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								tsconfig.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| { | ||||
|   "files": [], | ||||
|   "references": [ | ||||
|     { "path": "./tsconfig.app.json" }, | ||||
|     { "path": "./tsconfig.node.json" } | ||||
|   ] | ||||
| } | ||||
							
								
								
									
										24
									
								
								tsconfig.node.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								tsconfig.node.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | ||||
| { | ||||
|   "compilerOptions": { | ||||
|     "tsBuildInfoFile": "./node_modules/.tmp/tsconfig.node.tsbuildinfo", | ||||
|     "target": "ES2022", | ||||
|     "lib": ["ES2023"], | ||||
|     "module": "ESNext", | ||||
|     "skipLibCheck": true, | ||||
|  | ||||
|     /* Bundler mode */ | ||||
|     "moduleResolution": "bundler", | ||||
|     "allowImportingTsExtensions": true, | ||||
|     "isolatedModules": true, | ||||
|     "moduleDetection": "force", | ||||
|     "noEmit": true, | ||||
|  | ||||
|     /* Linting */ | ||||
|     "strict": true, | ||||
|     "noUnusedLocals": true, | ||||
|     "noUnusedParameters": true, | ||||
|     "noFallthroughCasesInSwitch": true, | ||||
|     "noUncheckedSideEffectImports": true | ||||
|   }, | ||||
|   "include": ["vite.config.ts"] | ||||
| } | ||||
| @@ -1,37 +1,8 @@ | ||||
| import svelte from '@svitejs/vite-plugin-svelte'; | ||||
| import windiCSS from 'vite-plugin-windicss'; | ||||
| import { defineConfig } from 'vite'; | ||||
| // | ||||
| export default defineConfig(({ command, mode }) => { | ||||
| 	const isProduction = mode === 'production'; | ||||
| 	return { | ||||
| 		base: './', | ||||
| 		build: { | ||||
| 			polyfillDynamicImport: false, | ||||
| 			cssCodeSplit: false, | ||||
| 			minify: isProduction | ||||
| 		}, | ||||
| 		plugins: [ | ||||
| 			windiCSS({ | ||||
| 				//@ts-ignore | ||||
| 				verbose: true, | ||||
| 				silent: false, | ||||
| 				debug: true, | ||||
| 				config: 'tailwind.config.js', // tailwind config file path (optional) | ||||
| 				compile: false, // false: interpretation mode; true: compilation mode | ||||
| 				prefix: 'windi-', // set compilation mode style prefix | ||||
| 				globalPreflight: true, // set preflight style is global or scoped | ||||
| 				globalUtility: true // set utility style is global or scoped | ||||
| 			}), | ||||
| 			svelte({ | ||||
| 				//@ts-ignore | ||||
| 				hot: !isProduction, | ||||
| 				emitCss: true, | ||||
| 				extensions: [ '.md', '.svx', '.svelte' ], | ||||
| 				preprocess: [ | ||||
| 					// | ||||
| 				] | ||||
| 			}) | ||||
| 		] | ||||
| 	}; | ||||
| }); | ||||
| import { defineConfig } from 'vite' | ||||
| import tailwindcss from '@tailwindcss/vite' | ||||
| import { svelte } from '@sveltejs/vite-plugin-svelte' | ||||
|  | ||||
| // https://vite.dev/config/ | ||||
| export default defineConfig({ | ||||
|   plugins: [svelte(), tailwindcss()], | ||||
| }) | ||||
|   | ||||
		Reference in New Issue
	
	Block a user