playwright-mcp/src/tools/configure.ts
Ryan Malloy b2462593bc feat: add comprehensive device emulation with geolocation, locale, timezone, permissions, and colorScheme
- Added browser_list_devices tool to show 143+ available device profiles organized by category (iPhone, iPad, Pixel, Galaxy, Desktop, Other)
- Enhanced browser_configure tool with device emulation using Playwright's device descriptors database
- Added support for geolocation coordinates with accuracy settings
- Implemented locale and timezone configuration for internationalization testing
- Added colorScheme preference (light/dark/no-preference) for accessibility testing
- Included permissions management for various browser APIs (geolocation, notifications, camera, microphone)
- Device emulation properly overrides individual viewport/userAgent settings when specified
- All context options are properly applied and browser context is recreated with new settings

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-08-11 06:06:43 -06:00

185 lines
7.7 KiB
TypeScript

/**
* Copyright (c) Microsoft Corporation.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import { z } from 'zod';
import { devices } from 'playwright';
import { defineTool } from './tool.js';
import type { Context } from '../context.js';
import type { Response } from '../response.js';
const configureSchema = z.object({
headless: z.boolean().optional().describe('Whether to run the browser in headless mode'),
viewport: z.object({
width: z.number(),
height: z.number(),
}).optional().describe('Browser viewport size'),
userAgent: z.string().optional().describe('User agent string for the browser'),
device: z.string().optional().describe('Device to emulate (e.g., "iPhone 13", "iPad", "Pixel 5"). Use browser_list_devices to see available devices.'),
geolocation: z.object({
latitude: z.number().min(-90).max(90),
longitude: z.number().min(-180).max(180),
accuracy: z.number().min(0).optional().describe('Accuracy in meters (default: 100)')
}).optional().describe('Set geolocation coordinates'),
locale: z.string().optional().describe('Browser locale (e.g., "en-US", "fr-FR", "ja-JP")'),
timezone: z.string().optional().describe('Timezone ID (e.g., "America/New_York", "Europe/London", "Asia/Tokyo")'),
colorScheme: z.enum(['light', 'dark', 'no-preference']).optional().describe('Preferred color scheme'),
permissions: z.array(z.string()).optional().describe('Permissions to grant (e.g., ["geolocation", "notifications", "camera", "microphone"])')
});
const listDevicesSchema = z.object({});
export default [
defineTool({
capability: 'core',
schema: {
name: 'browser_list_devices',
title: 'List available devices for emulation',
description: 'Get a list of all available device emulation profiles including mobile phones, tablets, and desktop browsers. Each device includes viewport, user agent, and capabilities information.',
inputSchema: listDevicesSchema,
type: 'readOnly',
},
handle: async (context: Context, params: z.output<typeof listDevicesSchema>, response: Response) => {
try {
const deviceList = Object.keys(devices).sort();
const deviceCount = deviceList.length;
// Organize devices by category for better presentation
const categories = {
'iPhone': deviceList.filter(d => d.includes('iPhone')),
'iPad': deviceList.filter(d => d.includes('iPad')),
'Pixel': deviceList.filter(d => d.includes('Pixel')),
'Galaxy': deviceList.filter(d => d.includes('Galaxy')),
'Desktop': deviceList.filter(d => d.includes('Desktop')),
'Other': deviceList.filter(d =>
!d.includes('iPhone') &&
!d.includes('iPad') &&
!d.includes('Pixel') &&
!d.includes('Galaxy') &&
!d.includes('Desktop')
)
};
let result = `Available devices for emulation (${deviceCount} total):\n\n`;
for (const [category, deviceNames] of Object.entries(categories)) {
if (deviceNames.length > 0) {
result += `**${category}:**\n`;
deviceNames.forEach(device => {
const deviceInfo = devices[device];
result += `${device} - ${deviceInfo.viewport.width}x${deviceInfo.viewport.height}${deviceInfo.isMobile ? ' (mobile)' : ''}${deviceInfo.hasTouch ? ' (touch)' : ''}\n`;
});
result += '\n';
}
}
result += 'Use browser_configure with the "device" parameter to emulate any of these devices.';
response.addResult(result);
} catch (error) {
throw new Error(`Failed to list devices: ${error}`);
}
},
}),
defineTool({
capability: 'core',
schema: {
name: 'browser_configure',
title: 'Configure browser settings',
description: 'Change browser configuration settings like headless/headed mode, viewport size, user agent, device emulation, geolocation, locale, timezone, color scheme, or permissions for subsequent operations. This will close the current browser and restart it with new settings.',
inputSchema: configureSchema,
type: 'destructive',
},
handle: async (context: Context, params: z.output<typeof configureSchema>, response: Response) => {
try {
const currentConfig = context.config;
const changes: string[] = [];
// Track what's changing
if (params.headless !== undefined) {
const currentHeadless = currentConfig.browser.launchOptions.headless;
if (params.headless !== currentHeadless) {
changes.push(`headless: ${currentHeadless}${params.headless}`);
}
}
if (params.viewport) {
const currentViewport = currentConfig.browser.contextOptions.viewport;
if (!currentViewport || currentViewport.width !== params.viewport.width || currentViewport.height !== params.viewport.height) {
changes.push(`viewport: ${currentViewport?.width || 'default'}x${currentViewport?.height || 'default'}${params.viewport.width}x${params.viewport.height}`);
}
}
if (params.userAgent) {
const currentUA = currentConfig.browser.contextOptions.userAgent;
if (params.userAgent !== currentUA) {
changes.push(`userAgent: ${currentUA || 'default'}${params.userAgent}`);
}
}
if (params.device) {
if (!devices[params.device]) {
throw new Error(`Unknown device: ${params.device}. Use browser_list_devices to see available devices.`);
}
changes.push(`device: emulating ${params.device}`);
}
if (params.geolocation) {
changes.push(`geolocation: ${params.geolocation.latitude}, ${params.geolocation.longitude}${params.geolocation.accuracy || 100}m)`);
}
if (params.locale) {
changes.push(`locale: ${params.locale}`);
}
if (params.timezone) {
changes.push(`timezone: ${params.timezone}`);
}
if (params.colorScheme) {
changes.push(`colorScheme: ${params.colorScheme}`);
}
if (params.permissions && params.permissions.length > 0) {
changes.push(`permissions: ${params.permissions.join(', ')}`);
}
if (changes.length === 0) {
response.addResult('No configuration changes detected. Current settings remain the same.');
return;
}
// Apply the configuration changes
await context.updateBrowserConfig({
headless: params.headless,
viewport: params.viewport,
userAgent: params.userAgent,
device: params.device,
geolocation: params.geolocation,
locale: params.locale,
timezone: params.timezone,
colorScheme: params.colorScheme,
permissions: params.permissions,
});
response.addResult(`Browser configuration updated successfully:\n${changes.map(c => `${c}`).join('\n')}\n\nThe browser has been restarted with the new settings.`);
} catch (error) {
throw new Error(`Failed to update browser configuration: ${error}`);
}
},
}),
];