惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

H
Help Net Security
J
Java Code Geeks
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
H
Hackread – Cybersecurity News, Data Breaches, AI and More
V
Visual Studio Blog
G
Google Developers Blog
V
V2EX
The Register - Security
The Register - Security
博客园 - 三生石上(FineUI控件)
云风的 BLOG
云风的 BLOG
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
CTFtime.org: upcoming CTF events
CTFtime.org: upcoming CTF events
博客园_首页
S
SegmentFault 最新的问题
博客园 - Franky
Martin Fowler
Martin Fowler
Stack Overflow Blog
Stack Overflow Blog
A
About on SuperTechFans
人人都是产品经理
人人都是产品经理
aimingoo的专栏
aimingoo的专栏
罗磊的独立博客
C
Check Point Blog
MyScale Blog
MyScale Blog
T
The Blog of Author Tim Ferriss
MongoDB | Blog
MongoDB | Blog
The GitHub Blog
The GitHub Blog
Last Week in AI
Last Week in AI
Microsoft Azure Blog
Microsoft Azure Blog
IT之家
IT之家
F
Fortinet All Blogs
Jina AI
Jina AI
P
Proofpoint News Feed
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
阮一峰的网络日志
阮一峰的网络日志
B
Blog
L
LangChain Blog
月光博客
月光博客
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
宝玉的分享
宝玉的分享
博客园 - 【当耐特】
T
Tailwind CSS Blog
酷 壳 – CoolShell
酷 壳 – CoolShell
Microsoft Security Blog
Microsoft Security Blog
WordPress大学
WordPress大学
奇客Solidot–传递最新科技情报
奇客Solidot–传递最新科技情报
B
Blog RSS Feed
博客园 - 聂微东
Hugging Face - Blog
Hugging Face - Blog
M
MIT News - Artificial intelligence
GbyAI
GbyAI

DEV Community

