From 0c0f4fc90cae0240a397182fa8ffaa3147b016f2 Mon Sep 17 00:00:00 2001 From: vednoc Date: Sun, 15 Oct 2023 16:14:25 +0200 Subject: [PATCH] feat(ui): show errors for client-side validation Thank you, @a0eoc, for the help. Fix #243 Fix #273 --- web/typescript/page/edit.ts | 17 ++++++++++++++--- web/views/review/create.tmpl | 3 +++ web/views/style/add.tmpl | 18 +++++++++++++----- web/views/style/edit.tmpl | 16 ++++++++++++---- web/views/style/import.tmpl | 16 ++++++++++++---- web/views/user/account.tmpl | 4 +++- 6 files changed, 57 insertions(+), 17 deletions(-) diff --git a/web/typescript/page/edit.ts b/web/typescript/page/edit.ts index b41da424..d7ab9760 100644 --- a/web/typescript/page/edit.ts +++ b/web/typescript/page/edit.ts @@ -1,10 +1,21 @@ export function checkMaxLength() { type input = HTMLTextAreaElement | HTMLInputElement; + type error = HTMLParagraphElement; const validate = (el: input, max: number) => { - el.setCustomValidity(el.value.length > max - ? `Your input must be up to ${max} characters.` - : ''); + const curr = el.value.length; + if (curr > max) { + el.setCustomValidity(`Your input must be up to ${max} characters.`); + message(el, `Input is too long. Characters used: ${curr}/${max}`); + } else { + el.setCustomValidity(''); + message(el, ''); + } + } + + const message = (el: input, msg: string) => { + const e = document.querySelector(`.danger.${el.name}`) as error; + if (e) e.innerText = msg; } [...document.querySelectorAll('[maxlength]')].forEach((el: input) => { diff --git a/web/views/review/create.tmpl b/web/views/review/create.tmpl index 2d418676..d81dc6f0 100644 --- a/web/views/review/create.tmpl +++ b/web/views/review/create.tmpl @@ -30,6 +30,9 @@ style="min-height: 120px" aria-describedby="comment-hint" >{{ .Review.Comment }} +
diff --git a/web/views/style/add.tmpl b/web/views/style/add.tmpl index 3b23ef92..19a4e165 100644 --- a/web/views/style/add.tmpl +++ b/web/views/style/add.tmpl @@ -29,8 +29,10 @@ required maxlength="50" type="text" name="name" id="name" value="{{ .Style.Name }}" placeholder="e.g. UserStyles.world tweaks"> - {{ with .err.Name }}{{ end }} - {{ with .dupName }}{{ end }} + + {{ with .dupName }}{{ end }} Short description of what your userstyle does in plain text. Will be used for SEO and rich embeds. @@ -40,7 +42,9 @@ aria-describedby="description-hint" placeholder="e.g. UI experiments for UserStyles.world." >{{ .Style.Description }} - {{ with .err.Description }}{{ end }} + Features, requirements, instructions, links, changelog, etc. Supports {{ template "partials/markdown" . }} syntax. @@ -50,7 +54,9 @@ aria-describedby="notes-hint" placeholder="e.g. Please raise issues in the GitHub repository." >{{ .Style.Notes }} - {{ with .err.Notes }}{{ end }} + Link to userstyle's homepage, bug tracker, etc. @@ -93,7 +99,9 @@ aria-describedby="category-hint" value="{{ .Style.Category }}" placeholder="e.g. test.userstyles.world -> userstyles"> - {{ with .err.Category }}{{ end }} + - {{ with .err.Description }}{{ end }} + Features, requirements, instructions, links, changelog, etc. Supports {{ template "partials/markdown" . }} syntax. @@ -45,7 +49,9 @@ aria-describedby="notes-hint" placeholder="e.g. Please raise issues in the GitHub repository." >{{ .Style.Notes }} - {{ with .err.Notes }}{{ end }} + Link to userstyle's homepage, bug tracker, etc. @@ -113,7 +119,9 @@ aria-describedby="category-hint" value="{{ .Style.Category }}" placeholder="e.g. test.userstyles.world -> userstyles"> - {{ with .err.Category }}{{ end }} + - {{ with .err.Description }}{{ end }} + Features, requirements, instructions, links, changelog, etc. Supports {{ template "partials/markdown" . }} syntax. @@ -46,7 +50,9 @@ aria-describedby="notes-hint" placeholder="e.g. Please raise issues in the GitHub repository." >{{ .Style.Notes }} - {{ with .err.Notes }}{{ end }} + Link to userstyle's homepage, bug tracker, etc. @@ -89,7 +95,9 @@ aria-describedby="category-hint" value="{{ .Style.Category }}" placeholder="e.g. test.userstyles.world -> userstyles"> - {{ with .err.Category }}{{ end }} + - {{ with .errBio }}{{ end }} +