Logo

dev-resources.site

for different kinds of informations.

7 Ways to Refactor Your Flutter Application

Published at
1/7/2025
Categories
flutter
dart
refactoring
appdev
Author
matsch1
Categories
4 categories in total
flutter
open
dart
open
refactoring
open
appdev
open
Author
7 person written this
matsch1
open
7 Ways to Refactor Your Flutter Application

Refactoring is a vital part of maintaining and improving your Flutter application.
It ensures your codebase remains clean, consistent, and efficient as your app grows.
In this article, we’ll explore seven practical ways to refactor your Flutter application.

1. Use Parameters in Widgets for Consistency

When building widgets, hardcoding values like padding or fontSize can lead to inconsistencies. Instead, pass these values as parameters to make widgets reusable and consistent.

Example: Use parameters instead of hardcoded values

class GreetingWidget extends StatelessWidget {
  final String name;
  final double _fontSize = 20;
  final double _paddingVal = 10;

  const GreetingWidget({
    required this.name,
    super.key,
  });

  @override
  Widget build(BuildContext context) {
    final String greeting = 'Hello, $name!'; // Internal parameter

    return Padding(
      padding: EdgeInsets.all(_paddingVal),
      child: Column(children: [
        Text(
          greeting,
          style: TextStyle(fontSize: _fontSize),
        ),
        Text(
          "nice to see you!",
          style: TextStyle(fontSize: _fontSize),
        ),
      ]),
    );
  }
}
Enter fullscreen mode Exit fullscreen mode

By defining paddingVal and fontSize as internal parameters, the GreetingWidget can easily maintained.


2. Create a Global Parameter File

The next step to creating internal parameters for consistency, is creating global parameters for consistency.
All values that should remain consistent throughout your app, such as colors, scaling factors, or button sizes, store them in a global file.
This file doesn't need a special formatting. You can easily make it like this.

Example: Global Parameters File

// wrapperBox
double boxWidthFactor = 0.9;
double paddingVal = 10;
double borderWidth = 3;
double borderRadius = 10;
double boxHeaderTextSize = 16;
Enter fullscreen mode Exit fullscreen mode

To use this global parameters in different files. Just import the global parameter file at the beginnging.

Usage:

import 'package:<appName>/common/src/globals.dart';

Text(
  'Hello, World!',
  style: TextStyle(fontSize: boxHeaderTextSize),
);
Enter fullscreen mode Exit fullscreen mode

This approach ensures consistency across the entire app and simplifies updates.


3. Organize Your Dart Files Effectively

A well-structured lib directory improves code readability and maintainability. A common structure is feature-based organization:

Example File Structure

lib/
|-- features/
|   |-- home/
|   |   |-- home_screen.dart
|   |   |-- home_controller.dart
|-- shared/
|   |-- classes/
|   |-- funcs/
|   |-- widgets/
|   |-- utils/
|   |-- themes.dart
Enter fullscreen mode Exit fullscreen mode

Refer to Flutter Professional Folder Structure: Feature-first or Layer-first? for an in-depth guide to file organization.


4. Create Custom Widgets for Reusability

If you find yourself duplicating similar widgets, extract them into custom widgets. This reduces code duplication and improves maintainability.

Example: Extracting a Custom Widget

class CustomCard extends StatelessWidget {
  final String title;
  final String subtitle;
  final double _fontSize = 12;

  const CustomCard({
    required this.title,
    required this.subtitle,
    super.key,
  });

  @override
  Widget build(BuildContext context) {
    return Card(
      child: Padding(
        padding: EdgeInsets.all(paddingVal),
        child: Column(children: [
          ListTile(
            title: Text(title),
            subtitle: Text(subtitle),
          ),
          Text(
            "Hello hacker",
            style: TextStyle(fontSize: _fontSize),
          ),
          Text(
            "nice to see you!",
            style: TextStyle(fontSize: _fontSize),
          ),
        ]),
      ),
    );
  }
}
Enter fullscreen mode Exit fullscreen mode

Usage:

CustomCard(title: 'Flutter', subtitle: 'Custom Widgets');
Enter fullscreen mode Exit fullscreen mode

5. Differentiate Internal and External Parameters/Methods

When working with Flutter, distinguishing between internal and external parameters or methods is essential.
Internal parameters or methods are private to the widget and typically have an underscore (_) prefix,
while external parameters or methods are exposed to other widgets or parts of the app and lack this prefix.

When to Use an Underscore

  • Use an underscore (_) for private properties or methods that should not be accessed outside the widget.
  • Internal parameters or methods are intended for use within the widget's implementation only.

Example: Internal Parameter with Underscore

class CounterWidget extends StatefulWidget {
  @override
  _CounterWidgetState createState() => _CounterWidgetState();
}

class _CounterWidgetState extends State<CounterWidget> {
  int _count = 0; // Internal parameter

  void _increment() { // Internal method
    setState(() {
      _count++;
    });
  }

  @override
  Widget build(BuildContext context) {
    return Column(
      children: [
        Text('Count: $_count'),
        ElevatedButton(onPressed: _increment, child: Text('Increment')),
      ],
    );
  }
}
Enter fullscreen mode Exit fullscreen mode

In this example, _count and _increment are internal and should not be accessed outside _CounterWidgetState.

When to Avoid an Underscore

  • Avoid underscores for public properties or methods that are meant to be used by parent widgets or other components.
  • External parameters or methods define the API of your widget and should be accessible.

Example: External Parameters

class GreetingWidget extends StatelessWidget {
  final String name; // External parameter

  GreetingWidget({required this.name});

  @override
  Widget build(BuildContext context) {
    return Text('Hello, $name!');
  }
}
Enter fullscreen mode Exit fullscreen mode

Here, name is an external parameter passed by the parent widget, defining the widget's behavior.

