data:image/s3,"s3://crabby-images/d5933/d59331f5c63e59422b98eba6b315e1f0b2be6351" alt="A Chrome DevTools screenshot of Journey"
Journey is a clientside map-based web app meant to help empower me to cycle for fun and fitness.
The main motivation came from my desire to explore Seattle’s bicycle network and trail system.
My motivations were threefold:
-
Lower the activation energy needed to set out on a bike ride by picking a new point of interest
-
Manage time constraints by using isochrones to gauge how far away my destination should be
-
Improve perceptions of terrain by exaggerating elevations and not using a top-down perspective
The web app is currently live over at https://kennydurk.in/journey.
- Journey uses the Mapbox developer ecosystem as its foundation. Reasons being:
- I wanted to learn a programmatic way to work with maps, and MapboxGL JS is well-maintained.
- They have great developer relations, APIs, and documentation.
data:image/s3,"s3://crabby-images/b7702/b770276c7dd4612f0b703bf84f6e6f31ea383067" alt="A pitch of ~62.5° and terrain exaggeration of 4.0"
-
The app uses exaggerated terrain and an angulated pitch so as to highlight the surrounding elevation
-
It’s a single page app that takes up the entire screen for simplicity
-
Should be responsive and designed from a mobile-first perspective
-
On first visit, a camera animation is displayed to showcase the 3D aspect of the map and its terrain
-
User supplies their origin point, how long they would like to cycle, and the type of destination they would like to discover
Pictured at right: A pitch of ~62.5° and terrain exaggeration of 4.0
data:image/s3,"s3://crabby-images/4a719/4a71966ba16b4710c241f8eec81539fa2745f70c" alt="A sample isochrone showing all points a 30 min bike ride away from the Capitol Hill neighborhood"
-
Utilizes the concept of an isochrone, a ring around a given point that will convey the ideal range of the ride based on the user’s time constraints
-
Randomly selects a point along the ring and searches for destinations of the specified type in proximity to that point
-
Selects the first result and supplies routing instructions and depicts them on the map
Pictured at right: A sample isochrone showing all points a 30 min bike ride away from the Capitol Hill neighborhood.
A successful lookup and bike route to a nearby café in the Denver, CO metropolitan area.
Tool | Notes |
---|---|
Mapbox GL JS | Core JavaScript library for creating a WebGL based interactive map |
Isochrone API | Supplies the ring of points based on desired time commitment |
Geocoding API | Searches in proximity to a point for destinations of the desired type |
Directions Plugin | Mapbox-provided plugin, supplies routing results and instructions UI |
React JS | View layer, provides interactivity based on UI changes (e.g. toggles) |
Vite Bundler | Package manager that handles bundle minification and other build tasks |
Github Pages | Free web hosting for rapid prototyping and deployment |
A rough outline of the core user flow of the current web app.
- Refactoring the codebase to TypeScript
- Add unit testing for helper methods and React components
- Let the user determine their point along the ring
- Add more destination types (possibly switch a better geocoding API)
- Add handling for sparse regions where no results are discovered
- Improve the “point along a ring” approach, as this can be interfered with by shorelines
- Improve the “radius surrounding a point” search for points-of-interest
- Allow the user to specify region and show a camera animation showing their region’s terrain
- Add a gradient in the route line that shows terrain intensity
- Add a chart graphing elevation gain throughout the route
This is not an actively-maintained project, but PRs are nonetheless welcome 🙂
- Sign up for a Mapbox Developer API key and copy the key.
- Clone the repository onto your machine
- Create a file at the root level called
.env
and add the following:VITE_MAPBOX_KEY=yourApiKeyStringGoesHere
- Run
npm install
to obtain all the required dependencies - Run
npm run dev
and Vite will generate a port for you to visit locally