Skip to main content

programming - Is it possible to update Dynamic objects even when they are off screen?


I am using Dynamic and DynamicModule to display graphics/statistics generated from a long-running set of parallel tasks. The data to display is passed from the workers to the main kernel via a SharedFunction (which handles any of the concurrency issues), and then copied over from the SharedFunction variable to a local (on the FrontEnd, actually) variable that is then used to generate the graphics and statistics.


The copying from the SharedFunction variable to the local/FE variable happens within the Dynamic 'loop'.


The problem is that of course if the Dynamic object moves off the screen, then all of the copying from the SharedFunction variable to the local/FE variable stops. If I wait too long, then a whole lot of copying queues up, and when the Dynamic object regains visibility, the whole thing can hang while a lot of copying is being performed. I believe that the SharedFunction variable then gets locked while all of the copying occurs, and then the workers also get blocked, waiting for their chance to update the SharedFunction variable. So my entire parallel calculation gets delayed, defeating the benefits of using parallel.


Of course I introduced this two-stage mechanism - copying from the SharedFunction to the local/FE variable - in the hopes of avoiding this sort of issue as I know that some of my graphs take a long time to generate. I hoped to minimize the access to the SharedFunction variable, so I don't access it directly from the graphics routines. Nevertheless, like I say, the copying mechanism itself is also sufficiently costly as to now be its own bottleneck.


I have looked over the Advanced Dynamic Functionality section of the manual, and have seen the section concerning Refresh:



Refresh can also be used to cause updates at regular time intervals. It is important to understand that this is not a feature that should be used lightly. It is fundamental to the design of Dynamic that it does not need to update on any fixed schedule, because it simply always updates immediately whenever doing so would be useful. But there are some situations where this either cannot, or just unfortunately does not, happen.



But this only refers to the timing of updates, not how they are triggered.



And I think I have read every(!) entry here on MMA.SE concerning Dynamic/DynamicModuleand Refresh. However I am still pretty certain that there is no such thing as Dynamic updates when the Dynamic object is off the screen.


This question sounds like my question. However the consensus seems to be that trying to have Dynamic objects update in this fashion is a poor design. FrontEndDynamicExpression is mentioned, but it looks a little bit daunting.


As a simple example (nicked from somewhere here on MMA.SE) I have


DynamicModule[{showclock = True, ii = 0}, {Checkbox[Dynamic[showclock]], 
Dynamic[If[showclock, Refresh[ii++, UpdateInterval -> 0.1],
ToString[ii] <> "-No count"]]}]

which illustrates that neither Refresh nor UpdateInterval are useful for what I want (the updating of ii is stopped when the object moves off the screen).


The only thing I can think of here is to try one of the ScheduledTasks functions. I have avoided this only because I have never used one any of these functions so far, and keep hoping that Dynamic has some feature itself.


I think that my issue might be further aggravated by the fact that I am running my program on another machine via Remote Desktop, and I get the feeling that disconnecting the session - even though the Dynamic NB is still open and 'visible' - also triggers Dynamic to take a break.



My question therefore is: Is there any way to force Dynamic/DynamicModule to update even when the Dynamic object is not visible?


I apologize of course for not offering any sort of minimal working example or useful code.


My version is MMA 9 and Windows 8.1.



Answer



It's hard to say much without details but let's try:


globalVar = 0;

DynamicModule[{ localVar = 0},
Column[{
Button["Stop", RemoveScheduledTask@ScheduledTasks[]],

Dynamic@List@localVar
}]
,
Initialization :> (
RunScheduledTask[
If[localVar != globalVar, localVar = globalVar; FinishDynamic[]],
1, 2
]
)
]


This is kind of listener that checks if the globalVar was updated once per second. You can minimize the notebook and set globalVar=5 e.g. in new one.


Notes




  • ScheduledTasks have to be run from Initialization so the full scoped (by DM) symbol name is sent to the kernel.


    ScheduledTask with scoped variables




  • Dynamic@localVar alone is not enough to update it, just because there are some undocumented ways Dynamic handles just symbols and more nested expressions, so I added List.





  • localVar != globalVar may be time consuming so you can introduce additional variable/flag that will be set to 0 after assignment and to 1 by your external tasks after their are done.




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