From 0ef5d855c9c3692a9f5e4fddfd77ffdda373955c Mon Sep 17 00:00:00 2001
From: "google-labs-jules[bot]"
<161369871+google-labs-jules[bot]@users.noreply.github.com>
Date: Wed, 17 Jun 2026 05:06:34 +0000
Subject: [PATCH] =?UTF-8?q?=F0=9F=8E=A8=20Palette:=20Add=20async=20loading?=
=?UTF-8?q?=20states=20to=20authentication=20forms?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
Co-authored-by: singhaditya21 <53948039+singhaditya21@users.noreply.github.com>
---
.Jules/palette.md | 3 +++
web-demo/js/app.js | 30 ++++++++++++++++++++++++++++++
2 files changed, 33 insertions(+)
create mode 100644 .Jules/palette.md
diff --git a/.Jules/palette.md b/.Jules/palette.md
new file mode 100644
index 0000000..2915120
--- /dev/null
+++ b/.Jules/palette.md
@@ -0,0 +1,3 @@
+## 2026-06-17 - Adding loading states to async form submissions
+**Learning:** When adding visual loading states to vanilla JS form submit buttons, accessing `e.submitter` provides a reliable way to target the specific button used, avoiding issues with generalized button selectors. A `finally` block with awaitable promises is crucial for guaranteed state cleanup, especially since users can submit forms multiple ways (enter key, pointer click).
+**Action:** Use `e.submitter` safely in form submission event handlers and ensure `aria-busy`, `disabled`, and `innerHTML` restoration is wrapped in `try/finally` blocks.
diff --git a/web-demo/js/app.js b/web-demo/js/app.js
index 11508de..37d32e4 100644
--- a/web-demo/js/app.js
+++ b/web-demo/js/app.js
@@ -144,6 +144,15 @@ class ClimaAI {
e.preventDefault();
const email = document.getElementById('loginEmail').value;
const password = document.getElementById('loginPassword').value;
+ const submitBtn = e.submitter;
+ let originalText = '';
+
+ if (submitBtn) {
+ originalText = submitBtn.innerHTML;
+ submitBtn.innerHTML = '⏳ Logging in...';
+ submitBtn.disabled = true;
+ submitBtn.setAttribute('aria-busy', 'true');
+ }
try {
this.showToast('Logging in...', 'info');
@@ -155,6 +164,12 @@ class ClimaAI {
this.checkSubscription();
} catch (error) {
this.showToast(error.message || 'Login failed', 'error');
+ } finally {
+ if (submitBtn) {
+ submitBtn.innerHTML = originalText;
+ submitBtn.disabled = false;
+ submitBtn.removeAttribute('aria-busy');
+ }
}
}
@@ -163,6 +178,15 @@ class ClimaAI {
const name = document.getElementById('registerName').value;
const email = document.getElementById('registerEmail').value;
const password = document.getElementById('registerPassword').value;
+ const submitBtn = e.submitter;
+ let originalText = '';
+
+ if (submitBtn) {
+ originalText = submitBtn.innerHTML;
+ submitBtn.innerHTML = '⏳ Signing up...';
+ submitBtn.disabled = true;
+ submitBtn.setAttribute('aria-busy', 'true');
+ }
try {
this.showToast('Creating account...', 'info');
@@ -174,6 +198,12 @@ class ClimaAI {
this.checkSubscription();
} catch (error) {
this.showToast(error.message || 'Registration failed', 'error');
+ } finally {
+ if (submitBtn) {
+ submitBtn.innerHTML = originalText;
+ submitBtn.disabled = false;
+ submitBtn.removeAttribute('aria-busy');
+ }
}
}