Skip to main content

plotting - Can I use a custom (Frame)Ticks specification to define the PlotRange?


The documentation for Ticks and FrameTicks shows how one can define a function for custom ticks or frame ticks, for example:


niceTicks[min_, max_, n_: 7] := FindDivisions[{min, max}, n]

After creating some example data


fakedata1 = FoldList[0.99 #1 + #2 &, 1., 
RandomVariate[NormalDistribution[0, 1], {100}]];

We can see that FrameTicks specifications will allow for functions, and will automatically pass the minimum and maximum of the data set as arguments.



ListLinePlot[fakedata1, Frame -> True, PlotRange -> All, PlotRangePadding -> 0, 
FrameTicks -> {{niceTicks, None}, {Automatic, None}}]

enter image description here


Similarly one could use:


ListLinePlot[fakedata1, Frame -> True, PlotRange -> All, PlotRangePadding -> 0, 
FrameTicks -> {{niceTicks[##, 10] &, None}, {Automatic, None}}]

enter image description here


As an aside, when using data points specified in terms of x,y coordinates, as in DateListPlot and related functions, the FrameTicks specification will correctly pass the minimum and maximum of the x or y coordinates as appropriate, which is pretty cool.



My question is how one can link the PlotRange of the plot to the ticks determined by such a custom function. In this particular instance, the FindDivisions function called by niceTicks ranges from -2 to 10, but the plot only takes up as much space as the data require, plus any PlotRangePadding.


niceTicks[Min@fakedata1, Max@fakedata1]
(* {-2, 0, 2, 4, 6, 8, 10} *)

If I want the plot to have ticks effectively at the very top and bottom of the frame like this, using all the ticks output by my niceTicks function, I need to specify the PlotRange explicitly using the same niceTicks function:


ListLinePlot[fakedata1, Frame -> True, 
PlotRange -> niceTicks[Min@fakedata1, Max@fakedata1][[{1, -1}]],
PlotRangePadding -> 0, FrameTicks -> {{niceTicks, None}, {Automatic, None}}]

enter image description here



I'm sure that calling an auxiliary function twice like this doesn't add a lot of overhead, but it does require getting explicit minima and maxima from the data, which is a bit clunky when you consider multiple series, dated data for DateListPlots etc.


Is there a way to pass the maximum and minimum tick values determined in the Ticks or FrameTicks option expression to the PlotRange option automatically? It would be more elegant code with fewer special cases to code for. I'm building this into a suite of custom plotting functions, where a tick specification like the last picture shown is a requirement.


It would also be nice to pass the same tickmarks to GridLines somehow, so that Automatic gridlines automatically matched up with the specified ticks.



Answer



I thought something like what @Mr Wizard suggested might work but it seems that PlotRange is called earlier in the evaluation sequence. You can however match the GridLines ok:


niceTicks[min_, max_, n_: 7] := (yRange = {min, max}; 
FindDivisions[{min, max}, n, Method -> "ExtendRange"])

Dynamic@ListLinePlot[fakedata1,
FrameTicks -> {{niceTicks[##, 3] &, None}, {Automatic, None}},

GridLines -> {None, niceTicks[##, 3] &}, Frame -> True,
PlotRangePadding -> 0, PlotLabel -> yRange]

Unfortunately using PlotRange->{Automatic,yRange} doesn't work.


Comments

Popular posts from this blog

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...

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...

How to remap graph properties?

Graph objects support both custom properties, which do not have special meanings, and standard properties, which may be used by some functions. When importing from formats such as GraphML, we usually get a result with custom properties. What is the simplest way to remap one property to another, e.g. to remap a custom property to a standard one so it can be used with various functions? Example: Let's get Zachary's karate club network with edge weights and vertex names from here: http://nexus.igraph.org/api/dataset_info?id=1&format=html g = Import[ "http://nexus.igraph.org/api/dataset?id=1&format=GraphML", {"ZIP", "karate.GraphML"}] I can remap "name" to VertexLabels and "weights" to EdgeWeight like this: sp[prop_][g_] := SetProperty[g, prop] g2 = g // sp[EdgeWeight -> (PropertyValue[{g, #}, "weight"] & /@ EdgeList[g])] // sp[VertexLabels -> (# -> PropertyValue[{g, #}, "name"]...