Authentication Security Deep Dive: From Brute Force to Salted Hashing (With Java Examples) Why AI Systems Don’t Fail — They Drift Spilling beans for how i learn for exam😁"Reinforcement Learning Cheat Sheet" I Replaced Chrome with Safari for AI Browser Automation. Here's What Broke (and What Finally Worked) How Python Borrows Other People's Work The $40 Architecture: Processing 1 Billion API Requests with 99.99% Uptime Vibe Coding: A Workflow Guide (From Zero to SaaS) Most webhook security guides protect the wrong side. The scary part is delivery. Headless CMS for TanStack Start: Build a Blog with Cosmic EU Age Verification App "Hacked in 2 Minutes" — What Actually Happened Comfy Cloud’s delete function does not actually remove files Running AI Models on GPU Cloud Servers: A Beginner Guide Event-driven media intelligence with AWS Step Functions and Bedrock I scored 500 AI prompts across 8 quality dimensions — here's what broke How to Call Google Gemini API from Next.js (Free Tier, No Backend Needed) The Portal Protocol: Reclaiming Human Connection in the Age of AI How to Fix Your Team's Scattered Knowledge Problem With a Self-Hosted Forum Intro to tc Cloud Functors: A Graph-First Mental Model for the Modern Cloud Designing Multi-Tenant Backends With Both Ownership and Team Access I Built a Neumorphic CSS Library with 77+ Components — Here's What I Learned PostgreSQL Performance Optimization: Why Connection Pooling Is Critical at Scale Cómo construí un SaaS multi-rubro para gestionar expensas en Argentina con FastAPI + Vue 3 🚀 I Built an Ethical Hacking Scanner Tool – Open Source Project I Replaced /usage and /context in Claude Code With a Single Statusline A Pythonic Way to Handle Emails (IMAP/SMTP) with Auto-Discovery and AI-Ready Design I Collected 8.9 Million Polymarket Price Points — Here's What I Found About How Markets Really Move EcoTrack AI — Carbon Footprint Tracker & Dashboard Everyone's Using AI. No One Agrees How. 5 self-hosted ebook managers worth trying in 2026 Building Your First AI Agent with LangChain: From Chatbot to Autonomous Assistant Common SOC 2 Failures (Real World) Stop Vibe-Checking Your AI App: A Practical Guide to Evals How to Use SonarQube and SonarScanner Locally to Level Up Your Code Quality Your Next To-Do App Is Dead — I Replaced Mine with an OpenClaw AI Sign a Nostr event in 60 lines of Python using coincurve — no nostr-sdk, no nbxplorer, no rust toolchain ITGC Audit Explained Like You’re in Big 4 Patch Tuesday abril 2026: Microsoft parcha 163 vulnerabilidades y un zero-day en SharePoint Stop scraping everything: a better way to track competitor price changes Listing on MCPize + the Official MCP Registry while routing payments OUTSIDE the marketplace — how I kept 100% of my x402 revenue Building an AI-Powered Risk Intelligence System Using Serverless Architecture Why We Ripped Function Overloading Out of Our AI Toolchain Testing AI-Generated Code: How to Actually Know If It Works SaaS Churn Is Killing Your Business. Here Is What to Do About It (Without a Support Team) The Speed of AI Is No Longer Linear - And Self-Improving Models Are Why How to Implement RBAC for MCP Tools: A Practical Guide for Engineering Teams From Standard Quote to Persuasive Proposal: AI Automation for Arborists I built a CLI that scaffolds complete multi-tenant SaaS apps Axios CVE-2025–62718: The Silent SSRF Bug That Could Be Hiding in Your Node.js App Right Now The dashboard that ended our friendship Data Pipelines Explained Simply (and How to Build Them with Python) The Hidden Cost of AI Systems Nobody Talks About. undefined vs undeclared, and how typeof behaves Switching from file-based jobs to NATS/Kafka in Rust without changing code io_uring Adventures: Rust Servers That Love Syscalls Why Agentic AI is Killing the Traditional Database The POUR principles of web accessibility for developers and designers Quantum Neural Network 3D — A Deep Dive into Interactive WebGL Visualization How To Install Caveman In Codex On macOS And Windows Automation Pipeline Reliability: Why Your Workflow Breaks When Nobody Is Watching I Built an 'Open World' AI Coding Agent — It Works From ANY Folder From Freelancing to Product: A Tech Service Company's SaaS Transformation China's AI Giants: Adding Tencent Hunyuan & ByteDance Doubao to AI University (74 Providers) On the Vibe Coders and Their Lies clerk: Auto-Summarize Your Claude Code Sessions AI Weekly — 2026/04/10–04/17 | The Model Lockdown Is Here, but the Toolchain Is the Real Battleground AI 週報 — 2026/04/10–2026/04/17 模型封鎖潮來了,但工具鏈才是真戰場 Maybe this is how Open-Source apps are born... 🚀 Fine-Tune LLMs with LoRA and QLoRA: 2026 Guide tRPC v11 + Next.js App Router: End-to-End Type Safety Without the Boilerplate ShadCN UI in 2026: Why I Stopped Installing Component Libraries and Started Owning My Components SaaS Billing in React Server Components: Stripe + Supabase Without a Single `useEffect` Join our DEV Weekend Challenge — $1,000 in Prizes Across TEN winners! Submissions Due April 20 at 6:59 AM UTC. Implementing FSRS Spaced Repetition in Flutter + Supabase — Adding Memory Science to an AI Learning App "I Texted My Localhost From the Train — Claude Code Fixed the Bug Before I Got Home" I Built a Sales Prep AI and It Went Deeper Than Expected Design to Code #2: One JSON, Eleven Outputs Solving the 100M-Row Problem: A Summary Table Pattern for High-Volume Push Notification Logs Flutter Web With Wasm: What Actually Changes For Developers I Built 50 Royalty-Free Soundtracks for My Side Project in a Weekend Using AI Music Generation The Vibe Coding Security Checklist: 7 Things to Check Before You Ship Stop Letting Googlebot Guess Fix Your React App's SEO Right Desconstruindo o Streaming do LinkedIn: Como Criar um Engine de Extração de Vídeo de Alta Performance com HLS e FFmpeg (EDA Part-1) EDA (Exploratory Data Analysis) Explained With Real Life — Why Looking at Your Data Is the Most Important Step in Machine Learning Brand Relationship Management at Scale: Our 4-Touch Outreach System for 200+ Brands Why String.fromEnvironment() Might Return an Empty String in Dart JGuardrails 1.0.0 — Hardening Java LLM Apps Against Jailbreaks, Toxicity, and Prompt Injection Plan and Schedule a Full Week of Threads Content From One Claude Conversation Coding Cat Oran Ep3, Five Tables Changed Everything Updated: BFF Pattern I'm done watching freelancers get buried by 200 proposals. So I'm building the alternative. This is my first post BFS Algorithm in Java Step by Step Tutorial with Examples Tracking LLM Pricing Monthly: An Open Dataset for 22 AI Models How We Measure Content ROI on a Comparison Site: Revenue Attribution Without Perfect Data Introducing Nova AI Ops: The AI-Native Operating System for SRE Teams I built a free desktop video downloader for Windows — Grabbit How Talkie OCR Helps Vision-Impaired & Dyslexic Users Read the World Around Them VRCFaceTracking安装和iPhone面捕配置教程,有bug Even CrowdStrike Can't See Your Agents The Automation Gold Rush: What n8n Workflows and Claude Are Opening Up for Developers Right Now
I Just Wanted to Know Where My Browsing Time Went. Five Years Later, Firefox Recommended It.
victor zhang · 2026-05-15 · via DEV Community

