blob: f607d3483f3386835a4cca891bd23660b0640796 [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
george822446ed02007-03-10 08:55:35 +000042void ScaledDIBSectionBuffer::setScale(int scale_) {
43 if (scale == scale_ || scale_ <= 0) return;
george822ac935e2006-05-29 13:57:39 +000044
george822446ed02007-03-10 08:55:35 +000045 scale = scale_;
46 if (scale == 100) scaling = false;
george82b2f8ae52006-09-16 14:41:12 +000047 else scaling = true;
george824ff66752006-11-20 15:55:05 +000048
49 // FIXME:
50 // Calculate the scale weight tabs must be in the ScalePixelBuffer class
51 freeWeightTabs();
52 calculateScaledBufferSize();
george822446ed02007-03-10 08:55:35 +000053 scaleFilters.makeWeightTabs(scaleFilterID, src_width, scaled_width, &xWeightTabs);
54 scaleFilters.makeWeightTabs(scaleFilterID, src_height, scaled_height, &yWeightTabs);
george824ff66752006-11-20 15:55:05 +000055
george82e4b3c2c2006-09-05 06:43:28 +000056 recreateBuffers();
Constantin Kaplinsky729598c2006-05-25 05:12:25 +000057}
58
Constantin Kaplinsky1ae2eb02006-05-26 05:24:24 +000059void ScaledDIBSectionBuffer::setPF(const PixelFormat &pf_) {
george82e4b3c2c2006-09-05 06:43:28 +000060 if (memcmp(&(ScaledPixelBuffer::pf), &pf_, sizeof(pf_)) == 0) return;
george82e3341e72006-08-02 15:23:39 +000061
george8243d1fa02006-11-26 11:18:38 +000062 if (!pf_.trueColour) throw rfb::UnsupportedPixelFormatException();
george820ba8cb72006-09-23 07:07:17 +000063
george82b2f8ae52006-09-16 14:41:12 +000064 pf = pf_;
Constantin Kaplinsky1ae2eb02006-05-26 05:24:24 +000065 if (scaling) {
george82b2f8ae52006-09-16 14:41:12 +000066 if (src_buffer) {
67 src_buffer->setPF(pf);
68 } else {
69 src_buffer = new ManagedPixelBuffer(pf, src_width, src_height);
george82e4b3c2c2006-09-05 06:43:28 +000070 src_data = &(src_buffer->data);
71 }
george82b2f8ae52006-09-16 14:41:12 +000072 if (memcmp(&(DIBSectionBuffer::getPF()), &RGB24, sizeof(PixelFormat)) != 0) {
73 DIBSectionBuffer::setPF(RGB24);
74 }
75 } else {
76 DIBSectionBuffer::setPF(pf);
Constantin Kaplinsky1ae2eb02006-05-26 05:24:24 +000077 }
Constantin Kaplinsky729598c2006-05-25 05:12:25 +000078}
79
80void ScaledDIBSectionBuffer::setSize(int src_width_, int src_height_) {
george82e3341e72006-08-02 15:23:39 +000081 if (src_width == src_width_ && src_height == src_height_) return;
Constantin Kaplinsky729598c2006-05-25 05:12:25 +000082 src_width = src_width_;
83 src_height = src_height_;
george824ff66752006-11-20 15:55:05 +000084
85 // FIXME:
86 // Calculate the scale weight tabs must be in the ScalePixelBuffer class
george823a1982e2007-11-04 07:35:51 +000087 recreateRowAccum();
george824ff66752006-11-20 15:55:05 +000088 freeWeightTabs();
89 calculateScaledBufferSize();
george822446ed02007-03-10 08:55:35 +000090 scaleFilters.makeWeightTabs(scaleFilterID, src_width, scaled_width, &xWeightTabs);
91 scaleFilters.makeWeightTabs(scaleFilterID, src_height, scaled_height, &yWeightTabs);
george824ff66752006-11-20 15:55:05 +000092
george82e4b3c2c2006-09-05 06:43:28 +000093 recreateBuffers();
Constantin Kaplinsky729598c2006-05-25 05:12:25 +000094}
95
george82770bbbc2007-03-12 10:48:09 +000096void ScaledDIBSectionBuffer::setScaleWindowSize(int width, int height) {
97 if (scaled_width == width && scaled_height == height) return;
98
99 freeWeightTabs();
100
101 scaled_width = width_ = width;
102 scaled_height = height_ = height;
103
104 if (scaled_width == src_width && scaled_height == src_height) scaling = false;
105 else scaling = true;
106 scale_ratio_x = (double)scaled_width / src_width;
107 scale_ratio_y = (double)scaled_height / src_height;
108 scale = (int)(scale_ratio_x * 100);
109
110 // FIXME:
111 // Calculate the scale weight tabs must be in the ScalePixelBuffer class
112 scaleFilters.makeWeightTabs(scaleFilterID, src_width, scaled_width, &xWeightTabs);
113 scaleFilters.makeWeightTabs(scaleFilterID, src_height, scaled_height, &yWeightTabs);
114
115 recreateBuffers();
116}
117
Constantin Kaplinsky729598c2006-05-25 05:12:25 +0000118void ScaledDIBSectionBuffer::recreateScaledBuffer() {
george82b2f8ae52006-09-16 14:41:12 +0000119 if (scaling && memcmp(&(DIBSectionBuffer::getPF()), &RGB24, sizeof(PixelFormat)) != 0) {
120 DIBSectionBuffer::setPF(RGB24);
george820ba8cb72006-09-23 07:07:17 +0000121 } else if (!scaling && (memcmp(&(DIBSectionBuffer::getPF()), &pf, sizeof(PixelFormat)) != 0)){
122 DIBSectionBuffer::setPF(pf);
george82b2f8ae52006-09-16 14:41:12 +0000123 } else {
george82e3341e72006-08-02 15:23:39 +0000124 DIBSectionBuffer::recreateBuffer();
george82e3341e72006-08-02 15:23:39 +0000125 }
Constantin Kaplinsky729598c2006-05-25 05:12:25 +0000126}
127
george82e4b3c2c2006-09-05 06:43:28 +0000128void ScaledDIBSectionBuffer::recreateBuffers() {
george820ba8cb72006-09-23 07:07:17 +0000129 // Recreate the source pixel buffer
130 if (src_width && src_height && pf.depth > 0) {
george82e4b3c2c2006-09-05 06:43:28 +0000131 if (scaling) {
george82b2f8ae52006-09-16 14:41:12 +0000132 if (src_buffer) {
george82e4b3c2c2006-09-05 06:43:28 +0000133 if (src_buffer->width() != src_width || src_buffer->width() != src_height)
134 src_buffer->setSize(src_width, src_height);
george82e4b3c2c2006-09-05 06:43:28 +0000135 } else {
george820ba8cb72006-09-23 07:07:17 +0000136 src_buffer = new ManagedPixelBuffer(pf, src_width, src_height);
george82e4b3c2c2006-09-05 06:43:28 +0000137 src_data = &(src_buffer->data);
george820ba8cb72006-09-23 07:07:17 +0000138 if (data) memcpy(src_buffer->data, data, src_width * src_height * (getPF().bpp/8));
george82e4b3c2c2006-09-05 06:43:28 +0000139 }
140 }
george820ba8cb72006-09-23 07:07:17 +0000141 }
142 // Recreate the scaled pixel buffer
143 recreateScaledBuffer();
144 if (scaling && src_buffer && data) scaleRect(Rect(0, 0, src_width, src_height));
145 else if (!scaling && src_buffer) {
146 if (src_buffer->data && data) memcpy(data, src_buffer->data, src_buffer->area() * (getPF().bpp/8));
147 delete src_buffer;
148 src_buffer = 0;
149 src_data = 0;
george82e4b3c2c2006-09-05 06:43:28 +0000150 }
151}
152
george8296b17b92006-09-05 15:44:35 +0000153void ScaledDIBSectionBuffer::calculateScaledBufferSize() {
154 ScaledPixelBuffer::calculateScaledBufferSize();
155 width_ = scaled_width;
156 height_ = scaled_height;
157}
158
Constantin Kaplinsky729598c2006-05-25 05:12:25 +0000159void ScaledDIBSectionBuffer::fillRect(const Rect &dest, Pixel pix) {
160 if (scaling) {
161 src_buffer->fillRect(dest, pix);
162 scaleRect(dest);
163 } else {
164 DIBSectionBuffer::fillRect(dest, pix);
165 }
166}
167
168void ScaledDIBSectionBuffer::imageRect(const Rect &dest, const void* pixels, int stride) {
169 if (scaling) {
170 src_buffer->imageRect(dest, pixels, stride);
171 scaleRect(dest);
172 } else {
173 DIBSectionBuffer::imageRect(dest, pixels, stride);
174 }
175}
176
177void ScaledDIBSectionBuffer::copyRect(const Rect &dest, const Point &move_by_delta) {
178 if (scaling) {
179 src_buffer->copyRect(dest, move_by_delta);
180 scaleRect(dest);
181 } else {
182 DIBSectionBuffer::copyRect(dest, move_by_delta);
183 }
184}
185
186void ScaledDIBSectionBuffer::maskRect(const Rect& r, const void* pixels, const void* mask_) {
187 if (scaling) {
188 src_buffer->maskRect(r, pixels, mask_);
189 scaleRect(r);
190 } else {
191 DIBSectionBuffer::maskRect(r, pixels, mask_);
192 }
193}
194
195void ScaledDIBSectionBuffer::maskRect(const Rect& r, Pixel pixel, const void* mask_) {
196 if (scaling) {
197 src_buffer->maskRect(r, pixel, mask_);
198 scaleRect(r);
199 } else {
200 DIBSectionBuffer::maskRect(r, pixel, mask_);
201 }
202}