Microscopy, Python, and Open Source
data:image/s3,"s3://crabby-images/7d5c0/7d5c05519e6385d08e2bb54208df7be24b04b5c8" alt="".png)
At Ramona, we care deeply about how images are captured and displayed. Over the last few years, we’ve leveraged and contributed to open source software mostly by opening bug reports, suggesting fixes, and helping package scientific software for the community at large.
Today, we celebrate our first sponsorship of an open source project, the Pygfx project! Pygfx, pronounced “py-graphics”, is a Python library with the purpose to bring powerful and reliable visualization to the scientific python ecosystem. If you are interested in python and image visualization, we recommend you go check out the website. At Ramona, we leverage its capabilities to display our Gigapixel data right to your screen allowing you to interact with every single picture captured by the MCAM®.
As a celebration, we choose to study a common problem relatable to anybody trying to visualize content to a screen and realize that as they enlarge their image, their data starts to look blurry. In this blog post, we decide to focus on one of the simplest images, that of a logo, our logo!!! As we alluded to, if you try to zoom into our 128 x 128 pixel logo, you’ll quickly see that the quality appears to degrade!!
data:image/s3,"s3://crabby-images/358b3/358b3b1fbf2530052a8ed4900dbe6350eeac320c" alt=""
In fact, the quality doesn’t degrade, but rather the finite resolution of the underlying image shows its limits! However, at Ramona, our mission is to make you forget about pixels and bits and let you focus on biology, science, and discovery! One popular solution is to use vector graphics to draw in a scalable fashion — our marketing team loves to use those to make our swag look amazing!
In order to celebrate our new sponsorship of the Pygfx projects, we are deciding to redraw the logo you see on the MCAM® software in a pixel perfect fashion using some of the latest capabilities of the library!
data:image/s3,"s3://crabby-images/358b3/358b3b1fbf2530052a8ed4900dbe6350eeac320c" alt=""
The before and after are quite stark, read along if you want to learn how we achieved this!
How many pixels should your logo have?
When acquiring microscopy data, one can often estimate the number of pixels required for a given experiment by considering the size of the biology of interest. However when drawing a picture on a monitor, one cannot always determine the resolution of the display used to render this image. Many times, one has to guess an adequate resolution for an image, save the image in a PNG format to avoid compression artifacts, and hope that most users don’t zoom in too much to notice any inconsistencies at edges. However, with 4k and 8k screens, scientists can often zoom much more than they used to! It would be nice to have infinite pixels!!!! But for that we must decompose the logo into a mathematical formula that can be computed based on the final resolution.
In computer graphics, one strategy often used is based on signed distance field, SDF for short. The idea behind an SDF is to provide a formula for the distance from any pixel to the edge of the shape. If one can find a closed form formula, then the drawing can have “infinite” resolution giving us sharp edges. Pixels that are “inside the shape” are tagged as “negative” by convention. This allows one to color pixels with a negative distance with one color, and pixels with a positive distance in another. For more information we recommend users read Rougier’s Paper on creating pixel perfect markers for scientific communication on the GPU which can be retrieved on this link as well as some great posts and videos by Inigo Quilez found here.
The examples below use the Pygfx engine developed by Almar Klein and the rest of the Pygfx team!
The Ramona Logo
The first step is to decompose the Ramona Logo into a series of primitive shapes for which we can write down a formula.
We will break up the logo into 4 different components:
- The 3 filled circles
- The ring in the lower right hand side
- The circle connectors
- The outer perimeter
We will define a formula for each of these and combine all four (4) formulas into a single one for the final rendering.
data:image/s3,"s3://crabby-images/1fdc2/1fdc27c78dd8615b6164a065fb7795f63baf4071" alt=""
data:image/s3,"s3://crabby-images/c2386/c23864deb7e208afda01370f668373c24049d67b" alt=""
Starting it all off with Pygfx
We will want to create our WebGPU canvas to be able to start drawing. For this we will create a small script that places a single point on the scene and use Pygfx markers to define the shape. Because we want to be able to interact with the logo, we will use a few of Pygfx’s features to allow us to get closer and further away from the logo. This is the same technology that allows you to zoom in and out of the massive images captured by Ramona’s MCAM®!
data:image/s3,"s3://crabby-images/71452/7145225461da823767f9d0b2a8b58ec6c5d313c4" alt="".webp)
Python
data:image/s3,"s3://crabby-images/3a41c/3a41c36732cf02edda5bfa015877bcd3508b5961" alt=""
In the code above, we’ll start to specify the parameter custom_sdf with the mathematical function we will construct.
The three circles
data:image/s3,"s3://crabby-images/6474f/6474f02bb68a96f3e9c207c5d9f0fb0d1b19d6ab" alt=""
where p
is the position on the screen that we are trying to paint, and r
is the radius of our circle. In English, this gives us an equation that takes on a value of 0 exactly at the radius of the circle, negative values for the point inside the radius, and positive values for points outside the radius. If you want to experience what the SDF looks like in an interactive fashion, I can point you to IQ’s online shader website (link here) called where draws the SDF of a perfect circle. The blue regions denote the “negative” distances, while the orange denotes the “positive” distances. By clicking on the image, a circle with radius equal to the absolute distance will be drawn.
To draw all 3 circles, we will use a key concept behind signed distance field for non-intersecting shapes: the signed distance field can be computed by taking the minimum of the signed distance of each independent shape. To write this formula for Pygfx to use, we must write a short piece of code in WGSL, a modern programming language for GPU rendering, that expresses the mathematics required to evaluate the SDF above. In Python the full code used to draw our three circles looks like:
Python
data:image/s3,"s3://crabby-images/4bf88/4bf885603dcbf9b2cdcc6e07c354612f1182b5ef" alt=""
data:image/s3,"s3://crabby-images/5a9e9/5a9e9e9fa1b6a2f976b8053950058b127bc86693" alt=""
For brevity, we’ll start to omit the python code portion in future examples and only include the contents of the custom_sdf string variable which defines our SDF.
The ring
To define a ring, we’ll take the formula for a circle and make it an annular shape of thickness t.
data:image/s3,"s3://crabby-images/8de83/8de83fd160a7be2ef02b0d97f3c56e8da14a1524" alt=""
We take care to subtract the thickness from the radius of this shape so that the outer edge of the ring has the same size as the previous three circles.
let coord_use = coord / size * 2;
let thickness = 0.1;
let thickness_half = 0.05;
let radius = 1. / 3.;
let gap = 0.05;
let circle_offset = radius + gap;
let bottom_right_circle = length(coord_use - vec2<f32>(circle_offset, -circle_offset)) - (radius - thickness_half);
let bottom_right_ring = abs(bottom_right_circle) - thickness_half;
return bottom_right_ring * size;
data:image/s3,"s3://crabby-images/ee8c8/ee8c8d8e2ecebad67e8772fb50247e915997a846" alt=""
The circle connectors
The small circle connectors in our logo were quite challenging for me to define mathematically. In one approximation, we can take the connections to be the empty spaces between circles that are tangent to the original 3 circles. My inspiration came from looking at coins on my desk.the original proportions
data:image/s3,"s3://crabby-images/7bffc/7bffc93546fb3ef377f7370a322de2fb00b850db" alt=""
If the coins are of correct size, and appropriately arranged, then they start to resemble the 3 connected circles. However, before we proceed, we want to find a mathematical relationship between the circles as a function of their radius, the gap between adjacent circles, and the width of the connector at its narrowest point. This took me back to high school algebra!
data:image/s3,"s3://crabby-images/52a73/52a73e9656d675da8bda5551d606a24bc59a6316" alt=""
Taking our two formulas:
data:image/s3,"s3://crabby-images/75b87/75b87537e40cdac52805237006b7b6547c0b033d" alt=""
Along with the identity
data:image/s3,"s3://crabby-images/ee2be/ee2bea4892643acb9dabf7f56d41582f4c724248" alt=""
We find the final relationship
data:image/s3,"s3://crabby-images/74e9d/74e9da77d404e0072e118f0e0c6f841e95594544" alt=""
We can use this to draw an attempt of our concept on the screen.
data:image/s3,"s3://crabby-images/a5d51/a5d51e84a6bb4f3336f156a0aacefaed336bb8e9" alt=""
data:image/s3,"s3://crabby-images/7d5c0/7d5c05519e6385d08e2bb54208df7be24b04b5c8" alt="".png)
At Ramona, we care deeply about how images are captured and displayed. Over the last few years, we’ve leveraged and contributed to open source software mostly by opening bug reports, suggesting fixes, and helping package scientific software for the community at large.
Today, we celebrate our first sponsorship of an open source project, the Pygfx project! Pygfx, pronounced “py-graphics”, is a Python library with the purpose to bring powerful and reliable visualization to the scientific python ecosystem. If you are interested in python and image visualization, we recommend you go check out the website. At Ramona, we leverage its capabilities to display our Gigapixel data right to your screen allowing you to interact with every single picture captured by the MCAM®.
As a celebration, we choose to study a common problem relatable to anybody trying to visualize content to a screen and realize that as they enlarge their image, their data starts to look blurry. In this blog post, we decide to focus on one of the simplest images, that of a logo, our logo!!! As we alluded to, if you try to zoom into our 128 x 128 pixel logo, you’ll quickly see that the quality appears to degrade!!
data:image/s3,"s3://crabby-images/358b3/358b3b1fbf2530052a8ed4900dbe6350eeac320c" alt=""
In fact, the quality doesn’t degrade, but rather the finite resolution of the underlying image shows its limits! However, at Ramona, our mission is to make you forget about pixels and bits and let you focus on biology, science, and discovery! One popular solution is to use vector graphics to draw in a scalable fashion — our marketing team loves to use those to make our swag look amazing!
In order to celebrate our new sponsorship of the Pygfx projects, we are deciding to redraw the logo you see on the MCAM® software in a pixel perfect fashion using some of the latest capabilities of the library!
data:image/s3,"s3://crabby-images/358b3/358b3b1fbf2530052a8ed4900dbe6350eeac320c" alt=""
The before and after are quite stark, read along if you want to learn how we achieved this!
How many pixels should your logo have?
When acquiring microscopy data, one can often estimate the number of pixels required for a given experiment by considering the size of the biology of interest. However when drawing a picture on a monitor, one cannot always determine the resolution of the display used to render this image. Many times, one has to guess an adequate resolution for an image, save the image in a PNG format to avoid compression artifacts, and hope that most users don’t zoom in too much to notice any inconsistencies at edges. However, with 4k and 8k screens, scientists can often zoom much more than they used to! It would be nice to have infinite pixels!!!! But for that we must decompose the logo into a mathematical formula that can be computed based on the final resolution.
In computer graphics, one strategy often used is based on signed distance field, SDF for short. The idea behind an SDF is to provide a formula for the distance from any pixel to the edge of the shape. If one can find a closed form formula, then the drawing can have “infinite” resolution giving us sharp edges. Pixels that are “inside the shape” are tagged as “negative” by convention. This allows one to color pixels with a negative distance with one color, and pixels with a positive distance in another. For more information we recommend users read Rougier’s Paper on creating pixel perfect markers for scientific communication on the GPU which can be retrieved on this link as well as some great posts and videos by Inigo Quilez found here.
The examples below use the Pygfx engine developed by Almar Klein and the rest of the Pygfx team!
The Ramona Logo
The first step is to decompose the Ramona Logo into a series of primitive shapes for which we can write down a formula.
We will break up the logo into 4 different components:
- The 3 filled circles
- The ring in the lower right hand side
- The circle connectors
- The outer perimeter
We will define a formula for each of these and combine all four (4) formulas into a single one for the final rendering.
data:image/s3,"s3://crabby-images/1fdc2/1fdc27c78dd8615b6164a065fb7795f63baf4071" alt=""
data:image/s3,"s3://crabby-images/c2386/c23864deb7e208afda01370f668373c24049d67b" alt=""
Starting it all off with Pygfx
We will want to create our WebGPU canvas to be able to start drawing. For this we will create a small script that places a single point on the scene and use Pygfx markers to define the shape. Because we want to be able to interact with the logo, we will use a few of Pygfx’s features to allow us to get closer and further away from the logo. This is the same technology that allows you to zoom in and out of the massive images captured by Ramona’s MCAM®!
data:image/s3,"s3://crabby-images/71452/7145225461da823767f9d0b2a8b58ec6c5d313c4" alt="".webp)
Python
data:image/s3,"s3://crabby-images/3a41c/3a41c36732cf02edda5bfa015877bcd3508b5961" alt=""
In the code above, we’ll start to specify the parameter custom_sdf with the mathematical function we will construct.
The three circles
data:image/s3,"s3://crabby-images/6474f/6474f02bb68a96f3e9c207c5d9f0fb0d1b19d6ab" alt=""
where p
is the position on the screen that we are trying to paint, and r
is the radius of our circle. In English, this gives us an equation that takes on a value of 0 exactly at the radius of the circle, negative values for the point inside the radius, and positive values for points outside the radius. If you want to experience what the SDF looks like in an interactive fashion, I can point you to IQ’s online shader website (link here) called where draws the SDF of a perfect circle. The blue regions denote the “negative” distances, while the orange denotes the “positive” distances. By clicking on the image, a circle with radius equal to the absolute distance will be drawn.
To draw all 3 circles, we will use a key concept behind signed distance field for non-intersecting shapes: the signed distance field can be computed by taking the minimum of the signed distance of each independent shape. To write this formula for Pygfx to use, we must write a short piece of code in WGSL, a modern programming language for GPU rendering, that expresses the mathematics required to evaluate the SDF above. In Python the full code used to draw our three circles looks like:
Python
data:image/s3,"s3://crabby-images/4bf88/4bf885603dcbf9b2cdcc6e07c354612f1182b5ef" alt=""
data:image/s3,"s3://crabby-images/5a9e9/5a9e9e9fa1b6a2f976b8053950058b127bc86693" alt=""
For brevity, we’ll start to omit the python code portion in future examples and only include the contents of the custom_sdf string variable which defines our SDF.
The ring
To define a ring, we’ll take the formula for a circle and make it an annular shape of thickness t.
data:image/s3,"s3://crabby-images/8de83/8de83fd160a7be2ef02b0d97f3c56e8da14a1524" alt=""
We take care to subtract the thickness from the radius of this shape so that the outer edge of the ring has the same size as the previous three circles.
let coord_use = coord / size * 2;
let thickness = 0.1;
let thickness_half = 0.05;
let radius = 1. / 3.;
let gap = 0.05;
let circle_offset = radius + gap;
let bottom_right_circle = length(coord_use - vec2<f32>(circle_offset, -circle_offset)) - (radius - thickness_half);
let bottom_right_ring = abs(bottom_right_circle) - thickness_half;
return bottom_right_ring * size;
data:image/s3,"s3://crabby-images/ee8c8/ee8c8d8e2ecebad67e8772fb50247e915997a846" alt=""
The circle connectors
The small circle connectors in our logo were quite challenging for me to define mathematically. In one approximation, we can take the connections to be the empty spaces between circles that are tangent to the original 3 circles. My inspiration came from looking at coins on my desk.the original proportions
data:image/s3,"s3://crabby-images/7bffc/7bffc93546fb3ef377f7370a322de2fb00b850db" alt=""
If the coins are of correct size, and appropriately arranged, then they start to resemble the 3 connected circles. However, before we proceed, we want to find a mathematical relationship between the circles as a function of their radius, the gap between adjacent circles, and the width of the connector at its narrowest point. This took me back to high school algebra!
data:image/s3,"s3://crabby-images/52a73/52a73e9656d675da8bda5551d606a24bc59a6316" alt=""
Taking our two formulas:
data:image/s3,"s3://crabby-images/75b87/75b87537e40cdac52805237006b7b6547c0b033d" alt=""
Along with the identity
data:image/s3,"s3://crabby-images/ee2be/ee2bea4892643acb9dabf7f56d41582f4c724248" alt=""
We find the final relationship
data:image/s3,"s3://crabby-images/74e9d/74e9da77d404e0072e118f0e0c6f841e95594544" alt=""
We can use this to draw an attempt of our concept on the screen.
data:image/s3,"s3://crabby-images/a5d51/a5d51e84a6bb4f3336f156a0aacefaed336bb8e9" alt=""