Skip to content
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

Thoughts about mobx-vue #3

Open
nighca opened this issue Jun 6, 2018 · 8 comments
Open

Thoughts about mobx-vue #3

nighca opened this issue Jun 6, 2018 · 8 comments

Comments

@nighca
Copy link

nighca commented Jun 6, 2018

Hi there! I've read the docs and came with some ideas:

  1. Mostly what we want is to connect Store instance with Component instance/element, instead of to connect Store instance with Component class/definition. The later may introduce issues when the same Component class/definition used (instantiated) multiple times - different Component instance/element will share the same state. Such behavior suits global-state-situation, while not local-viewModal-state-situation.

  2. Maybe connect is not a good idea. connect defines extra properties on Vue vm in an implicit way. IMO, it will be better not to provide the ability of connect. Offering ability of observer allows developer to organize their state in more explicit way, maybe something like:

    @observer // assuming that we provide `observer` like mobx-react
    @Component()
    export default class App extends Vue {
      viewModel = new ViewModel() // which means `viewModel.age` in template
    }

What do you think?

@kuitos kuitos added the question Further information is requested label Jun 6, 2018
@kuitos
Copy link
Member

kuitos commented Jun 7, 2018

It's so great that met someone thought the same!

  1. Usually we use store in a singleton way, which means they are global and that is what redux and vuex does actually. So I think maybe you need a workaround to solve the shared-state problem(also needs in redux and vuex scenario):
@Connect(store)
@Component()
export default class App extends Vue {
    mounted() {
        this.initStore()
    }
    
    destroyed() {
        this.destroyStore()
    }
}

That is not my favour too so maybe I should support initializing the state automatically.

class ViewModel() {}
@Connect(ViewModel)
@Component()
export default class App extends Vue {}

But the risk of this workaround is that Store and ViewModel are different conceptually, the former lives singleton and the later lives around the component lifecycle, but they are defined the same way(class recipe). For a bindings/connector we shouldn't introduce too much extra concepts imo.

Or you can try the special solution with DI which mmlpx provided, you can take a glance althought it not documented well yet😄.

  1. Yes observer is more semantical than connect in a reactive system. But what I concern is, since we have take over the data and methods definition with a mobx model, why we need to define a extra field to be the member of component instance? which is against the experience of vue users.

    Mobx model should be equivalent to the component instance/vm conceptually, that means we can access the model directly instead of through a vm member.

    Definitely, observer is more semantical for a reactive system, I need to think it over.

@xrado
Copy link

xrado commented Jun 7, 2018

What about connecting multiple stores?

@kuitos
Copy link
Member

kuitos commented Jun 7, 2018

@xrado good question! Actually that is what made me in a dilemma.
I think one component should bind with one ViewModel, and the ViewModel could composed by multiple store and itself local states IMO, which mmlpx dose and that matching the standard MVVM philosophy.
But now I realized that I should not export any values of my own to users, be unopinioned is the basic principle of mobx and mobx-vue.

@nighca @xrado I will realize a version with observer and custom initialization these days. like:

@Observer
@Component() 
export default class App extends Vue {
  store = store
  vm = new ViewModel()
}

and using in template:

<div>{{store.user}} {{vm.loading}}</div>

Thanks for your feedback!

@kuitos
Copy link
Member

kuitos commented Jun 8, 2018

Released v2.0.0! @nighca @xrado

@nighca
Copy link
Author

nighca commented Jun 8, 2018

@kuitos 👍 Very impressive productivity.

To me, it is reasonable to provide another API like @Connect(ViewModel) and do class instantiation automatically. As it is more relevant to DI thing, maybe such an API and DI solution for Vue-Mobx can be provided together.

@kuitos
Copy link
Member

kuitos commented Jun 8, 2018

@nighca Automatic instantiation will introduce several new issues which should not simply construct the ViewModel with new, also applies to di system. Look such a scenario:

class ViewModel {
    id: number;
    @observable users = [];

    constructor(id: number) {
        this.id = id;
    }
    
    async onInit() {
        this.users = await http.get(`/users/${this.id}`)
    }
}

We need provide the ability to pass parameters to constructor, maybe it looks like this:

Connect(ViewModel, 10)

And things are moving forward. The parameter cames from vue instance, such as a router query string. We need support another usage:

Connect(ViewModel, vm => [vm.$route.query.userId])

In fact that are what mmlpx provided already!😉.

With mmlpx you can initialize states like this:

import { inject } from 'mmlpx';
class App extends Vue {
    
    @inject()
    vm1: ViewModel;
    
    @inject(ViewModel)
    vm2;
    
    @inject(ViewModel, 10)
    vm3;
    
    @inject(ViewModel, vm => [vm.$routes.userId])
    vm4;
}

DI should be an independent solution for who don't wanna initilize dependencies manually, and applies to any other framework imo. But Connect also required to me, maybe I should provide a more convenient instantiation way for mobx-vue by mmlpx-vue, and documented as best practice😄.

@nighca
Copy link
Author

nighca commented Jun 8, 2018

@kuitos Yes, what I mean is just what mmlpx does.

But I'm not sure if mmlpx already suits Vue context well. Maybe there is some more work needed, and if so, it is fine to provide these things within mobx-vue (instead of some other package like mmlpx-vue), just like that mobx-react provides Provider/inject. Connect is just another high level function based on Observer.

@jswhisperer
Copy link

I'm kinda wondering what direction this project might take with a "new vue" on the horizon. Since vue seems to move away from classes and therefore decorators is there a happy path to using mobx with vue in 2020?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

No branches or pull requests

4 participants