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

add ImageViz for mapbox ImageSource #61

Merged
merged 15 commits into from
Mar 16, 2018
Merged

add ImageViz for mapbox ImageSource #61

merged 15 commits into from
Mar 16, 2018

Conversation

vincentsarago
Copy link
Collaborator

This PR tries to add an ImageViz method to enable Mapbox GL static ImageSource.

There is couple things that need to be discussed:

  • do we need to pass data here
    class ImageViz(MapViz):
    """Create a image viz"""
    def __init__(self,
    data,
    image,
    coordinates,
    *args,
    **kwargs):
    """Construct a Mapviz object
  • What kind of image input we want to allow ?
  • I don't know why but image or coordinates needs to be flipped to be displayed correctly on the map

Preview

capture d ecran 2018-03-07 a 11 21 25

cc @dnomadb @ryanbaumann

@vincentsarago vincentsarago self-assigned this Mar 7, 2018
@dnomadb
Copy link

dnomadb commented Mar 7, 2018

I don't know why but image or coordinates needs to be flipped to be displayed correctly on the map

GL JS explicitly takes the order of UL, UR, LR, LL https://www.mapbox.com/mapbox-gl-js/style-spec#sources-image, not W, S, E, N

What kind of image input we want to allow ?

It seems that this would be most useful for displaying manipulated data, which in most cases would be rows, cols, depth numpy ndarrays.

coordinates = [... ]
img = imread('img.jpg')

visualization = ImageViz(img, coordinates, ...)

visualization.show()

Beyond what this might add in terms of ndarray ==> image data encoding within the library, to be more effectively used we'd probably have to add a utility to take data ndarrays and apply a color ramp (which could be done via `matplotlib):

from matplotlib import cm


# data scaled to 0-1
data

# using the viridis colormap
img = (cm.viridis(data) * np.iiinfo(np.uint8).max).astype(np.uint8)

@ryanbaumann
Copy link
Contributor

This is awesome @vincentsarago! Let me know when you're ready for 👀 on a review.

center: {{ center }},
zoom: {{ zoom }},
transformRequest: (url, resourceType)=> {
if ( url.slice(0,22) == 'https://api.mapbox.com' ) {
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We should use regex to be more robust in catching requests to the Mapbox api -- older styles may still have

  • api.tiles.mapbox.com
  • a.tiles.mapbox.com
  • b.tiles ..

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@dnomadb good flag. I'll include this change in a separate PR, since it spans all visualization types.

@@ -276,3 +276,33 @@ def add_unique_template_variables(self, options):
clusterRadius=self.clusterRadius,
clusterMaxZoom=self.clusterMaxZoom
))


class ImageViz(MapViz):
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@vincentsarago could we call this class an ImageLayer? it looks layer-like to me and I think Layer could tell a user a lot about how to use the class.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Right now it's a proper MapViz, but I think at the end we could make it a layer meaning that instead of having a ImageViz, we could add a addImage function to the MapViz class.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Agree, a change to use layer nomenclature makes sense here. I'd propose a renaming of the API from *Viz to *Layer in a separate PR from this one, as a part of #34.

@vincentsarago
Copy link
Collaborator Author

Ok I like the idea of using Matplotlib (+Pillow) if this is what the users are use to do.

Here is what I propose

  • add an utility function to encode the image (ndarray) to a base64 string (plus prefix)
# mapboxgl.utils.img_encode

# `img_encode` is a wrapper arround `matplotlib.image.imsave` 
# https://matplotlib.org/api/image_api.html#matplotlib.image.imsave

import base64
from io import BytesIO

def img_encode(arr, **kwargs):
    """Encode ndarray to base64 string image data

    Parameters
    ----------
    arr: ndarray (rows, cols, depth)
    kwargs: passed directly to matplotlib.image.imsave
    """
    sio = BytesIO()
    imsave(sio, arr, **kwargs)
    sio.seek(0)
    img_format = kwargs['format'] if kwargs.get('format') else 'png'
    img_str = base64.b64encode(sio.getvalue()).decode()
    
    return 'data:image/{};base64,{}'.format(img_format, img_str)
  • document the use of imread to read image file
img = imread('./mosaic.jpg')
img = np.mean(img[::10, ::10], axis=2)
img = img_encode(img, format='png', cmap='gnuplot')

# Coordinates must be an array in the form of [UL, UR, LR, LL]
coordinates = [
    [-123.40515640309, 38.534294809274336],
    [-115.92938988349292, 38.534294809274336],
    [-115.92938988349292, 32.08296982365502], 
    [-123.40515640309, 32.08296982365502]]

# Create the viz from the dataframe
viz = ImageViz(None, img, coordinates, access_token=token,
                height='400px',
                center = (-119, 35),
                zoom = 5)
viz.show()

@dnomadb @ryanbaumann @sgillies

Copy link
Contributor

@ryanbaumann ryanbaumann left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@vincentsarago looks great! Made a few comments, all optional. Made a small commit to aid running the example locally for the first time.

img_format = kwargs['format'] if kwargs.get('format') else 'png'
img_str = base64.b64encode(sio.getvalue()).decode()

return 'data:image/{};base64,{}'.format(img_format, img_str)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@vincentsarago it may be good to include an optional filename output parameter here. This would encourage users to output a local file with the encoded image data so that the mapboxgl viz class can load the image data asynchronously (versus stick it into a large HTML blob).

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In fact I think because mapboxgl can read local files we can totally remove this utility function and show the user it can use imsave

"\n",
" map.addSource('image', {\n",
" 'type': 'image',\n",
" 'url': '
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@vincentsarago let's change this to an async URL load from a local file, if possible, to reduce file size of the notebook and encourage async loading of images.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Oh well I didn't knew mapboxgl could read a local file 👌

@@ -276,3 +276,33 @@ def add_unique_template_variables(self, options):
clusterRadius=self.clusterRadius,
clusterMaxZoom=self.clusterMaxZoom
))


class ImageViz(MapViz):
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Agree, a change to use layer nomenclature makes sense here. I'd propose a renaming of the API from *Viz to *Layer in a separate PR from this one, as a part of #34.

mapboxgl/viz.py Outdated

def __init__(self,
data,
image,
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@vincentsarago could we simplify the syntax by just passing the pixels or URL for an image to the required data parameter, instead of adding a new parameter?

@ryanbaumann ryanbaumann mentioned this pull request Mar 11, 2018
@@ -14,7 +14,8 @@
"\n",
"```\n",
"pip install mapboxgl\n",
"pip install Pillow numpy\n",
"pip install imread\n",
Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@ryanbaumann imread comes from matplotlib which is already installed

install_requires=['jinja2', 'geojson', 'colour', 'matplotlib'],
so no need to install imread

@@ -14,7 +14,8 @@
"\n",
"```\n",
"pip install mapboxgl\n",
"pip install Pillow numpy\n",
"pip install imread\n",
"pip install numpy\n",
Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

same here, numpy will be installed by default when installing mapboxgl-jupyter

@vincentsarago
Copy link
Collaborator Author

ok going back to @dnomadb point earlier, maybe ImageViz could accept 2 kind of input:

  • path (string), could be url, local path or event base64 encoded image
  • numpy array

If a numpy array is given ImageViz will encode it for the user

cc @ryanbaumann

@vincentsarago
Copy link
Collaborator Author

this is ready @ryanbaumann I'll merge it when the tests will be over !

@vincentsarago vincentsarago merged commit 20ca8fe into master Mar 16, 2018
@vincentsarago vincentsarago deleted the image branch March 16, 2018 16:46
@ryanbaumann
Copy link
Contributor

🎉 @vincentsarago thanks!

@ryanbaumann ryanbaumann mentioned this pull request Mar 25, 2018
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants