Engineering is the art of building things within constraints. Without constraints, you aren’t really doing engineering. Constraints can involve cost, time, attention, tools, or materials. Avoiding “feature creep” is crucial — it adds unnecessary complexity. Here’s an excerpt describing the challenge facing the engineer. The engineer’s task is to identify, understand, and interpret design constraints. It is usually not enough to build a technically successful product; it must also meet further requirements.
Power vs practicality
In developing Fulcrum, we consistently work within strict boundaries. We strive to balance power and flexibility with practicality and usability. Such constraints lead to a better product—if you can’t have everything, you prioritize what’s essential. This approach helps prevent “feature creep,” ensuring the focus remains on necessary features rather than excessive additions.
Microsoft Office, exemplifying “feature creep”
The Balancing act
The practice of balancing is also relevant to our customers. Fulcrum is used by hundreds of organizations in the context of their own business rules and processes. Instead of engineering a software product, our users are engineering a solution to their problem using the Fulcrum app builder, custom workflow rules, reporting, and analysis, all customizable to fit the goals of the business. When given a box of tools to build yourself a solution to a problem, the temptation is high to try to make it do and solve everything. But with each increase in power or complexity, usability of your system takes a hit in the form of added burden on your end users to understand the complex system — they’re there to use your tool for a task, finish the job, and go home.
This balance between power and usability is related to my last post on treating causes rather than symptoms of pain. Trying too hard to make a tool solve every potential problem in one step can (and almost always does) lead to overcomplicating the result, to the detriment of everyone.
In our case as a product development and design team, a powerful suite of options without extremely tight attention on implementation runs the risk of becoming so complex that the lion’s share of users can’t even figure it out. GitHub’s Ben Balter recently wrote a great piece on the risks of optimizing your product for edge cases1:
No product is going to satisfy 100% of user needs, although it’s sure tempting to try. If a 20%-er requests a feature that isn’t going to be used by the other 80%, there’s no harm in just making it a non-default option, right?
The cost of adding more
We have a motto at GitHub, part of the GitHub Zen, that “anything added dilutes everything else.” In reality, there is always a non-zero cost to adding that extra option. Most immediately, it’s the time you spend building feature A, instead of building feature B. A bit beyond that, it’s the cognitive burden you’ve just added to each user’s onboarding experience as they try to grok how to use the thing you’ve added (and if they should). In the long run, it’s much more than maintenance. Complexity begets complexity, meaning each edge case you account for today, creates many more edge cases down the line.
This is relevant to anyone building something to solve a problem, not just software products. Put this in the context of a Fulcrum data collection workflow. The steps might look something like this:
- Analyze your requirements to figure out what data is required at what stage in the process.
- Build an app in Fulcrum around those needs.
- Deploy to field teams.
- Collect data.
- Run reports or analysis.
What we notice a surprising amount of the time is an enormous investment in step 2, sometimes to the exclusion of much effort on the other stages of the workflow. With each added field on a survey, requirement for data entry, overly-specific validation, you add potential hang-ups for end users responsible for actually collecting data. With each new requirement, usability suffers. People do this for good reason — they’re trying to accommodate those edge cases, the occasions where you do need to collect this one additional piece of info, or validate something against a specific requirement. Do this enough times, however, and your implementation is all about addressing the edge problems, not the core problem.
When you’re building a tool to solve a problem, think about how you may be impacting the core solution when you add knobs and settings for the edge cases. Best-fit solutions require testing your product against the complete ideal life cycle of usage. Start with something simple and gradually add complexity as needed, rather than the reverse.
- Ben’s blog is an excellent read if you’re into software and the relationship to government and enterprise. ↩