<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:dc="http://purl.org/dc/elements/1.1/">
  <channel>
    <title>Forem: Trần Nguyên Châu</title>
    <description>The latest articles on Forem by Trần Nguyên Châu (@chautnus).</description>
    <link>https://forem.com/chautnus</link>
    <image>
      <url>https://media2.dev.to/dynamic/image/width=90,height=90,fit=cover,gravity=auto,format=auto/https:%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F3931686%2F7a6266e6-8f17-4d56-9c5e-0f22edb90b48.jpg</url>
      <title>Forem: Trần Nguyên Châu</title>
      <link>https://forem.com/chautnus</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://forem.com/feed/chautnus"/>
    <language>en</language>
    <item>
      <title>I built a Chrome Extension that saves product images + context directly to Google Drive &amp; Sheets</title>
      <dc:creator>Trần Nguyên Châu</dc:creator>
      <pubDate>Fri, 22 May 2026 23:37:24 +0000</pubDate>
      <link>https://forem.com/chautnus/i-built-a-chrome-extension-that-saves-product-images-context-directly-to-google-drive-sheets-317m</link>
      <guid>https://forem.com/chautnus/i-built-a-chrome-extension-that-saves-product-images-context-directly-to-google-drive-sheets-317m</guid>
      <description>&lt;h2&gt;
  
  
  I built a Chrome Extension that saves product images + context directly to Google Drive &amp;amp; Sheets
&lt;/h2&gt;

&lt;p&gt;Most product research workflows are secretly broken. You just don't notice until you're drowning in 400 unnamed screenshots and a half-filled spreadsheet at 11pm.&lt;/p&gt;

&lt;p&gt;Four problems. One tool.&lt;/p&gt;

&lt;blockquote&gt;
&lt;p&gt;❌ Manual saving steals hours you don't have&lt;br&gt;
❌ Scrapers are unstable and dump garbage images with zero context&lt;br&gt;
❌ Images without context become completely unsearchable over time&lt;br&gt;
❌ Your workflow breaks the moment you switch devices&lt;/p&gt;
&lt;/blockquote&gt;

&lt;p&gt;Here's how &lt;strong&gt;ImageSnap&lt;/strong&gt; solves each one.&lt;/p&gt;




&lt;h2&gt;
  
  
  ❌ Pain #1: Manual saving is a productivity black hole
&lt;/h2&gt;

&lt;p&gt;The typical workflow:&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;Find a product image you want&lt;/li&gt;
&lt;li&gt;Right-click → Save As&lt;/li&gt;
&lt;li&gt;Rename the file so it makes sense later&lt;/li&gt;
&lt;li&gt;Navigate to the right folder&lt;/li&gt;
&lt;li&gt;Open your spreadsheet&lt;/li&gt;
&lt;li&gt;Manually type in the product name, price, URL, supplier, notes...&lt;/li&gt;
&lt;li&gt;Repeat. 50 times. Every session.&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;You're not doing research. You're doing data entry.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;✅ How ImageSnap fixes it:&lt;/strong&gt; One click opens a side panel. The product name, price, and URL are already filled in — scraped automatically from the page. Hit &lt;strong&gt;Snap&lt;/strong&gt; and the image goes to Google Drive, the row goes to Google Sheets. The whole thing takes 5 seconds.&lt;/p&gt;




&lt;h2&gt;
  
  
  ❌ Pain #2: Scrapers are unstable and give you garbage
&lt;/h2&gt;

&lt;p&gt;Automated scrapers sound great in theory. In practice:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;They break every time the site updates its HTML&lt;/li&gt;
&lt;li&gt;They pull hundreds of irrelevant images — icons, banners, ads, lazy-loaded placeholders&lt;/li&gt;
&lt;li&gt;You get a dump of raw data with no context about &lt;em&gt;why&lt;/em&gt; you saved anything&lt;/li&gt;
&lt;li&gt;Running them requires technical setup most researchers don't want to deal with&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;&lt;strong&gt;✅ How ImageSnap fixes it:&lt;/strong&gt; You stay in control. You browse naturally, and when you find something worth saving, you capture it intentionally. No garbage images. No broken selectors. No maintenance. You decide what gets saved and why.&lt;/p&gt;




&lt;h2&gt;
  
  
  ❌ Pain #3: Images without context become unsearchable
&lt;/h2&gt;

&lt;p&gt;This is the silent killer. You save 300 product images over two weeks. Then you need to find &lt;em&gt;"that ceramic supplier from last Tuesday with the MOQ under 500."&lt;/em&gt;&lt;/p&gt;

