// // "$Id: Fl_Copy_Surface.cxx 11898 2016-08-27 15:17:02Z manolo $" // // Copy-to-clipboard code for the Fast Light Tool Kit (FLTK). // // Copyright 1998-2016 by Bill Spitzak and others. // // This library is free software. Distribution and use rights are outlined in // the file "COPYING" which should have been included with this file. If this // file is missing or damaged, see the license at: // // http://www.fltk.org/COPYING.php // // Please report all bugs and problems on the following page: // // http://www.fltk.org/str.php // #include #include #if defined(__APPLE__) #include Fl_Quartz_Surface_::Fl_Quartz_Surface_(int w, int h) : Fl_System_Printer(), width(w), height(h) { } int Fl_Quartz_Surface_::printable_rect(int *w, int *h) { *w = width; *h = height; return 0; } const char *Fl_Quartz_Surface_::class_id = "Fl_Quartz_Surface_"; #elif defined(WIN32) Fl_GDI_Surface_::Fl_GDI_Surface_() : Fl_Paged_Device() { driver(new Fl_GDI_Graphics_Driver); depth = 0; } Fl_GDI_Surface_::~Fl_GDI_Surface_() { delete driver(); } void Fl_GDI_Surface_::translate(int x, int y) { GetWindowOrgEx(fl_gc, origins+depth); SetWindowOrgEx(fl_gc, origins[depth].x - x, origins[depth].y - y, NULL); if (depth < sizeof(origins)/sizeof(POINT)) depth++; else Fl::warning("Fl_GDI_Surface_: translate stack overflow!"); } void Fl_GDI_Surface_::untranslate() { if (depth > 0) depth--; SetWindowOrgEx(fl_gc, origins[depth].x, origins[depth].y, NULL); } const char *Fl_GDI_Surface_::class_id = "Fl_GDI_Surface_"; #endif const char *Fl_Copy_Surface::class_id = "Fl_Copy_Surface"; /** Constructor. \param w and \param h are the width and height of the clipboard surface in pixels where drawing will occur. */ Fl_Copy_Surface::Fl_Copy_Surface(int w, int h) : Fl_Surface_Device(NULL) { width = w; height = h; #ifdef __APPLE__ helper = new Fl_Quartz_Surface_(width, height); driver(helper->driver()); prepare_copy_pdf_and_tiff(w, h); oldgc = fl_gc; #elif defined(WIN32) helper = new Fl_GDI_Surface_(); driver(helper->driver()); oldgc = fl_gc; // exact computation of factor from screen units to EnhMetaFile units (0.01 mm) HDC hdc = GetDC(NULL); int hmm = GetDeviceCaps(hdc, HORZSIZE); int hdots = GetDeviceCaps(hdc, HORZRES); int vmm = GetDeviceCaps(hdc, VERTSIZE); int vdots = GetDeviceCaps(hdc, VERTRES); int dhr = GetDeviceCaps(hdc, DESKTOPHORZRES); // true number of pixels on display ReleaseDC(NULL, hdc); float factorw = (100.f * hmm) / hdots; float factorh = (100.f * vmm) / vdots; // Global display scaling factor: 1, 1.25, 1.5, 1.75, etc... float scaling = dhr/float(hdots); RECT rect; rect.left = 0; rect.top = 0; rect.right = (LONG)(w * factorw / scaling); rect.bottom = (LONG)(h * factorh / scaling); gc = CreateEnhMetaFile (NULL, NULL, &rect, NULL); if (gc != NULL) { SetTextAlign(gc, TA_BASELINE|TA_LEFT); SetBkMode(gc, TRANSPARENT); } #else // Xlib helper = new Fl_Xlib_Surface_(); driver(helper->driver()); Fl::first_window()->make_current(); oldwindow = fl_xid(Fl::first_window()); xid = fl_create_offscreen(w,h); Fl_Surface_Device *present_surface = Fl_Surface_Device::surface(); set_current(); fl_color(FL_WHITE); fl_rectf(0, 0, w, h); present_surface->set_current(); #endif } /** Destructor. */ Fl_Copy_Surface::~Fl_Copy_Surface() { #ifdef __APPLE__ complete_copy_pdf_and_tiff(); fl_gc = oldgc; delete (Fl_Quartz_Surface_*)helper; #elif defined(WIN32) if(oldgc == fl_gc) oldgc = NULL; HENHMETAFILE hmf = CloseEnhMetaFile (gc); if ( hmf != NULL ) { if ( OpenClipboard (NULL) ){ EmptyClipboard (); SetClipboardData (CF_ENHMETAFILE, hmf); CloseClipboard (); } DeleteEnhMetaFile(hmf); } DeleteDC(gc); fl_gc = oldgc; delete (Fl_GDI_Surface_*)helper; #else // Xlib fl_pop_clip(); unsigned char *data = fl_read_image(NULL,0,0,width,height,0); fl_window = oldwindow; _ss->set_current(); Fl::copy_image(data,width,height,1); delete[] data; fl_delete_offscreen(xid); delete (Fl_Xlib_Surface_*)helper; #endif } /** Copies a widget in the clipboard \param widget any FLTK widget (e.g., standard, custom, window, GL view) to copy \param delta_x and \param delta_y give the position in the clipboard of the top-left corner of the widget */ void Fl_Copy_Surface::draw(Fl_Widget* widget, int delta_x, int delta_y) { helper->print_widget(widget, delta_x, delta_y); } void Fl_Copy_Surface::set_current() { #if defined(__APPLE__) || defined(WIN32) fl_gc = gc; fl_window = (Window)1; Fl_Surface_Device::set_current(); #else fl_window=xid; _ss = Fl_Surface_Device::surface(); Fl_Surface_Device::set_current(); fl_push_no_clip(); #endif } #if defined(__APPLE__) size_t Fl_Copy_Surface::MyPutBytes(void* info, const void* buffer, size_t count) { CFDataAppendBytes ((CFMutableDataRef) info, (const UInt8 *)buffer, count); return count; } void Fl_Copy_Surface::init_PDF_context(int w, int h) { CGRect bounds = CGRectMake(0, 0, w, h ); pdfdata = CFDataCreateMutable(NULL, 0); CGDataConsumerRef myconsumer; #if MAC_OS_X_VERSION_MAX_ALLOWED >= 1040 if (&CGDataConsumerCreateWithCFData != NULL) { myconsumer = CGDataConsumerCreateWithCFData(pdfdata); // 10.4 } else #endif { static CGDataConsumerCallbacks callbacks = { Fl_Copy_Surface::MyPutBytes, NULL }; myconsumer = CGDataConsumerCreate ((void*) pdfdata, &callbacks); } gc = CGPDFContextCreate (myconsumer, &bounds, NULL); CGDataConsumerRelease (myconsumer); } void Fl_Copy_Surface::prepare_copy_pdf_and_tiff(int w, int h) { init_PDF_context(w, h); if (gc == NULL) return; CGRect bounds = CGRectMake(0, 0, w, h ); CGContextBeginPage (gc, &bounds); CGContextTranslateCTM(gc, 0, h); CGContextScaleCTM(gc, 1.0f, -1.0f); CGContextSaveGState(gc); } void Fl_Copy_Surface::draw_decorated_window(Fl_Window* win, int delta_x, int delta_y) { int bt = win->decorated_h() - win->h(); draw(win, delta_x, bt + delta_y ); // draw the window content if (win->border()) { // draw the window title bar CGContextSaveGState(gc); CGContextTranslateCTM(gc, delta_x, bt + delta_y); CGContextScaleCTM(gc, 1, -1); Fl_X::clip_to_rounded_corners(gc, win->w(), bt); void *layer = Fl_X::get_titlebar_layer(win); if (layer) { CGColorSpaceRef cspace = CGColorSpaceCreateDeviceRGB(); // for unknown reason, rendering the layer to the Fl_Copy_Surface pdf graphics context does not work; // we use an auxiliary bitmap context CGContextRef auxgc = CGBitmapContextCreate(NULL, win->w(), bt, 8, 0, cspace, kCGImageAlphaPremultipliedLast); CGColorSpaceRelease(cspace); CGContextTranslateCTM(auxgc, 0, bt); CGContextScaleCTM(auxgc, 1, -1); Fl_X::draw_layer_to_context(layer, auxgc, win->w(), bt); Fl_RGB_Image *image = new Fl_RGB_Image((const uchar*)CGBitmapContextGetData(auxgc), win->w(), bt, 4, CGBitmapContextGetBytesPerRow(auxgc)); // 10.2 image->draw(0, 0); delete image; CGContextRelease(auxgc); } else { CGImageRef img = Fl_X::CGImage_from_window_rect(win, 0, -bt, win->w(), bt); CGContextDrawImage(gc, CGRectMake(0, 0, win->w(), bt), img); CFRelease(img); } CGContextRestoreGState(gc); } } #else /** Copies a window and its borders and title bar to the clipboard. \param win an FLTK window to copy \param delta_x and \param delta_y give the position in the clipboard of the top-left corner of the window's title bar */ void Fl_Copy_Surface::draw_decorated_window(Fl_Window* win, int delta_x, int delta_y) { helper->draw_decorated_window(win, delta_x, delta_y, this); } #endif // __APPLE__ #if !(defined(__APPLE__) || defined(WIN32) || defined(FL_DOXYGEN)) /* graphics driver that translates all graphics coordinates before calling Xlib */ class Fl_translated_Xlib_Graphics_Driver_ : public Fl_Xlib_Graphics_Driver { int offset_x, offset_y; // translation between user and graphical coordinates: graphical = user + offset unsigned depth; // depth of translation stack int stack_x[20], stack_y[20]; // translation stack allowing cumulative translations public: static const char *class_id; const char *class_name() {return class_id;}; Fl_translated_Xlib_Graphics_Driver_() { offset_x = 0; offset_y = 0; depth = 0; } virtual ~Fl_translated_Xlib_Graphics_Driver_() {}; void translate_all(int dx, int dy) { // reversibly adds dx,dy to the offset between user and graphical coordinates stack_x[depth] = offset_x; stack_y[depth] = offset_y; offset_x = stack_x[depth] + dx; offset_y = stack_y[depth] + dy; push_matrix(); translate(dx, dy); if (depth < sizeof(stack_x)/sizeof(int)) depth++; else Fl::warning("%s: translate stack overflow!", class_id); } void untranslate_all() { // undoes previous translate_all() if (depth > 0) depth--; offset_x = stack_x[depth]; offset_y = stack_y[depth]; pop_matrix(); } void rect(int x, int y, int w, int h) { Fl_Xlib_Graphics_Driver::rect(x+offset_x, y+offset_y, w, h); } void rectf(int x, int y, int w, int h) { Fl_Xlib_Graphics_Driver::rectf(x+offset_x, y+offset_y, w, h); } void xyline(int x, int y, int x1) { Fl_Xlib_Graphics_Driver::xyline(x+offset_x, y+offset_y, x1+offset_x); } void xyline(int x, int y, int x1, int y2) { Fl_Xlib_Graphics_Driver::xyline(x+offset_x, y+offset_y, x1+offset_x, y2+offset_y); } void xyline(int x, int y, int x1, int y2, int x3) { Fl_Xlib_Graphics_Driver::xyline(x+offset_x, y+offset_y, x1+offset_x, y2+offset_y, x3+offset_x); } void yxline(int x, int y, int y1) { Fl_Xlib_Graphics_Driver::yxline(x+offset_x, y+offset_y, y1+offset_y); } void yxline(int x, int y, int y1, int x2) { Fl_Xlib_Graphics_Driver::yxline(x+offset_x, y+offset_y, y1+offset_y, x2+offset_x); } void yxline(int x, int y, int y1, int x2, int y3) { Fl_Xlib_Graphics_Driver::yxline(x+offset_x, y+offset_y, y1+offset_y, x2+offset_x, y3+offset_y); } void line(int x, int y, int x1, int y1) { Fl_Xlib_Graphics_Driver::line(x+offset_x, y+offset_y, x1+offset_x, y1+offset_y); } void line(int x, int y, int x1, int y1, int x2, int y2) { Fl_Xlib_Graphics_Driver::line(x+offset_x, y+offset_y, x1+offset_x, y1+offset_y, x2+offset_x, y2+offset_y); } void draw(const char* str, int n, int x, int y) { Fl_Xlib_Graphics_Driver::draw(str, n, x+offset_x, y+offset_y); } void draw(int angle, const char *str, int n, int x, int y) { Fl_Xlib_Graphics_Driver::draw(angle, str, n, x+offset_x, y+offset_y); } void rtl_draw(const char* str, int n, int x, int y) { Fl_Xlib_Graphics_Driver::rtl_draw(str, n, x+offset_x, y+offset_y); } void draw(Fl_Pixmap *pxm, int XP, int YP, int WP, int HP, int cx, int cy) { XP += offset_x; YP += offset_y; translate_all(-offset_x, -offset_y); Fl_Xlib_Graphics_Driver::draw(pxm, XP, YP, WP,HP,cx,cy); untranslate_all(); } void draw(Fl_Bitmap *bm, int XP, int YP, int WP, int HP, int cx, int cy) { XP += offset_x; YP += offset_y; translate_all(-offset_x, -offset_y); Fl_Xlib_Graphics_Driver::draw(bm, XP, YP, WP,HP,cx,cy); untranslate_all(); } void draw(Fl_RGB_Image *img, int XP, int YP, int WP, int HP, int cx, int cy) { XP += offset_x; YP += offset_y; translate_all(-offset_x, -offset_y); Fl_Xlib_Graphics_Driver::draw(img, XP, YP, WP,HP,cx,cy); untranslate_all(); } void draw_image(const uchar* buf, int X,int Y,int W,int H, int D=3, int L=0) { X += offset_x; Y += offset_y; translate_all(-offset_x, -offset_y); Fl_Xlib_Graphics_Driver::draw_image(buf, X, Y, W,H,D,L); untranslate_all(); } void draw_image(Fl_Draw_Image_Cb cb, void* data, int X,int Y,int W,int H, int D=3) { X += offset_x; Y += offset_y; translate_all(-offset_x, -offset_y); Fl_Xlib_Graphics_Driver::draw_image(cb, data, X, Y, W,H,D); untranslate_all(); } void draw_image_mono(const uchar* buf, int X,int Y,int W,int H, int D=1, int L=0) { X += offset_x; Y += offset_y; translate_all(-offset_x, -offset_y); Fl_Xlib_Graphics_Driver::draw_image_mono(buf, X, Y, W,H,D,L); untranslate_all(); } void draw_image_mono(Fl_Draw_Image_Cb cb, void* data, int X,int Y,int W,int H, int D=1) { X += offset_x; Y += offset_y; translate_all(-offset_x, -offset_y); Fl_Xlib_Graphics_Driver::draw_image_mono(cb, data, X, Y, W,H,D); untranslate_all(); } void copy_offscreen(int x, int y, int w, int h, Fl_Offscreen pixmap, int srcx, int srcy) { Fl_Xlib_Graphics_Driver::copy_offscreen(x+offset_x, y+offset_y, w, h,pixmap,srcx,srcy); } void push_clip(int x, int y, int w, int h) { Fl_Xlib_Graphics_Driver::push_clip(x+offset_x, y+offset_y, w, h); } int not_clipped(int x, int y, int w, int h) { return Fl_Xlib_Graphics_Driver::not_clipped(x + offset_x, y + offset_y, w, h); }; int clip_box(int x, int y, int w, int h, int& X, int& Y, int& W, int& H) { int retval = Fl_Xlib_Graphics_Driver::clip_box(x + offset_x, y + offset_y, w,h,X,Y,W,H); X -= offset_x; Y -= offset_y; return retval; } void pie(int x, int y, int w, int h, double a1, double a2) { Fl_Xlib_Graphics_Driver::pie(x+offset_x,y+offset_y,w,h,a1,a2); } void arc(int x, int y, int w, int h, double a1, double a2) { Fl_Xlib_Graphics_Driver::arc(x+offset_x,y+offset_y,w,h,a1,a2); } void polygon(int x0, int y0, int x1, int y1, int x2, int y2) { Fl_Xlib_Graphics_Driver::polygon(x0+offset_x,y0+offset_y,x1+offset_x,y1+offset_y,x2+offset_x,y2+offset_y);} void polygon(int x0, int y0, int x1, int y1, int x2, int y2, int x3, int y3) { Fl_Xlib_Graphics_Driver::polygon(x0+offset_x,y0+offset_y,x1+offset_x,y1+offset_y,x2+offset_x,y2+offset_y,x3+offset_x,y3+offset_y); } void loop(int x0, int y0, int x1, int y1, int x2, int y2) {Fl_Xlib_Graphics_Driver::loop(x0+offset_x,y0+offset_y,x1+offset_x,y1+offset_y,x2+offset_x,y2+offset_y);} void loop(int x0, int y0, int x1, int y1, int x2, int y2, int x3, int y3) { Fl_Xlib_Graphics_Driver::loop(x0+offset_x,y0+offset_y,x1+offset_x,y1+offset_y,x2+offset_x,y2+offset_y,x3+offset_x,y3+offset_y); } void point(int x, int y) { Fl_Xlib_Graphics_Driver::point(x+offset_x, y+offset_y); } }; const char *Fl_translated_Xlib_Graphics_Driver_::class_id = "Fl_translated_Xlib_Graphics_Driver_"; void Fl_Xlib_Surface_::translate(int x, int y) { ((Fl_translated_Xlib_Graphics_Driver_*)driver())->translate_all(x, y); } void Fl_Xlib_Surface_::untranslate() { ((Fl_translated_Xlib_Graphics_Driver_*)driver())->untranslate_all(); } Fl_Xlib_Surface_::Fl_Xlib_Surface_() : Fl_Paged_Device() { driver(new Fl_translated_Xlib_Graphics_Driver_()); } Fl_Xlib_Surface_::~Fl_Xlib_Surface_() { delete driver(); } const char *Fl_Xlib_Surface_::class_id = "Fl_Xlib_Surface_"; #endif // // End of "$Id: Fl_Copy_Surface.cxx 11898 2016-08-27 15:17:02Z manolo $". //