Rebuilding TaxiRouter SG

Chee Aun speaking at Talk.JS about TaxiRouter SG in 2016

In 2016, I built TaxiRouter SG and wrote an article about it. I even gave a talk on Singapore JS meetup. Heck I even got featured on Digital News Asia, Channel 8 News and the GovTech conference! 😅

It's been two years and I still find it unbelievable that such a simple little web app could get me this far. Since then, I had a few ideas on improving it but slowly neglecting it as I got busy with too many things. My days of peace were taken away when someone reported to me that the map disappeared from the site, on February 15.

It turns out that the map was not loading on all my transportation web apps, including BusRouter SG and RailRouter SG! 😱 I immediately suspect that something must be broken on Google Maps since all of them uses the Google Maps JavaScript API. Perhaps my API key stopped working or my apps have exceeded the free daily usage limits?

Turns out, it's this:

map.addListener('bounds_changed', function(){
   var bounds = map.getBounds();
   if (bounds.intersects(mapBounds)){
     ...
   }
});

Since my web apps mostly revolve around Singapore, I always have to write a piece of code to detect if the user panned outside of Singapore's boundary. It could be unintentional too if the user outside of Singapore clicked on the 'Current location' (or 'My Location') button that utilizes the browser's geolocation feature.

Anyway, looking back at the code, the critical error happens when the bounds variable returns as null and it doesn't have the intersects method. It didn't break before, so somewhere along the Google Maps JavaScript API releases on February kind of broke it and rendered all my apps useless 😅

Immediately, I applied all the quick hotfixes. 🔥

Scratching my own itch

After the quick fixes, obviously I looked at my apps and started to have that itch again. Revamping TaxiRouter.SG has always been on my mind for the past 2 years.

Why? This.

TaxiRouter SG site, first version

In the first version, I spent too much of my time trying to optimize the performance of the thousands of taxi markers on Google Maps. I was proud that I devised a pretty smart way of rendering them on high zoom levels, but not on the lower zoom levels.

The markers ended up hidden on low zoom levels and taxi stands are rendered by default instead, as shown in the screenshot above. For first-time users, this is a bit confusing when they see a map that should show actual taxis but they won't see any of them until they zoom in.

Individual taxis on TaxiRouter SG, higher zoom levels

Other sites, for example, Joseph Tang's taxi heat map uses a heatmap (duh), and Yongquan's taxi availability visualisation uses a hexbin map.

Joseph Tang's taxi heat map
Yongquan's taxi availability visualisation, using hexbin map

These are very useful to easily spot the density of the populated available taxis. They work very well on low zoom levels, but unfortunately not so much on higher zoom levels.

Ultimately, here's the ideal case:

  • On low zoom levels, show a heatmap.
  • On high zoom levels, show individual taxis, not grouped or coalesced.
  • Taxi stands only appear on high zoom levels because they are not useful on low zoom levels anyway.
  • Zero performance issues.

Challenge accepted

I've always been quite dissatisfied with the Google Maps JavaScript API V3 due to its usage of raster tiles. Meanwhile on the user-facing Google Maps web site, it uses vector tiles, rendered in WebGL, resulting in super smooth zooms and technically reduced data transfer.

Last year, I got my first taste on another WebGL-based vector-tiled map JavaScript library called Mapbox GL JS. Also last year, I remembered that Mapbox finally introduced heatmaps that are dynamically generated, powered by WebGL and could render smoothly at 60 fps while panning and zooming! One of the highlighted features is the application of heatmap-opacity to fade out the heatmap for a more detailed look when zooming in!

Comparatively speaking, the same features could be done on Google Maps JavaScript API as well. The differences would be:

  • Google Maps JS API V3 only allows fixed zoom levels (1, 2, 3), which explains the "lag" when it needs to fade in or out the tiles between the zoom levels. Mapbox GL JS allows floating-number zoom levels (1.234, 2.887, 3.147) because… vector 😉
  • Markers performance on Mapbox GL JS is so much better, as the data layer is passed to the GPU, as if you're building a game.
  • Customizing the map (markers and styles) based on zoom levels is quite troublesome on Google Maps JS API. On Mapbox GL JS, expressions are super powerful for data-driven styling. Transitions between property values can be automagically interpolated and zoom_changed events are not needed!

