Skip to main content

graphics - Consistent Plot Styles across multiple MMA files and data sets


I am starting to write a thesis for which I want to use MMA for all my plotting needs, mostly because a lot of the basic analysis has been done there. To ensure that I am applying styles consistently (plot colors, image size, label size, etc.), I need a method which I can use across multiple files.


There are several types of plots that I will need:



  1. ListPlot / Plot

  2. ListDensityPlot / DensityPlot

  3. ListContourPlot / ContourPlot



I can think of several way of doing this:



  1. Define my own functions thPlot, thListPlot, etc which have my default styles applied

  2. Define my own style options into a variable, and use some combination of FilterRules and Options to make sure that options from that variable are applied to the right type of plot

  3. Set default options using Default for the different types of plots.


I am assuming that each of my datasets (which may have one or more plots) will be self-contained mathematica files that I can re-run. So, if I decide to change the style in some central location (via one of the methods above) I can re-run the MMA files and get the plots in the new style.


What have people done in the past?



Answer



For an example of plot option customization that I am still rather proud of please see:

How to change the default ColorData used in Mathematica's Plot?


For general customization I think you already outlined some good options. I personally favor the custom function method for maximum control. Another, perhaps cleaner method that I crudely copy from Leonid works well if you can pool options for all plot types in one list ($myoptions):


Update: changed withOptions to use the "injector pattern" so as not to disrupt functions if withOptions is used indiscriminately (withOptions[ 1 + 1 ]).


$myoptions = {Filling -> {1 -> {{2}, {Red, Purple}}}};

SetAttributes[withOptions, HoldFirst]

withOptions[head_Symbol[body__]] :=
FilterRules[$myoptions, Options[head]] /. {opts___} :> head[body, opts]


withOptions @ Plot[{Sin @ Log @ x, Cos[3 x]}, {x, 0, Pi}]

Mathematica graphics




Here I posted another method in answer to specifying plot ranges, but the method is very general and can be used for any options or parameters. It works by creating an UpValue assignment for the symbol given that applies to any head. This could of course be further restricted to apply to only one head or a list of heads. One could also include FilterRules as shown in the code above, depending on the effect desired. For example:


SetAttributes[setOpts, HoldAllComplete]

Quiet[
setOpts[s_Symbol, pat_: _, spec__?OptionQ] :=
s /: (h : pat)[pre__, s, post___] :=

FilterRules[{spec}, Options[h]] /. {opts___} :> h[pre, opts, post],
Optional::"opdef"
]

Now a single option as used above is assigned to op1:


setOpts[op1, Filling -> {1 -> {{2}, {Red, Purple}}}]

And used in Plot:


 Plot[{Sin @ Log @ x, Cos[3 x]}, {x, 0, Pi}, op1]


Mathematica graphics


Alternatively the options can be restricted to a particular head or head pattern. Here a different set of options, also named op1, is defined only for ParametricPlot:


setOpts[op1, ParametricPlot, Frame -> True, PlotStyle -> Thick, Axes -> None, 
MeshShading -> {Red, Blue}, Mesh -> 15]

ParametricPlot[{{2 Cos[t], 2 Sin[t]}, {2 Cos[t], Sin[t]}}, {t, 0, 2 Pi}, op1]

enter image description here


Using op1 in Plot will still produce the result shown earlier.


You could also use Alternatives in the pattern, e.g: Plot | ListPlot | Histogram or more complication patterns with conditions, etc.



Comments

Popular posts from this blog

mathematical optimization - Minimizing using indices, error: Part::pkspec1: The expression cannot be used as a part specification

I want to use Minimize where the variables to minimize are indices pointing into an array. Here a MWE that hopefully shows what my problem is. vars = u@# & /@ Range[3]; cons = Flatten@ { Table[(u[j] != #) & /@ vars[[j + 1 ;; -1]], {j, 1, 3 - 1}], 1 vec1 = {1, 2, 3}; vec2 = {1, 2, 3}; Minimize[{Total@((vec1[[#]] - vec2[[u[#]]])^2 & /@ Range[1, 3]), cons}, vars, Integers] The error I get: Part::pkspec1: The expression u[1] cannot be used as a part specification. >> Answer Ok, it seems that one can get around Mathematica trying to evaluate vec2[[u[1]]] too early by using the function Indexed[vec2,u[1]] . The working MWE would then look like the following: vars = u@# & /@ Range[3]; cons = Flatten@{ Table[(u[j] != #) & /@ vars[[j + 1 ;; -1]], {j, 1, 3 - 1}], 1 vec1 = {1, 2, 3}; vec2 = {1, 2, 3}; NMinimize[ {Total@((vec1[[#]] - Indexed[vec2, u[#]])^2 & /@ R...

functions - Get leading series expansion term?

Given a function f[x] , I would like to have a function leadingSeries that returns just the leading term in the series around x=0 . For example: leadingSeries[(1/x + 2)/(4 + 1/x^2 + x)] x and leadingSeries[(1/x + 2 + (1 - 1/x^3)/4)/(4 + x)] -(1/(16 x^3)) Is there such a function in Mathematica? Or maybe one can implement it efficiently? EDIT I finally went with the following implementation, based on Carl Woll 's answer: lds[ex_,x_]:=( (ex/.x->(x+O[x]^2))/.SeriesData[U_,Z_,L_List,Mi_,Ma_,De_]:>SeriesData[U,Z,{L[[1]]},Mi,Mi+1,De]//Quiet//Normal) The advantage is, that this one also properly works with functions whose leading term is a constant: lds[Exp[x],x] 1 Answer Update 1 Updated to eliminate SeriesData and to not return additional terms Perhaps you could use: leadingSeries[expr_, x_] := Normal[expr /. x->(x+O[x]^2) /. a_List :> Take[a, 1]] Then for your examples: leadingSeries[(1/x + 2)/(4 + 1/x^2 + x), x] leadingSeries[Exp[x], x] leadingSeries[(1/x + 2 + (1 - 1/x...

What is and isn't a valid variable specification for Manipulate?

I have an expression whose terms have arguments (representing subscripts), like this: myExpr = A[0] + V[1,T] I would like to put it inside a Manipulate to see its value as I move around the parameters. (The goal is eventually to plot it wrt one of the variables inside.) However, Mathematica complains when I set V[1,T] as a manipulated variable: Manipulate[Evaluate[myExpr], {A[0], 0, 1}, {V[1, T], 0, 1}] (*Manipulate::vsform: Manipulate argument {V[1,T],0,1} does not have the correct form for a variable specification. >> *) As a workaround, if I get rid of the symbol T inside the argument, it works fine: Manipulate[ Evaluate[myExpr /. T -> 15], {A[0], 0, 1}, {V[1, 15], 0, 1}] Why this behavior? Can anyone point me to the documentation that says what counts as a valid variable? And is there a way to get Manpiulate to accept an expression with a symbolic argument as a variable? Investigations I've done so far: I tried using variableQ from this answer , but it says V[1...