Skill: Browser Screenshot
Take focused screenshots of specific regions on web pages β a Reddit post, a tweet, an article section, a chart, etc. β not just a full-page dump.
Prerequisite: agent-browser must be installed and Chrome must have remote debugging enabled. See references/agent-browser-setup.md if unsure.
Overview
This skill handles the full pipeline:
- Research the best page to screenshot (web search, fetch)
- Navigate to the right page in the browser
- Locate the target element/region on the page
- Capture a focused, cropped screenshot of just that region
Hard Rule: No Full-Screen Screenshots
NEVER output an uncropped full-viewport or full-page screenshot as a final result. Full screenshots contain too much noise (nav bars, sidebars, ads, unrelated content) and are unsuitable as article illustrations. Every screenshot MUST be cropped to a focused region.
Step 0: Research β Find and Validate Sources Before Opening the Browser
The browser is for capturing, not for browsing. Before opening anything in Chrome, use text-based tools (WebSearch, WebFetch) to find candidate pages, read their content, and decide which ones are actually worth screenshotting.
Research-First Workflow
- WebSearch to find candidate pages for the topic
- WebFetch each candidate to read its text content β check if it has the information/visual you need
- Evaluate: Is this page worth a screenshot? Does it have a clear, focused region that would work as an illustration?
- Only then open the browser to capture the screenshot
This saves significant time β most candidate pages won't be worth screenshotting, and you can eliminate them without the overhead of browser navigation.
When to Use Browser-First Instead
Skip the WebSearch/WebFetch phase and go directly to Chrome browsing when:
- The target platform requires login β Reddit, LinkedIn, X/Twitter, and other social platforms often gate content behind login walls. If the user's Chrome session is already logged in, use the browser directly.
- The user specifies a platform with a clear search need β e.g., "find a Reddit post about X" or "screenshot a tweet about Y". Go straight to the platform's search in Chrome.
- WebFetch returns blocked/incomplete content β some sites aggressively block non-browser requests. If you get a 403, a CAPTCHA page, or stripped content, switch to Chrome.
In these cases, Chrome browsing replaces WebSearch β navigate to the platform's search page, browse results, and evaluate pages visually before deciding what to screenshot.
Page Selection Strategy
The right page depends on the context of the article and how recent/notable the subject is:
| Subject Type |
Best Page to Find |
How to Find It |
| New model/feature launch (< 6 months) |
Official blog post announcing it |
WebSearch "<model name>" site:<vendor-domain> blog |
| Established product (> 6 months) |
Product landing page or docs overview |
WebSearch "<model name>" official page |
| Open-source model |
HuggingFace model card or GitHub repo |
Direct URL: huggingface.co/<org>/<model> |
| API service |
API documentation page |
WebSearch "<service name>" API docs |
Note: This table lists common subject types but is not exhaustive. Apply the same research-first strategy to any subject type β find the most authoritative and visually clean source page for the topic at hand.
What Makes a Good Screenshot Source
Core principle: Less is more. Focus on content, not chrome.
A good screenshot source contains a focused, self-contained piece of information β a paragraph of text, a key quote, a data table, a diagram. It should NOT be a busy page full of buttons, navigation, sidebars, and interactive elements.
- Prefer: A section of a blog post with a clear heading and 1-2 paragraphs of text. A single chart or diagram. A model card header with name and description. A quote or key finding.
- Avoid: Full landing pages with CTAs and navigation. Dashboard views with multiple panels. Pages dominated by UI controls (buttons, dropdowns, forms) rather than readable content.
- Official blog posts are ideal: they have hero images, prominent titles, and concise descriptions designed for sharing
- Product landing pages can work but only if you crop to the hero section β ignore the rest
- HuggingFace model cards are reliable for open-source models: consistent layout, model name + description always at top
- API docs are acceptable fallback: show the product name and key specs
Rule of thumb: If the region you plan to capture contains more interactive UI elements (buttons, links, nav items) than readable text content, it's a bad crop. Find a more content-rich region, or pick a different page entirely.
Pre-Flight URL Validation
Before opening in the browser, validate URLs with WebFetch (lightweight HEAD/GET) to avoid wasting time on 404s or redirects:
WebFetch: <candidate-url>
β Check status code, title, and content snippet
β If 404 or redirect to unrelated page, try next candidate
Region Selection Strategy
Think about what the article reader needs to see in this screenshot:
| Article Context |
What to Capture |
Target Region |
| Introducing a model in a lineup |
Model name + key tagline/description |
Blog hero section or HF model card header |
| Comparing capabilities |
Feature highlights or spec table |
Blog section showing specs/features |
| Discussing a specific feature |
The feature description |
Relevant section heading + 1-2 paragraphs |
| Showing a product/service |
Brand identity + value prop |
Landing page hero (title + subtitle + visual) |
The screenshot should make the reader think "ah, that's what this model/product is" β not "what am I looking at?"
Step 1: Navigate to the Target Page
Always Start by Listing Tabs
agent-browser --auto-connect tab list
Check if the page is already open. Reuse existing tabs β they have login sessions and correct state.
Navigation by Input Type
| User Provides |
Strategy |
| Direct URL |
agent-browser --auto-connect open <url> |
| Search query |
open https://www.google.com/search?q=<encoded-query> β find and click the best result |
| Platform + topic |
Construct platform search URL (see below) β locate target content |
| Vague description |
Google search β evaluate results β navigate to best match |
Platform-Specific Search URLs
| Platform |
Search URL Pattern |
| Reddit |
https://www.reddit.com/search/?q=<query> |
| X / Twitter |
https://x.com/search?q=<query> |
| LinkedIn |
https://www.linkedin.com/search/results/content/?keywords=<query> |
| Hacker News |
https://hn.algolia.com/?q=<query> |
| GitHub |
https://github.com/search?q=<query> |
| YouTube |
https://www.youtube.com/results?search_query=<query> |
Wait for Page Load
After navigation, wait for content to settle:
agent-browser --auto-connect wait --load networkidle
Note: Some sites (Reddit, X, LinkedIn) never reach networkidle. If open already shows the page title in its output, skip the wait. Use wait 2000 as a safe alternative.
Step 2: Locate the Target Region
This is the critical step. The goal is to find a CSS selector that precisely wraps the content to capture.
Primary Method: DOM Selector Discovery
-
Take an annotated screenshot to understand the page layout:
agent-browser --auto-connect screenshot --annotate
-
Take a snapshot to see the page's accessibility tree:
agent-browser --auto-connect snapshot -i
-
Identify the target container element. Look for:
- Semantic HTML containers:
<article>, <main>, <section>
- Platform-specific components (see Platform Selectors)
- Data attributes:
[data-testid="..."], [data-id="..."]
-
Verify with get box to confirm the element has a reasonable bounding box:
agent-browser --auto-connect get box "<selector>"
This returns { x, y, width, height }. Sanity-check:
- Width should be > 100px and < viewport width
- Height should be > 50px
- If the box is the entire page, the selector is too broad β refine it
-
If the selector is hard to find, use eval to explore the DOM:
agent-browser --auto-connect eval "document.querySelector('article')?.getBoundingClientRect()"
Platform Selectors
Common container selectors for popular platforms:
| Platform |
Target |
Typical Selector |
| Reddit |
A post |
shreddit-post, [data-testid="post-container"] |
| X / Twitter |
A tweet |
article[data-testid="tweet"] |
| LinkedIn |
A feed post |
.feed-shared-update-v2 |
| Hacker News |
A story + comments |
#hnmain .fatitem |
| GitHub |
A repo card |
[data-hpc], .repository-content |
| YouTube |
Video player area |
#player-container-outer |
| Generic article |
Main content |
article, main, [role="main"], .post-content, .article-body |
These selectors may change over time. Always verify with get box before using.
Multiple Matching Elements
If the selector matches multiple elements (e.g., multiple tweets on a timeline), narrow it down:
agent-browser --auto-connect get count "article[data-testid='tweet']"
agent-browser --auto-connect eval --stdin <<'EOF'
const posts = document.querySelectorAll('article[data-testid="tweet"]');
for (let i = 0; i < posts.length; i++) {
const text = posts[i].textContent.substring(0, 80);
console.log(i, text);
}
EOF
Then target a specific one using :nth-of-type(N) or a unique parent selector.
Step 3: Capture the Focused Screenshot
Method A: Scroll + Viewport Screenshot (Preferred for Viewport-Sized Targets)
Best when the target element fits within the viewport.
agent-browser --auto-connect scrollintoview "<selector>"
agent-browser --auto-connect wait 500
agent-browser --auto-connect screenshot /tmp/browser-screenshot-raw.png
Then crop using the bounding box (see Cropping).
Method B: Full-Page Screenshot + Crop (For Any Size Target)
Best when the target might be larger than the viewport or when precise cropping is needed.
agent-browser --auto-connect screenshot --full /tmp/browser-screenshot-full.png
agent-browser --auto-connect get box "<selector>"
Then crop (see Cropping).
Cropping
Use ImageMagick (magick on IMv7, convert is deprecated) to crop the screenshot to the target region. Add padding for visual breathing room.
Retina Display Handling
Critical: On macOS Retina displays, screenshots are captured at 2x resolution. A 1728x940 viewport produces a 3456x1880 image. You MUST account for this:
-
Detect the scale factor: Compare viewport size vs actual image dimensions:
magick identify /tmp/screenshot.png
-
Multiply get box coordinates by the scale factor before cropping:
SCALE=2
X=$((200 * SCALE))
Y=$((450 * SCALE))
W=$((680 * SCALE))
H=$((520 * SCALE))
PADDING=$((16 * SCALE))
Crop Command
magick /tmp/browser-screenshot-full.png \
-crop $((W + PADDING*2))x$((H + PADDING*2))+$((X - PADDING))+$((Y - PADDING)) \
+repage \
<output-path>.png
Important: get box returns floating-point values. Round them to integers before passing to ImageMagick.
Padding: Use 12β20px (viewport px). Increase to ~30px if the target has a distinct visual boundary (card, bordered box). Use 0 if the user wants a tight crop.
Output Path
- If the user specifies an output path, use that
- Otherwise, save to a descriptive name in the current directory, e.g.,
reddit-post-screenshot.png, tweet-screenshot.png
Step 4: Verify the Result
After cropping, read the output image to verify it captured the right content:
If the crop is wrong (missed content, too much whitespace, wrong element), adjust the selector or bounding box and retry.
Fallback: Visual Highlight Confirmation
When DOM-based location is uncertain β the selector might be wrong, multiple candidates exist, or the target is ambiguous β use JS-injected highlighting to visually confirm before cropping.
How It Works
-
Inject a highlight border on the candidate element:
agent-browser --auto-connect eval --stdin <<'EOF'
(function() {
const el = document.querySelector('<selector>');
if (!el) { console.log('NOT_FOUND'); return; }
el.style.outline = '4px solid red';
el.style.outlineOffset = '2px';
el.scrollIntoView({ block: 'center' });
})();
EOF
-
Take a screenshot and visually inspect:
agent-browser --auto-connect screenshot /tmp/highlight-check.png
Read the screenshot to check if the red border surrounds the correct content.
-
If correct