On 21 February, I posted some sneak peeks of the all-new TaxiRouter SG:

TaxiRouter SG, showing user's current location and taxis around
TaxiRouter SG showing taxis overlapping with heatmap

I was pretty excited to see that it works!

At this point, I don't really get why or how it could work at all. The taxi availability API already returns the results in GeoJSON so it's pretty straightforward as the format is primarily the standard data source for Mapbox GL JS. Unlike the first version of TaxiRouter SG, I don't have to fix flickering issues when refreshing all markers, reusing or recycling existing markers, or use d3.queue(50) which runs at most 50 tasks concurrently among the thousands of markers!

Mapbox GL JS has successfully deprecated all the efforts I've done with Google Maps JS API two years ago! It feels like a cheat code when all it does is to push everything (mostly?) to the GPU 😅😅😅

All the little details

On first load, TaxiRouter SG shows the whole Singapore with the heatmap layer on it. The heatmap colors are pretty self-explanatory with red showing the most densely populated taxis among the yellow portions.

TaxiRouter SG site

I manage to maintain all existing features from the first version:

  • "My location" or current location button, which will plot the user's location so that they can use it to locate taxis surrounding them. I've also added a bonus feature, a compass indicator which resembles very closely to the one on Google Maps.
  • Stationary taxis, which I labeled for taxis that didn't move for a minute, are still there. Instead of 50% opacity, I changed them to orange-colored taxis and purposely put more attention on them.
  • Taxi stands are available on higher zoom levels.
  • Bottom right panel indicator for nearby taxis and taxi stands, within a 5-minute walk, are there as well.

As the map is zoomed in, individual taxis will slowly and partially fade in.

TaxiRouter SG, zoomed in from heatmap layer to show individual taxis

Do note that the heatmap layer is still there, but the taxi markers are on top of it. As the map is further zoomed in, taxi stands will start appearing.

TaxiRouter SG, zoomed in to show taxi stands

When the markers start appearing, they seem to automatically disappear when it's detected that they may collide with other markers or labels around them. In fact, at these zoom levels, not all markers are displayed yet! Only some of them are rendered, as each marker are carefully shown without colliding one another.

This wonderful magic is all done by Mapbox GL JS's icon-allow-overlap and icon-ignore-placement. These properties are turned off by default, which means that the markers will not overlap with other symbols and will auto-disappear if they collide with other symbols 💥. The markers also have a default padding of 2 pixels via the icon-padding property, which kind of gives some space in between markers 💥.

Finally as the map is zoomed in to much higher levels, these properties are turned on to show all taxis (everything, no more hiding).

TaxiRouter SG, zoomed in to show all taxis, road-directions, POIs and buildings

The heatmap layer disappears at this zoom level as it's no longer useful. Taxis and taxi stands overlap one another.

There are two new features:

  • Traffic layer is shown to give a rough idea on where to easily grab taxis and avoid congestion. At first I apply the traffic layer on top of the Mapbox Dark layer but later I found a much better layer called navigation-preview-night-v2. It includes the traffic congestion layer by default, shows road direction symbols (the arrows), and even renders more Points of Interests (POIs)!
  • Buildings are rendered to give a clearer picture of the surroundings, which can be useful for people to identify stopping points especially when using cab-hailing apps.

Oh yeah, the buildings are 3-dimensional, a feature that was introduced by Mapbox in 2016. Users can zoom, rotate and tilt the map up to 60 degrees! Buildings are styled to be slightly opaque so that they don't block the labels and symbols behind them. 💥💥💥

TaxiRouter SG showing 3D buildings
TaxiRouter SG, showing taxis at the Changi Airport
TaxiRouter SG, showing taxis at Marina Bay Sands

I think most people would agree that this is the best version of TaxiRouter SG yet.

The relaunch

On 26 February, around 10 days after the disappearing-map issue, I publicly relaunched TaxiRouter SG V2. It's still open-sourced on GitHub and any feedback is always welcomed.

I hope that this web app continues to serve both as a visualization tool and a utility that's useful for everyone, especially the locals and travellers in Singapore. 😀🚕🇸🇬