-
Notifications
You must be signed in to change notification settings - Fork 3.3k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Replace MVVM with MVC for the architecture pages #11438
Comments
That's so elegant way, I wish I could see that when I was starting with Flutter |
I appreciate the discussion on Flutter architecture, but I disagree with replacing MVVM with MVC. First, Flutter’s architecture guide is a big step toward standardization—having a consistent guide is better than none. Second, while you argue MVVM isn’t suitable for Flutter because it lacks binding, the same strict logic applies to MVC: Flutter’s UI acts as the controller, making MVC just as incompatible. Third, Android’s architecture isn’t explicitly MVVM, so suggesting a shift away due to Compose misrepresents its guidance. Finally, I believe MVP is a better fit than both MVC and MVVM for Flutter. Simply renaming “ViewModel” to “Presenter” would align with definitions while keeping the guide intact. Perhaps the request could be reframed to propose MVP instead? |
I think the entirety of the Bluesky thread you linked to actually provides a number of very compelling counterpoints against this issue and I have to agree that this entire debate feels entirely unproductive and just catering to the (very strong) opinions of a tiny minority of folks. edit: I would also point out that the author is encouraging people to give a thumbs up to this issue on social media which is perhaps leading it to appear that there is a level of support for this which may not actually exist though it’s impossible to know people’s motivations, I just point it out to say that this is something to be considered. |
I tend to think these kind of topics are very nit-picky with acronyms, and there are history were acronyms like these are slightly changed to accommodate the usage and even heavily debated on with no real conclusion that's fully agreed on.
Here is an example how Django uses a acronym with their own changes django docs From wikipedia about mvvm
which I would say Flutter uses the boilerplate approach by using builders and access to the listenable, but that is just my interpretation of it. MVP is a fully different approach where the presenter has access to the view, which is not the case with MVVM or MVC, to me that is a real difference that would change the way the architecture is implemented.
MY OPINION At the end of the day I am open for whatever, the course we have uses the same architecture (MVVM, and some differences in convention for services, repositories, and going simplicity first). But yeah, if we end up all agreeing that MVC describes this convention better we will just Also, as already mentioned in this issue both Compose world and SwiftUI world has been going the MVVM direction so if this goes through expect some confusion. |
Hi all, thanks @alefwd for creating this ticket.
I looked at the code of the Pokemon app example provided here, and the I've seen a similar pattern in the past called "the Riverpod architecture" which is probably a better name for it. Code with Andrea has a good comparison between "The Riverpod Architecture" and MVC here: https://codewithandrea.com/articles/comparison-flutter-app-architectures/#comparison-with-mvc I personally think MVVM is still the best way to describe the design pattern, even if the Wikipedia's definition of the pattern doesn't fully match it because it doesn't have Data Binding. This naming is also supported by other learning materials, for example in the Flutter Engineering book by Majid Haijan, his implementation of MVVM is very similar to the one in the Flutter Architecture Guidelines. But, if we are looking for a more accurate naming, it ain't MVC either. |
@miquelbeltran The same naming is also in the book "Flutter Design Patterns and Best Practices" |
@afterwire This comment itself, whether you're aware of it or not, is suggesting the reader that the support on this issue is backed up by people who have no idea what the issue is about and blindly went to the issue to give a like just because they saw it on social media. It should be simple to understand that the people who agree with this issue tend not to write down their own motivations if they're the same as what's written by the author, in which leaving a like is sufficient. On the other hand, those who disagree with this issue are more to be expected to write down their reasoning of what they disagree with. While you're just expressing concern and not accusations, I'd suggest you reconsider writing such comment as the reader could easily interpret it as an invalidation of the level of support here, which may introduce an unnecessary bias into the discussion. |
I don’t mean it in an accusatory way that something particularly nefarious is going on but I do at the same time think that the drive by like without really considering the issue is actually exactly what is happening here. The original thread I referenced actually went into some detail where it was pointed out that the reasons for why MVVM make sense is actually a nuanced and multifaceted issue where the idea of how closely it maps to the Wikipedia article is just one of many factors and it was repeatedly pointed out that there was a total absence of real world evidence to suggest this was anything other than the personal preferences of a small handful of people. It wasn’t even hinted at in the issue that there was an opposing set of viewpoints or considerations here so kind of by definition it seemed rather difficult to see it as a particularly nuanced topic and seemed to fit a pattern of drive by likes that was worth point out without pointing fingers because to be clear I don’t think anyone did anything wrong here at all but I do think the context is important. Edit: I also just find it to be a really weird premise to turn around and tell both the Flutter and Android teams that they don’t know what they are talking about when it comes to their own products and how to talk about them. |
@afterwire my idea is to improve the Flutter's docs about architecture, and you can disagree with me and I'm fine with that, but my intentions are honest. I don't think the de-facto reference for Flutter architecture is an MVVM. I personally couldn't care less if it came from the Android team, the Flutter team, the Pope or God directly, if I find it incorrect, I'll point it out. If I'm proved wrong, so that what we're actually using it is MVVM, then I'm totally fine with changing my mind. Until then, I think using MVVM is very misleading because of the reason I explained. |
Maybe we’re trying to forcefully fit Flutter into a pre-existing and well-known paradigm in order to conform to the norms of the industry as well as existing architectures which were designed for older types of technologies, and it makes us feel warm and fuzzy as we don’t have to yet learn another pattern. I believe Flutter has been such a revolutionary framework and owes its success by not standing next to legacy technologies like Microsoft’s Xamarin or Android’s XML because it is a modern paradigm (declarative vs imperative). If we stick to the Wikipedia definition of MVC, and if we revisit MVVM and what Microsoft intended it to be - a model-binder, I still feel that Flutter is slightly closer to the MVVM in the case that the ViewModel does not know anything about the View and the way it implements the Command Pattern, which makes it for a truly unidirectional data flow (data flows down while events go up). However, if we’re in the renaming discussion, maybe it should be called MVVN (Model-View-View-Notifier) since it aligns a bit closer to what’s doing: notifying listening views to rebuild themselves upon state changes. I'm just happy the Flutter team added this documentation which was long overdue and much needed by the community. Call it one way or the other - the community won already. |
I really like this name Roman, it's at the same time descriptive and unambiguous! |
Some favor calling it MVVM, some MVC, others MVU (Model View Updater, which is Microsoft naming for this architecture). Whatever you call it, it orchestrates and coordinates updates between the Model and the View. I like to think it "controls" what goes where, and simply call it a controller for that reason. However, since it in Flutter typically is, and especially in this guide, based on a Notifier, calling it MVVN sounds fine too, good suggestion @romanejaquez. I don't mind calling it an MVU or MVC either, but an MVVM it is not, for all the "binding" reasons mentioned earlier 🤷🏻♂️ 😄 |
It sounds like the majority of the responses in this thread are in favor of recommending the same pattern, but disagree on the name. I think MVVM is a better name than MVC, because Flutter is a reactive UI framework, and we want developers to avoid thinking in terms of "controlling" the view, and instead get them to think about the view as a function of the state (whether that state is a
It's true that other frameworks use data binding to publish changes from the view model to the view, but in the Flutter docs we are moving in the direction of recommending using ChangeNotifier, a class with ValueNotifiers, or simply a StatefulWidget for your ViewModel. Perhaps we could make it more clear. For these docs, the exact structure of your code is less important, you may prefer to use Streams, ChangeNotifiers, or your own custom Dart classes to implement the design pattern we are recommending. The important point is that the ViewModel publishes that something changed, and the view needs to listen to the ViewModel so that it can rebuild based on the new data. When something happens in the view, it needs a reference to the ViewModel so it can respond to user input. |
💯
When writing these docs, I debated whether I should explicitly mention MVVM. I considered ignoring it altogether and not using any existing architecture acronym. I decided to use MVVM because it's a frame of reference that most developers are already familiar with, which makes the content easier to digest. Using an unfamiliar acronym like MVVN requires far more explanation. I would have to (ironically) explain MVVM more in depth than I did, because I'd need to start with a well known concept (VM) as the frame of reference to then move to the unfamiliar concept (VN). The reader would need to know MVVM intimately in order to understand the trivial differences between VM and VN. This would be a distraction and ultimately hurt the readers ability to internalize the important parts of the docs. I actually like the idea of MVVN quite a bit in theory, but I'm not convinced (yet) that it would improve the readers ability to learn this design pattern, and that is the only goal of this project. As always, I encourage everyone to continue this conversation, and I'm open to being wrong. edit: This argument applies to all suggestions of changing the docs to support a different acronym. In order to consider that, the new acronym must:
|
Agreed 100%. As was mentioned elsewhere when this topic came up, it’s incredibly difficult to find any real life evidence that anyone can point to that would suggest this is causing anyone real confusion in practice rather than a strong personal preference of a small number of people. That’s not to say that it’s a perfect one to one mapping with everybody’s idea of what MVVM might look like in other contexts but as you mentioned it does help provide an actively useful frame of reference upon which people can build upon and not feel overwhelmed by thinking this is some obscure Flutter specific pattern that they have to learn. I’d also make the argument that there are only a tiny number of scenarios where the differences between this and say the Wikipedia version that was referenced have any kind of relevance to whatever the topic may be. I think the entire premise of the term “design pattern” actually encapsulates the concept very clearly that there is variation between implementations that adapt to the specific problem set at hand. In terms of the overall tradeoffs I think this actually makes the most sense of all the options presented so far. |
@miquelbeltran thanks for the reply, I try to enter into details without being too demanding
so last night you gave me some thinking, I had to find some old notes, and it turns that the actual implementation I use of the controller, a ChangeNotifier containing a UiState (so a state holder), comes from the Flutter's skeleton (new) template. There the SettingsController is defined, and it holds the theme of the app, and when changed it triggers the UI rebuild. Now 2 points:
to this:
the part I like here is that the user's actions are triggered having a reference to the controller, while the UI needs to have a reference to the state obtained through the controller |
@johnpryan just to clarify, the controller doesn't have a reference to the view at all, and the controller doesn't control the view, it controls the model. You can clearly see it here, there's no connection between the view and the controller: the view and the "controller" are instead connected, and heavily connected as they're data-binded, in an MVVM, as you can clearly see here: so please remember when you talk about MVVM, don't have in mind the code which is presented in Flutter's guide, because that is not an MVVM, which is the whole point of this discussion. What we implemented is not an MVC either (see my previous message), but is actually an MVN, this is the correct term we should introduce and use, it expresses the reactiveness, it gets rid of the legacy of the data-binding / xml we don't have, and it doesn't have much overhead to learn as it's quite easy to grasp IMHO |
The MVC pattern comes from Smalltalk's interpretation, but in other frameworks like Cocoa, the role of the controller changed, and became the intermediary between the model and the view. When I'm talking about MVC, I should probably call it Model-view-presenter instead. So are we recommending MVC? Or are we recommending MVVM? Or something else? I see two main criteria for MVVM:
It's true that Flutter does not support two-way data binding. Instead, it's a reactive framework. You don't need binding if the view layer is updates when
A controller has a reference to the view, but a view-model "notifies" the view when something changed, and the data-binding system updates the screen to show the changed data. From an architecture point of view, I think MVVM is a good recommendation so that developers can separate their business logic from their view logic. If you're writing app-specific code (business logic), it's a good idea to separate it out so you can test it independently. If you're writing UI code, you probably want to make it as reusable as possible if you are building a large app. |
fyi, i don't know what i'm doing and i didn't understand why it was called MVVM. it actually confused me because i didn't know what MVVM was (there's a lot of us new to this stuff). i thought models didn't have any moving parts. i thought models were just guides/rules to reference and keep things tight and organized if something is telling something what happened or what to do, then ya it's notifying something. it's delivering a message. if something else can receive instructions and commands then ya it's a receiver or listener or watcher or part of a messaging system. if someone told me there was this model that a view referenced and that view had a messenger to tell something else what happened or what to do, then i would have learned this way faster just through a simple choice of words. what about MVM? really, what else does it need? |
Model View Signal? |
@jtkeyva I think the argument here is more along the lines of do you find a term which is known by a huge amount of people but varies by an admittedly fairly obscure and not very relevant part (I.e this entire argument about data binding) or do you try and come up with some new term that nobody knows but maybe more accurately reflects the small amount of conversations where that point of difference might be relevant. To me personally, it’s not even close in terms of trade offs. Nobody’s level of comprehension is improved with a new term that nobody knows unless you then go on to present a lot of background text around the intricacies of MV* patterns and how this one is 95% the same as this other one but then is different in this one very specific way. At that point you are off on an entire side quest that has zero to do with the actual purpose of these guides, made everything longer and less focused and picked up almost no meaningful benefits. Like at most I would say this topic maybe would justify 2-3 sentences that perhaps point out that unlike in some other frameworks that rely on markup for the UI as a part of their MVVM architecture and therefore need this concept of data binding in order to implement the pattern, Flutter is a reactive framework which actually simplifies things a bit and here’s how it looks... But to me the idea that we would even consider throwing away the bridge that we could offer a huge number of developers by associating this to a term that they are in many cases already familiar with for no good reason is an objectively terrible set of trade offs for me when you think about it for a moment. |
To me the fact that we, willingly and as part of the Flutter community, decide not to care about a name which doesn't reflect the pattern we're actually using, just not to confuse the users, is simply disgusting. And I think it will create even more confusion on the long run |
This conversation is no longer productive and careening toward nasty. This is not the place to throw around insults. I've set a reminder for myself to unlock this post in 2 weeks. |
Page URL
https://docs.flutter.dev/app-architecture/guide#mvvm
Page source
https://github.com/flutter/website/blob/main/src/content/app-architecture/guide.md
Describe the problem
I understand this request is opinable, and debatable, because it depends on the definitions we use, but when I presented Flutter Architecture Components, which was inspired by Android Architecture Components, I realised the Android's MVVM model doesn't apply to Flutter, while the MVC is more representative of how Flutter works.
Let's stick to some generic definition of the MVVM and MVC patterns.
1 We can see from this picture:
that an MVVM requires data binding between the view and the view model. This is old legacy, coming from Android using xml as binder between the UI and the data (side note: now with Compose, someone should tell the Android team to reconsider the MVVM pattern as well).
"Declarative data and command-binding are implicit in the MVVM pattern. In the Microsoft solution stack, the binder is a markup language called XAML.[10] The binder frees the developer from being obliged to write boiler-plate logic to synchronize the view model and view. When implemented outside of the Microsoft stack, the presence of a declarative data binding technology is what makes this pattern possible,[5][11] and without a binder, one would typically use MVP or MVC instead and have to write more boilerplate (or generate it with some other tool)"
2 We can see from this picture:
that the MVC represents more precisely (than MVVM) the cycle where the user clicks the UI, which access the controller to manipulate the model, and after the manipulation the UI is notified and gets updated.
"If user input prompts a change in a model, the controller will signal the model to change, but the model is then responsible for telling its views to update"
"The controller responds to the user input and performs interactions on the data model objects. The controller receives the input, optionally validates it and then passes the input to the model"
3 This is argumented not only by me here, but also by Christian here, by Thomas here, and many others. I think these 3 comments from Ahmed summarise it well.
4 I have a repo with an entire app build with MVC here, if someone wants to have a look.
Expected fix
In my best wishes, I would like to see the MVVM taken away from the guide, and replaced with MVC
Additional context
No response
I would like to fix this problem.
The text was updated successfully, but these errors were encountered: