blob: a3eb4fb3cf7e31489807e5fea7165e8f0b6345ec [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;
george823e43ed52007-11-05 17:13:10 +000044 if (!(getPixelFormat().trueColour) && scale_ != 100) throw rfb::UnsupportedPixelFormatException();
george82c79c7bb2007-11-05 11:22:14 +000045 ScaledPixelBuffer::setScale(scale_);
george822446ed02007-03-10 08:55:35 +000046 if (scale == 100) scaling = false;
george82b2f8ae52006-09-16 14:41:12 +000047 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
george823e43ed52007-11-05 17:13:10 +000054 if (!pf_.trueColour && isScaling()) throw rfb::UnsupportedPixelFormatException();
george820ba8cb72006-09-23 07:07:17 +000055
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;
Constantin Kaplinsky729598c2006-05-25 05:12:25 +000074 src_width = src_width_;
75 src_height = src_height_;
george824ff66752006-11-20 15:55:05 +000076
77 // FIXME:
78 // Calculate the scale weight tabs must be in the ScalePixelBuffer class
george823a1982e2007-11-04 07:35:51 +000079 recreateRowAccum();
george824ff66752006-11-20 15:55:05 +000080 freeWeightTabs();
81 calculateScaledBufferSize();
george822446ed02007-03-10 08:55:35 +000082 scaleFilters.makeWeightTabs(scaleFilterID, src_width, scaled_width, &xWeightTabs);
83 scaleFilters.makeWeightTabs(scaleFilterID, src_height, scaled_height, &yWeightTabs);
george824ff66752006-11-20 15:55:05 +000084
george82e4b3c2c2006-09-05 06:43:28 +000085 recreateBuffers();
Constantin Kaplinsky729598c2006-05-25 05:12:25 +000086}
87
george82770bbbc2007-03-12 10:48:09 +000088void ScaledDIBSectionBuffer::setScaleWindowSize(int width, int height) {
89 if (scaled_width == width && scaled_height == height) return;
90
91 freeWeightTabs();
92
93 scaled_width = width_ = width;
94 scaled_height = height_ = height;
95
96 if (scaled_width == src_width && scaled_height == src_height) scaling = false;
97 else scaling = true;
98 scale_ratio_x = (double)scaled_width / src_width;
99 scale_ratio_y = (double)scaled_height / src_height;
100 scale = (int)(scale_ratio_x * 100);
101
102 // FIXME:
103 // Calculate the scale weight tabs must be in the ScalePixelBuffer class
104 scaleFilters.makeWeightTabs(scaleFilterID, src_width, scaled_width, &xWeightTabs);
105 scaleFilters.makeWeightTabs(scaleFilterID, src_height, scaled_height, &yWeightTabs);
106
107 recreateBuffers();
108}
109
Constantin Kaplinsky729598c2006-05-25 05:12:25 +0000110void ScaledDIBSectionBuffer::recreateScaledBuffer() {
george82b2f8ae52006-09-16 14:41:12 +0000111 if (scaling && memcmp(&(DIBSectionBuffer::getPF()), &RGB24, sizeof(PixelFormat)) != 0) {
112 DIBSectionBuffer::setPF(RGB24);
george820ba8cb72006-09-23 07:07:17 +0000113 } else if (!scaling && (memcmp(&(DIBSectionBuffer::getPF()), &pf, sizeof(PixelFormat)) != 0)){
114 DIBSectionBuffer::setPF(pf);
george82b2f8ae52006-09-16 14:41:12 +0000115 } else {
george82e3341e72006-08-02 15:23:39 +0000116 DIBSectionBuffer::recreateBuffer();
george82e3341e72006-08-02 15:23:39 +0000117 }
Constantin Kaplinsky729598c2006-05-25 05:12:25 +0000118}
119
george82e4b3c2c2006-09-05 06:43:28 +0000120void ScaledDIBSectionBuffer::recreateBuffers() {
george820ba8cb72006-09-23 07:07:17 +0000121 // Recreate the source pixel buffer
122 if (src_width && src_height && pf.depth > 0) {
george82e4b3c2c2006-09-05 06:43:28 +0000123 if (scaling) {
george82b2f8ae52006-09-16 14:41:12 +0000124 if (src_buffer) {
george82e4b3c2c2006-09-05 06:43:28 +0000125 if (src_buffer->width() != src_width || src_buffer->width() != src_height)
126 src_buffer->setSize(src_width, src_height);
george82e4b3c2c2006-09-05 06:43:28 +0000127 } else {
george820ba8cb72006-09-23 07:07:17 +0000128 src_buffer = new ManagedPixelBuffer(pf, src_width, src_height);
george82e4b3c2c2006-09-05 06:43:28 +0000129 src_data = &(src_buffer->data);
george820ba8cb72006-09-23 07:07:17 +0000130 if (data) memcpy(src_buffer->data, data, src_width * src_height * (getPF().bpp/8));
george82e4b3c2c2006-09-05 06:43:28 +0000131 }
132 }
george820ba8cb72006-09-23 07:07:17 +0000133 }
134 // Recreate the scaled pixel buffer
135 recreateScaledBuffer();
136 if (scaling && src_buffer && data) scaleRect(Rect(0, 0, src_width, src_height));
137 else if (!scaling && src_buffer) {
138 if (src_buffer->data && data) memcpy(data, src_buffer->data, src_buffer->area() * (getPF().bpp/8));
139 delete src_buffer;
140 src_buffer = 0;
141 src_data = 0;
george82e4b3c2c2006-09-05 06:43:28 +0000142 }
143}
144
george8296b17b92006-09-05 15:44:35 +0000145void ScaledDIBSectionBuffer::calculateScaledBufferSize() {
146 ScaledPixelBuffer::calculateScaledBufferSize();
147 width_ = scaled_width;
148 height_ = scaled_height;
149}
150
Constantin Kaplinsky729598c2006-05-25 05:12:25 +0000151void ScaledDIBSectionBuffer::fillRect(const Rect &dest, Pixel pix) {
152 if (scaling) {
153 src_buffer->fillRect(dest, pix);
154 scaleRect(dest);
155 } else {
156 DIBSectionBuffer::fillRect(dest, pix);
157 }
158}
159
160void ScaledDIBSectionBuffer::imageRect(const Rect &dest, const void* pixels, int stride) {
161 if (scaling) {
162 src_buffer->imageRect(dest, pixels, stride);
163 scaleRect(dest);
164 } else {
165 DIBSectionBuffer::imageRect(dest, pixels, stride);
166 }
167}
168
169void ScaledDIBSectionBuffer::copyRect(const Rect &dest, const Point &move_by_delta) {
170 if (scaling) {
171 src_buffer->copyRect(dest, move_by_delta);
172 scaleRect(dest);
173 } else {
174 DIBSectionBuffer::copyRect(dest, move_by_delta);
175 }
176}
177
178void ScaledDIBSectionBuffer::maskRect(const Rect& r, const void* pixels, const void* mask_) {
179 if (scaling) {
180 src_buffer->maskRect(r, pixels, mask_);
181 scaleRect(r);
182 } else {
183 DIBSectionBuffer::maskRect(r, pixels, mask_);
184 }
185}
186
187void ScaledDIBSectionBuffer::maskRect(const Rect& r, Pixel pixel, const void* mask_) {
188 if (scaling) {
189 src_buffer->maskRect(r, pixel, mask_);
190 scaleRect(r);
191 } else {
192 DIBSectionBuffer::maskRect(r, pixel, mask_);
193 }
194}