Skip to main content

graphics - How to combine ArrayPlots?


Define:


fat    = ArrayPlot[RandomReal[1, {20, 10}], ColorFunction->"Rainbow"];

skinny = ArrayPlot[RandomReal[1, {20, 1}], ColorFunction->"Rainbow"];

What must one do to display fat and skinny side-by-side, so that



  • the bottom and top edges of fat line up vertically with the bottom and top edges (respectively) of skinny (IOW, both fat and skinny appear as having the same height)

  • fat appears 10x wider than skinny


(GraphicsRow[{fat, skinny}] produces a ridiculous monster featuring a tiny fat next to a gigantic skinny. Pretty much everything else I've tried produces the same nonsense.)



Answer



GraphicsGrid makes too many assumptions about similarity of sizes. For this kind of case, use Grid. I'd also recommend using the PlotRangePadding option in fat so that the two are exactly the same height. I'm not sure why the one-column case skinny doesn't use plot range padding, but it seems to be the way it works. (If you turn off the frame using Frame->False, you do need to turn off PlotRangePadding for both graphics.)



fat = ArrayPlot[RandomReal[1, {20, 10}], ColorFunction -> "Rainbow", 
PlotRangePadding -> 0];
skinny = ArrayPlot[RandomReal[1, {20, 1}], ColorFunction -> "Rainbow"];

Grid[{{fat, skinny}}, Spacings -> -0.2]

aligned perfectly


This is what it looks like without the Spacings option.


just a bit of space





As for producing the desired plot type, MatrixPlot does the job with just a few custom option values. Consider the following example. First, I create some fake data:


testdata = 
RandomVariate[TriangularDistribution[{0, 1}, 0.2], {30, 30}];

Some custom ticks will be handy, too. Notice that I use the list-valued third position to specify that the ticks will be zero length.


xtix = Array[{#, Rotate["E" <> ToString[#], 3 Pi/2], {0, 0}} &, {30}];
ytix = Array[{#, "F" <> ToString[#], {0, 0}} &, {30}];

To get the look shown in the example plot in your question, you need an appropriate usage of Blend, together with the ColorFunctionScaling->False option (if the data are already scaled between zero and 1, or True if they aren't). Notice the form of Blend that sets particular colors to particular values. This is in the documentation but little used.


MatrixPlot[testdata, ColorFunctionScaling -> False, 

ColorFunction -> (Blend[{{0, Red}, {0.2, Black}, {1, Green}}, #] &),
PlotRangePadding -> 0,
FrameTicks -> {{ytix, None}, {xtix, None}}]

enter image description here


It would be easy to automate this further, either by building this into a custom function, or using SetOptions to set some of those options to MatrixPlot by default. However, I can imagine that you might want to tweak the ColorFunction depending on how your data are distributed.


As for building up arrays of these graphic elements, you need to investigate the ImageSize and AspectRatio options, as well as the original question which your previous question was a duplicate of. Once that is done, it is easy to put together something that looks a lot like the Wikipedia example.


First, define a custom color Blend:


myblend = (Blend[{{-1, Red}, {0, Black}, {0.5, Black}, {1, Green}}, #] &);


Then load in the HierarchicalClustering package to access the DendrogramPlot function. We can then do something like this (I used a different sized random data set to the previous case):


Grid[{{DendrogramPlot[Transpose@testdata, AspectRatio -> 1/5, 
ImageSize -> 240, ImagePadding -> {{15, 0}, {0, 0}}],
Null}, {MatrixPlot[testdata, ColorFunctionScaling -> False,
AspectRatio -> 1, ColorFunction -> myblend,
FrameStyle -> AbsoluteThickness[0], PlotRangePadding -> 0,
FrameTicks -> {{ytix, None}, {xtix, None}},
BaseStyle -> {FontFamily -> "Helvetica Neue", FontSize -> 8},
ImageSize -> 250],
DendrogramPlot[testdata, AspectRatio -> 5, Orientation -> Right,

ImageSize -> 45, ImagePadding -> {{0, 0}, {20, 0}}]}},
Spacings -> {0, -0.2}]

enter image description here


To summarize, yes, I do think that Mathematica can do this, and in a relatively straightforward way, too. If it is not specifically included as a custom function in the application, it is probably because there are so many different fields in the world that use Mathematica, and they can't build it all into the main application. I agree that it would be nice if there were more user-distributed custom packages around. This has already been discussed on the site but as yet not much has come of it.


Comments

Popular posts from this blog

front end - keyboard shortcut to invoke Insert new matrix

I frequently need to type in some matrices, and the menu command Insert > Table/Matrix > New... allows matrices with lines drawn between columns and rows, which is very helpful. I would like to make a keyboard shortcut for it, but cannot find the relevant frontend token command (4209405) for it. Since the FullForm[] and InputForm[] of matrices with lines drawn between rows and columns is the same as those without lines, it's hard to do this via 3rd party system-wide text expanders (e.g. autohotkey or atext on mac). How does one assign a keyboard shortcut for the menu item Insert > Table/Matrix > New... , preferably using only mathematica? Thanks! Answer In the MenuSetup.tr (for linux located in the $InstallationDirectory/SystemFiles/FrontEnd/TextResources/X/ directory), I changed the line MenuItem["&New...", "CreateGridBoxDialog"] to read MenuItem["&New...", "CreateGridBoxDialog", MenuKey["m", Modifiers-...

How to thread a list

I have data in format data = {{a1, a2}, {b1, b2}, {c1, c2}, {d1, d2}} Tableform: I want to thread it to : tdata = {{{a1, b1}, {a2, b2}}, {{a1, c1}, {a2, c2}}, {{a1, d1}, {a2, d2}}} Tableform: And I would like to do better then pseudofunction[n_] := Transpose[{data2[[1]], data2[[n]]}]; SetAttributes[pseudofunction, Listable]; Range[2, 4] // pseudofunction Here is my benchmark data, where data3 is normal sample of real data. data3 = Drop[ExcelWorkBook[[Column1 ;; Column4]], None, 1]; data2 = {a #, b #, c #, d #} & /@ Range[1, 10^5]; data = RandomReal[{0, 1}, {10^6, 4}]; Here is my benchmark code kptnw[list_] := Transpose[{Table[First@#, {Length@# - 1}], Rest@#}, {3, 1, 2}] &@list kptnw2[list_] := Transpose[{ConstantArray[First@#, Length@# - 1], Rest@#}, {3, 1, 2}] &@list OleksandrR[list_] := Flatten[Outer[List, List@First[list], Rest[list], 1], {{2}, {1, 4}}] paradox2[list_] := Partition[Riffle[list[[1]], #], 2] & /@ Drop[list, 1] RM[list_] := FoldList[Transpose[{First@li...

plotting - How to draw lines between specified dots on ListPlot?

I would like to create a plot where I have unconnected dots and some connected. So far, I have figured out how to draw the dots. My code is the following: ListPlot[{{1, 1}, {2, 2}, {3, 3}, {4, 4}, {1, 4}, {2, 5}, {3, 6}, {4, 7}, {1, 7}, {2, 8}, {3, 9}, {4, 10}, {1, 10}, {2, 11}, {3, 12}, {4,13}, {2.5, 7}}, Ticks -> {{1, 2, 3, 4}, None}, AxesStyle -> Thin, TicksStyle -> Directive[Black, Bold, 12], Mesh -> Full] I have thought using ListLinePlot command, but I don't know how to specify to the command to draw only selected lines between the dots. Do have any suggestions/hints on how to do that? Thank you. Answer One possibility would be to use Epilog with Line : ListPlot[ {{1, 1}, {2, 2}, {3, 3}, {4, 4}, {1, 4}, {2, 5}, {3, 6}, {4, 7}, {1, 7}, {2, 8}, {3, 9}, {4, 10}, {1, 10}, {2, 11}, {3, 12}, {4, 13}, {2.5, 7}}, Ticks -> {{1, 2, 3, 4}, None}, AxesStyle -> Thin, TicksStyle -> Directive[Black, Bold, 12], Mesh -> Full, Epilog -> { Line[ ...