view_model_kit 0.2.2 copy "view_model_kit: ^0.2.2" to clipboard
view_model_kit: ^0.2.2 copied to clipboard

A simple and easy ViewModel and state management Library. support ViewModel's state hold, Global state management. (Container's state hold)

view_model_kit #

pub package

A simple and easy ViewModel and state management Library.

How to use #

see examples.

1. Setting StatefulWidget and Create ViewModel #

main_page.dart

class MainPage extends StatefulWidget {
  const MainPage({super.key});

  @override
  State<MainPage> createState() => _MainPageState();
}

class _MainPageState extends StateWithViewModel<MainPage, MainViewModel> {
  @override
  MainViewModel createViewModel() => MainViewModel();

  @override
  Widget build(BuildContext context) {
    return Scaffold();
  }
}

main_view_model.dart

class MainViewModel extends BaseViewModel {
  @override
  void onReady() {
    // onReady called only once when the ViewModel is ready. (first build)
  }
}

2. Use R<V> or RList<V> with ViewModel #

class MainViewModel extends BaseViewModel {
  late final _count = createMutable(0); // can modify. (MutableR<V>)
  R<int> get count => _count; // can not modify. (R<V>)
  late final _todos = createMutableList<Todo>(); // or createMutableList<Todo>([]);
  RList<Todo> get todos => _todos; // can not modify. (RList<V>)

  @override
  void onReady() async {
    _count.value = 1; // auto rebuild.
    _count.value++; // auto rebuild.
    // count.value = 1; // error. because type of count is R<V>. use instead of MutableR<V>.value = 1;

    final todos = await getTodosFromRemote();

    _todos.addAll([...todos]); // auto rebuild.
    // todo.addAll([...todos]); // error. because type of count is RList<V>. use instead of MutableRList<V>.addAll;

    //but...
    print(_todos.value); // ok
    print(_todos.value.addAll([...todos])); // not rebuild. use instead of todos.addAll([...todos]);
  }
}

3. Use R<V> or RList<V> with Widget (at StateWithViewModel) #

@override
Widget build(BuildContext context) {
  return Scaffold(
    body: Center(
      child: Column(chlidren: [
        Text("${viewModel.count.value}"),
        Text("${viewModel.todos.value}"),
      ]),
    ),
  );
}

4. Minimal rebuild #

@override
Widget build(BuildContext context) {
  return Scaffold(
    body: Center(
      child: Column(chlidren: [
        SelectBuilder(
          rx: viewModel.count,
          builder: (context, count) =>
              Text("$count (rebuild only this widget)"), // rebuild only (when updated)
        ),
        Text("${viewModel.count.value} (not rebuild)"),
      ]),
    ),
  );
}

5. Global State Manage Container #

in this example, using GetIt.

main_container.dart

class MainContainer extends BaseContainer {
  late final _globalCount = createMutable(0);
  R<int> get globalCount => _globalCount;
  
  void incrementGlobalCount() {
    _globalCount.value++;
  }
}

main.dart

void main() {
  GetIt.instance.registerLazySingleton(() => MainContainer());
  // or, registerLazySingleton(MainContainer.new);
  runApp(MyApp());
}

counter1_page.dart

class Counter1Page extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    final container = GetIt.instance.get<MainContainer>();
    return Scaffold(
      body: Center(child: SelectBuilder(
        rx: container.globalCount,
        builder: (context, value) => ElevatedButton(
            child: Text("Page1 count up : $value"),
            onPressed: () => container.incrementGlobalCount()),
        ),
      ),
    );
  }
}

counter2_page.dart

class Counter2Page extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    final container = GetIt.instance.get<MainContainer>();
    return Scaffold(
      body: Center(child: SelectBuilder(
        rx: container.globalCount,
        builder: (context, value) => ElevatedButton(
            child: Text("Page2 count up : $value"),
            onPressed: () => container.incrementGlobalCount()),
        ),
      ),
    );
  }
}

6. Observer #

class ObserveTestPage extends StatefulWidget {
  const ObserveTestPage({super.key});

  @override
  State<ObserveTestPage> createState() => _ObserveTestPageState();
}

class _ObserveTestPageState extends StateWithViewModel<ObserveTestPage, ObserveTestViewModel> {
  @override
  ObserveTestViewModel createViewModel() => ObserveTestViewModel();
  
  final mainContainer = GetIt.instance.get<MainContainer>();

  @override
  void initState() {
    super.initState();
    /// viewModel.counter lifecycle is same as this Stateful Widget. not has observer.
    viewModel.counter.observe((v) => print("counter value changed! : $v"));
    
    /// but, mainContainer.globalCount lifecycle is very different. (not depend on this Stateful Widget)
    /// that's means globalCount must be `cancelObserve` when dispose.
    mainContainer.globalCount.observe(observeGlobalCount);
  }
  
  void observeGlobalCount(int v) {
    print("globalCount value changed! : $v");
  }

  @override
  void dispose() {
    mainContainer.globalCount.cancelObserve(observeGlobalCount);
    super.dispose();
  }
  
  @override
  Widget build(BuildContext context) {
    return const Placeholder();
  }
}
2
likes
150
points
33
downloads

Publisher

verified publishernote11.dev

Weekly Downloads

A simple and easy ViewModel and state management Library. support ViewModel's state hold, Global state management. (Container's state hold)

Repository (GitHub)
View/report issues

Documentation

API reference

License

BSD-3-Clause (license)

Dependencies

flutter

More

Packages that depend on view_model_kit