views:

349

answers:

4
+10  A: 

MVC is an interesting abstraction, but has some problems.

In reality, the controller and view are often paired--even though in theory you should be able to replace either one without the other, in reality the interface mechanisms to different views are so different that the controller & view are combined.

The best description I've seen relating to Java is that the view is your swing components so your portion of the view code is nothing but placing those components on the screen.

Your controller is the rest of that class, the listeners and the rest of your code that interacts with the view.

My suggestion would be to not worry too much about isolating the view and controller, but that said, I am totally behind keeping a very strong separation between the model and the view/controller.

EDIT/Advanced: I have used a pattern where controller and view are isolated and it is more flexible, but it tends to be a lot more work. I think Struts uses the binding model--if you want to see some abstraction techniques you might look there or search for stuff about "binding" swing controls.

Bill K
+1 I totally agree.
neuro
+1 This also tallies with my experience.
Zarkonnen
(Though if anyone *can* make a case for how to sanely separate the view and the controller, I would be very interested.)
Zarkonnen
Thanks for your answer Bill. One of my fear is, as you realized, is "Writing bad code". Do think my implementation is acceptable? One more thing: " [...] that the view is your swing components and your view code is nothing but placing those components on the screen. [...]"Did you meant " [...] your controller code is nothing [...]"?
Henrik P. Hessel
It is possible to separate the view and the controller, but IMHO not as clean as can be achieved in a web application.
Praveen Angyan
Yeah, I used the word model a few times when I was thinking controller. Thought I caught them all on review. Also your code looks fine--I've seen much worse. If you had to develop 50 screens with dozens of controls on each I'd say seriously look into creating a system that uses binding, but for just a couple screens, the traditional techniques work fine.
Bill K
@Praveen Not really, on a web app, your "View" is "The Web". If you switch that out to another view (say swing controls, or a database front-end or a CLI, I bet you are going to replace most of your controller code as well). MVC only seems easier on the web because you assume you are never switching views.
Bill K
@Zarkonnen - Sure, have a Controller thread that monitors an Event Queue. Both the Models and the View place things onto this queue whenever they have an action. As items are popped off of the queue certain actions are performed by the controller, and messages are sent to either the view or model (depending on the message).
amischiefr
@Zarkonnen - Consider a table view. The controller contains a list of model objects, and the controller implements a "TableDataSource" interface that the table view defines. The table view isn't tied to the controller, it's tied to TableDataSource. That creates a fair amount of separation by using dependency injection.
Tom Dalling
+1  A: 

From my perspective, this is something I ran into when I first tried to separate out my Models, Views and Controllers in my first desktop application.

With Web Applications, the MVC pattern fits VERY naturally because of the innate nature of the web, but unfortunately it's not possible to fit a pure MVC patter to a desktop app, where the Operating system plays an innate role in notifications. This usually leads to the pattern being implemented as you've shown in your diagram.

However the pattern that you've shown really needs to be implemented like this, I think (I've switched over to .NET after a brief affair with Java, so please keep that in mind):

public class ContactManagerMainScreenModel
{
    ContactManagerMainScreen v;
    // Loading Local Storage
    LocalContactStorage LocalContactStorage = new LocalContactStorage();
    // Favorite list
    boolean showFavoritesList = true;

    public void register(ContactManagerMainScreen v)
    {
        this.v = v;
    }

    public void ShowOrHideFavoritesList()
    {
        if(showFavoritesList) 
        {
            showFavoritesList = false;
            v.RefreshView(this);
        }
        else
        {
            showFavoritesList = true;
            v.RefreshView(this);
        }
    }
}

Meanwhile the controller would be responsible for receiving user actions. So if you have a button that says "Toggle Favorites", this would cause the controller to call _model.ShowOrHideFavoritesList(). The model would update itself and ask the view to refresh itself using it's new state.

The view would now be free from the dependency on the controller.

Praveen Angyan
Again, thanks for your answer. Your implementation would lead to a model which directly controls the UI itself. Would you suggest dropping the entire controller?
Henrik P. Hessel
No, please read my last two paragraphs after the code. The controller is responsible for handling ALL user notifications. Here's another example. Let's say the user clicks on "Add New Contact", the controller would be responsible for calling _model.AddNewContact(string firstName, string lastName). The model would then add this contact to it's own internal state and then call RefreshView(this) on the view.
Praveen Angyan
What I mean is that the controller is still responsible for capturing all user notifications and then decidiing how to update the model. The model then always calls RefreshView(this) on the view which then redisplays itself...
Praveen Angyan
Understood :) +1
Henrik P. Hessel
+2  A: 

I don't think that diagram is very good, and probably makes things a lot more confusing.

The controller should be responsible for giving the Model to the View. The Model should contain nothing more than simple accessors for your data. Any need to interact with the Model - or change any of it's values - should take place thru the Controller.

This way, the View only needs to know how to render/present the Model to the user. So any operations on the Model - things like saveFavoriteContactsToLocalStorage() - should be methods of the Controller, not the View class. Also, the Controller should not need a reference to the View to be constructed - I think that ends up reversing the intended order of the entire MVC pattern.

matt b
I believe his diagram is correct. The problem is that MVC isn't a good abstraction and every toolkit tweaks it in such a way as to make it fit the way they designed it. Some do a good job, some not so good--the one thing that is for sure is that if you change the view, you're going to end up changing the controller (which is against the theory).
Bill K
Perhaps. I'm only really familar with using MVC in web contexts, in which case the Controller would simply return the View and Model to use (as in Spring MVC). I suppose it might be quite different on the desktop.
matt b
+1  A: 

I agree with Bill K. Standard MVC is not that important when dealing with rich client. It is a great model when writting web apps as events (clicks in your browser) are quite different than your view (HTML render).

With standard GUI, a better model is something called PAC (presentation/abstraction/Control). Here the presentation is (the view + the event handler), the controller manage the exchanges between the presentation and the abstraction. And the abstraction is your business model.

This way you have a controller that manage the link between the GUI part (view + user events) and your abstraction. You have a PAC agent for any piece of GUI you develop and a clean separation between them.

Another article about something better than MVC related to PAC : HMVC. Quite old, but it is practical and it helps understanding things.

neuro