&lt;p&gt;Good luck.&lt;/p&gt;

&lt;p&gt;A folder of &lt;code&gt;IMG_20260521_143022.jpg&lt;/code&gt; files tells you nothing. Even if you renamed them, you've lost the price, the URL, the supplier name, the category — everything that made that image worth saving in the first place.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;✅ How ImageSnap fixes it:&lt;/strong&gt; Every image is captured with structured metadata — price, URL, title, custom fields you define per category. It all lives in Google Sheets, which means you can filter, sort, and search across everything instantly. The image and its context are permanently linked by ID. You'll find that ceramic supplier in 10 seconds.&lt;/p&gt;




&lt;h2&gt;
  
  
  ❌ Pain #4: Your workflow breaks the moment you switch devices
&lt;/h2&gt;

&lt;p&gt;You save images on your laptop. You spot products on your phone during lunch. You reference your research on a tablet in a meeting. Three devices, three separate systems, constant friction.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;✅ How ImageSnap fixes it:&lt;/strong&gt; Everything syncs to your Google Drive and Google Sheets in real time. Open your sheet on any device — phone, tablet, laptop — and your entire research library is there, with images and context intact. The extension works on desktop. The web dashboard works everywhere. One workflow, every device, any time.&lt;/p&gt;




&lt;h2&gt;
  
  
  The technical side (for the devs)
&lt;/h2&gt;

&lt;p&gt;The project is a monorepo with two independent build targets:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Web dashboard&lt;/strong&gt; → Next.js 15 + React 19 + Tailwind CSS 4&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Chrome extension&lt;/strong&gt; → Vite + TypeScript, Manifest V3&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Both share a &lt;code&gt;src/shared/&lt;/code&gt; layer — same auth logic, type definitions, and Google API helpers across both contexts.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;src/
├── shared/        ← shared between web + extension
│   ├── lib/       ← google-auth, sheets, drive helpers
│   └── hooks/     ← useAppData, etc.
├── web/           ← React app (popup + web dashboard)
└── extension/     ← manifest, content_script
app/               ← Next.js routes + API
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;One thing that caught me off guard: &lt;strong&gt;extension popups are completely stateless&lt;/strong&gt;. Every time the popup closes, the JS context is destroyed — including any in-memory auth tokens.&lt;/p&gt;

&lt;p&gt;My first instinct was to use an httpOnly session cookie via the API. It silently broke because cookies with &lt;code&gt;sameSite: lax&lt;/code&gt; don't get sent in cross-origin fetch calls from the &lt;code&gt;chrome-extension://&lt;/code&gt; protocol. The fix was &lt;code&gt;chrome.storage.local&lt;/code&gt;:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="c1"&gt;// After successful login&lt;/span&gt;
&lt;span class="nf"&gt;saveTokenToExtStorage&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;token&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;profile&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;email&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;

&lt;span class="c1"&gt;// On every popup reopen — no server roundtrip needed&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;stored&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nf"&gt;loadTokenFromExtStorage&lt;/span&gt;&lt;span class="p"&gt;();&lt;/span&gt;
&lt;span class="k"&gt;if &lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;stored&lt;/span&gt;&lt;span class="p"&gt;?.&lt;/span&gt;&lt;span class="nx"&gt;token&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;profile&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nf"&gt;getUserInfo&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;stored&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;token&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
  &lt;span class="k"&gt;if &lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;profile&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="cm"&gt;/* restore state instantly */&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;On the data side, ImageSnap stores &lt;strong&gt;nothing&lt;/strong&gt; on its own servers except subscription status. Your images live in your Google Drive. Your metadata lives in your Google Sheets. You own everything.&lt;/p&gt;




&lt;h2&gt;
  
  
  Try it
&lt;/h2&gt;

&lt;p&gt;ImageSnap is live on the Chrome Web Store with a free tier (30 captures/month).&lt;/p&gt;

&lt;p&gt;👉 &lt;a href="https://www.imagesnap.cloud" rel="noopener noreferrer"&gt;imagesnap.cloud&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;If you're doing ecommerce research, sourcing, competitor tracking, or content creation — give it a try and let me know what you think in the comments.&lt;/p&gt;

</description>
      <category>webdev</category>
      <category>productivity</category>
      <category>opensource</category>
      <category>showdev</category>
    </item>
    <item>
      <title>How AI made my workflow more mouse-driven than keyboard-driven</title>
      <dc:creator>Trần Nguyên Châu</dc:creator>
      <pubDate>Thu, 14 May 2026 16:57:00 +0000</pubDate>
      <link>https://forem.com/chautnus/how-ai-made-my-workflow-more-mouse-driven-than-keyboard-driven-2cb6</link>
      <guid>https://forem.com/chautnus/how-ai-made-my-workflow-more-mouse-driven-than-keyboard-driven-2cb6</guid>
      <description>&lt;p&gt;A small shift happened in my workflow over the last year, and I didn’t fully notice it until recently:&lt;/p&gt;