I didn't start this project because I wanted to build a productivity app.

It began with a smaller, almost embarrassing question.

Where did my time go?

Not in the philosophical sense. I just wanted to know why a browser tab opened “for a minute” could quietly take an hour with it. At the end of the day, the feeling was always blurry: I knew some time had disappeared, but I could not point to where, when, or how much.

So I built a browser extension for myself.

It recorded how long I stayed on each website, counted visits, and showed a simple chart. That was the whole product: a small mirror for my browser.

Dashboard showing browsing time patterns

Five years after the first commit, that small tool has become Time Tracker - Web Habit Builder: an open-source browser extension for Chrome, Edge, Firefox, and Firefox for Android. Across browser stores, it has maintained a 4.9+ rating, and on Firefox it has earned Mozilla's rare Recommended badge.

I don't really think of this as a growth story.

It's more about what happens when a private little mirror keeps running into real users, real data, real browsers, and real habits.

1. The first version answered the wrong question

The first version answered a simple question:

How much time did I spend on each website?

It was useful, in a narrow way.

The project started in 2021. After I shared it on a small tech forum in 2022, I noticed something awkward. People installed it, opened the numbers for a few days, and then some of them drifted away.

The extension was working. The data was correct.

But correctness was not enough.

A number can be honest and still feel empty. “You spent 3 hours on YouTube today” doesn't automatically make anyone more aware. Sometimes it only adds guilt. Sometimes it fades into background noise. A number alone doesn't explain a pattern.

That was the first time I realized:

Tracking is not the value. Recognition is the value.

So the product slowly moved away from raw records and toward something users could actually recognize.

Compared with the version I wrote about in 2022, the biggest change is not more charts. The product now has two surfaces with different jobs.

The popup became a small daily cockpit: quick enough to answer what is happening right now. The dashboard became slower and more reflective: top sites, long-term trends, calendar heatmaps, and recent activity timelines.

The point is not to make the data look impressive. It is to make a messy browsing day recognizable.

Your day is not just “too much browsing.” It has a shape.

You open it and notice: this month was heavier than last month. The busiest time is not when you thought it was. One site is harmless on weekdays, then quietly takes over the weekend.

2. Users did not ask for more charts. They asked for help.

Once people could see their behavior, the next request became obvious.

Can it stop me?

That was when the product stopped being easy.

A time tracker is mostly passive. It observes. It records. It shows things.

A site blocker is different. It interrupts the user.

And the strange thing about this type of blocker is that the user is on both sides of the system. The calm version of the user creates a rule. The tired version of the user tries to bypass it.

At first, the limit feature was simple: set a daily limit for a website, and show a blocking page when the time is used up.

Then the edge cases started arriving.

Some users wanted weekly limits. Some wanted per-session limits. Some wanted weekday rules, URL-level blocking, strict mode, delayed unlocks, or a “five more minutes” option that was helpful but not too easy.

The rule system grew from a small condition into a real engine.

Limit rules

The change that helped most was much smaller: a warning.

In early versions, when time ran out, the block page appeared suddenly. It was technically correct, but emotionally rough. Users were interrupted at the worst possible moment.

Blocked page

Now the extension can notify the user before the limit is reached.

A few minutes of warning changes the relationship. The blocker no longer feels like a trap. It feels more like a reminder from the earlier version of yourself — the version that made the rule before the impulse arrived.

That small detail taught me more than some much larger rewrites.

For behavior-related products, friction matters. But the timing of friction matters even more.

3. A good default is quieter than onboarding

Over the next few years, the extension accumulated many features: categories, virtual sites, timelines, side panels, dark mode, import, sync, mobile support, and localization.

At that point, an onboarding flow would have been easy to justify.

I still didn't add one.

