blob: 5c9611cdbc11f9f00073ddc1dc0b44c18d832c1b [file] [log] [blame]
Pierre Ossmanac13abe2014-02-07 14:46:26 +01001/* Copyright 2011-2014 Pierre Ossman for Cendio AB
Pierre Ossmanc18753c2011-06-17 07:35:56 +00002 *
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
Peter Åstrandc359f362011-08-23 12:04:46 +000019#ifdef HAVE_CONFIG_H
20#include <config.h>
21#endif
22
Pierre Ossmanc18753c2011-06-17 07:35:56 +000023#include <ApplicationServices/ApplicationServices.h>
24
25#include <FL/Fl_Window.H>
26#include <FL/x.H>
27
28#include <rfb/LogWriter.h>
29#include <rfb/Exception.h>
30
Pierre Ossman8ca4c1d2014-09-22 12:54:26 +020031#include "i18n.h"
Pierre Ossmanc18753c2011-06-17 07:35:56 +000032#include "OSXPixelBuffer.h"
33
34using namespace rfb;
35
Pierre Ossmanac13abe2014-02-07 14:46:26 +010036static rfb::LogWriter vlog("OSXPixelBuffer");
Pierre Ossmanc18753c2011-06-17 07:35:56 +000037
Pierre Ossmanac13abe2014-02-07 14:46:26 +010038OSXPixelBuffer::OSXPixelBuffer(int width, int height) :
39 PlatformPixelBuffer(rfb::PixelFormat(32, 24, false, true,
40 255, 255, 255, 16, 8, 0),
Pierre Ossman2e5a1062014-01-30 17:57:27 +010041 width, height, NULL, width),
Pierre Ossman41a0c152016-12-29 15:57:21 +010042 image(NULL)
Pierre Ossmanc18753c2011-06-17 07:35:56 +000043{
44 CGColorSpaceRef lut;
Pierre Ossman41a0c152016-12-29 15:57:21 +010045 CGDataProviderRef provider;
Pierre Ossmanc18753c2011-06-17 07:35:56 +000046
Pierre Ossmanac13abe2014-02-07 14:46:26 +010047 data = new rdr::U8[width * height * format.bpp/8];
48 if (data == NULL)
Pierre Ossman86750632014-10-10 13:33:19 +020049 throw rfb::Exception(_("Not enough memory for framebuffer"));
Pierre Ossmanac13abe2014-02-07 14:46:26 +010050
Pierre Ossman41a0c152016-12-29 15:57:21 +010051 lut = CGDisplayCopyColorSpace(kCGDirectMainDisplay);
52 if (!lut) {
53 vlog.error(_("Could not get screen color space. Using slower fallback."));
54 lut = CGColorSpaceCreateDeviceRGB();
55 if (!lut)
56 throw rfb::Exception("CGColorSpaceCreateDeviceRGB");
57 }
Pierre Ossmanc18753c2011-06-17 07:35:56 +000058
Pierre Ossman41a0c152016-12-29 15:57:21 +010059 provider = CGDataProviderCreateWithData(NULL, data,
60 width * height * 4, NULL);
61 if (!provider)
62 throw rfb::Exception("CGDataProviderCreateWithData");
63
64 image = CGImageCreate(width, height, 8, 32, width * 4, lut,
65 kCGImageAlphaNoneSkipFirst | kCGBitmapByteOrder32Little,
66 provider, NULL, false, kCGRenderingIntentDefault);
Pierre Ossmanc18753c2011-06-17 07:35:56 +000067 CGColorSpaceRelease(lut);
Pierre Ossman41a0c152016-12-29 15:57:21 +010068 CGDataProviderRelease(provider);
69 if (!image)
70 throw rfb::Exception("CGImageCreate");
Pierre Ossmanc18753c2011-06-17 07:35:56 +000071}
72
73
Pierre Ossmanac13abe2014-02-07 14:46:26 +010074OSXPixelBuffer::~OSXPixelBuffer()
Pierre Ossmanc18753c2011-06-17 07:35:56 +000075{
Pierre Ossman41a0c152016-12-29 15:57:21 +010076 CGImageRelease(image);
Pierre Ossmanac13abe2014-02-07 14:46:26 +010077 delete [] data;
Pierre Ossmanc18753c2011-06-17 07:35:56 +000078}
79
80
Pierre Ossmanac13abe2014-02-07 14:46:26 +010081void OSXPixelBuffer::draw(int src_x, int src_y, int x, int y, int w, int h)
Pierre Ossmanc18753c2011-06-17 07:35:56 +000082{
Pierre Ossmanc18753c2011-06-17 07:35:56 +000083 CGContextRef gc;
Pierre Ossman41a0c152016-12-29 15:57:21 +010084 CGRect rect;
Pierre Ossmanc18753c2011-06-17 07:35:56 +000085
Pierre Ossman41a0c152016-12-29 15:57:21 +010086 gc = fl_gc;
Pierre Ossmanc18753c2011-06-17 07:35:56 +000087
88 CGContextSaveGState(gc);
89
Pierre Ossman41a0c152016-12-29 15:57:21 +010090 // macOS Coordinates are from bottom left, not top left
91 src_y = height() - (src_y + h);
92 y = Fl_Window::current()->h() - (y + h);
Pierre Ossmanc18753c2011-06-17 07:35:56 +000093
Pierre Ossman41a0c152016-12-29 15:57:21 +010094 // Reset the transformation matrix back to the default identity
95 // matrix as otherwise we get a massive performance hit
96 CGContextConcatCTM(gc, CGAffineTransformInvert(CGContextGetCTM(gc)));
97
98 // We have to use clipping to partially display an image
99 rect.origin.x = x;
100 rect.origin.y = y;
Pierre Ossmanc18753c2011-06-17 07:35:56 +0000101 rect.size.width = w;
102 rect.size.height = h;
103
104 CGContextClipToRect(gc, rect);
Pierre Ossmanc18753c2011-06-17 07:35:56 +0000105
106 rect.origin.x = x - src_x;
Pierre Ossman41a0c152016-12-29 15:57:21 +0100107 rect.origin.y = y - src_y;
Pierre Ossmanc18753c2011-06-17 07:35:56 +0000108 rect.size.width = width();
Pierre Ossman41a0c152016-12-29 15:57:21 +0100109 rect.size.height = height();
Pierre Ossmanc18753c2011-06-17 07:35:56 +0000110
Pierre Ossmancb9eefa2012-08-17 13:37:42 +0000111 CGContextDrawImage(gc, rect, image);
Pierre Ossmanc18753c2011-06-17 07:35:56 +0000112
113 CGContextRestoreGState(gc);
114}