&lt;p&gt;I don’t type nearly as much anymore.&lt;/p&gt;

&lt;p&gt;A lot of my day used to be spent writing — code, messages, docs, search queries, notes.&lt;br&gt;
Now a much bigger portion of it is spent reading, reviewing, comparing, monitoring, and switching between contexts.&lt;/p&gt;

&lt;p&gt;AI is a big reason why.&lt;/p&gt;

&lt;p&gt;Instead of generating every draft from scratch, I’m now often:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;reviewing AI output&lt;/li&gt;
&lt;li&gt;comparing versions&lt;/li&gt;
&lt;li&gt;checking whether something is correct&lt;/li&gt;
&lt;li&gt;jumping between browser tabs, apps, terminals, and docs&lt;/li&gt;
&lt;li&gt;monitoring results instead of manually producing every line myself
That sounds like a software change.
But for me, it also became a physical interaction change.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;My hand is on the mouse most of the day now.&lt;/p&gt;

&lt;p&gt;And that exposed something I hadn’t really questioned before: most of my productivity actions are still designed around the keyboard.&lt;/p&gt;

&lt;p&gt;Things like:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;taking a screenshot&lt;/li&gt;
&lt;li&gt;switching apps&lt;/li&gt;
&lt;li&gt;snapping windows&lt;/li&gt;
&lt;li&gt;opening a bookmarked URL&lt;/li&gt;
&lt;li&gt;triggering a repeated shortcut&lt;/li&gt;
&lt;li&gt;moving between “work surfaces” quickly&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;None of those are difficult.&lt;br&gt;
But when your workflow becomes more mouse-heavy and less typing-heavy, reaching back to the keyboard starts to feel less like a shortcut and more like a tiny interruption.&lt;/p&gt;

&lt;p&gt;That’s the part that surprised me.&lt;/p&gt;

&lt;p&gt;Keyboard shortcuts used to feel seamless because I was already typing all the time.&lt;br&gt;
Now I’m often reading something on screen, following an AI-generated draft, comparing outputs, or scanning multiple windows. In that mode, my mouse is already “active,” while the keyboard feels secondary.&lt;/p&gt;

&lt;p&gt;So the old shortcut model still works.&lt;br&gt;
It just no longer fits the flow as naturally.&lt;/p&gt;

&lt;p&gt;That mismatch is what pushed me to build a small Windows tool for myself: RightWheel.&lt;/p&gt;

&lt;p&gt;The idea is simple:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;hold right-click, scroll, and trigger an action.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;I started using it for the handful of things I kept doing over and over:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;switching tasks&lt;/li&gt;
&lt;li&gt;launching common shortcuts&lt;/li&gt;
&lt;li&gt;opening frequently used links&lt;/li&gt;
&lt;li&gt;taking screenshots&lt;/li&gt;
&lt;li&gt;reducing the constant mouse-keyboard-mouse transition&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;What interested me most wasn’t just the tool itself.&lt;br&gt;
It was the realization behind it:&lt;/p&gt;

&lt;p&gt;AI may be changing not only what we do on computers, but how we physically interact with them.&lt;/p&gt;

&lt;p&gt;If more of our work becomes:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;reviewing instead of drafting&lt;/li&gt;
&lt;li&gt;orchestrating instead of manually producing&lt;/li&gt;
&lt;li&gt;navigating instead of typing&lt;/li&gt;
&lt;li&gt;supervising systems instead of directly operating every step&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;...then it makes sense that our input habits might shift too.&lt;/p&gt;

&lt;p&gt;The assumption that the keyboard is always the primary productivity surface may come from an earlier shape of digital work.&lt;/p&gt;

&lt;p&gt;I’m curious whether other people are feeling this too.&lt;/p&gt;

&lt;p&gt;Have AI tools changed your mouse-to-keyboard ratio?&lt;br&gt;
Have you adapted your setup in any way — better mouse software, macros, Stream Deck, window managers, voice input, custom gestures?&lt;/p&gt;

&lt;p&gt;I’d love to know what changed in your own day-to-day workflow.&lt;/p&gt;

</description>
      <category>ai</category>
      <category>productivity</category>
      <category>shortcut</category>
      <category>automation</category>
    </item>
  </channel>
</rss>
