Skip to content
Advertisement

Scaling shape with frame size

I’m new to Java and I’m playing around with a simple GUI example. After having received a good answer to my last question, I tried to take it one step further and to scale the shape with the frame size (with fixed aspect ratio). Indeed, when I change the window size, the shape scales with it. However, the shape never fills the frame to a maximum, there is always quite a lot of space at the right and bottom margins. Here’s the code:

JavaScript

The scaling factor seems to be ok (2000 * 0.6415 = 1283). My guess is that although the calculation itself is right, I apply the scaling factor at the wrong place. Any help with this will be appreciated.

EDIT: Here is a screenshot of the window that I get to make the problem clearer:

shape with large margins at the right and at the bottom

EDIT2: Here is the updated code (slightly renamed):

JavaScript

This is what the result looks like with shape.setRect(0, 0, 2000, 500):

Shape size 2000x500

This is what the result looks like with shape.setRect(0, 0, 800, 500):

Shape size 2000x800

If I replace return minFactor by return 1.0 (and stay with shape.setRect(0, 0, 2000, 500)), I get this result:

Shape size 2000x500, scaling factor 1.0

If I replace return minFactor by return 0.9 (and stay with shape.setRect(0, 0, 2000, 500)), I get this result:

enter image description here

If I replace return minFactor by return 0.9 and change to shape.setRect(0, 0, 3000, 500), I get the same visible result as with return 1.0 and shape.setRect(0, 0, 2000, 500).

Advertisement

Answer

there is always quite a lot of space at the right and bottom margins.

Works fine for me.

That is there is not extra space as long as the aspect ration of the frame is the same as that of the rectangle.

If you want the rectangle to always fill the space available, then you would need a separate scaling factor for the width and the height.

Other comments:

JavaScript

I’m not sure why you are using a translate, but if you do then your preferred size calculation is wrong. You would need to add 3 to both the width and height, otherwise 3 pixels will be cut off when you paint the component at it preferred size.

JavaScript

Don’t reference the content pane or the frame. Custom painting for the panel is based on the size of the panel. Also, the panel may not always be added directly to the content pane.

So the code should be:

JavaScript

Also since you don’t need the frame the constructor for your panel should be:

JavaScript

In your application if you ever do need access to the frame you can use:

JavaScript

Edit:

You conditionally set the “minFactor”:

JavaScript

If you always want the aspect ratio preserved then do it unconditionally:

JavaScript

Edit2:

JavaScript

The purpose of invoking super.paintComponent() is to clear the background before doing custom painting to make sure you have no painting artifacts. Typically this is done as the first statement in the method. I’m not sure why you are applying a transform to the Graphics before invoking the method.

So I would suggest the code should be:

JavaScript

Edit 3:

I then tried to compile and execute your code via the command prompt and the gap is gone.

I also use the command line to compile/test which got me thinking about another question I comment on recently: How to make a picture not blurry in Java

Sure enough this is a related problem.

When I use:

JavaScript

I see the desired behaviour. However when I use:

JavaScript

I see the behaviour you describe. So this would lead me to believe that Eclipse is doing something similar.

However, I don’t know how to resolve the problem so that the painting is done correctly whether scaling is used or not.

Edit 4:

Just found the solution. The Graphics object already contains the scaling factor for the application and you are resetting the factor. Instead you need to adjust the factor:

JavaScript
User contributions licensed under: CC BY-SA
2 People found this is helpful
Advertisement