blob: 20d0f89a94aabec68a04c1ca1328f21c40204072 [file] [log] [blame]
Constantin Kaplinsky729598c2006-05-25 05:12:25 +00001/* Copyright (C) 2006 TightVNC Team. All Rights Reserved.
2 *
3 * This is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License as published by
5 * the Free Software Foundation; either version 2 of the License, or
6 * (at your option) any later version.
7 *
8 * This software is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
12 *
13 * You should have received a copy of the GNU General Public License
14 * along with this software; if not, write to the Free Software
15 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307,
16 * USA.
17 *
18 * TightVNC distribution homepage on the Web: http://www.tightvnc.com/
19 *
20 */
21
22// -=- ScaledDIBSectionBuffer.cxx
23
24#include <math.h>
25
26#include <rfb_win32/ScaledDIBSectionBuffer.h>
27
28using namespace rfb;
29using namespace win32;
30
george82b2f8ae52006-09-16 14:41:12 +000031const PixelFormat RGB24(32, 24, 0, 1, 255, 255, 255, 16, 8, 0);
32
Constantin Kaplinsky729598c2006-05-25 05:12:25 +000033ScaledDIBSectionBuffer::ScaledDIBSectionBuffer(HWND window)
34 : src_buffer(0), scaling(false), DIBSectionBuffer(window) {
george82df9f31c2006-09-16 11:06:07 +000035 scaled_data = &data;
Constantin Kaplinsky729598c2006-05-25 05:12:25 +000036}
37
38ScaledDIBSectionBuffer::~ScaledDIBSectionBuffer() {
39 if (src_buffer) delete src_buffer;
40}
41
george82bf0adb32006-07-29 10:29:41 +000042void ScaledDIBSectionBuffer::setScaleRatio(double scale_ratio_) {
george820ba8cb72006-09-23 07:07:17 +000043 if (scale_ratio == scale_ratio_ || scale_ratio <= 0) return;
george822ac935e2006-05-29 13:57:39 +000044
george82b2f8ae52006-09-16 14:41:12 +000045 scale_ratio = scale_ratio_;
46 if (scale_ratio == 1) scaling = false;
47 else scaling = true;
george82e4b3c2c2006-09-05 06:43:28 +000048 recreateBuffers();
Constantin Kaplinsky729598c2006-05-25 05:12:25 +000049}
50
Constantin Kaplinsky1ae2eb02006-05-26 05:24:24 +000051void ScaledDIBSectionBuffer::setPF(const PixelFormat &pf_) {
george82e4b3c2c2006-09-05 06:43:28 +000052 if (memcmp(&(ScaledPixelBuffer::pf), &pf_, sizeof(pf_)) == 0) return;
george82e3341e72006-08-02 15:23:39 +000053
george820ba8cb72006-09-23 07:07:17 +000054 if (!pf_.trueColour) throw rfb::UnsupportedPixelFormatException();
55
george82b2f8ae52006-09-16 14:41:12 +000056 pf = pf_;
Constantin Kaplinsky1ae2eb02006-05-26 05:24:24 +000057 if (scaling) {
george82b2f8ae52006-09-16 14:41:12 +000058 if (src_buffer) {
59 src_buffer->setPF(pf);
60 } else {
61 src_buffer = new ManagedPixelBuffer(pf, src_width, src_height);
george82e4b3c2c2006-09-05 06:43:28 +000062 src_data = &(src_buffer->data);
63 }
george82b2f8ae52006-09-16 14:41:12 +000064 if (memcmp(&(DIBSectionBuffer::getPF()), &RGB24, sizeof(PixelFormat)) != 0) {
65 DIBSectionBuffer::setPF(RGB24);
66 }
67 } else {
68 DIBSectionBuffer::setPF(pf);
Constantin Kaplinsky1ae2eb02006-05-26 05:24:24 +000069 }
Constantin Kaplinsky729598c2006-05-25 05:12:25 +000070}
71
72void ScaledDIBSectionBuffer::setSize(int src_width_, int src_height_) {
george82e3341e72006-08-02 15:23:39 +000073 if (src_width == src_width_ && src_height == src_height_) return;
74
Constantin Kaplinsky729598c2006-05-25 05:12:25 +000075 src_width = src_width_;
76 src_height = src_height_;
george82e4b3c2c2006-09-05 06:43:28 +000077 recreateBuffers();
Constantin Kaplinsky729598c2006-05-25 05:12:25 +000078}
79
80void ScaledDIBSectionBuffer::recreateScaledBuffer() {
george82b2f8ae52006-09-16 14:41:12 +000081 calculateScaledBufferSize();
82 if (scaling && memcmp(&(DIBSectionBuffer::getPF()), &RGB24, sizeof(PixelFormat)) != 0) {
83 DIBSectionBuffer::setPF(RGB24);
george820ba8cb72006-09-23 07:07:17 +000084 } else if (!scaling && (memcmp(&(DIBSectionBuffer::getPF()), &pf, sizeof(PixelFormat)) != 0)){
85 DIBSectionBuffer::setPF(pf);
george82b2f8ae52006-09-16 14:41:12 +000086 } else {
george82e3341e72006-08-02 15:23:39 +000087 DIBSectionBuffer::recreateBuffer();
george82e3341e72006-08-02 15:23:39 +000088 }
Constantin Kaplinsky729598c2006-05-25 05:12:25 +000089}
90
george82e4b3c2c2006-09-05 06:43:28 +000091void ScaledDIBSectionBuffer::recreateBuffers() {
george820ba8cb72006-09-23 07:07:17 +000092 // Recreate the source pixel buffer
93 if (src_width && src_height && pf.depth > 0) {
george82e4b3c2c2006-09-05 06:43:28 +000094 if (scaling) {
george82b2f8ae52006-09-16 14:41:12 +000095 if (src_buffer) {
george82e4b3c2c2006-09-05 06:43:28 +000096 if (src_buffer->width() != src_width || src_buffer->width() != src_height)
97 src_buffer->setSize(src_width, src_height);
george82e4b3c2c2006-09-05 06:43:28 +000098 } else {
george820ba8cb72006-09-23 07:07:17 +000099 src_buffer = new ManagedPixelBuffer(pf, src_width, src_height);
george82e4b3c2c2006-09-05 06:43:28 +0000100 src_data = &(src_buffer->data);
george820ba8cb72006-09-23 07:07:17 +0000101 if (data) memcpy(src_buffer->data, data, src_width * src_height * (getPF().bpp/8));
george82e4b3c2c2006-09-05 06:43:28 +0000102 }
103 }
george820ba8cb72006-09-23 07:07:17 +0000104 }
105 // Recreate the scaled pixel buffer
106 recreateScaledBuffer();
107 if (scaling && src_buffer && data) scaleRect(Rect(0, 0, src_width, src_height));
108 else if (!scaling && src_buffer) {
109 if (src_buffer->data && data) memcpy(data, src_buffer->data, src_buffer->area() * (getPF().bpp/8));
110 delete src_buffer;
111 src_buffer = 0;
112 src_data = 0;
george82e4b3c2c2006-09-05 06:43:28 +0000113 }
114}
115
george8296b17b92006-09-05 15:44:35 +0000116void ScaledDIBSectionBuffer::calculateScaledBufferSize() {
117 ScaledPixelBuffer::calculateScaledBufferSize();
118 width_ = scaled_width;
119 height_ = scaled_height;
120}
121
Constantin Kaplinsky729598c2006-05-25 05:12:25 +0000122void ScaledDIBSectionBuffer::fillRect(const Rect &dest, Pixel pix) {
123 if (scaling) {
124 src_buffer->fillRect(dest, pix);
125 scaleRect(dest);
126 } else {
127 DIBSectionBuffer::fillRect(dest, pix);
128 }
129}
130
131void ScaledDIBSectionBuffer::imageRect(const Rect &dest, const void* pixels, int stride) {
132 if (scaling) {
133 src_buffer->imageRect(dest, pixels, stride);
134 scaleRect(dest);
135 } else {
136 DIBSectionBuffer::imageRect(dest, pixels, stride);
137 }
138}
139
140void ScaledDIBSectionBuffer::copyRect(const Rect &dest, const Point &move_by_delta) {
141 if (scaling) {
142 src_buffer->copyRect(dest, move_by_delta);
143 scaleRect(dest);
144 } else {
145 DIBSectionBuffer::copyRect(dest, move_by_delta);
146 }
147}
148
149void ScaledDIBSectionBuffer::maskRect(const Rect& r, const void* pixels, const void* mask_) {
150 if (scaling) {
151 src_buffer->maskRect(r, pixels, mask_);
152 scaleRect(r);
153 } else {
154 DIBSectionBuffer::maskRect(r, pixels, mask_);
155 }
156}
157
158void ScaledDIBSectionBuffer::maskRect(const Rect& r, Pixel pixel, const void* mask_) {
159 if (scaling) {
160 src_buffer->maskRect(r, pixel, mask_);
161 scaleRect(r);
162 } else {
163 DIBSectionBuffer::maskRect(r, pixel, mask_);
164 }
165}