Skip to main content

plotting - DynamicModule not working when additional code is included


I have read the Wolfram documentation on Dynamic and understood enough to suspect that what I am trying to do may not be possible, so I am hoping I am wrong.


Here is the base example that works very nicely:


DynamicModule[

{n1 = 10},
Panel[
Column[{
InputField[Dynamic[n1]],
Dynamic[
Plot[Sin[Pi*n1*x], {x, 1, 3},
GridLines -> Automatic,
ImageSize -> 300]
]
}]

]
]

It creates an input dialog textbox that I can edit. When I enter a new number (3 in this case) and press return the frequency of the Sin wave changes as we would expect:


Sin.jpg


On the other hand, this code does not work, or works only partially:


Clear[list];
list = {"red", "yellow", "blue"};
DynamicModule[
{n1 = 10},

Column[{
InputField[Dynamic[colour], String],
If[list[[1]] == colour,
n1 = 1,
0],
Dynamic[
Plot[Sin[Pi*n1*x], {x, 1, 3},
GridLines -> Automatic,
ImageSize -> 300]
]

}]
]

In my real code I actually have a Do loop around the If statement, but since the same problem and behaviour is exhibited by this simpler example I kept things simple. The idea is that I would like to be able to enter a string, do a search on that string by cycling through an array of names, pick the index that correponds to a match, and then use that index in the plot.


This code, instead, does not respond when I enter "red" (without the quotes), as it should. However, if I execute the cell again then it remembers the new value for colour and updates the frequency. In other words, it's not dynamic. I tried different things but nothing worked. Making colour one of the module variables, as my instinct would prompt me to do, loses even the updating when re-executing the cell since now colour is not even a global variable anymore and does not remember the last value I entered. What am I doing wrong?


Thanks


Edit:


Based on @anderstood's reply I was able to go further and encountered a interesting puzzle. The new code now is


Clear[list];
list = {"red", "yellow", "blue"};

DynamicModule[
{n1 = 10, colour = "red"},
Panel[
Column[
{
InputField[Dynamic[colour], String],
Dynamic[
Do[
If[list[[i]] == colour,
Print["i = ", i];

n1 = i],
{i, 1, 3}
]
],
Dynamic[
Plot[Sin[Pi*n1*x], {x, 1, 3},
GridLines -> Automatic,
ImageSize -> 300]
]
}

]
]
]

Note that I am not using n1 = 0 in the If statement because now I am using the Do loop. Interestingly, using a Break once the desired colour has been found does not work because Break does not seem to be able to get out of the Dynamic environment (my guess). The corresponding output shows an annoying 'Null' as the second entry in the column:


Blue.jpg


Here I found some inspiration in Answer 7 to this question How to avoid returning a Null if there is no "else" condition in an If contruct, provided by @F'x. Using


DeleteCases[{hello, apple, Null, 34}, Null]

works and returns



{hello, apple, 34}.

However, inserting DeleteCases around the curly brackets inside Column[] in the code above does NOT get rid of the Null. Interestingly, I do not think this Null is coming from the If statement (as is the case in the other question), but I think it is coming from Dynamic. In any case, DeleteCases has no effect. I can't imagine why, other than Dynamic is somehow interfering. Note that I tried putting DeleteCases and more Dynamics in different places in this code but nothing worked.


If it's possible to get rid of this Null in the output panel it would be nice. Any suggestions much appreciated.



Answer



The following seems to work:


Clear[list];
list = {"red", "yellow", "blue"};
DynamicModule[{n1 = 10},
Column[{InputField[Dynamic[colour], String],

Dynamic[If[list[[1]] == colour, n1 = 1, n1 = 0]],
Dynamic@Plot[Sin[Pi*n1*x], {x, 1, 3}, GridLines -> Automatic,
ImageSize -> 300]}]]

Note the n1 = 0 in the If loop, so that n1 does not keep its value of 1 if entry is "blue" or "yellow". Also, the If loop is in a new Dynamic[] since it has to be dynamically updated.


Mathematica graphics


Mathematica graphics


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