Best Practices

  • Use underscores (_) for private/internal properties or methods to encapsulate implementation details.
  • Keep external parameters clean and intuitive to create a clear API for your widgets.

6. Scale Widgets Using Screen Size

To make your app responsive, scale widgets based on screen size using the MediaQuery class.

Example: Responsive Padding

class ResponsiveBox extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    double screenWidth = MediaQuery.of(context).size.width;

    return SizedBox(
      width: screenWidth*0.8;
      child: Text('Responsive Box'),
    );
  }
}
Enter fullscreen mode Exit fullscreen mode

This ensures your UI adapts gracefully to different screen sizes and prevents your app from overflow.


7. Usage of Providers: Managing State Effectively

The provider package simplifies state management by allowing widgets to listen to changes and rebuild accordingly.
This is necessary if you need the same data in different widgets (read or write).
Here are two popular provider type I already used:

ChangeNotifierProvider

Use ChangeNotifierProvider for managing mutable state.
For example a quiz game app with a QuizProvider.
If some widget modifies the quiz data with the help of the QuizProvider, every Consumer of the quizd data will be notfied.

Example: Quiz Game App

import 'package:provider/provider.dart';

const GameUI({
  super.key,
});

@override
Widget build(BuildContext context) {

  return MultiProvider(
    providers: [
      ChangeNotifierProvider(
        create: (context) => QuizProvider(),
      ),
    ],
    child: const Scaffold(
      appBar: CustomAppBar(),
      body: Body(),
    ),
  );
}
class QuizProvider with ChangeNotifier {
  late QuizGame _quizGame;
  QuizProvider({});

  void loadGame() async {
    var questions = await fetchAllQuestions();
    _quizGame =
        QuizGame();
    _quizGame.init();
    notifyListeners();
  }

  void selectAnswer(int selectedAnswers) {
    _quizGame.currentQuestion.selectedAnswer = selectedAnswers;
    notifyListeners();
  }

  void submitAnswer() async {
    _quizGame.submitAnswer();
    notifyListeners();
  }

  void nextQuestion() async {
    // await Future.delayed(const Duration(milliseconds: 200));
    _quizGame.nextQuestion();
    notifyListeners();
  }
}
Enter fullscreen mode Exit fullscreen mode

The QuizProvider data can be accessed using context.watch<QuizProvider>() or manipulated using context.read<QuizProvider>().submitAnswer()

FutureProvider

Use FutureProvider for widgets that depend on asynchronous data.

Example: Fetching User Data

import 'package:flutter/material.dart';
import 'package:provider/provider.dart';

void main() {
  runApp(
    FutureProvider<User>(
      create: (context) => fetchUser(),
      initialData: User(name: 'Loading...', email: 'Loading...'),
      child: MyApp(),
    ),
  );
}

class User {
  final String name;
  final String email;

  User({required this.name, required this.email});
}

Future<User> fetchUser() async {
  await Future.delayed(Duration(seconds: 2)); // Simulate network delay
  return User(name: 'John Doe', email: '[email protected]');
}

class MyApp extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      home: UserScreen(),
    );
  }
}

class UserScreen extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    final user = Provider.of<User>(context);

    return Scaffold(
      appBar: AppBar(title: Text('User Info')),
      body: Center(
        child: Column(
          mainAxisAlignment: MainAxisAlignment.center,
          children: [
            Text('Name: ${user.name}', style: TextStyle(fontSize: 24)),
            SizedBox(height: 10),
            Text('Email: ${user.email}', style: TextStyle(fontSize: 18)),
          ],
        ),
      ),
    );
  }
}
Enter fullscreen mode Exit fullscreen mode

Refactoring doesn’t have to be overwhelming. By applying these seven techniques, you can improve your Flutter application’s maintainability, scalability, and overall quality.

appdev Article's
30 articles in total
Favicon
"When the student is ready, the teacher will appear." ~ Lao Tzu's Tao Te Ching
Favicon
7 Ways to Refactor Your Flutter Application
Favicon
Redis: Powering Real-Time Applications with Unmatched Performance
Favicon
🎉 We're Just About Ready to Start Beta Testing! And you can join in! 🎉
Favicon
The important thing is not to stop questioning
Favicon
How to Develop an OTT App like Netflix?
Favicon
Every question...
Favicon
Is Linux really Good??
Favicon
App Development: A Comprehensive Guide to Building Successful Apps
Favicon
Exploring Three Major core components of React Native
Favicon
I created a whole macOS app with Flutter just using AI, here is my report
Favicon
AI In Education
Favicon
Remodel Your Business Operations with Custom ERP Solutions
Favicon
What I learned from my NodeJS course (part 2)
Favicon
What I learned from my NodeJS course (part 1)
Favicon
HyperTAG, an Open-Source Telegram Bot That Summarizes Links & YouTube Videos with AI, and Also Does a Lot More!
Favicon
Expert Tips for Creating a Seamless and Scalable Food Ordering Platform
Favicon
iOS App Development Process: A Step-by-Step Guide
Favicon
Light Path Learning Demo!
Favicon
How Can Enterprise Mobile App Development Be the Future of Your Business Growth?
Favicon
Ultimate Guide to Building a Successful Mobile App
Favicon
The Roadmap to Modernization: Effective Technology Migration Strategies
Favicon
Benefits of Outsource Python Development for Startups and Enterprises
Favicon
What to Look for in the Best Medical Software Development Companies
Favicon
Animation in Flutter
Favicon
#learningflutter I am working on MCQ app. When i try to run my app.
Favicon
Is Flutter frontend or backend?
Favicon
Some Flutter tips and tricks for developers
Favicon
Best IDE for Flutter framework?
Favicon
Game monetization pitfalls that prevent developers from maximizing app revenue

Featured ones: