Skip to main content

image processing - Color scheme used by Colorize?


The Colorize function is often used to display the results produced by MorphologicalComponents:


i = ImageTake[
ExampleData[{"TestImage", "Mandrill"}], {1, -1, 10}, {1, -1, 10}];
b = Binarize[i, .5];
mc = MorphologicalComponents[b];
Colorize[mc]


mandrill taken


The colors represent the integer index values of each foreground component, so the magenta area is 2, the black background is 0, 42 is browney-orange, 19 is violet, and - well, my question is, what is this color scheme?


I'm working on images like the following, to try to understand morphological images, but a legend-type approach might be better:


Show[
ArrayPlot[mc,
Epilog -> {
MapIndexed[{Text[Style[#1, 10, Gray],
Reverse[#2 - {.5, .5}]]} &,
Reverse[mc],

{2}]}, PlotRangePadding -> 0],
Colorize[mc]
]

labeled



Thanks to stefan's amazing detective work, I've got a way to list the colors, although this isn't available until you run Colorize once or set up the colors another way first:


colorizeColors[n_] := 
Part[RGBColor[N@#1/256, N@#2/256, N@#3/256] & @@
Image`ColorOperationsDump`hashcolor[#1] & /@ Range[1, 100], n]

Graphics[Table[{colorizeColors[x], Rectangle[{x, 0}, {x + 1, 1}],
White, Text[x, {x + 0.5, -.5}]}, {x, 1, 30}], Background -> Black,
ImageSize -> 800]

colorize legend



Answer



How does Colorize work


Colorize does use DarkRainbow as a default color scheme.


What Colorize does first is setting up a hashtable with RGB values from DarkRainbow through DataPaclets`ColorDataDump`


The next operation is setting up a list of indices from the original image, iterating through the internal hash-table to fill a list of color triplets and call an internal mapping function, to join it all together.



I'll show here an example for your result from your MorphologicalComponents.


1) create the indices table


indices = Cases[Tally[Flatten[mc]][[All, 1]], _?Positive]  

2) allocate a color list


colors = ConstantArray[0, {Max[0, indices], 3}]

3) fill the colors list with the hashcolor values


((colors[[#]] = Quiet[Image`ColorOperationsDump`hashcolor[#1]]) &) /@ indices;


4) join it all together to create an image


Image[Image`ColorOperationsDump`mapFunc2D1[mc, colors, indices],
Byte, ColorSpace -> "RGB"]

enter image description here


There is one problem with this approach. I didn't find out, how the internal hash-table is generated with the values from DarkRainbow. What that means is, that if my approach should succeed, there must've been an execution of Colorize before.


Agreed, that setting up the internal hash-table is a critical step in the coloring process of Colorize, it does by all means solve your legend problem.


Update: Setting up the internal hash-table


I wrote before that I couldn't find out how the internal hash-table was generated. Well, now I know how:


Caution! This makes heavy use of undocumented functionality and may be deprecated.



The internal data-structures are set up from the first time any function is called with Colorize and/or HighlighImage functionality.


What they do is, they call but do not evaluate them. This seems to be enough in order to initialize the internal data structures, they just construct the object.


System`Dump`AutoLoad[Hold[Colorize], 
Hold[Colorize, HighlightImage,
Image`ColorOperationsDump`colorspaceToColor,
Image`ColorOperationsDump`colorToColorspace],
"ImageProcessing`ColorOpsColorize`"
]

So if you evaluate this code block everything else I described works just perfectly.



Update 2: Cleaner Initialisation


Although the above code block works I didn't like it. There must be another way to initialise the internal data-structures for Colorize. Something that does not rely on undocumented functionality which may change in future versions.


It seems to me that a load of Mathematica implementation is loaded from a dump file, the first time it gets used.


Let's check that for Colorize:


ClearAttributes[Colorize, ReadProtected]
Information[Colorize]

This sequence yields the same code that I listed above. Obviously this is the initialisation sequence specific for Colorize (and HighlighImage).


In C++/Java there is something called a Constructor. So my question was is there a way to avoid the above ugly code block and simply initialize the Colorize functionality prior any usage, meaning calling a Constructor.


Actually it is. Evaluate!



What this means is. You can completely replace the above System`Dump sequence with a simple call to Evaluate in order to initialise the internal data-structures:


Evaluate[Colorize]; (*construct the object and initialise internal data-structures*)

Done. As long as WRI continues with their load on demand paradigm this will hold true. Btw. this works on version 8 as well.


Comments

Popular posts from this blog

plotting - Plot 4D data with color as 4th dimension

I have a list of 4D data (x position, y position, amplitude, wavelength). I want to plot x, y, and amplitude on a 3D plot and have the color of the points correspond to the wavelength. I have seen many examples using functions to define color but my wavelength cannot be expressed by an analytic function. Is there a simple way to do this? Answer Here a another possible way to visualize 4D data: data = Flatten[Table[{x, y, x^2 + y^2, Sin[x - y]}, {x, -Pi, Pi,Pi/10}, {y,-Pi,Pi, Pi/10}], 1]; You can use the function Point along with VertexColors . Now the points are places using the first three elements and the color is determined by the fourth. In this case I used Hue, but you can use whatever you prefer. Graphics3D[ Point[data[[All, 1 ;; 3]], VertexColors -> Hue /@ data[[All, 4]]], Axes -> True, BoxRatios -> {1, 1, 1/GoldenRatio}]

plotting - Filling between two spheres in SphericalPlot3D

Manipulate[ SphericalPlot3D[{1, 2 - n}, {θ, 0, Pi}, {ϕ, 0, 1.5 Pi}, Mesh -> None, PlotPoints -> 15, PlotRange -> {-2.2, 2.2}], {n, 0, 1}] I cant' seem to be able to make a filling between two spheres. I've already tried the obvious Filling -> {1 -> {2}} but Mathematica doesn't seem to like that option. Is there any easy way around this or ... Answer There is no built-in filling in SphericalPlot3D . One option is to use ParametricPlot3D to draw the surfaces between the two shells: Manipulate[ Show[SphericalPlot3D[{1, 2 - n}, {θ, 0, Pi}, {ϕ, 0, 1.5 Pi}, PlotPoints -> 15, PlotRange -> {-2.2, 2.2}], ParametricPlot3D[{ r {Sin[t] Cos[1.5 Pi], Sin[t] Sin[1.5 Pi], Cos[t]}, r {Sin[t] Cos[0 Pi], Sin[t] Sin[0 Pi], Cos[t]}}, {r, 1, 2 - n}, {t, 0, Pi}, PlotStyle -> Yellow, Mesh -> {2, 15}]], {n, 0, 1}]

plotting - Mathematica: 3D plot based on combined 2D graphs

I have several sigmoidal fits to 3 different datasets, with mean fit predictions plus the 95% confidence limits (not symmetrical around the mean) and the actual data. I would now like to show these different 2D plots projected in 3D as in but then using proper perspective. In the link here they give some solutions to combine the plots using isometric perspective, but I would like to use proper 3 point perspective. Any thoughts? Also any way to show the mean points per time point for each series plus or minus the standard error on the mean would be cool too, either using points+vertical bars, or using spheres plus tubes. Below are some test data and the fit function I am using. Note that I am working on a logit(proportion) scale and that the final vertical scale is Log10(percentage). (* some test data *) data = Table[Null, {i, 4}]; data[[1]] = {{1, -5.8}, {2, -5.4}, {3, -0.8}, {4, -0.2}, {5, 4.6}, {1, -6.4}, {2, -5.6}, {3, -0.7}, {4, 0.04}, {5, 1.0}, {1, -6.8}, {2, -4.7}, {3, -1....