Skip to content
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

Add support for specifying the aspect ratio of the plot area #88

Merged
merged 9 commits into from
Apr 3, 2021

Conversation

alex-hhh
Copy link
Collaborator

@alex-hhh alex-hhh commented Mar 8, 2021

Plot will calculate the plot area such that it fills up the entire available size of the plot, but this creates a problem for plots which need to maintain their "shape". For example, by default when plotting a circle, you get an ellipse:

(require plot)
(plot-background "lightblue")
(plot-x-label #f)
(plot-y-label #f)

(plot (polar (lambda (t) 1)) #:width 400 #:height 200)

circle-non-ar

This pull request introduces the #:aspect-ratio parameter for all plot functions and a corresponding plot-aspect-ratio parameter, which allows specifying the ratio between the width and height of the plot area. For example, by specifying an aspect ratio of 1, the plot will actually look like a circle.

(plot (polar (lambda (t) 1)) #:width 400 #:height 200 #:aspect-ratio 1/1)

circle-with-ar

The aspect ratio setting also works for 3D plots.

@alex-hhh alex-hhh marked this pull request as draft March 8, 2021 09:07
@bdeket
Copy link
Contributor

bdeket commented Mar 8, 2021

Hi,
I didn't manage to test it yet and will do so tonight, but if I understand correctly the aspect ratio is defined in terms of the plot-area width and height.
I would like to suggest to instead define this in terms of the x and y (and z) axis so that when you are drawing an ellipse you don't need to calculate the right aspect ratio every time you change the shape.
An alternative is to add an option axis-constrained? ( I think maple did this ) so you have only two options: maximize the available area or keep axiss with same unit step size

@alex-hhh
Copy link
Collaborator Author

alex-hhh commented Mar 8, 2021

Hi @bdeket , thanks for the feedback. Yes the aspect ratio is defined in terms of the plot area, this seemed more natural to me, especially when using aspect ratios of 4:3 and 16:10. It is also simpler to think about this for 3d plots, where the aspect ratio applies to the view.

Just to make sure I understand what you are suggesting, since I am not familiar with maple. Are you suggesting that the aspect ratio would be defined in terms of plot units, such that, for example, with an aspect ratio of 1, this plot would remain a circle, even though the x-axis range is 4 but the y axis range is 2?

(plot (polar (lambda (t) 1)) #:x-min -2 #:x-max 2 #:y-min -1 #:y-max 1)

@bdeket
Copy link
Contributor

bdeket commented Mar 8, 2021

Indeed, that was the idea.
I suggested this because the only use case I could think of to use the aspect-ratio is exactly that: keeping plot units in x and y (z) direction exactly in step.
But maybe you have another use case, and this question is orthogonal to that.

I think in maple this same ratio for the plot units was also enforced when you started zooming in and out, but I'm not sure anymore (I would consider it a plus)

@samth
Copy link
Member

samth commented Mar 8, 2021

A different approach here might be to allow controlling two kinds of widths/heights separately -- the plot itself and the entire surface. That would allow achieving your result by specifying the plotted area to have size 200x200. This incidentally might be useful for creating plots that are aligned when combined as picts.

@rfindler
Copy link
Member

rfindler commented Mar 8, 2021

Just to chime in with @samth 's comments, one of the things I have struggled with when using plot (mostly everythign with plot is amazingly smooth sailing!) is when I have multiple plots that I want to put into the same figure and I want the mapping of plot coordinates -> drawn coordinates to be the same for all the plots and I want to put a bunch of plots together into a figure (by doing stuff like hbl-append with the picts that come out of the plot library). Things like the axis labels being different sizes foils me, however. If these changes help with that usecase, that'd be fantastic. Also, a related request might be a few examples that explain how to make this usecase work well in the docs would also be very welcome so people who are using plot for papers find it and know what to do.

@bdeket
Copy link
Contributor

bdeket commented Mar 8, 2021

wrt "maintaining shape"
I think I still don't completely understand the intent. Is this to address the comment of #85 or to address #7 , or something else entirely?

(plot3d (polar3d (λ (ϕ θ) 1)) #:width 400 #:height 200 #:altitude 0)

b1

(plot3d (polar3d (λ (ϕ θ) 1)) #:width 400 #:height 200  #:altitude 0 #:aspect-ratio 1/1)

b2

If the aspect ratio was in terms of plot-units, the second picture would show a ball(/circle) whatever the view angle


Also, when I start zooming in on your second circle (aspect ratio 1/1) it looks like something goes wrong with calculating the optimum size (too much white space at bottom and left):
zoomed

@alex-hhh
Copy link
Collaborator Author

alex-hhh commented Mar 8, 2021

I understand that many of people want #7 and #84 to be implemented and that many people would like to be able to align the plot areas of several plots, for the purpose of writing publications. This pull request does not address any of those issues. Those issues are constantly on my mind, but as of today, I don't know how to implement them in a way that it does not create more problems than it solves. The main problem with them is that the plot package has many use cases, and so far, all proposed solutions handle one or two use cases well, while creating potential problems for other use cases.


The purpose of this feature is to address the question in #85, as well as the problem I had when I wrote my Plot Animations blog post, where I had to manually adjust the size of the plot, until the circle looked like a circle. As @bdeket noted, #85 is not addressed perfectly...

To a certain extent can also be used as an example of how many files have to change to add one single parameter to the plot API, which I think is one drawback of the current plot API...

The problem is addressed in such a way that, I feel it is easy for someone with a programming background to use the feature and it works reasonably well with all use cases of the plot package. In particular:

  • I would expect an aspect ratio of 1:1 to result in a square plot area, and I think this is what most people would expect. If the aspect ratio would be in plot units, as @bdeket suggested, a plot with an X range of 2 and an Y range of 1 would be a rectangle with an aspect ratio of 1:1, which is unintuitive. @bdeket your suggestion of maintaining "square" plot units is also useful, and perhaps could be added as a separate plot parameter, perhaps in a different PR
  • specifying the plot area size separately, as @samth suggested, will put additional burden on the user to control the layout, and there will be no help from the plot package in determining what the optimal fill of the entire plot image would be, especially when taking into account labels and axes. The size of the plot area would have to be constantly tweaked as the plot data (and therefore labels) change -- this might be acceptable when creating plots for a publication, where the data is fixed, but won't work in any other use case. It is also unclear how the plot area would grow when a plot frame created by plot-frame is resized by the user.

@bdeket , I will look into the issue you noticed with the "deep zooming"

@rfindler
Copy link
Member

rfindler commented Mar 8, 2021

Indeed. My bad @alex-hhh . Sorry.

@alex-hhh
Copy link
Collaborator Author

alex-hhh commented Mar 9, 2021

Hi @bdeket , this is the result of zooming in excessively with the current plot package (without the changes in this PR). It looks like something is wrong with calculating the plot area size, but it is not introduced by this PR.

(require plot)
(plot-background "lightblue")
(plot-x-label #f)
(plot-y-label #f)

(plot (polar (lambda (t) 1)) #:width 400 #:height 200)

image

@alex-hhh
Copy link
Collaborator Author

alex-hhh commented Mar 9, 2021

With regards to the 3D plots being distorted, this is caused by the fact that the projection itself is not considering the aspect ratio, I had a brief look at the code in plot-area.rkt for the 3D plots and it looks like the code is doing "manual" calculations instead of defining "world->view", "view->projection" and "projection->screen" matrices, so it is not easy to change to take the aspect ratio into account -- it would be a nice improvement to rewrite this aspect of the 3D plots. This would also make 3D plots faster, since going from 3D world coordinates to DC coordinates would be a single matrix multiplication instead of a chain of transformations, as it is now.

This refactoring would also have to be done if we implement the "plot units" based aspect ratio.

@alex-hhh alex-hhh marked this pull request as ready for review March 29, 2021 12:14
@alex-hhh alex-hhh merged commit c457141 into racket:master Apr 3, 2021
@alex-hhh alex-hhh deleted the ah/aspect-ratio branch April 3, 2021 04:31
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants