Building Dynamic UIs: Implementing Role-Based Access Control and Feature Flags in Flutter
The Flutter news you actually need
No spam, ever. Unsubscribe in one click.
Building an app that serves multiple user types? Whether you’re creating a marketplace for buyers and sellers, a SaaS platform with admin and user tiers, or an internal tool with varying permissions, you’ll quickly face a core architectural question: how do you manage different UI versions and feature sets without creating a maintenance nightmare?
The naive approach—duplicating entire pages for each role—leads to fragile, bloated code. Change a common element? You must update it in multiple places. This is where implementing a clean system for Role-Based Access Control (RBAC) and Feature Flags becomes essential. It lets you conditionally render UI and manage logic based on user permissions, keeping your codebase DRY, testable, and scalable.
Let’s build a practical system from the ground up.
1. Defining Your Permission Model
First, we need a source of truth for user roles and permissions. An enum is a perfect, type-safe starting point.
enum UserRole {
guest,
customer,
seller,
admin,
}
// Often, a user model will hold this
class AppUser {
final String id;
final String name;
final UserRole role;
// ... other fields
const AppUser({
required this.id,
required this.name,
required this.role,
});
}
For more granular control beyond roles, you can model specific feature permissions. This is useful for feature flags, where you might enable a beta feature for only 10% of users, regardless of role.
class FeatureFlags {
final bool canUseAdvancedDashboard;
final bool canEditGlobalContent;
final bool isNewMessagingUiEnabled;
const FeatureFlags({
this.canUseAdvancedDashboard = false,
this.canEditGlobalContent = false,
this.isNewMessagingUiEnabled = false,
});
}
In a real app, this FeatureFlags object would typically be populated from your backend or a remote configuration service.
2. Making Permissions Accessible
You need to provide this data to your widget tree. Using a state management solution like Provider, Riverpod, or Bloc is ideal. Here’s a simple example using Provider:
// A simple state holder
class UserSession extends ChangeNotifier {
AppUser? _currentUser;
FeatureFlags _flags = const FeatureFlags();
AppUser? get currentUser => _currentUser;
UserRole get currentRole => _currentUser?.role ?? UserRole.guest;
FeatureFlags get featureFlags => _flags;
void login(AppUser user, FeatureFlags flags) {
_currentUser = user;
_flags = flags;
notifyListeners();
}
void logout() {
_currentUser = null;
_flags = const FeatureFlags();
notifyListeners();
}
// Helper method
bool hasRole(UserRole role) => currentRole == role;
}
void main() {
runApp(
ChangeNotifierProvider(
create: (_) => UserSession(),
child: const MyApp(),
),
);
}
3. The Core Building Blocks: Conditional Widgets
Now for the fun part: building reusable widgets that react to permissions.
A. The Simple Visibility Wrapper
This widget shows or hides its child based on a condition. It’s perfect for hiding a specific button or section.
class ConditionalVisibility extends StatelessWidget {
final bool isVisible;
final Widget child;
final bool maintainState; // Keep widget alive when invisible
const ConditionalVisibility({
super.key,
required this.isVisible,
required this.child,
this.maintainState = false,
});
@override
Widget build(BuildContext context) {
return Visibility(
visible: isVisible,
maintainState: maintainState,
maintainAnimation: maintainState,
maintainInteractivity: maintainState,
child: child,
);
}
}
// Usage in a ProductPage
// ...
Consumer<UserSession>(
builder: (context, session, _) {
return ConditionalVisibility(
isVisible: session.hasRole(UserRole.seller) ||
session.hasRole(UserRole.admin),
child: FilledButton(
onPressed: () => _editProduct(),
child: const Text('Edit Listing'),
),
);
},
),
// ...
B. The RoleGate Widget
This widget renders one of two widgets based on the user’s role. It’s more explicit for swapping whole sections.
class RoleGate extends StatelessWidget {
final UserRole requiredRole;
final Widget grantedChild;
final Widget? deniedChild;
const RoleGate({
super.key,
required this.requiredRole,
required this.grantedChild,
this.deniedChild,
});
@override
Widget build(BuildContext context) {
final session = context.watch<UserSession>();
final hasAccess = session.hasRole(requiredRole);
if (hasAccess) {
return grantedChild;
}
return deniedChild ?? const SizedBox.shrink(); // Or a placeholder
}
}
// Usage: Show a seller's dashboard tab only to sellers
RoleGate(
requiredRole: UserRole.seller,
grantedChild: const SellerDashboardTab(),
deniedChild: const UpgradePromptCard(), // Nice UX for denied users
),
C. Feature Flag Widget
Similar to RoleGate, but for feature toggles.
class FeatureFlag extends StatelessWidget {
final bool flag;
final Widget enabledChild;
final Widget? disabledChild;
const FeatureFlag({
super.key,
required this.flag,
required this.enabledChild,
this.disabledChild,
});
@override
Widget build(BuildContext context) {
return flag ? enabledChild : (disabledChild ?? const SizedBox.shrink());
}
}
// Usage
Consumer<UserSession>(
builder: (context, session, _) {
return FeatureFlag(
flag: session.featureFlags.isNewMessagingUiEnabled,
enabledChild: const NewMessagingInterface(),
disabledChild: const LegacyMessagingInterface(),
);
},
);
4. Structuring Your Pages
With these tools, your page logic stays clean and centralized.
class ProductDetailPage extends StatelessWidget {
final Product product;
const ProductDetailPage({super.key, required this.product});
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(title: Text(product.name)),
body: Padding(
padding: const EdgeInsets.all(16.0),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
// Common UI for ALL users
ProductImage(url: product.imageUrl),
const SizedBox(height: 16),
Text(product.description, style: Theme.of(context).textTheme.bodyLarge),
const SizedBox(height: 24),
ProductPriceWidget(price: product.price),
const Spacer(),
// Dynamic Action Bar based on role
Consumer<UserSession>(
builder: (context, session, _) {
if (session.hasRole(UserRole.customer)) {
return BuyNowButton(product: product);
} else if (session.hasRole(UserRole.seller)) {
return SellerActionBar(product: product);
} else {
return SignInPrompt();
}
},
),
],
),
),
);
}
}
Common Pitfalls & Best Practices
- Don’t Check Permissions Too Deeply: Avoid scattering raw
if (user.role == Role.admin)checks deep in your business logic. Centralize checks in helper widgets or services. - Secure Your Backend: UI-based controls are for UX only. Always enforce permissions on your backend API. A user can modify a local app state or reverse-engineer your build.
- Plan for State: When you hide a widget with
Visibility(maintainState: false), it’s disposed. If it’s a form with user input, usemaintainState: trueor structure your state to persist separately. - Test Thoroughly: Write widget tests that mock different user sessions to ensure your UI correctly renders for each role and feature flag combination.
By adopting this layered approach—a clear permission model, accessible state, and reusable conditional widgets—you build a Flutter app that can gracefully adapt to any user, today or in the future. Your code remains clean, your features are easy to toggle, and adding a new user role becomes a straightforward task instead of a refactoring odyssey.
This blog is produced with the assistance of AI by a human editor. Learn more
Related Posts
Optimizing Flutter UI Performance: Best Practices for Date Formatting and Expensive Operations
Developers often face performance bottlenecks when performing expensive operations like date formatting directly within Flutter's `build` method, especially in fast-scrolling lists. This post will delve into common pitfalls, explain why these operations are costly, and provide practical strategies for optimizing UI performance by caching formatters, using `initState`, and leveraging `compute` for background processing without blocking the UI.
Optimizing Your Flutter Dev Setup: IDEs, Simulators, and AI Tools for Peak Productivity
Flutter developers frequently seek to refine their development environments. This post will dive into popular IDE choices like VS Code and Android Studio, discuss best practices for managing iOS and Android simulators (including in-IDE options), and explore the practical integration of AI tools for code generation and problem-solving to boost overall efficiency.
Demystifying Flutter Performance: Practical Strategies for Large-Scale Apps
Flutter's performance is often blamed for issues in complex applications, but the real culprits are usually architectural decisions, inefficient widget rebuilds, and unoptimized resource handling. This post will dive into common performance bottlenecks in large Flutter apps, providing actionable strategies for profiling, optimizing state management, handling images and network requests efficiently, and leveraging CI/CD for continuous performance monitoring.