My rule is simple: if a feature must be explained before it can be used, the interaction probably needs more work.

I ended up designing the extension in layers. After installation, tracking starts automatically. Click the icon, and you get today's overview. Open the dashboard, and you see longer-term patterns. Go into options, and you can customize a lot.

Power users can configure a lot, but configuration doesn't stand at the entrance. That is one of the most important product decisions in the project.

Category timeline

4. Privacy was not a feature checkbox. It shaped the architecture.

The extension sits close to sensitive data.

Browsing history is personal. Even if the product doesn't care about the content of pages, the pattern itself says a lot about a person.

So from the beginning, I avoided accounts, servers, and analytics collection. The data stays local by default.

Later, users asked for cross-device backup and remote query. Instead of building my own cloud service, I added user-controlled sync options:

  • GitHub Gist
  • Obsidian local REST API
  • WebDAV

That means users bring their own storage. The extension provides the integration, but doesn't become the data owner.

This decision made some things harder. There is no central backend to debug. Different WebDAV servers behave differently. Obsidian users have different local setups. Gist has its own API limits and edge cases.

But it fits the product.

A tool about personal habits shouldn't quietly become another place where personal history is collected.

5. Long-term users and real browsers changed the engineering priorities

The product looked simple when the data was small and the browser target was mostly Chrome.

Then the history started doing what history does: it piled up.

After two or three years of daily tracking, browsing history was no longer a small settings object. It had become time-series data: large, structured, and constantly queried. Reports became slower. Popup loading became heavier. The product was starting to punish the users who had trusted it the longest.

So in v4.0, I moved tracking data to IndexedDB. The goal was not to show off a new architecture. The goal was quieter: heavy users should update, keep their history, and simply feel that the product had become lighter.

Firefox brought a similar reminder from another direction. Browser extensions only look standardized from a distance. Background behavior, sidebar APIs, Android support, and Manifest V3 details all differ enough to matter.

Supporting Firefox well meant making the cross-browser design more explicit: separate platform behavior where needed, fewer assumptions from one browser, and more testing around the places where users actually feel the difference.

The Mozilla Recommended badge came much later. I didn't build the product for that badge. But looking back, these boring engineering decisions probably mattered: safer data migration, faster reports, fewer platform assumptions, and a product experience that feels native instead of merely compatible.

6. Community changed the product more than metrics did

The product has no growth team, no ads, and no onboarding funnel.

Most improvements came from users saying very specific things.

A user wants to merge several domains because one service uses many subdomains. Another user wants to import data from another extension. Someone wants to track local files. Someone wants WebDAV. Someone wants Firefox Android. Someone wants the interface to feel native in their own language.

Those requests were not abstract “user needs.” They were real situations.

Translation is the clearest example. The project now has a Crowdin-based workflow, with English as the required source locale and partial translations allowed to fall back safely.

Crowdin

Today it has locale targets for 14 languages.

That detail matters because internationalization is not just uploading strings. The extension still has to map browser language codes, validate placeholders, generate extension metadata, and handle layout direction for RTL languages.

This changed how I think about feedback.

A public review is nice. But a detailed issue, a screenshot, or a carefully described routine is often more valuable. It tells you where the product is meeting real life.

7. Today, it is no longer just a tracker

This is the part that changed the most since the early days.

The tool started as a way to answer one narrow question: where did my browsing time go?

Now, that answer is only the beginning. The project has become less about measuring time itself, and more about helping users notice the patterns around it.

It's a small shift in wording, but it changed how I think about the product.

A tracker records what happened. A habit system helps users recognize what keeps happening, and gives them a few ways to respond without handing their browsing history to someone else's backend.

That is the current shape of the project: still simple enough to install and understand in a minute, but mature enough for people who want to work with their browsing behavior over time.

8. What I would do differently

If I were to start again, I would spend less time thinking about the feature list, and more time thinking about how quickly a new user could get one useful answer.

A user should be able to install it, open the popup, and understand something about today without reading a guide.

I would still add advanced options, but I would place them behind natural paths: reports after people have data, rules after people notice a pattern, sync after people care about history, import after people are ready to switch.

There are technical things I would do earlier, of course: IndexedDB, integration tests, and more explicit cross-browser design.

But the simpler lesson is the one I keep coming back to: don't make users pay attention before the product has earned it. Let the default experience carry the first mile, and treat real user routines as product input, not just support work.

Links


If you are building a small product that started as a personal tool, I would love to hear what changed when real users walked in.