diff --git a/.changeset/kind-rice-clean.md b/.changeset/kind-rice-clean.md
new file mode 100644
index 0000000000..c68248112f
--- /dev/null
+++ b/.changeset/kind-rice-clean.md
@@ -0,0 +1,5 @@
+---
+"astro": patch
+---
+
+Fixes some built-in apps of the dev toolbar not closing when clicking the page
diff --git a/.github/workflows/scripts.yml b/.github/workflows/scripts.yml
index e3b9db4992..f7c5305e07 100644
--- a/.github/workflows/scripts.yml
+++ b/.github/workflows/scripts.yml
@@ -7,6 +7,7 @@ on:
- "main"
paths:
- "packages/astro/src/runtime/client/**/*"
+ - "!packages/astro/src/runtime/client/dev-toolbar/**/*"
# Automatically cancel in-progress actions on the same branch
concurrency:
diff --git a/packages/astro/e2e/dev-toolbar.test.js b/packages/astro/e2e/dev-toolbar.test.js
index 3afdf9036d..c3e5528fb1 100644
--- a/packages/astro/e2e/dev-toolbar.test.js
+++ b/packages/astro/e2e/dev-toolbar.test.js
@@ -69,6 +69,10 @@ test.describe('Dev Toolbar', () => {
await page.click('#go-to-b');
await consolePromise;
+ toolbar = page.locator('astro-dev-toolbar');
+ appButton = toolbar.locator('button[data-app-id="astro:home"]');
+ await appButton.click();
+
astroAppCanvas = toolbar.locator('astro-dev-toolbar-app-canvas[data-app-id="astro:home"]');
astroToolbarCards = await astroAppCanvas.locator('astro-dev-toolbar-card');
await page.waitForSelector('astro-dev-toolbar-card');
@@ -290,4 +294,18 @@ test.describe('Dev Toolbar', () => {
expect(serverRenderTime).not.toBe(null);
expect(clientRenderTime).not.toBe(null);
});
+
+ test('can quit apps by clicking outside the window', async ({ page, astro }) => {
+ await page.goto(astro.resolveUrl('/'));
+
+ const toolbar = page.locator('astro-dev-toolbar');
+ for (const appId of ['astro:home', 'astro:audit', 'astro:xray', 'astro:settings']) {
+ const appButton = toolbar.locator(`button[data-app-id="${appId}"]`);
+ await appButton.click();
+
+ await expect(appButton).toHaveClass('item active');
+ await page.click('body');
+ await expect(appButton).not.toHaveClass('active');
+ }
+ });
});
diff --git a/packages/astro/src/runtime/client/dev-toolbar/apps/astro.ts b/packages/astro/src/runtime/client/dev-toolbar/apps/astro.ts
index 27a7b22873..85dd3b6fe5 100644
--- a/packages/astro/src/runtime/client/dev-toolbar/apps/astro.ts
+++ b/packages/astro/src/runtime/client/dev-toolbar/apps/astro.ts
@@ -1,7 +1,7 @@
import type { DevToolbarApp, DevToolbarMetadata } from '../../../../@types/astro.js';
import { isDefinedIcon, type Icon } from '../ui-library/icons.js';
import { colorForIntegration, iconForIntegration } from './utils/icons.js';
-import { createWindowElement } from './utils/window.js';
+import { closeOnOutsideClick, createWindowElement } from './utils/window.js';
const astroLogo =
'';
@@ -34,7 +34,6 @@ export default {
createCanvas();
document.addEventListener('astro:after-swap', createCanvas);
- document.addEventListener('astro:after-swap', fetchIntegrationData);
eventTarget.addEventListener('app-toggled', async (event) => {
resetDebugButton();
@@ -45,6 +44,8 @@ export default {
}
});
+ closeOnOutsideClick(eventTarget);
+
function fetchIntegrationData() {
fetch('https://astro.build/api/v1/dev-overlay/', {
cache: 'no-cache',
@@ -372,8 +373,11 @@ export default {
resetDebugButton();
}, 3500);
});
-
canvas.append(windowComponent);
+
+ // If we have integration data, rebuild that part of the UI as well
+ // as it probably mean that the user had already open the app in this session (ex: view transitions)
+ if (integrationData) refreshIntegrationList();
}
function resetDebugButton() {
diff --git a/packages/astro/src/runtime/client/dev-toolbar/apps/audit/index.ts b/packages/astro/src/runtime/client/dev-toolbar/apps/audit/index.ts
index dfaa7dc88d..e07e6c6ac8 100644
--- a/packages/astro/src/runtime/client/dev-toolbar/apps/audit/index.ts
+++ b/packages/astro/src/runtime/client/dev-toolbar/apps/audit/index.ts
@@ -1,3 +1,4 @@
+import { finder } from '@medv/finder';
import type { DevToolbarApp, DevToolbarMetadata } from '../../../../../@types/astro.js';
import type { DevToolbarHighlight } from '../../ui-library/highlight.js';
import {
@@ -6,9 +7,8 @@ import {
getElementsPositionInDocument,
positionHighlight,
} from '../utils/highlight.js';
-import { createWindowElement } from '../utils/window.js';
+import { closeOnOutsideClick, createWindowElement } from '../utils/window.js';
import { a11y } from './a11y.js';
-import { finder } from '@medv/finder';
import { perf } from './perf.js';
const icon =
@@ -72,26 +72,7 @@ export default {
document.addEventListener('astro:after-swap', async () => lint());
document.addEventListener('astro:page-load', async () => refreshLintPositions);
- function onPageClick(event: MouseEvent) {
- const target = event.target as Element | null;
- if (!target) return;
- if (!target.closest) return;
- if (target.closest('astro-dev-toolbar')) return;
- eventTarget.dispatchEvent(
- new CustomEvent('toggle-app', {
- detail: {
- state: false,
- },
- })
- );
- }
- eventTarget.addEventListener('app-toggled', (event: any) => {
- if (event.detail.state === true) {
- document.addEventListener('click', onPageClick, true);
- } else {
- document.removeEventListener('click', onPageClick, true);
- }
- });
+ closeOnOutsideClick(eventTarget);
async function lint() {
audits.forEach(({ highlightElement }) => {
diff --git a/packages/astro/src/runtime/client/dev-toolbar/apps/settings.ts b/packages/astro/src/runtime/client/dev-toolbar/apps/settings.ts
index 7fa1993090..42d1d46d9e 100644
--- a/packages/astro/src/runtime/client/dev-toolbar/apps/settings.ts
+++ b/packages/astro/src/runtime/client/dev-toolbar/apps/settings.ts
@@ -1,6 +1,6 @@
import type { DevToolbarApp } from '../../../../@types/astro.js';
import { settings, type Settings } from '../settings.js';
-import { createWindowElement } from './utils/window.js';
+import { closeOnOutsideClick, createWindowElement } from './utils/window.js';
interface SettingRow {
name: string;
@@ -49,11 +49,13 @@ export default {
id: 'astro:settings',
name: 'Settings',
icon: 'gear',
- init(canvas) {
+ init(canvas, eventTarget) {
createSettingsWindow();
document.addEventListener('astro:after-swap', createSettingsWindow);
+ closeOnOutsideClick(eventTarget);
+
function createSettingsWindow() {
const windowElement = createWindowElement(
`