Skip to main content

plotting - Module inside of Manipulate Example


Here is a very nice example from The Student's Introduction to MATHEMATICA.


Manipulate[Module[{x, y},

ContourPlot[Exp[-x^2 - y^2] + x y, {x, -1, 1}, {y, -1, 1},
Contours -> 20,
Epilog ->
Dynamic[{Arrow[{pt,
pt + {y - 2 E^(-x^2 - y^2) x, x - 2 E^(-x^2 - y^2) y} /. {x ->
pt[[1]], y -> pt[[2]]}}]}]
]],
{{pt, {.5, .5}}, Locator,
Appearance -> Graphics[{Red, Disk[]}, ImageSize -> 5]}]


Which produces a wonderful demonstration.


enter image description here


My next step was to try the following:


Clear[x, y, f];
f = E^(-x^2 - y^2) + x y;
Grad[f, {x, y}];

Then in the next cell, I tried:


Manipulate[Module[{x, y},
ContourPlot[f, {x, -1, 1}, {y, -1, 1},

Contours -> 20,
Epilog ->
Dynamic[{Arrow[{pt,
pt + Grad[f, {x, y}] /. {x -> pt[[1]], y -> pt[[2]]}}]}]
]],
{{pt, {.5, .5}}, Locator,
Appearance -> Graphics[{Red, Disk[]}, ImageSize -> 5]}]

Which gave only this:


enter image description here



I gave Evaluate a try but that didn't work. So I tried removing the Module.


Manipulate[
ContourPlot[f, {x, -1, 1}, {y, -1, 1},
Contours -> 20,
Epilog ->
Dynamic[{Arrow[{pt,
pt + Grad[f, {x, y}] /. {x -> pt[[1]], y -> pt[[2]]}}]}]
],
{{pt, {.5, .5}}, Locator,
Appearance -> Graphics[{Red, Disk[]}, ImageSize -> 5]}]


And that worked.


enter image description here


But try typing


x=12

in the next cell and watch what happens to the arrow.


Finally I tried wrapping everything with a DynamicModule to see if it would prevent the x=12 issue in the notebook. First, this cell.


Clear[x, y, f];
f = E^(-x^2 - y^2) + x y;

Grad[f, {x, y}];

Then:


DynamicModule[{x, y},
Manipulate[
ContourPlot[f, {x, -1, 1}, {y, -1, 1},
Contours -> 20,
Epilog ->
Dynamic[{Arrow[{pt,
pt + Grad[f, {x, y}] /. {x -> pt[[1]], y -> pt[[2]]}}]}]

],
{{pt, {.5, .5}}, Locator,
Appearance -> Graphics[{Red, Disk[]}, ImageSize -> 5]}]]

This only produced:


enter image description here


As folks know, there has been a lot of discussion on not using Module inside of Manipulate and this is probably a good example of why not, but there is a lot of stuff happening here that I don't understand and could use some discussion explaining some of the issues:




  1. Why does Dynamic in the first code just update the arrow and not the contour plot.





  2. Why doesn't f = E^(-x^2 - y^2) + x y; and Grad[f, {x, y}]; work in the second piece of code?




  3. Why doesn't DynamicModule work in the last piece of code?




  4. And what is the best way to protect the arrow if a student type x=12 in their notebook?





Answers to Questions #3 and #4:


I should have defined the function f in the body of my dynamic module.


DynamicModule[{x, y, f},
f = E^(-x^2 - y^2) + x y;
Manipulate[
ContourPlot[f, {x, -1, 1}, {y, -1, 1}, Contours -> 20,
Epilog ->
Dynamic[{Arrow[{pt,
pt + Grad[f, {x, y}] /. {x -> pt[[1]],

y -> pt[[2]]}}]}]], {{pt, {.5, .5}}, Locator,
Appearance -> Graphics[{Red, Disk[]}, ImageSize -> 5]}]]

This works and x is protected.



Answer



Both Module and DynamicModule are shadowing the global variables x and y in the example in which you use them. The demonstration is best written without using either Module or DynamicModule.


Manipulate[
ContourPlot[f, {x, -1, 1}, {y, -1, 1}, Contours -> 20,
Epilog -> Dynamic[Arrow[{pt, pt + grad /. {x -> pt[[1]], y -> pt[[2]]}}]]],
{f, None},

{grad, None},
{{pt, {.5, .5}}, Locator, Appearance -> Graphics[{Red, Disk[]}, ImageSize -> 5]},
TrackedSymbols :> {pt},
Initialization :> (
f = E^(-x^2 - y^2) + x y;
grad = Grad[f, {x, y}])]

plot


Update


Sorry that I was careless about the testing of my code. The issue that you raise in your comment can be fixed by using pure functions. I do need to introduce Module in the fix.



My reworking of your example still keeps everything localized, Specifying controls that are non-functioning and invisible, like func and grad, is a useful trick for creating localized variables in Manipulate expressions,


x = 12; y = 42; func = 1; grad = 0;
Manipulate[
ContourPlot[func[x, y], {x, -1, 1}, {y, -1, 1},
Contours -> 20,
Epilog -> Dynamic[Arrow[{pt, pt + grad[pt[[1]], pt[[2]]]}]]],
{func, None},
{grad, None},
{{pt, {.5, .5}}, Locator, Appearance -> Graphics[{Red, Disk[]}, ImageSize -> 5]},
TrackedSymbols :> {pt},

Initialization :> (
func = (E^(-#1^2 - #2^2) + #1 #2 &);
grad =
With[{
g = Module[{x, y},
Grad[func[x, y], {x, y}] /. {x -> #1, y -> #2}]},
Function[g]])]

The code taken from The Student's Introduction to MATHEMATICA works because it doesn't define functions, but uses expressions for both the function and the gradient. I don't like that approach because it is too rigidly coupled to a particular function. With my fixed code you only need to redefine func to introduce a new function. For example


Manipulate[

ContourPlot[func[x, y], {x, -1, 1}, {y, -1, 1},
Contours -> 20,
Epilog -> Dynamic[Arrow[{pt, pt + grad[pt[[1]], pt[[2]]]}]]],
{func, None},
{grad, None},
{{pt, {0, .1}}, Locator, Appearance -> Graphics[{Red, Disk[]}, ImageSize -> 5]},
TrackedSymbols :> {pt},
Initialization :> (
func = (E^(#1^2 - #2^2) &);
grad =

With[{
g = Module[{x, y},
Grad[func[x, y], {x, y}] /. {x -> #1, y -> #2}]},
Function[g]])]

new-func


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