-
Notifications
You must be signed in to change notification settings - Fork 21
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Enable ZX Live to be run from inside of a notebook. #161
Enable ZX Live to be run from inside of a notebook. #161
Conversation
9a8e25b
to
d443487
Compare
This is very cool! |
Right now the instance of the graph in ZXLive and that in pyzx are the same. So if you make a change in ZXLive, then this is also reflected in pyzx, which is what you want. However, you can also make a change in pyzx, and this also changes the graph in ZXLive, but this is not reflected in the visuals. It is probably too hard to make ZXLive just 'know' when a change happened, but maybe we can add a command like 'update_graph' that just updates all the graphics, which you can call when you are done doing stuff in pyzx? |
This could maybe even the form of an environment, like:
Or perhaps even better, whenever the ZXLive window regains focus, we call a function that redraws the graph. |
What is the intended workflow when a user opens ZX Live from a notebook to edit a graph, and then opens Proof mode? Is it intended that the final state of the graph after the proof steps be saved back into the notebook? What shouldn't happen is that two tabs (the original edit panel plus the new proof panel) both point back to the original graph, as it's then confusing if the user makes changes to both and then saves them. I think only one tab should be connected to the original graph, in which case maybe the edit panel should be locked or disabled or even closed if the user opens a proof panel, when ZX Live in in embed mode. But maybe it would be confusing if the app behaved (too) differently in standalone mode vs. embedded mode. (It's also necessary to handle the case where the user calls ZX Live to edit the same graph twice. This shouldn't be allowed, and should result in an error message, or maybe just bringing the existing tab into focus.)
I think the least confusing thing would be that each ZX Live window (not tab) is associated with a single graph from the notebook, and whatever you did inside that window is tied to the original graph that was used to open it. The way that Qt applications are implemented in ipython, there is a singleton instance of the app per kernel. And since ZX Live is implemented with one main window for the app, that means that each new graph is opened in a new tab inside that main window. It would have to be changed to allow opening multiple windows in addition to multiple tabs, if it's desirable to have one window per original notebook graph object. Or maybe it's fine to just have different graphs open in different tabs, the way it is now. |
It's the "same" only in the sense that if you make a change in ZX Live and click File->Save, the changes are saved back to the graph in the notebook. But the changes are not "live". I think it's less confusing/dangerous if no data is actually changed in the notebook until/unless the user deliberately requests the changes be saved. It is possible to make it so that changes made in ZX Live are immediately saved back to the notebook. In The problem is that ZX Live (or any Qt application) is opened asynchronously, so that the notebook continues to run. So the user could have both the notebook and ZX Live open, and make changes in both, and ZX Live is not aware of the changes made to the graph in the notebook.
The PR is just a proof-of-concept but there are two ways to go with this. Either we make a copy of the graph initially when it's imported into ZX Live, to break the link explicitly (so that changes made in the notebook no longer affect anything in ZX Live, and the only way to send the edits back to the notebook is via File->Save); or we somehow try to detect or allow the user to inform us if the graph is changed in the notebook, and update the internal state and visuals in ZX Live. I think the former is clearer/cleaner, and that the latter leaves many possible opportunities for bugs due to things being out of sync.
What's a use case in which a user edits a graph in ZX Live, but continues to make changes using pyzx in the notebook, and needs to update the opened graph in ZX Live? Instead of updating the graph, which is complicated, wouldn't it make more sense to just open a new tab with the changed graph? |
I'll experiment with keeping changes between the notebook and ZX Live in sync, but I suspect it's going to lead to weird edge cases, like what happens if the user makes changes to both and the update in both directions happen at the same time. There's just all kinds of possibilities for bugs because of race conditions. |
That's true, I didn't think of what would happen if you would make changes in both settings.
If we want to go the route of syncing the ZXLive and pyzx graph so that a change in one updates the other, I think we should let the pyzx changes take precedence. So if I make some change in ZXLive, which I don't save, and then I make a change in pyzx, this should overwrite my changes in ZXLive. I think this is the cleanest way because the zxlive application is called by pyzx |
... and it turns out to be a huge mess due to the ability to undo/redo. Any updates made in the notebook would have to also be saved on the undo stack. That's complex enough without introducing an external source of possible edits. |
I think the solution which would lead to the fewest potential bugs down the road is to copy the input graph, work entirely on that within ZX Live, and return some kind of object to the caller from which they can extract additional information. The caller should be informed that this returned object can change due to actions taken in the embedded ZX Live app, but the connection to the original graph has been severed and they can do whatever they want with it. I think this solution has the clearest semantics which can be understood by a user. That also solves the problem of what happens if you open the same graph object twice: ZX Live simply has two copies of the graph in different tabs, which don't affect each other. This can even be considered a feature. Maybe the user wants to compare different sets of edits (or proof steps or whatever) to the same original graph. However, the downside is that the graph is copied, which requires memory. Is that an issue for larger graphs? (But I don't imagine that a graphical tool would be very useful for a graph with millions of vertices, for instances.) The API design question here is then what kind of information is needed to be communicated from ZX Live back to the notebook? For example, the returned object can bundle the working copy of the input graph, as well as other things produced from it like proofs. p.s. Is it "ZX Live" or "ZXLive"? I've seen both but it should be kept consistent in the documentation and also in the app's own dialogs. |
I think that solution sounds good. Something like The memory cost of the copying is not important, as I can't imagine you would want to use ZXLive for a reallly large graph. |
The |
Okay, I've added a way for the user to get (a copy of) the graph by name. Also, if the user edits a graph whose name already exists in a tab, that tab is replaced (but the action can be undo from within ZXLive). However, I'm not sure what to do if the name is not unique. Right now, it just acts on the first tab with that name. |
No description provided.