Skip to content

eugenkiss/static-mobx-routing

Repository files navigation

Static MobX Routing Example Project

This project is an example of a statically typed and more controllable approach to routing web-based client-side apps by leveraging MobX and TypeScript. I tried to make this example sufficiently complex such that more interesting routing scenarious can be examined. Play with the live instance.

Note, I do not claim that this approach is perfect. I am very much open to suggestions and discussions—one of the main reasons I created it in the first place. Nonetheless, I have been happy with this approach so far.

Run Instructions

Simply run npm install and later npm start. For yarn it's yarn install and yarn start. Go to http://localhost:5001/ to see the app.

Even though this projects mimics a client-side app that communicates with a remote backend, there is no need run a local server. The reason is that this project contains an embedded mock server implementation. Application state is not persisted across reloads of the app.

Advantages

The following are the advantages of this routing and project structuring approach:

  • Type system and tooling support
  • UI is completely derived from state
    • More control over routing and its interaction with other state (such as caches)
    • Simpler data flow (e.g. app state changes transparently lead to route changes)

Please read “How to Decouple State and UI” for background information.

Motivation

I have never been satisfied with routing in client-side apps. When I was writing Android apps the conventional code needed to change from one conceptual screen to another (routing code) seemed unnecessarily convoluted. Don't get me started on this whole Fragment/Activity fiasco. Eventually, I found an approach to routing code with which I was satisfied and that actually was statically typed (a shoutout to Conductor as it made things way better even though it was not strictly necessary).

When I started writing React apps the consensus was to use React Router. Even with its version 3, routing code felt complicated and limiting compared to the approach I have employed in my Android code. At some point, I started using MobX for state management and change propagation which, by the way, is all that I wished for after I have written my master's thesis on GUI Programming. Although the non-routing code became much better with MobX there was still something off about the routing code. By chance, I read the terrific article “How to Decouple State and UI” by MobX's author Michel Weststrate which was eye-opening to say the least. In conventional React routing code URL changes do not directly change app state but first lead to mounting components which then change the app state themselves. This goes against the paradigm of deriving the UI completely from the app state and has several negative consequences. Based on this article I even created a presentation (slides).

Despite the benefits of the MobX-based routing code approach presented in the article there were still two things that bothered me about it and that were better in my previous Android approach. Namely, the static type system was not leveraged and code that felt should be close together (as you would most of the time change it simultaneously) was separated. In order to see if I could improve upon these points and also to test if that approach works in more complicated scenarios, I created this project.

Hopefully, this project leads to discussions and new insights. Maybe you see potentials for improving the approach which I'd like to hear about. Maybe you disagree with this approach and in that case I would be very interested in arguments against it.

Overview

As described in the Motivation section, I essentially set out to create a “sufficiently” complicated example that takes the spirit of the “How to Decouple State and UI” article and improves it by leveraging static type information and restructuring routing code so that things that should be close together are in fact close together. To be specific, I created dedicated classes for routes. Other features, to achieve the goal of ”sufficient complicatedness”, include:

  • Fast screen transitions
  • Embedded mock server with adjustable parameters (delay, variance, error rate)
  • Skeleton loading and handling loading states in general
  • Client-side caching
  • Login/logout handling
  • Handling error cases (e.g. failed request)
  • Handling race conditions
  • Prevention of screen transitions
  • Handling query parameters
  • History (de)serialization
  • Scroll position restoration
  • Form handling
  • Animations

You can have a look at the live instance and play around with the app. Make sure to adjust the mock server parameters to see the handling of error cases. Try going back and forth in the history with different scroll positions. Note history manipulations where, for example, an new post route entry is replaced with the resulting post route after successful saving. Try changing the search query paramter and note that there will never be two successive search routes in the history. Try to increase the delay and variance to their max values and quickly click on a post then go back and click on another post and repeat. If you do that for a while and you open the developer console you should see whenever a race condition was prevented. If not clear, it is of course not a real application but more of a demo so that not much effort has been put into its visual design.

The most relevant files with respect to routing code are store.tsx, router.tsx and history.tsx. The entry point is index.tsx.

For simplicity (and for other reasons) I do not employ dependency injection for the store and other things.

I came to the conclusion that (re)storing scroll positions and other soft or component-specific UI state should be the responsibility of the respective React component. As the component knows best when to restore UI state, e.g. after asynchronous loading has finished, it feels natural to let it take the responsibility. In Android there are specific lifecycle methods to achieve that. The idea is to save, e.g., the scroll position of the component to the current history entry with a unique id and to restore it if the user goes back to that history entry.

A shoutout to the following great (but arguably lesser known) projects that I used to great effect in this one:

  • styled components. As React has been the “revelation” for client-side apps in general, so is styled components when it comes to an app's styling concerns.
  • FromState. Forms have been a pain point in React code since its inceptions. FormState seems to me one of the best approaches from playing around with it and comparing it to others.

If you see something that can be improved, please let me know (GitHub Issue)!

Open Problems

It should be possible to make a library out of this project. I'm in the process of doing that.

The @action annotation is not really doing what it is supposed to do. The reason is the transformation that TypeScript applies to async functions (see https://mobx.js.org/refguide/action.html). Something like https://github.com/mobxjs/babel-plugin-mobx-deep-action but directly for TypeScript would be great. Afaik, TypeScript will soon allow such plugins.

In Android and Java land there is Retrofit which I find to be a very good HTTP client library. In this project I essentially created my own ad-hoc one based on fetch. It works fine, however a TypeScript version of Retrofit would be useful I think.

Some other questions in my mind are:

  • Has the approach to state transitions of buttons depending on requests states downsides?
  • Is there a better approach here to handle caching and race conditions? Has this RequestState approach downsides. MobX-Utils's fromPromise could be used as well and could also be used to handle race conditions. But it seems to be more limited. For example, if I want to handle the case of a cache-hit and still want to load data (i.e. show cached text but indicate with a loading animation that fresh data still needs to be fetched).
  • What are arguments for using MobX-Utils's createViewModel for Post instead of the current approach?
  • Should MobX provide debounced computed properties (see search.tsx)?

Another very annoying problem is the following situation. Let's say you navigate to a post /posts/0. Then you manually change the path in the url bar to /posts/1. If you press the back button now (to get back to /posts/0) the history cursor is not updated because the popstate event is not fired. Instead, it seems the browser completely reloads the page. I tried very hard to find a solution but I don't know what to do about it.

In general, the fact that you cannot ask the browser for the current history index is irritatingly limiting. See also this video by React Router's creators mentioning the problem. For this reason, UI states for a history entry are saved per URL. The implication is that if you have two entries in the history with the same URL they will share the same UI states. I think this is a fair compromise in practice. Still, the implementation is not pretty mainly because there is no hook to get the URL before it is changed.

Misc

The generated text is inspired by “Escape from New York” ;).

Related Projects

As mentioned already several times, the main inspiration is the article “How to Decouple State and UI”.

MobX Router is another great project which is similarly inspired by the aforementioned article but takes a slightly different aproach.

About

Example project that uses MobX and TypeScript for routing (WIP). http://static-mobx-routing.surge.sh

Topics

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published