summaryrefslogtreecommitdiffstats
path: root/remoting/android/java/src/org/chromium/chromoting/DesktopCanvas.java
blob: a0149cfc7fdac1bb473c42f5fac6490c3452bb1a (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
// Copyright 2015 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

package org.chromium.chromoting;

import android.graphics.PointF;
import android.graphics.RectF;

/**
 * This class is responsible for transforming the desktop image matrix.
 */
public class DesktopCanvas {
    /**
     * Maximum allowed zoom level - see {@link #repositionImageWithZoom()}.
     */
    private static final float MAX_ZOOM_FACTOR = 100.0f;

    private final DesktopViewInterface mViewer;
    private final RenderData mRenderData;

    /**
     * The current cursor position is stored here as a float so that the desktop image can be
     * positioned with sub-pixel accuracy to give a smoother panning animation at high zoom levels.
     */
    private PointF mCursorPosition = new PointF();

    /**
     * Represents the amount of vertical space in pixels used by the soft input device and
     * accompanying system UI.
     */
    private int mInputMethodOffsetY = 0;

    /**
     * Represents the amount of horizontal space in pixels used by the soft input device and
     * accompanying system UI.
     */
    private int mInputMethodOffsetX = 0;

    public DesktopCanvas(DesktopViewInterface viewer, RenderData renderData) {
        mViewer = viewer;
        mRenderData = renderData;
    }

    /**
     * Returns the position of the cursor.
     *
     * @return A point representing the position of the cursor.
     */
    public PointF getCursorPosition() {
        return new PointF(mCursorPosition.x, mCursorPosition.y);
    }

    /**
     * Sets the position of the cursor which is used for rendering.
     *
     * @param newX The new value of the x coordinate.
     * @param newY The new value of the y coordinate
     */
    public void setCursorPosition(float newX, float newY) {
        mCursorPosition.set(newX, newY);
    }

    /**
     * Sets the offset values used to calculate the space used by the current soft input method.
     *
     * @param offsetX The space used by the soft input method UI on the right edge of the screen.
     * @param offsetY The space used by the soft input method UI on the bottom edge of the screen.
     */
    public void setInputMethodOffsetValues(int offsetX, int offsetY) {
        mInputMethodOffsetX = offsetX;
        mInputMethodOffsetY = offsetY;
    }

    /** Repositions the image by zooming it such that the complete image fits on the screen. */
    public void resizeImageToFitScreen() {
        synchronized (mRenderData) {
            // Protect against being called before the image has been initialized.
            if (mRenderData.imageWidth == 0 || mRenderData.imageHeight == 0) {
                return;
            }

            float screenToImageScale = 1.0f;
            float[] imageSize = {mRenderData.imageWidth, mRenderData.imageHeight};
            mRenderData.transform.mapVectors(imageSize);

            // If the image is smaller than the screen in both dimensions, then we want
            // to scale it up to fit.
            boolean scaleImageUp = imageSize[0] < mRenderData.screenWidth
                    && imageSize[1] < mRenderData.screenHeight;

            // If the image is larger than the screen in any dimension, we want to
            // shrink it to fit.
            boolean scaleImageDown = imageSize[0] > mRenderData.screenWidth
                    || imageSize[1] > mRenderData.screenHeight;

            if (scaleImageUp || scaleImageDown) {
                // Displayed image is too small or too large to fit the screen dimensions.
                // Apply the minimum scale needed to fit both the width and height.
                screenToImageScale =
                        Math.min((float) mRenderData.screenWidth / mRenderData.imageWidth,
                                 (float) mRenderData.screenHeight / mRenderData.imageHeight);
                mRenderData.transform.setScale(screenToImageScale, screenToImageScale);
            }
        }

        repositionImage();
    }

    /**
     * Repositions the image by translating it (without affecting the zoom level) to place the
     * cursor close to the center of the screen.
     */
    public void repositionImage() {
        synchronized (mRenderData) {
            float adjustedScreenWidth = mRenderData.screenWidth - mInputMethodOffsetX;
            float adjustedScreenHeight = mRenderData.screenHeight - mInputMethodOffsetY;

            // Get the current cursor position in screen coordinates.
            float[] cursorScreen = {mCursorPosition.x, mCursorPosition.y};
            mRenderData.transform.mapPoints(cursorScreen);

            // Translate so the cursor is displayed in the middle of the screen.
            mRenderData.transform.postTranslate(
                    (float) adjustedScreenWidth / 2 - cursorScreen[0],
                    (float) adjustedScreenHeight / 2 - cursorScreen[1]);

            // Get the coordinates of the desktop rectangle (top-left/bottom-right corners) in
            // screen coordinates. Order is: left, top, right, bottom.
            RectF rectScreen = new RectF(0, 0, mRenderData.imageWidth, mRenderData.imageHeight);
            mRenderData.transform.mapRect(rectScreen);

            float leftDelta = rectScreen.left;
            float rightDelta = rectScreen.right - mRenderData.screenWidth + mInputMethodOffsetX;
            float topDelta = rectScreen.top;
            float bottomDelta = rectScreen.bottom - mRenderData.screenHeight + mInputMethodOffsetY;
            float xAdjust = 0;
            float yAdjust = 0;

            if (rectScreen.right - rectScreen.left < adjustedScreenWidth) {
                // Image is narrower than the screen, so center it.
                xAdjust = -(rightDelta + leftDelta) / 2;
            } else if (leftDelta > 0 && rightDelta > 0) {
                // Panning the image left will show more of it.
                xAdjust = -Math.min(leftDelta, rightDelta);
            } else if (leftDelta < 0 && rightDelta < 0) {
                // Pan the image right.
                xAdjust = Math.min(-leftDelta, -rightDelta);
            }

            // Apply similar logic for yAdjust.
            if (rectScreen.bottom - rectScreen.top < adjustedScreenHeight) {
                yAdjust = -(bottomDelta + topDelta) / 2;
            } else if (topDelta > 0 && bottomDelta > 0) {
                yAdjust = -Math.min(topDelta, bottomDelta);
            } else if (topDelta < 0 && bottomDelta < 0) {
                yAdjust = Math.min(-topDelta, -bottomDelta);
            }

            mRenderData.transform.postTranslate(xAdjust, yAdjust);

            mViewer.transformationChanged();
        }
    }

    /**
     * Repositions the image by translating and zooming it, to keep the zoom level within sensible
     * limits. The minimum zoom level is chosen to avoid black space around all 4 sides. The
     * maximum zoom level is set arbitrarily, so that the user can zoom out again in a reasonable
     * time, and to prevent arithmetic overflow problems from displaying the image.
     */
    public void repositionImageWithZoom() {
        synchronized (mRenderData) {
            // Avoid division by zero in case this gets called before the image size is initialized.
            if (mRenderData.imageWidth == 0 || mRenderData.imageHeight == 0) {
                return;
            }

            // Zoom out if the zoom level is too high.
            float currentZoomLevel = mRenderData.transform.mapRadius(1.0f);
            if (currentZoomLevel > MAX_ZOOM_FACTOR) {
                mRenderData.transform.setScale(MAX_ZOOM_FACTOR, MAX_ZOOM_FACTOR);
            }

            // Get image size scaled to screen coordinates.
            float[] imageSize = {mRenderData.imageWidth, mRenderData.imageHeight};
            mRenderData.transform.mapVectors(imageSize);

            if (imageSize[0] < mRenderData.screenWidth && imageSize[1] < mRenderData.screenHeight) {
                // Displayed image is too small in both directions, so apply the minimum zoom
                // level needed to fit either the width or height.
                float scale = Math.min((float) mRenderData.screenWidth / mRenderData.imageWidth,
                                       (float) mRenderData.screenHeight / mRenderData.imageHeight);
                mRenderData.transform.setScale(scale, scale);
            }
        }

        repositionImage();
    }
}