| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154 |
- #!/usr/bin/env python3
- """
- verify.py — Playwright封装,用于验证claude-design产出的HTML
- Usage:
- python verify.py path/to/design.html # 基础:打开+截图+抓控制台错误
- python verify.py design.html --viewports 1920x1080,375x667 # 多viewport
- python verify.py deck.html --slides 10 # 幻灯片逐页截(前10张)
- python verify.py design.html --output ./screenshots/ # 输出目录
- python verify.py design.html --show # 非headless,打开真实浏览器
- 依赖:
- pip install playwright
- playwright install chromium
- """
- import argparse
- import sys
- import os
- import time
- from pathlib import Path
- def parse_viewport(s):
- w, h = s.split('x')
- return {'width': int(w), 'height': int(h)}
- def verify_html(html_path, viewports=None, slides=0, output_dir=None, show=False, wait=2000):
- try:
- from playwright.sync_api import sync_playwright
- except ImportError:
- print("ERROR: playwright未安装。")
- print("运行: pip install playwright && playwright install chromium")
- sys.exit(1)
- html_path = Path(html_path).resolve()
- if not html_path.exists():
- print(f"ERROR: 文件不存在: {html_path}")
- sys.exit(1)
- if output_dir is None:
- output_dir = html_path.parent / 'screenshots'
- output_dir = Path(output_dir)
- output_dir.mkdir(parents=True, exist_ok=True)
- file_url = html_path.as_uri()
- stem = html_path.stem
- if viewports is None:
- viewports = [{'width': 1440, 'height': 900}]
- console_errors = []
- page_errors = []
- with sync_playwright() as p:
- browser = p.chromium.launch(headless=not show)
- for viewport in viewports:
- context = browser.new_context(viewport=viewport, device_scale_factor=2)
- page = context.new_page()
- page.on("console", lambda msg: console_errors.append(f"[{msg.type}] {msg.text}") if msg.type in ("error", "warning") else None)
- page.on("pageerror", lambda err: page_errors.append(str(err)))
- print(f"\n→ 打开 {file_url} @ {viewport['width']}x{viewport['height']}")
- page.goto(file_url, wait_until='networkidle')
- page.wait_for_timeout(wait)
- if slides > 0:
- for i in range(slides):
- screenshot_path = output_dir / f"{stem}-slide-{str(i + 1).zfill(2)}.png"
- page.screenshot(path=str(screenshot_path), full_page=False)
- print(f" ✓ slide {i+1} → {screenshot_path.name}")
- if i < slides - 1:
- page.keyboard.press('ArrowRight')
- page.wait_for_timeout(500)
- else:
- suffix = f"-{viewport['width']}x{viewport['height']}" if len(viewports) > 1 else ""
- screenshot_path = output_dir / f"{stem}{suffix}.png"
- page.screenshot(path=str(screenshot_path), full_page=False)
- print(f" ✓ 截图 → {screenshot_path.name}")
- full_path = output_dir / f"{stem}{suffix}-full.png"
- page.screenshot(path=str(full_path), full_page=True)
- print(f" ✓ 完整页 → {full_path.name}")
- if show:
- print(" (浏览器窗口保持打开,按Enter关闭...)")
- input()
- context.close()
- browser.close()
- print("\n" + "=" * 50)
- print("验证报告")
- print("=" * 50)
- if page_errors:
- print(f"\n❌ Page Errors ({len(page_errors)}):")
- for e in page_errors:
- print(f" - {e}")
- else:
- print("\n✅ 无JavaScript错误")
- if console_errors:
- print(f"\n⚠️ Console Errors/Warnings ({len(console_errors)}):")
- for e in console_errors[:20]:
- print(f" - {e}")
- if len(console_errors) > 20:
- print(f" ... 还有{len(console_errors) - 20}条")
- else:
- print("✅ Console干净")
- print(f"\n📸 截图保存至: {output_dir}")
- return 0 if not page_errors else 1
- def main():
- parser = argparse.ArgumentParser(
- description="Verify HTML design outputs with Playwright",
- formatter_class=argparse.RawDescriptionHelpFormatter,
- )
- parser.add_argument("html_path", help="HTML file path")
- parser.add_argument("--viewports", default="1440x900",
- help="逗号分隔的viewport列表,格式 WxH(默认 1440x900)")
- parser.add_argument("--slides", type=int, default=0,
- help="幻灯片模式:截取前N张(需要HTML支持ArrowRight翻页)")
- parser.add_argument("--output", default=None,
- help="输出目录(默认HTML所在目录的screenshots/)")
- parser.add_argument("--show", action="store_true",
- help="非headless,打开真实浏览器窗口")
- parser.add_argument("--wait", type=int, default=2000,
- help="打开页面后等待的毫秒数(默认2000)")
- args = parser.parse_args()
- viewports = [parse_viewport(v) for v in args.viewports.split(",")]
- return verify_html(
- html_path=args.html_path,
- viewports=viewports,
- slides=args.slides,
- output_dir=args.output,
- show=args.show,
- wait=args.wait,
- )
- if __name__ == "__main__":
- sys.exit(main())
|