Skip to main content

dynamic - How to have and update a varying number of variables in Manipulate?


The OP in the question, How to return the unevaluated variable names from a list of variables which have been declared, raised a side issue in an edit, whose solution seemed far afield of the main question. I thought I would post my take on the question and self-answer, too.



The question is how to have an indeterminate number of variables in a Manipulate, each controlled by a slider and be able to update these values in the code of the Manipulate, say, by some numerical routine such as FindFit. There are several Q&A on a coding a variable number of variables (see below), but none I could find also addressed updating the indefinite number of controls.


Here is a example, modified from the linked question, that has a variable number of sliders. The sliders appear to work, but the button that executes FindFit causes errors and the model[] function is not plotted, even on the initial evaluation.


ClearAll[testplotfit, model, a, b];

data = Table[{x, 8 x^3 - 7 x^2 - 10 x + 1 + RandomReal[{-5, 5}]}, {x, -2, 2, 0.1}];

model[v_, terms_, x_] := FromDigits[Reverse@Array[v, terms], x];

SetAttributes[testplotfit, HoldRest];
testplotfit[terms_, data_] :=

Manipulate[
Column[{
Dynamic[Button["Compute", (* FindFit[] button *)
Block[{x}, With[{sol = FindFit[
data,
model[b, terms, x],
Transpose@{Array[b, terms], Array[a, terms]},
x]},
Do[a[i] = b[i] /. sol, {i, terms}]]]
]],

Show[ (* plots: *)
Plot[ (* model *)
model[a, terms, x] /.
Array[HoldPattern[a[#]] -> a[#] &, terms],
{x, Min@data[[All, 1]], Max@data[[All, 1]]},
PlotStyle -> Black, PlotRange -> MinMax /@ Transpose@data],
ListPlot[data, PlotStyle -> Red], (* data *)
ImageSize -> 300, PlotRangePadding -> Scaled[.05]
]
}],

Evaluate[ (* controls *)
Sequence @@ Table[With[{i = i}, {{a[i], 1, Subscript["a", i]}, -10, 10,
Appearance -> "Open"}], {i, 2}]]
]

Example:


testplotfit[2, data]

Mathematica graphics


What would be even better is if I could vary the number of terms with a slider and have the number of sliders change automatically.





A variable number of sliders and other controls comes up in several Q&A (and there are more):




Answer



Here is one way:


ClearAll[testplotfit];
testplotfit[terms0_, data0_, maxterms_: 10] :=
Manipulate[
Show[
Plot[

model[a, terms, x] /. Array[HoldPattern[a[#]] -> a[#] &, terms],
{x, Min@data[[All, 1]], Max@data[[All, 1]]},
PlotStyle -> Black, PlotRange -> MinMax /@ Transpose@data],
ListPlot[data, PlotStyle -> Red],
ImageSize -> 300, PlotRangePadding -> Scaled[.05]
],

{{a, a}, None}, (* {a,a} initializes a=a instead of a=0 *)
{{data, data0}, None}, (* only one copy of data is stored *)
{{terms, terms0}, 1, maxterms, 1},

Dynamic@ Column@ Table[
With[{i = i},
Control[{{a, a, Dynamic@Subscript["a", i]}, -10, 10,
Manipulator[Dynamic[a[i]], ##2, Appearance -> "Open"] &}]
],
{i, terms}],
Dynamic[Button["Compute",
Block[{x, b}, With[{sol = FindFit[
data,
model[b, terms, x],

Transpose@{Array[b, terms], Array[a, terms]},
x]}, (* TBD: test FindFit succeeded *)
Do[a[i] = b[i] /. sol, {i, terms}]]]
]],
Initialization :> (Do[a[i] = 0., {i, maxterms}])
]

Example:


testplotfit[4, data]



Remarks:


A major problem is that control declarations of the form {a[1],..}, {a[2},..} etc. results in the expressions a[1], a[2],... being replaced by symbols of the form $nnn$$ where nnn is some serial number. These replacements are done literally and lexically throughout the Manipulate code, so that expressions like a[i] and a[#] can never refer to the control variable that is constructed. That is why Plot and FindFit[] did not work in the OP. Note I avoided such declarations above.


I localized several variables (a, b, x, and data) for security. Also the use of data in the Manipulate means the data set (the argument data0) is stored in only one place in the Manipulate[] output, instead of being copied into all the references.


I used a trick, {{a, a}, None}, which subverts the default initialization a = 0 by initializing a to the uninitialized a, a symbol.


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