-
Notifications
You must be signed in to change notification settings - Fork 27
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
Pretty printing: _repr_html_ and _repr_svg_ #93
Comments
is there no existing code for pretty printing json files in general that we can leverage? |
Sure, but I'm thinking something more akin to msaf's plots that render out a table for the annotation objects. Doing A really cool version of this would combine all annotations into a single table-like widget, where each row corresponds to an annotation. This way, you could easily get a visual digest of all annotations in a compact space. One can also imagine having different rendering helpers for different namespace types, so that events draw like events and regions draw like regions. (Even fancier, pitches and chords draw like piano roll, etc.) |
So basically a jupyter version of sonic visualiser :P Yes, that would be cool. |
Yup! But it could also be used outside of notebook environments, eg, for a web-based visualizer. |
I think I can see where you're going with this... and I like it. |
Thinking a bit more about this, it might be tricky to properly handle overlapping intervals, eg:
should render as two rows that overlap by half horizontally, but
should render as one row. Ideally, a css layout engine should do this for us, but I'm not convinced that it will work. |
Why should the second example render as one row? One could argue that each On Fri, Jan 29, 2016 at 10:05 AM, Brian McFee [email protected]
|
That's not something you can easily deduce from the tabular annotations, since there are no constraints about mutual exclusivity between observations. |
I see your point. I can think of three potential solutions (which could all
That said, I think it's at least worth considering leveraging some of the On Fri, Jan 29, 2016 at 11:04 AM, Brian McFee [email protected]
|
I would really prefer to avoid the following here:
The former for obvious reasons. The latter because it can complicate sharing of notebooks if you have external or cross-site loading of dependencies. The instrument-per-row idea is fine too, but I see that as an embellishment that should come much later, once we have the basic formatting implemented. |
I did some snooping around here, and it seems like there isn't a great way to encode this directly in html. So we have two options:
Neither of these sound particularly good to me, but i'm leaning toward 2 because it keeps the python code simple. Algorithmically, I have a method in mind to detect overlapping intervals and pack them into as few parallel timelines as possible. Once that step is done, this example illustrates the kind of layout I have in mind. |
Alternatively, we can impelement This would still have the downside of forcing the layout logic into python, but using SVG instead of CSS for the positioning would be much better. If we attach the proper attributes to the svg elements, it would still be relatively easy to hook in via javascript to have interactions in #19. |
This could get really really heavy on huge jams files. I'm currently using JAMS for storing really high resolution time series spectral features. It'd be difficult to display them properly, and it'd be computationally intensive to reduce them on the client side. Maybe it'd be a good idea to restrict the kinds of annotations that can be displayed in this view? |
How are you storing these features? In blobs/vectors, or one of the annotation namespaces? In general, I think we're planning to support namespace-dependent rendering, so a "blob" annotation might simply show up as an interval with Otherwise, the common dense data types shouldn't pose much of a problem. E.g., rendering a pitch annotation as an svg curve might end up being large, but not insane. @hughrawlinson I'm wondering if you have any thoughts about canvas vs svg for this stuff? I'm leaning more toward svg because it would be dom-accessible and potentially interactive down-stream. (It seems to work well for d3, anyway!) |
I agree with SVG, it means you can take it out of the web context and render it in LaTeX or somewhere else that may be useful. It's also a nice format because it's easy to remove pieces in a text editor rather than having to fire up an image editor as you would have to with a rasterised image saved from a canvas. I'm storing my data in an annotation, which may be wrong. It'd be great to discuss that at some point if you wouldn't mind, but as for generally dealing with this kind of data just putting |
Great points; I hadn't thought of exporting to figure, but it would be pretty trivial this way!
Right. We may have relatively high sampling rates for curve annotations, but my understanding is that they're primarily there to capture abrupt changes, and the curves themselves tend to be relatively smooth otherwise. It wouldn't be hard to do some kind of compression (eg run-length encoding) to retain the visual characteristics of the annotation without losing too much.
I think this part is getting off-topic, and would be better handled in a separate thread. |
That last bit was definitely off topic, sorry! |
Coming back to this, it would be awesome to have piano-roll style viz for note-like annotations. A possible complication is that relevant namespaces (e.g. Another thing that would be awesome is the ability to overlay annotations for the purpose of visual comparison (e.g. references notes versus automatically transcribed notes). |
If we were doing this via d3, I'd say that it's all just a matter of how things are interpolated on display (quantization for notes vs cubic interp for f0).
Agree -- that would be awesome! |
But right now |
it can be, but generally you wouldn't do that ; f0 would live in pitch_hz. If you want to switch between the two, auto-convert before viz. |
Fair enough |
I did a bunch of thinking about this, and I have some slightly less scrambled ideas for how it might work. All horizontal element scaling should be relative to the total track duration, so that the SVG can stretch dynamically. I think for most namespaces, it makes sense for each observation to be an svg group, which itself contains one or more elements, along with some encoding of its raw properties ( For namespaces where intervals may overlap (eg, tags), we'll need some way of grouping the observations into non-overlapping subsets with the same value. I think we're just going to have to suck it up and do this logic manually, but it shouldn't be so bad. For chords, do we want to use chord labels, or convert into a chromagram with bass and root emphasized (say, by alpha and texture)? I would still color-code each chord by name in a consistent way. The grouping idea above works just as well here. Piano roll ought to be trivial. Continuous f0 is a sticky one -- i think it would be best encoded as a collection of polyline/paths, where each collection is a contiguous sequence of observations with no breaks in voicing. This breaks the grouping logic I mentioned above, but I'm not sure how else to do it. Some questions:
|
I think we do - without a time axis the usefulness of the visualization will be quite limited. Similarly, for (e.g.) piano-roll style viz it would be very useful to know the actual pitch of the notes, which requires ytick labels. |
Actually, why not use matplotlib? |
We could, but it's a pretty big dependency. It would also limit the amount of control we have on the svg coming out, since it adds a rendering abstraction layer in the middle. This would make it harder, I think, to generate svg that could be manipulated in-browser and backed out to jams again. |
I think it's relatively safe to assume anyone using python + jams will be using matplotlib too anyway.
Ah, I wasn't aware in-browser manipulation was on the table. So this would serve not just for visualization, but would in essence be a web-based interface for correcting annotations? Because then you also want to support sonification of the annotation, concurrent sonification of the annotation and an audio file, and probably a bunch of other stuff I haven't thought about. |
True, but it is bloated and complicated relative to what we need, i think.
Visualizing, first. The idea being that with a little bit of javascript, it would be easy to, say, make observations clickable and automatically seek to the corresponding point in the audio playback widget. Seymour does this, and it's super useful. Editing is a much more complicated beast, but it will have to happen eventually. I'm not sure that svg is the right tool for the job here, but I suspect you would want something that hooks into the DOM, is queryable, and flexible enough to contain all everything we need to make a jams annotation. Canvas is definitely not the answer.
We have that already. In a browser context, it would have to be done via remote query, but that seems totally doable. The alternative would be rewriting sonify in javascript, and that makes me seasick.
Not sure what you mean by this; mixing with the existing audio track? I have a notebook demo that does this. |
Just my $.02:
|
+1 @bmcfee, my memory is junk wrt to seymour ... do you have some kind demo On Wed, Mar 30, 2016 at 12:42 PM, Oriol Nieto [email protected]
|
Not really -- the server is password protected, and often down for reasons beyond mine (or anyone's) control. If you remind me in person, I can show it to you though. |
So, this is a little off-target, but since I needed it for a project, I've hacked up a very ad-hoc Not editable, depends on matplotlib, and a whole lot of other evils. But super quick and does the trick. |
@bmcfee I guess my question is: is it not worth developing a simple matplotlib-based viz (for every annotation namespace) while grander schemes are being hatched? The more I work with JAMS files the more I need annotation visualization. Pragmatically, viz-only is a PR we can probably manage short-term, whilst an editable interface is probably more involved. On a related note, would the plotly API (or the javascript library directly) be an interesting option? |
I'd prefer to minimize redundant work, but we also shouldn't let perfect be the enemy of good.. so if you have an idea for how to make this work, go for it. I haven't looked much at plotly -- I don't want to depend on their API, but plotly.js is MIT-licensed, so that could work. If we do go that route, it might be a good idea to just have everything live as an independent project (jams-viz), since it would invoke a pretty heavy set of dependencies. |
Not sure it's a good idea, but it's an idea... basically we'd have a jams.viz module (or some other name), and in it's simplest form it could have a single function that takes an annotation and plots it. Since every annotation has a namespace, we could have a separate visualization function that's called under the hood based on the annotation's namespace. That's basically it. Probably instead of taking a single annotation I'd support taking a list of annotations so that annotations can be plotted on top of each other. I'd probably only support overlapping annotations if they're of the same namespace to keep things simple, so perhaps first thing would be to sort the annotations by namespace, and then all annotations that have the same namespace get plotted together (or separately, could be an optional argument). It's another dependency, but it would also be nice to include mpld3 which would make all the plots zoomable/panable in jupyter. EDIT: I guess the user can just import mpld3 locally, no need to make it a dependency per-se. |
I'd like to avoid adding a heavy dependency (eg matplotlib) if we can avoid it. I'd also like to avoid piecemeal importing of submodules. Otherwise, yeah, that's exactly how I'd conceived of the viz module working -- pretty much exactly like sonify, or eval for that matter.
You don't even need that if all you want is pan/zoom: just use |
So what do you suggest? matplotlib seems like the most straight forward option (assuming we want good, not perfect)? |
Unless anyone has a better idea for the time being (?), I might take a stab at implementing a very simple matplotlib-based vis module. |
Just dumping this here for future reference: vega seems promising as a high-level language to making this kind of diagram, and vincent is a python layer on top of that. It doesn't add any dependencies we don't already have (ie, just pandas), so it might be a good option if/when it's stable. |
Vincent seems dead?
|
Punting this to 0.3 since display covers most of the preliminary use-cases. |
#149 implements a custom html renderer for annotation data to recover the previous behavior of I think it would be pretty simple to have a generic JObject renderer that uses collapsible list groups to give a compact view of the jams dom, but without any fancy visualization. Maybe I'll take a crack at this in the near future. |
Looks like the way to go about nested printing is via something like: <details>
<summary> top-level summary </summary>
detailed data dump
</details> |
I'd like to throw on a reimplementation of |
Fixed by merging #158 |
It would be nice if we could have better pretty-printing of jams objects (and annotations, specifically) for notebook environments, ie, by implementing a
_repr_html_()
method.The text was updated successfully, but these errors were encountered: