1 /* GTK - The GIMP Toolkit
2 * gtkprintoperation-win32.c: Print Operation Details for Win32
3 * Copyright (C) 2006, Red Hat, Inc.
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2 of the License, or (at your option) any later version.
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
15 * You should have received a copy of the GNU Lesser General Public
16 * License along with this library; if not, write to the
17 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
18 * Boston, MA 02111-1307, USA.
22 #define _WIN32_WINNT 0x0500
23 #define WINVER _WIN32_WINNT
36 #include <cairo-win32.h>
38 #include "gtkprintoperation-private.h"
39 #include "gtkprint-win32.h"
41 #include "gtkinvisible.h"
45 #include "gtkwin32embedwidget.h"
47 #define MAX_PAGE_RANGES 20
48 #define STATUS_POLLING_TIME 2000
50 #ifndef JOB_STATUS_RESTART
51 #define JOB_STATUS_RESTART 0x800
54 #ifndef JOB_STATUS_COMPLETE
55 #define JOB_STATUS_COMPLETE 0x1000
66 cairo_surface_t *surface;
67 GtkWidget *embed_widget;
68 } GtkPrintOperationWin32;
70 static void win32_poll_status (GtkPrintOperation *op);
72 static const GUID myIID_IPrintDialogCallback = {0x5852a2c3,0x6530,0x11d1,{0xb6,0xa3,0x0,0x0,0xf8,0x75,0x7b,0xf9}};
74 #if !defined (_MSC_VER) && !defined (MINGW64) && !defined (__MINGW64__)
76 #define INTERFACE IPrintDialogCallback
77 DECLARE_INTERFACE_ (IPrintDialogCallback, IUnknown)
79 STDMETHOD (QueryInterface)(THIS_ REFIID,LPVOID*) PURE;
80 STDMETHOD_ (ULONG, AddRef)(THIS) PURE;
81 STDMETHOD_ (ULONG, Release)(THIS) PURE;
82 STDMETHOD (InitDone)(THIS) PURE;
83 STDMETHOD (SelectionChange)(THIS) PURE;
84 STDMETHOD (HandleMessage)(THIS_ HWND,UINT,WPARAM,LPARAM,LRESULT*) PURE;
88 static UINT got_gdk_events_message;
91 run_mainloop_hook (HWND hdlg, UINT uiMsg, WPARAM wParam, LPARAM lParam)
93 if (uiMsg == WM_INITDIALOG)
95 gdk_win32_set_modal_dialog_libgtk_only (hdlg);
96 while (gtk_events_pending ())
97 gtk_main_iteration ();
99 else if (uiMsg == got_gdk_events_message)
101 while (gtk_events_pending ())
102 gtk_main_iteration ();
108 static GtkPageOrientation
109 orientation_from_win32 (short orientation)
111 if (orientation == DMORIENT_LANDSCAPE)
112 return GTK_PAGE_ORIENTATION_LANDSCAPE;
113 return GTK_PAGE_ORIENTATION_PORTRAIT;
117 orientation_to_win32 (GtkPageOrientation orientation)
119 if (orientation == GTK_PAGE_ORIENTATION_LANDSCAPE ||
120 orientation == GTK_PAGE_ORIENTATION_REVERSE_LANDSCAPE)
121 return DMORIENT_LANDSCAPE;
122 return DMORIENT_PORTRAIT;
125 static GtkPaperSize *
126 paper_size_from_win32 (short size)
132 case DMPAPER_LETTER_TRANSVERSE:
134 case DMPAPER_LETTERSMALL:
137 case DMPAPER_TABLOID:
144 case DMPAPER_STATEMENT:
147 case DMPAPER_EXECUTIVE:
148 name = "na_executive";
151 case DMPAPER_A3_TRANSVERSE:
155 case DMPAPER_A4SMALL:
156 case DMPAPER_A4_TRANSVERSE:
160 case DMPAPER_A5_TRANSVERSE:
167 case DMPAPER_B5_TRANSVERSE:
183 name = "na_number-9";
186 name = "na_number-10";
189 name = "na_number-11";
192 name = "na_number-12";
195 name = "na_number-14";
221 case DMPAPER_ENV_C65:
233 case DMPAPER_ENV_ITALY:
236 case DMPAPER_ENV_MONARCH:
239 case DMPAPER_ENV_PERSONAL:
240 name = "na_personal";
242 case DMPAPER_FANFOLD_US:
243 name = "na_fanfold-us";
245 case DMPAPER_FANFOLD_STD_GERMAN:
246 name = "na_fanfold-eur";
248 case DMPAPER_FANFOLD_LGL_GERMAN:
249 name = "na_foolscap";
254 case DMPAPER_JAPANESE_POSTCARD:
263 case DMPAPER_ENV_INVITE:
266 case DMPAPER_LETTER_EXTRA:
267 case DMPAPER_LETTER_EXTRA_TRANSVERSE:
268 name = "na_letter-extra";
270 case DMPAPER_LEGAL_EXTRA:
271 name = "na_legal-extra";
273 case DMPAPER_TABLOID_EXTRA:
276 case DMPAPER_A4_EXTRA:
277 name = "iso_a4-extra";
282 case DMPAPER_LETTER_PLUS:
283 name = "na_letter-plus";
285 case DMPAPER_A3_EXTRA:
286 case DMPAPER_A3_EXTRA_TRANSVERSE:
287 name = "iso_a3-extra";
289 case DMPAPER_A5_EXTRA:
290 name = "iso_a5-extra";
292 case DMPAPER_B5_EXTRA:
293 name = "iso_b5-extra";
305 return gtk_paper_size_new (name);
311 paper_size_to_win32 (GtkPaperSize *paper_size)
315 if (gtk_paper_size_is_custom (paper_size))
318 format = gtk_paper_size_get_name (paper_size);
320 if (strcmp (format, "na_letter") == 0)
321 return DMPAPER_LETTER;
322 if (strcmp (format, "na_ledger") == 0)
323 return DMPAPER_LEDGER;
324 if (strcmp (format, "na_legal") == 0)
325 return DMPAPER_LEGAL;
326 if (strcmp (format, "na_invoice") == 0)
327 return DMPAPER_STATEMENT;
328 if (strcmp (format, "na_executive") == 0)
329 return DMPAPER_EXECUTIVE;
330 if (strcmp (format, "iso_a2") == 0)
332 if (strcmp (format, "iso_a3") == 0)
334 if (strcmp (format, "iso_a4") == 0)
336 if (strcmp (format, "iso_a5") == 0)
338 if (strcmp (format, "iso_b4") == 0)
340 if (strcmp (format, "iso_b5") == 0)
342 if (strcmp (format, "na_quarto") == 0)
343 return DMPAPER_QUARTO;
344 if (strcmp (format, "na_10x14") == 0)
345 return DMPAPER_10X14;
346 if (strcmp (format, "na_number-9") == 0)
347 return DMPAPER_ENV_9;
348 if (strcmp (format, "na_number-10") == 0)
349 return DMPAPER_ENV_10;
350 if (strcmp (format, "na_number-11") == 0)
351 return DMPAPER_ENV_11;
352 if (strcmp (format, "na_number-12") == 0)
353 return DMPAPER_ENV_12;
354 if (strcmp (format, "na_number-14") == 0)
355 return DMPAPER_ENV_14;
356 if (strcmp (format, "na_c") == 0)
357 return DMPAPER_CSHEET;
358 if (strcmp (format, "na_d") == 0)
359 return DMPAPER_DSHEET;
360 if (strcmp (format, "na_e") == 0)
361 return DMPAPER_ESHEET;
362 if (strcmp (format, "iso_dl") == 0)
363 return DMPAPER_ENV_DL;
364 if (strcmp (format, "iso_c3") == 0)
365 return DMPAPER_ENV_C3;
366 if (strcmp (format, "iso_c4") == 0)
367 return DMPAPER_ENV_C4;
368 if (strcmp (format, "iso_c5") == 0)
369 return DMPAPER_ENV_C5;
370 if (strcmp (format, "iso_c6") == 0)
371 return DMPAPER_ENV_C6;
372 if (strcmp (format, "iso_c5c6") == 0)
373 return DMPAPER_ENV_C65;
374 if (strcmp (format, "iso_b6") == 0)
375 return DMPAPER_ENV_B6;
376 if (strcmp (format, "om_italian") == 0)
377 return DMPAPER_ENV_ITALY;
378 if (strcmp (format, "na_monarch") == 0)
379 return DMPAPER_ENV_MONARCH;
380 if (strcmp (format, "na_personal") == 0)
381 return DMPAPER_ENV_PERSONAL;
382 if (strcmp (format, "na_fanfold-us") == 0)
383 return DMPAPER_FANFOLD_US;
384 if (strcmp (format, "na_fanfold-eur") == 0)
385 return DMPAPER_FANFOLD_STD_GERMAN;
386 if (strcmp (format, "na_foolscap") == 0)
387 return DMPAPER_FANFOLD_LGL_GERMAN;
388 if (strcmp (format, "jpn_hagaki") == 0)
389 return DMPAPER_JAPANESE_POSTCARD;
390 if (strcmp (format, "na_9x11") == 0)
392 if (strcmp (format, "na_10x11") == 0)
393 return DMPAPER_10X11;
394 if (strcmp (format, "om_invite") == 0)
395 return DMPAPER_ENV_INVITE;
396 if (strcmp (format, "na_letter-extra") == 0)
397 return DMPAPER_LETTER_EXTRA;
398 if (strcmp (format, "na_legal-extra") == 0)
399 return DMPAPER_LEGAL_EXTRA;
400 if (strcmp (format, "na_arch") == 0)
401 return DMPAPER_TABLOID_EXTRA;
402 if (strcmp (format, "iso_a3-extra") == 0)
403 return DMPAPER_A3_EXTRA;
404 if (strcmp (format, "iso_a4-extra") == 0)
405 return DMPAPER_A4_EXTRA;
406 if (strcmp (format, "iso_a5-extra") == 0)
407 return DMPAPER_A5_EXTRA;
408 if (strcmp (format, "iso_b5-extra") == 0)
409 return DMPAPER_B5_EXTRA;
410 if (strcmp (format, "na_b-plus") == 0)
411 return DMPAPER_B_PLUS;
412 if (strcmp (format, "na_letter-plus") == 0)
413 return DMPAPER_LETTER_PLUS;
419 get_default_printer (void)
421 wchar_t *win32_printer_name = NULL;
422 gchar *printer_name = NULL;
425 GetDefaultPrinterW (NULL, &needed);
426 win32_printer_name = g_malloc ((gsize) needed * sizeof (wchar_t));
427 if (!GetDefaultPrinterW (win32_printer_name, &needed))
429 g_free (win32_printer_name);
432 printer_name = g_utf16_to_utf8 (win32_printer_name, -1, NULL, NULL, NULL);
433 g_free (win32_printer_name);
439 set_hard_margins (GtkPrintOperation *op)
441 double top, bottom, left, right;
442 GtkPrintOperationWin32 *op_win32 = op->priv->platform_data;
444 top = GetDeviceCaps (op_win32->hdc, PHYSICALOFFSETY);
445 bottom = GetDeviceCaps (op_win32->hdc, PHYSICALHEIGHT)
446 - GetDeviceCaps (op_win32->hdc, VERTRES) - top;
447 left = GetDeviceCaps (op_win32->hdc, PHYSICALOFFSETX);
448 right = GetDeviceCaps (op_win32->hdc, PHYSICALWIDTH)
449 - GetDeviceCaps (op_win32->hdc, HORZRES) - left;
451 _gtk_print_context_set_hard_margins (op->priv->print_context, top, bottom, left, right);
455 win32_start_page (GtkPrintOperation *op,
456 GtkPrintContext *print_context,
457 GtkPageSetup *page_setup)
459 GtkPrintOperationWin32 *op_win32 = op->priv->platform_data;
461 GtkPaperSize *paper_size;
464 devmode = GlobalLock (op_win32->devmode);
466 devmode->dmFields |= DM_ORIENTATION;
467 devmode->dmOrientation =
468 orientation_to_win32 (gtk_page_setup_get_orientation (page_setup));
470 paper_size = gtk_page_setup_get_paper_size (page_setup);
471 devmode->dmFields |= DM_PAPERSIZE;
472 devmode->dmFields &= ~(DM_PAPERWIDTH | DM_PAPERLENGTH);
473 devmode->dmPaperSize = paper_size_to_win32 (paper_size);
474 if (devmode->dmPaperSize == 0)
476 devmode->dmPaperSize = DMPAPER_USER;
477 devmode->dmFields |= DM_PAPERWIDTH | DM_PAPERLENGTH;
479 /* Lengths in DEVMODE are in tenths of a millimeter */
480 devmode->dmPaperWidth = gtk_paper_size_get_width (paper_size, GTK_UNIT_MM) * 10.0;
481 devmode->dmPaperLength = gtk_paper_size_get_height (paper_size, GTK_UNIT_MM) * 10.0;
484 ResetDCW (op_win32->hdc, devmode);
486 GlobalUnlock (op_win32->devmode);
488 set_hard_margins (op);
489 x_off = GetDeviceCaps (op_win32->hdc, PHYSICALOFFSETX);
490 y_off = GetDeviceCaps (op_win32->hdc, PHYSICALOFFSETY);
491 cairo_surface_set_device_offset (op_win32->surface, -x_off, -y_off);
493 StartPage (op_win32->hdc);
497 win32_end_page (GtkPrintOperation *op,
498 GtkPrintContext *print_context)
500 GtkPrintOperationWin32 *op_win32 = op->priv->platform_data;
502 cairo_surface_show_page (op_win32->surface);
504 EndPage (op_win32->hdc);
508 win32_poll_status_timeout (GtkPrintOperation *op)
510 GtkPrintOperationWin32 *op_win32 = op->priv->platform_data;
512 op_win32->timeout_id = 0;
513 /* We need to ref this, as setting the status to finished
514 might unref the object */
516 win32_poll_status (op);
518 if (!gtk_print_operation_is_finished (op))
519 op_win32->timeout_id = gdk_threads_add_timeout (STATUS_POLLING_TIME,
520 (GSourceFunc)win32_poll_status_timeout,
528 win32_end_run (GtkPrintOperation *op,
532 GtkPrintOperationWin32 *op_win32 = op->priv->platform_data;
534 HANDLE printerHandle = 0;
536 cairo_surface_finish (op_win32->surface);
538 EndDoc (op_win32->hdc);
540 if (op->priv->track_print_status)
542 devnames = GlobalLock (op_win32->devnames);
543 if (!OpenPrinterW (((gunichar2 *)devnames) + devnames->wDeviceOffset,
544 &printerHandle, NULL))
546 GlobalUnlock (op_win32->devnames);
549 GlobalFree (op_win32->devmode);
550 GlobalFree (op_win32->devnames);
552 cairo_surface_destroy (op_win32->surface);
553 op_win32->surface = NULL;
555 DeleteDC (op_win32->hdc);
557 if (printerHandle != 0)
559 op_win32->printerHandle = printerHandle;
560 win32_poll_status (op);
561 op_win32->timeout_id = gdk_threads_add_timeout (STATUS_POLLING_TIME,
562 (GSourceFunc)win32_poll_status_timeout,
566 /* Dunno what happened, pretend its finished */
567 _gtk_print_operation_set_status (op, GTK_PRINT_STATUS_FINISHED, NULL);
571 win32_poll_status (GtkPrintOperation *op)
573 GtkPrintOperationWin32 *op_win32 = op->priv->platform_data;
576 JOB_INFO_1W *job_info;
577 GtkPrintStatus status;
581 GetJobW (op_win32->printerHandle, op_win32->job_id,
582 1,(LPBYTE)NULL, 0, &needed);
583 data = g_malloc (needed);
584 ret = GetJobW (op_win32->printerHandle, op_win32->job_id,
585 1, (LPBYTE)data, needed, &needed);
591 job_info = (JOB_INFO_1W *)data;
592 win32_status = job_info->Status;
594 if (job_info->pStatus)
595 status_str = g_utf16_to_utf8 (job_info->pStatus,
596 -1, NULL, NULL, NULL);
599 (JOB_STATUS_COMPLETE | JOB_STATUS_PRINTED))
600 status = GTK_PRINT_STATUS_FINISHED;
601 else if (win32_status &
602 (JOB_STATUS_OFFLINE |
603 JOB_STATUS_PAPEROUT |
605 JOB_STATUS_USER_INTERVENTION))
607 status = GTK_PRINT_STATUS_PENDING_ISSUE;
608 if (status_str == NULL)
610 if (win32_status & JOB_STATUS_OFFLINE)
611 status_str = g_strdup (_("Printer offline"));
612 else if (win32_status & JOB_STATUS_PAPEROUT)
613 status_str = g_strdup (_("Out of paper"));
614 else if (win32_status & JOB_STATUS_PAUSED)
615 status_str = g_strdup (_("Paused"));
616 else if (win32_status & JOB_STATUS_USER_INTERVENTION)
617 status_str = g_strdup (_("Need user intervention"));
620 else if (win32_status &
621 (JOB_STATUS_BLOCKED_DEVQ |
624 status = GTK_PRINT_STATUS_FINISHED_ABORTED;
625 else if (win32_status &
626 (JOB_STATUS_SPOOLING |
627 JOB_STATUS_DELETING))
628 status = GTK_PRINT_STATUS_PENDING;
629 else if (win32_status & JOB_STATUS_PRINTING)
630 status = GTK_PRINT_STATUS_PRINTING;
632 status = GTK_PRINT_STATUS_FINISHED;
635 status = GTK_PRINT_STATUS_FINISHED;
639 _gtk_print_operation_set_status (op, status, status_str);
645 op_win32_free (GtkPrintOperationWin32 *op_win32)
647 if (op_win32->printerHandle)
648 ClosePrinter (op_win32->printerHandle);
649 if (op_win32->timeout_id != 0)
650 g_source_remove (op_win32->timeout_id);
655 get_parent_hwnd (GtkWidget *widget)
657 gtk_widget_realize (widget);
658 return gdk_win32_drawable_get_handle (widget->window);
662 devnames_to_settings (GtkPrintSettings *settings,
665 GtkPrintWin32Devnames *devnames = gtk_print_win32_devnames_from_win32 (hDevNames);
666 gtk_print_settings_set_printer (settings, devnames->device);
667 gtk_print_win32_devnames_free (devnames);
671 devmode_to_settings (GtkPrintSettings *settings,
676 devmode = GlobalLock (hDevMode);
678 gtk_print_settings_set_int (settings, GTK_PRINT_SETTINGS_WIN32_DRIVER_VERSION,
679 devmode->dmDriverVersion);
680 if (devmode->dmDriverExtra != 0)
682 char *extra = g_base64_encode (((char *)devmode) + sizeof (DEVMODEW),
683 devmode->dmDriverExtra);
684 gtk_print_settings_set (settings,
685 GTK_PRINT_SETTINGS_WIN32_DRIVER_EXTRA,
690 if (devmode->dmFields & DM_ORIENTATION)
691 gtk_print_settings_set_orientation (settings,
692 orientation_from_win32 (devmode->dmOrientation));
695 if (devmode->dmFields & DM_PAPERSIZE &&
696 devmode->dmPaperSize != 0)
698 GtkPaperSize *paper_size = paper_size_from_win32 (devmode->dmPaperSize);
701 gtk_print_settings_set_paper_size (settings, paper_size);
702 gtk_paper_size_free (paper_size);
704 gtk_print_settings_set_int (settings, "win32-paper-size", (int)devmode->dmPaperSize);
706 else if ((devmode->dmFields & DM_PAPERSIZE &&
707 devmode->dmPaperSize == 0) ||
708 ((devmode->dmFields & DM_PAPERWIDTH) &&
709 (devmode->dmFields & DM_PAPERLENGTH)))
711 GtkPaperSize *paper_size;
712 char *form_name = NULL;
713 if (devmode->dmFields & DM_FORMNAME)
714 form_name = g_utf16_to_utf8 (devmode->dmFormName,
715 -1, NULL, NULL, NULL);
716 if (form_name == NULL || form_name[0] == 0)
717 form_name = g_strdup (_("Custom size"));
719 /* Lengths in DEVMODE are in tenths of a millimeter */
720 paper_size = gtk_paper_size_new_custom (form_name,
722 devmode->dmPaperWidth / 10.0,
723 devmode->dmPaperLength / 10.0,
725 gtk_print_settings_set_paper_size (settings, paper_size);
726 gtk_paper_size_free (paper_size);
729 if (devmode->dmFields & DM_SCALE)
730 gtk_print_settings_set_scale (settings, devmode->dmScale);
732 if (devmode->dmFields & DM_COPIES)
733 gtk_print_settings_set_n_copies (settings,
736 if (devmode->dmFields & DM_DEFAULTSOURCE)
739 switch (devmode->dmDefaultSource)
751 case DMBIN_ENVMANUAL:
752 source = "envelope-manual";
766 case DMBIN_FORMSOURCE:
767 source = "form-source";
769 case DMBIN_LARGECAPACITY:
770 source = "large-capacity";
773 source = "large-format";
779 source = "small-format";
782 gtk_print_settings_set_default_source (settings, source);
783 gtk_print_settings_set_int (settings, "win32-default-source", devmode->dmDefaultSource);
786 if (devmode->dmFields & DM_PRINTQUALITY)
788 GtkPrintQuality quality;
789 switch (devmode->dmPrintQuality)
792 quality = GTK_PRINT_QUALITY_LOW;
795 quality = GTK_PRINT_QUALITY_NORMAL;
799 quality = GTK_PRINT_QUALITY_HIGH;
802 quality = GTK_PRINT_QUALITY_DRAFT;
805 gtk_print_settings_set_quality (settings, quality);
806 gtk_print_settings_set_int (settings, "win32-print-quality", devmode->dmPrintQuality);
809 if (devmode->dmFields & DM_COLOR)
810 gtk_print_settings_set_use_color (settings, devmode->dmColor == DMCOLOR_COLOR);
812 if (devmode->dmFields & DM_DUPLEX)
814 GtkPrintDuplex duplex;
815 switch (devmode->dmDuplex)
819 duplex = GTK_PRINT_DUPLEX_SIMPLEX;
821 case DMDUP_HORIZONTAL:
822 duplex = GTK_PRINT_DUPLEX_HORIZONTAL;
825 duplex = GTK_PRINT_DUPLEX_VERTICAL;
829 gtk_print_settings_set_duplex (settings, duplex);
832 if (devmode->dmFields & DM_COLLATE)
833 gtk_print_settings_set_collate (settings,
834 devmode->dmCollate == DMCOLLATE_TRUE);
836 if (devmode->dmFields & DM_MEDIATYPE)
839 switch (devmode->dmMediaType)
842 case DMMEDIA_STANDARD:
843 media_type = "stationery";
845 case DMMEDIA_TRANSPARENCY:
846 media_type = "transparency";
849 media_type = "photographic-glossy";
852 gtk_print_settings_set_media_type (settings, media_type);
853 gtk_print_settings_set_int (settings, "win32-media-type", devmode->dmMediaType);
856 if (devmode->dmFields & DM_DITHERTYPE)
859 switch (devmode->dmDitherType)
868 case DMDITHER_COARSE:
871 case DMDITHER_LINEART:
874 case DMDITHER_GRAYSCALE:
875 dither = "grayscale";
877 case DMDITHER_ERRORDIFFUSION:
878 dither = "error-diffusion";
881 gtk_print_settings_set_dither (settings, dither);
882 gtk_print_settings_set_int (settings, "win32-dither-type", devmode->dmDitherType);
885 GlobalUnlock (hDevMode);
889 dialog_to_print_settings (GtkPrintOperation *op,
890 LPPRINTDLGEXW printdlgex)
893 GtkPrintSettings *settings;
895 settings = gtk_print_settings_new ();
897 gtk_print_settings_set_print_pages (settings,
898 GTK_PRINT_PAGES_ALL);
899 if (printdlgex->Flags & PD_CURRENTPAGE)
900 gtk_print_settings_set_print_pages (settings,
901 GTK_PRINT_PAGES_CURRENT);
902 else if (printdlgex->Flags & PD_PAGENUMS)
903 gtk_print_settings_set_print_pages (settings,
904 GTK_PRINT_PAGES_RANGES);
906 if (printdlgex->nPageRanges > 0)
908 GtkPageRange *ranges;
909 ranges = g_new (GtkPageRange, printdlgex->nPageRanges);
911 for (i = 0; i < printdlgex->nPageRanges; i++)
913 ranges[i].start = printdlgex->lpPageRanges[i].nFromPage - 1;
914 ranges[i].end = printdlgex->lpPageRanges[i].nToPage - 1;
917 gtk_print_settings_set_page_ranges (settings, ranges,
918 printdlgex->nPageRanges);
922 if (printdlgex->hDevNames != NULL)
923 devnames_to_settings (settings, printdlgex->hDevNames);
925 if (printdlgex->hDevMode != NULL)
926 devmode_to_settings (settings, printdlgex->hDevMode);
928 gtk_print_operation_set_print_settings (op, settings);
932 devmode_from_settings (GtkPrintSettings *settings,
933 GtkPageSetup *page_setup)
938 GtkPaperSize *paper_size;
939 const char *extras_base64;
945 extras_base64 = gtk_print_settings_get (settings, GTK_PRINT_SETTINGS_WIN32_DRIVER_EXTRA);
947 extras = g_base64_decode (extras_base64, &extras_len);
949 hDevMode = GlobalAlloc (GMEM_MOVEABLE,
950 sizeof (DEVMODEW) + extras_len);
952 devmode = GlobalLock (hDevMode);
954 memset (devmode, 0, sizeof (DEVMODEW));
956 devmode->dmSpecVersion = DM_SPECVERSION;
957 devmode->dmSize = sizeof (DEVMODEW);
959 devmode->dmDriverExtra = 0;
960 if (extras && extras_len > 0)
962 devmode->dmDriverExtra = extras_len;
963 memcpy (((char *)devmode) + sizeof (DEVMODEW), extras, extras_len);
966 if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_WIN32_DRIVER_VERSION))
967 devmode->dmDriverVersion = gtk_print_settings_get_int (settings, GTK_PRINT_SETTINGS_WIN32_DRIVER_VERSION);
970 gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_ORIENTATION))
972 GtkPageOrientation orientation = gtk_print_settings_get_orientation (settings);
974 orientation = gtk_page_setup_get_orientation (page_setup);
975 devmode->dmFields |= DM_ORIENTATION;
976 devmode->dmOrientation = orientation_to_win32 (orientation);
980 paper_size = gtk_paper_size_copy (gtk_page_setup_get_paper_size (page_setup));
984 if (gtk_print_settings_has_key (settings, "win32-paper-size") &&
985 (size = gtk_print_settings_get_int (settings, "win32-paper-size")) != 0)
987 devmode->dmFields |= DM_PAPERSIZE;
988 devmode->dmPaperSize = size;
992 paper_size = gtk_print_settings_get_paper_size (settings);
996 devmode->dmFields |= DM_PAPERSIZE;
997 devmode->dmPaperSize = paper_size_to_win32 (paper_size);
998 if (devmode->dmPaperSize == 0)
1000 devmode->dmPaperSize = DMPAPER_USER;
1001 devmode->dmFields |= DM_PAPERWIDTH | DM_PAPERLENGTH;
1003 /* Lengths in DEVMODE are in tenths of a millimeter */
1004 devmode->dmPaperWidth = gtk_paper_size_get_width (paper_size, GTK_UNIT_MM) * 10.0;
1005 devmode->dmPaperLength = gtk_paper_size_get_height (paper_size, GTK_UNIT_MM) * 10.0;
1007 gtk_paper_size_free (paper_size);
1010 if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_SCALE))
1012 devmode->dmFields |= DM_SCALE;
1013 devmode->dmScale = gtk_print_settings_get_scale (settings);
1016 if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_N_COPIES))
1018 devmode->dmFields |= DM_COPIES;
1019 devmode->dmCopies = gtk_print_settings_get_n_copies (settings);
1022 if (gtk_print_settings_has_key (settings, "win32-default-source"))
1024 devmode->dmFields |= DM_DEFAULTSOURCE;
1025 devmode->dmDefaultSource = gtk_print_settings_get_int (settings, "win32-default-source");
1027 else if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_DEFAULT_SOURCE))
1029 devmode->dmFields |= DM_DEFAULTSOURCE;
1030 devmode->dmDefaultSource = DMBIN_AUTO;
1032 val = gtk_print_settings_get_default_source (settings);
1033 if (strcmp (val, "auto") == 0)
1034 devmode->dmDefaultSource = DMBIN_AUTO;
1035 if (strcmp (val, "cassette") == 0)
1036 devmode->dmDefaultSource = DMBIN_CASSETTE;
1037 if (strcmp (val, "envelope") == 0)
1038 devmode->dmDefaultSource = DMBIN_ENVELOPE;
1039 if (strcmp (val, "envelope-manual") == 0)
1040 devmode->dmDefaultSource = DMBIN_ENVMANUAL;
1041 if (strcmp (val, "lower") == 0)
1042 devmode->dmDefaultSource = DMBIN_LOWER;
1043 if (strcmp (val, "manual") == 0)
1044 devmode->dmDefaultSource = DMBIN_MANUAL;
1045 if (strcmp (val, "middle") == 0)
1046 devmode->dmDefaultSource = DMBIN_MIDDLE;
1047 if (strcmp (val, "only-one") == 0)
1048 devmode->dmDefaultSource = DMBIN_ONLYONE;
1049 if (strcmp (val, "form-source") == 0)
1050 devmode->dmDefaultSource = DMBIN_FORMSOURCE;
1051 if (strcmp (val, "large-capacity") == 0)
1052 devmode->dmDefaultSource = DMBIN_LARGECAPACITY;
1053 if (strcmp (val, "large-format") == 0)
1054 devmode->dmDefaultSource = DMBIN_LARGEFMT;
1055 if (strcmp (val, "tractor") == 0)
1056 devmode->dmDefaultSource = DMBIN_TRACTOR;
1057 if (strcmp (val, "small-format") == 0)
1058 devmode->dmDefaultSource = DMBIN_SMALLFMT;
1061 if (gtk_print_settings_has_key (settings, "win32-print-quality"))
1063 devmode->dmFields |= DM_PRINTQUALITY;
1064 devmode->dmPrintQuality = gtk_print_settings_get_int (settings, "win32-print-quality");
1066 else if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_RESOLUTION))
1068 devmode->dmFields |= DM_PRINTQUALITY;
1069 devmode->dmPrintQuality = gtk_print_settings_get_resolution (settings);
1071 else if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_QUALITY))
1073 devmode->dmFields |= DM_PRINTQUALITY;
1074 switch (gtk_print_settings_get_quality (settings))
1076 case GTK_PRINT_QUALITY_LOW:
1077 devmode->dmPrintQuality = DMRES_LOW;
1079 case GTK_PRINT_QUALITY_DRAFT:
1080 devmode->dmPrintQuality = DMRES_DRAFT;
1083 case GTK_PRINT_QUALITY_NORMAL:
1084 devmode->dmPrintQuality = DMRES_MEDIUM;
1086 case GTK_PRINT_QUALITY_HIGH:
1087 devmode->dmPrintQuality = DMRES_HIGH;
1092 if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_USE_COLOR))
1094 devmode->dmFields |= DM_COLOR;
1095 if (gtk_print_settings_get_use_color (settings))
1096 devmode->dmColor = DMCOLOR_COLOR;
1098 devmode->dmColor = DMCOLOR_MONOCHROME;
1101 if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_DUPLEX))
1103 devmode->dmFields |= DM_DUPLEX;
1104 switch (gtk_print_settings_get_duplex (settings))
1107 case GTK_PRINT_DUPLEX_SIMPLEX:
1108 devmode->dmDuplex = DMDUP_SIMPLEX;
1110 case GTK_PRINT_DUPLEX_HORIZONTAL:
1111 devmode->dmDuplex = DMDUP_HORIZONTAL;
1113 case GTK_PRINT_DUPLEX_VERTICAL:
1114 devmode->dmDuplex = DMDUP_VERTICAL;
1119 if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_COLLATE))
1121 devmode->dmFields |= DM_COLLATE;
1122 if (gtk_print_settings_get_collate (settings))
1123 devmode->dmCollate = DMCOLLATE_TRUE;
1125 devmode->dmCollate = DMCOLLATE_FALSE;
1128 if (gtk_print_settings_has_key (settings, "win32-media-type"))
1130 devmode->dmFields |= DM_MEDIATYPE;
1131 devmode->dmMediaType = gtk_print_settings_get_int (settings, "win32-media-type");
1133 else if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_MEDIA_TYPE))
1135 devmode->dmFields |= DM_MEDIATYPE;
1136 devmode->dmMediaType = DMMEDIA_STANDARD;
1138 val = gtk_print_settings_get_media_type (settings);
1139 if (strcmp (val, "transparency") == 0)
1140 devmode->dmMediaType = DMMEDIA_TRANSPARENCY;
1141 if (strcmp (val, "photographic-glossy") == 0)
1142 devmode->dmMediaType = DMMEDIA_GLOSSY;
1145 if (gtk_print_settings_has_key (settings, "win32-dither-type"))
1147 devmode->dmFields |= DM_DITHERTYPE;
1148 devmode->dmDitherType = gtk_print_settings_get_int (settings, "win32-dither-type");
1150 else if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_DITHER))
1152 devmode->dmFields |= DM_DITHERTYPE;
1153 devmode->dmDitherType = DMDITHER_FINE;
1155 val = gtk_print_settings_get_dither (settings);
1156 if (strcmp (val, "none") == 0)
1157 devmode->dmDitherType = DMDITHER_NONE;
1158 if (strcmp (val, "coarse") == 0)
1159 devmode->dmDitherType = DMDITHER_COARSE;
1160 if (strcmp (val, "fine") == 0)
1161 devmode->dmDitherType = DMDITHER_FINE;
1162 if (strcmp (val, "lineart") == 0)
1163 devmode->dmDitherType = DMDITHER_LINEART;
1164 if (strcmp (val, "grayscale") == 0)
1165 devmode->dmDitherType = DMDITHER_GRAYSCALE;
1166 if (strcmp (val, "error-diffusion") == 0)
1167 devmode->dmDitherType = DMDITHER_ERRORDIFFUSION;
1170 GlobalUnlock (hDevMode);
1176 dialog_from_print_settings (GtkPrintOperation *op,
1177 LPPRINTDLGEXW printdlgex)
1179 GtkPrintSettings *settings = op->priv->print_settings;
1180 const char *printer;
1182 if (settings == NULL)
1185 if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_PRINT_PAGES))
1187 GtkPrintPages print_pages = gtk_print_settings_get_print_pages (settings);
1189 switch (print_pages)
1192 case GTK_PRINT_PAGES_ALL:
1193 printdlgex->Flags |= PD_ALLPAGES;
1195 case GTK_PRINT_PAGES_CURRENT:
1196 printdlgex->Flags |= PD_CURRENTPAGE;
1198 case GTK_PRINT_PAGES_RANGES:
1199 printdlgex->Flags |= PD_PAGENUMS;
1203 if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_PAGE_RANGES))
1205 GtkPageRange *ranges;
1208 ranges = gtk_print_settings_get_page_ranges (settings, &num_ranges);
1210 if (num_ranges > MAX_PAGE_RANGES)
1211 num_ranges = MAX_PAGE_RANGES;
1213 printdlgex->nPageRanges = num_ranges;
1214 for (i = 0; i < num_ranges; i++)
1216 printdlgex->lpPageRanges[i].nFromPage = ranges[i].start + 1;
1217 printdlgex->lpPageRanges[i].nToPage = ranges[i].end + 1;
1221 printer = gtk_print_settings_get_printer (settings);
1223 printdlgex->hDevNames = gtk_print_win32_devnames_to_win32_from_printer_name (printer);
1225 printdlgex->hDevMode = devmode_from_settings (settings,
1226 op->priv->default_page_setup);
1230 IPrintDialogCallback iPrintDialogCallback;
1233 } PrintDialogCallback;
1236 static ULONG STDMETHODCALLTYPE
1237 iprintdialogcallback_addref (IPrintDialogCallback *This)
1239 PrintDialogCallback *callback = (PrintDialogCallback *)This;
1240 return ++callback->ref_count;
1243 static ULONG STDMETHODCALLTYPE
1244 iprintdialogcallback_release (IPrintDialogCallback *This)
1246 PrintDialogCallback *callback = (PrintDialogCallback *)This;
1247 int ref_count = --callback->ref_count;
1255 static HRESULT STDMETHODCALLTYPE
1256 iprintdialogcallback_queryinterface (IPrintDialogCallback *This,
1260 if (IsEqualIID (riid, &IID_IUnknown) ||
1261 IsEqualIID (riid, &myIID_IPrintDialogCallback))
1264 IUnknown_AddRef ((IUnknown *)This);
1270 return E_NOINTERFACE;
1274 static HRESULT STDMETHODCALLTYPE
1275 iprintdialogcallback_initdone (IPrintDialogCallback *This)
1280 static HRESULT STDMETHODCALLTYPE
1281 iprintdialogcallback_selectionchange (IPrintDialogCallback *This)
1286 static HRESULT STDMETHODCALLTYPE
1287 iprintdialogcallback_handlemessage (IPrintDialogCallback *This,
1294 PrintDialogCallback *callback = (PrintDialogCallback *)This;
1296 if (!callback->set_hwnd)
1298 gdk_win32_set_modal_dialog_libgtk_only (hDlg);
1299 callback->set_hwnd = TRUE;
1300 while (gtk_events_pending ())
1301 gtk_main_iteration ();
1303 else if (uMsg == got_gdk_events_message)
1305 while (gtk_events_pending ())
1306 gtk_main_iteration ();
1315 static IPrintDialogCallbackVtbl ipdc_vtbl = {
1316 iprintdialogcallback_queryinterface,
1317 iprintdialogcallback_addref,
1318 iprintdialogcallback_release,
1319 iprintdialogcallback_initdone,
1320 iprintdialogcallback_selectionchange,
1321 iprintdialogcallback_handlemessage
1324 static IPrintDialogCallback *
1325 print_callback_new (void)
1327 PrintDialogCallback *callback;
1329 callback = g_new0 (PrintDialogCallback, 1);
1330 callback->iPrintDialogCallback.lpVtbl = &ipdc_vtbl;
1331 callback->ref_count = 1;
1332 callback->set_hwnd = FALSE;
1334 return &callback->iPrintDialogCallback;
1338 plug_grab_notify (GtkWidget *widget,
1339 gboolean was_grabbed,
1340 GtkPrintOperation *op)
1342 EnableWindow (GetAncestor (GDK_WINDOW_HWND (widget->window), GA_ROOT),
1347 static BOOL CALLBACK
1348 pageDlgProc (HWND wnd, UINT message, WPARAM wparam, LPARAM lparam)
1350 GtkPrintOperation *op;
1351 GtkPrintOperationWin32 *op_win32;
1353 if (message == WM_INITDIALOG)
1355 PROPSHEETPAGEW *page = (PROPSHEETPAGEW *)lparam;
1358 op = GTK_PRINT_OPERATION ((gpointer)page->lParam);
1359 op_win32 = op->priv->platform_data;
1361 SetWindowLongPtrW (wnd, GWLP_USERDATA, (LONG_PTR)op);
1363 plug = _gtk_win32_embed_widget_new ((GdkNativeWindow) wnd);
1364 gtk_window_set_modal (GTK_WINDOW (plug), TRUE);
1365 op_win32->embed_widget = plug;
1366 gtk_container_add (GTK_CONTAINER (plug), op->priv->custom_widget);
1367 gtk_widget_show (op->priv->custom_widget);
1368 gtk_widget_show (plug);
1369 gdk_window_focus (plug->window, GDK_CURRENT_TIME);
1371 /* This dialog is modal, so we grab the embed widget */
1372 gtk_grab_add (plug);
1374 /* When we lose the grab we need to disable the print dialog */
1375 g_signal_connect (plug, "grab-notify", G_CALLBACK (plug_grab_notify), op);
1378 else if (message == WM_DESTROY)
1380 op = GTK_PRINT_OPERATION (GetWindowLongPtrW (wnd, GWLP_USERDATA));
1381 op_win32 = op->priv->platform_data;
1383 g_signal_emit_by_name (op, "custom-widget-apply", op->priv->custom_widget);
1384 gtk_widget_destroy (op_win32->embed_widget);
1385 op_win32->embed_widget = NULL;
1386 op->priv->custom_widget = NULL;
1390 op = GTK_PRINT_OPERATION (GetWindowLongPtrW (wnd, GWLP_USERDATA));
1391 op_win32 = op->priv->platform_data;
1393 return _gtk_win32_embed_widget_dialog_procedure (GTK_WIN32_EMBED_WIDGET (op_win32->embed_widget),
1394 wnd, message, wparam, lparam);
1400 static HPROPSHEETPAGE
1401 create_application_page (GtkPrintOperation *op)
1403 HPROPSHEETPAGE hpage;
1404 PROPSHEETPAGEW page;
1405 DLGTEMPLATE *template;
1408 WORD baseunitX, baseunitY;
1410 GtkRequisition requisition;
1411 const char *tab_label;
1413 /* Make the template the size of the custom widget size request */
1414 gtk_widget_size_request (op->priv->custom_widget, &requisition);
1416 base_units = GetDialogBaseUnits ();
1417 baseunitX = LOWORD (base_units);
1418 baseunitY = HIWORD (base_units);
1420 htemplate = GlobalAlloc (GMEM_MOVEABLE,
1421 sizeof (DLGTEMPLATE) + sizeof (WORD) * 3);
1422 template = GlobalLock (htemplate);
1423 template->style = WS_CHILDWINDOW | DS_CONTROL;
1424 template->dwExtendedStyle = WS_EX_CONTROLPARENT;
1426 template->x = MulDiv (0, 4, baseunitX);
1427 template->y = MulDiv (0, 8, baseunitY);
1428 template->cx = MulDiv (requisition.width, 4, baseunitX);
1429 template->cy = MulDiv (requisition.height, 8, baseunitY);
1431 array = (WORD *) (template+1);
1432 *array++ = 0; /* menu */
1433 *array++ = 0; /* class */
1434 *array++ = 0; /* title */
1436 memset (&page, 0, sizeof (page));
1437 page.dwSize = sizeof (page);
1438 page.dwFlags = PSP_DLGINDIRECT | PSP_USETITLE | PSP_PREMATURE;
1439 page.hInstance = GetModuleHandle (NULL);
1440 page.pResource = template;
1442 tab_label = op->priv->custom_tab_label;
1443 if (tab_label == NULL)
1444 tab_label = g_get_application_name ();
1445 if (tab_label == NULL)
1446 tab_label = _("Application");
1447 page.pszTitle = g_utf8_to_utf16 (tab_label,
1448 -1, NULL, NULL, NULL);
1449 page.pfnDlgProc = pageDlgProc;
1450 page.pfnCallback = NULL;
1451 page.lParam = (LPARAM) op;
1452 hpage = CreatePropertySheetPageW (&page);
1454 GlobalUnlock (htemplate);
1456 /* TODO: We're leaking htemplate here... */
1461 static GtkPageSetup *
1462 create_page_setup (GtkPrintOperation *op)
1464 GtkPrintOperationPrivate *priv = op->priv;
1465 GtkPageSetup *page_setup;
1466 GtkPrintSettings *settings;
1468 if (priv->default_page_setup)
1469 page_setup = gtk_page_setup_copy (priv->default_page_setup);
1471 page_setup = gtk_page_setup_new ();
1473 settings = priv->print_settings;
1476 GtkPaperSize *paper_size;
1478 if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_ORIENTATION))
1479 gtk_page_setup_set_orientation (page_setup,
1480 gtk_print_settings_get_orientation (settings));
1483 paper_size = gtk_print_settings_get_paper_size (settings);
1486 gtk_page_setup_set_paper_size (page_setup, paper_size);
1487 gtk_paper_size_free (paper_size);
1490 /* TODO: Margins? */
1496 GtkPrintOperationResult
1497 gtk_print_operation_run_without_dialog (GtkPrintOperation *op,
1500 GtkPrintOperationResult result;
1501 GtkPrintOperationWin32 *op_win32;
1502 GtkPrintOperationPrivate *priv;
1503 GtkPrintSettings *settings;
1504 GtkPageSetup *page_setup;
1506 HGLOBAL hDevMode = NULL;
1507 HGLOBAL hDevNames = NULL;
1509 const char *printer = NULL;
1510 double dpi_x, dpi_y;
1519 settings = priv->print_settings;
1521 op_win32 = g_new0 (GtkPrintOperationWin32, 1);
1522 priv->platform_data = op_win32;
1523 priv->free_platform_data = (GDestroyNotify) op_win32_free;
1524 printer = gtk_print_settings_get_printer (settings);
1528 /* No printer selected. Get the system default printer and store
1531 gchar *tmp_printer = get_default_printer ();
1534 result = GTK_PRINT_OPERATION_RESULT_ERROR;
1535 g_set_error_literal (&priv->error,
1537 GTK_PRINT_ERROR_INTERNAL_ERROR,
1538 _("No printer found"));
1541 gtk_print_settings_set_printer (settings, tmp_printer);
1542 printer = gtk_print_settings_get_printer (settings);
1543 g_free (tmp_printer);
1546 hDevNames = gtk_print_win32_devnames_to_win32_from_printer_name (printer);
1547 hDevMode = devmode_from_settings (settings, op->priv->default_page_setup);
1549 /* Create a printer DC for the print settings and page setup provided. */
1550 pdn = GlobalLock (hDevNames);
1551 pdm = GlobalLock (hDevMode);
1552 hDC = CreateDCW ((wchar_t*)pdn + pdn->wDriverOffset,
1553 (wchar_t*)pdn + pdn->wDeviceOffset,
1554 (wchar_t*)pdn + pdn->wOutputOffset,
1556 GlobalUnlock (hDevNames);
1557 GlobalUnlock (hDevMode);
1561 result = GTK_PRINT_OPERATION_RESULT_ERROR;
1562 g_set_error_literal (&priv->error,
1564 GTK_PRINT_ERROR_INTERNAL_ERROR,
1565 _("Invalid argument to CreateDC"));
1569 priv->print_context = _gtk_print_context_new (op);
1570 page_setup = create_page_setup (op);
1571 _gtk_print_context_set_page_setup (priv->print_context, page_setup);
1572 g_object_unref (page_setup);
1576 op_win32->surface = cairo_win32_printing_surface_create (hDC);
1577 dpi_x = (double) GetDeviceCaps (hDC, LOGPIXELSX);
1578 dpi_y = (double) GetDeviceCaps (hDC, LOGPIXELSY);
1580 cr = cairo_create (op_win32->surface);
1581 gtk_print_context_set_cairo_context (priv->print_context, cr, dpi_x, dpi_y);
1584 set_hard_margins (op);
1586 memset (&docinfo, 0, sizeof (DOCINFOW));
1587 docinfo.cbSize = sizeof (DOCINFOW);
1588 docinfo.lpszDocName = g_utf8_to_utf16 (op->priv->job_name, -1, NULL, NULL, NULL);
1589 docinfo.lpszOutput = NULL;
1590 docinfo.lpszDatatype = NULL;
1593 job_id = StartDocW (hDC, &docinfo);
1594 g_free ((void *)docinfo.lpszDocName);
1597 result = GTK_PRINT_OPERATION_RESULT_ERROR;
1598 g_set_error_literal (&priv->error,
1600 GTK_PRINT_ERROR_GENERAL,
1601 _("Error from StartDoc"));
1603 cairo_surface_destroy (op_win32->surface);
1604 op_win32->surface = NULL;
1608 result = GTK_PRINT_OPERATION_RESULT_APPLY;
1609 op_win32->hdc = hDC;
1610 op_win32->devmode = hDevMode;
1611 op_win32->devnames = hDevNames;
1612 op_win32->job_id = job_id;
1613 op->priv->print_pages = gtk_print_settings_get_print_pages (op->priv->print_settings);
1614 op->priv->num_page_ranges = 0;
1615 if (op->priv->print_pages == GTK_PRINT_PAGES_RANGES)
1616 op->priv->page_ranges = gtk_print_settings_get_page_ranges (op->priv->print_settings,
1617 &op->priv->num_page_ranges);
1618 op->priv->manual_num_copies = 1;
1619 op->priv->manual_collation = FALSE;
1620 op->priv->manual_reverse = FALSE;
1621 op->priv->manual_orientation = FALSE;
1622 op->priv->manual_scale = 1.0;
1623 op->priv->manual_page_set = GTK_PAGE_SET_ALL;
1624 op->priv->manual_number_up = 1;
1625 op->priv->manual_number_up_layout = GTK_NUMBER_UP_LAYOUT_LEFT_TO_RIGHT_TOP_TO_BOTTOM;
1627 op->priv->start_page = win32_start_page;
1628 op->priv->end_page = win32_end_page;
1629 op->priv->end_run = win32_end_run;
1632 if (!*do_print && hDC != NULL)
1635 if (!*do_print && hDevMode != NULL)
1636 GlobalFree (hDevMode);
1638 if (!*do_print && hDevNames != NULL)
1639 GlobalFree (hDevNames);
1644 GtkPrintOperationResult
1645 gtk_print_operation_run_with_dialog (GtkPrintOperation *op,
1650 LPPRINTDLGEXW printdlgex = NULL;
1651 LPPRINTPAGERANGE page_ranges = NULL;
1653 GtkWidget *invisible = NULL;
1654 GtkPrintOperationResult result;
1655 GtkPrintOperationWin32 *op_win32;
1656 GtkPrintOperationPrivate *priv;
1657 IPrintDialogCallback *callback;
1658 HPROPSHEETPAGE prop_page;
1664 op_win32 = g_new0 (GtkPrintOperationWin32, 1);
1665 priv->platform_data = op_win32;
1666 priv->free_platform_data = (GDestroyNotify) op_win32_free;
1670 invisible = gtk_invisible_new ();
1671 parentHWnd = get_parent_hwnd (invisible);
1674 parentHWnd = get_parent_hwnd (GTK_WIDGET (parent));
1676 printdlgex = (LPPRINTDLGEXW)GlobalAlloc (GPTR, sizeof (PRINTDLGEXW));
1679 result = GTK_PRINT_OPERATION_RESULT_ERROR;
1680 g_set_error_literal (&priv->error,
1682 GTK_PRINT_ERROR_NOMEM,
1683 _("Not enough free memory"));
1687 printdlgex->lStructSize = sizeof (PRINTDLGEXW);
1688 printdlgex->hwndOwner = parentHWnd;
1689 printdlgex->hDevMode = NULL;
1690 printdlgex->hDevNames = NULL;
1691 printdlgex->hDC = NULL;
1692 printdlgex->Flags = PD_RETURNDC | PD_NOSELECTION;
1693 if (op->priv->current_page == -1)
1694 printdlgex->Flags |= PD_NOCURRENTPAGE;
1695 printdlgex->Flags2 = 0;
1696 printdlgex->ExclusionFlags = 0;
1698 page_ranges = (LPPRINTPAGERANGE) GlobalAlloc (GPTR,
1699 MAX_PAGE_RANGES * sizeof (PRINTPAGERANGE));
1702 result = GTK_PRINT_OPERATION_RESULT_ERROR;
1703 g_set_error_literal (&priv->error,
1705 GTK_PRINT_ERROR_NOMEM,
1706 _("Not enough free memory"));
1710 printdlgex->nPageRanges = 0;
1711 printdlgex->nMaxPageRanges = MAX_PAGE_RANGES;
1712 printdlgex->lpPageRanges = page_ranges;
1713 printdlgex->nMinPage = 1;
1714 if (op->priv->nr_of_pages != -1)
1715 printdlgex->nMaxPage = op->priv->nr_of_pages;
1717 printdlgex->nMaxPage = 10000;
1718 printdlgex->nCopies = 1;
1719 printdlgex->hInstance = 0;
1720 printdlgex->lpPrintTemplateName = NULL;
1721 printdlgex->lpCallback = NULL;
1723 g_signal_emit_by_name (op, "create-custom-widget",
1724 &op->priv->custom_widget);
1725 if (op->priv->custom_widget) {
1726 prop_page = create_application_page (op);
1727 printdlgex->nPropertyPages = 1;
1728 printdlgex->lphPropertyPages = &prop_page;
1730 printdlgex->nPropertyPages = 0;
1731 printdlgex->lphPropertyPages = NULL;
1734 printdlgex->nStartPage = START_PAGE_GENERAL;
1735 printdlgex->dwResultAction = 0;
1737 dialog_from_print_settings (op, printdlgex);
1739 callback = print_callback_new ();
1740 printdlgex->lpCallback = (IUnknown *)callback;
1741 got_gdk_events_message = RegisterWindowMessage ("GDK_WIN32_GOT_EVENTS");
1743 hResult = PrintDlgExW (printdlgex);
1744 IUnknown_Release ((IUnknown *)callback);
1745 gdk_win32_set_modal_dialog_libgtk_only (NULL);
1747 if (hResult != S_OK)
1749 result = GTK_PRINT_OPERATION_RESULT_ERROR;
1750 if (hResult == E_OUTOFMEMORY)
1751 g_set_error_literal (&priv->error,
1753 GTK_PRINT_ERROR_NOMEM,
1754 _("Not enough free memory"));
1755 else if (hResult == E_INVALIDARG)
1756 g_set_error_literal (&priv->error,
1758 GTK_PRINT_ERROR_INTERNAL_ERROR,
1759 _("Invalid argument to PrintDlgEx"));
1760 else if (hResult == E_POINTER)
1761 g_set_error_literal (&priv->error,
1763 GTK_PRINT_ERROR_INTERNAL_ERROR,
1764 _("Invalid pointer to PrintDlgEx"));
1765 else if (hResult == E_HANDLE)
1766 g_set_error_literal (&priv->error,
1768 GTK_PRINT_ERROR_INTERNAL_ERROR,
1769 _("Invalid handle to PrintDlgEx"));
1771 g_set_error_literal (&priv->error,
1773 GTK_PRINT_ERROR_GENERAL,
1774 _("Unspecified error"));
1778 if (printdlgex->dwResultAction == PD_RESULT_PRINT ||
1779 printdlgex->dwResultAction == PD_RESULT_APPLY)
1781 result = GTK_PRINT_OPERATION_RESULT_APPLY;
1782 dialog_to_print_settings (op, printdlgex);
1785 result = GTK_PRINT_OPERATION_RESULT_CANCEL;
1787 if (printdlgex->dwResultAction == PD_RESULT_PRINT)
1791 double dpi_x, dpi_y;
1793 GtkPageSetup *page_setup;
1795 priv->print_context = _gtk_print_context_new (op);
1796 page_setup = create_page_setup (op);
1797 _gtk_print_context_set_page_setup (priv->print_context, page_setup);
1798 g_object_unref (page_setup);
1802 op_win32->surface = cairo_win32_printing_surface_create (printdlgex->hDC);
1804 dpi_x = (double)GetDeviceCaps (printdlgex->hDC, LOGPIXELSX);
1805 dpi_y = (double)GetDeviceCaps (printdlgex->hDC, LOGPIXELSY);
1807 cr = cairo_create (op_win32->surface);
1808 gtk_print_context_set_cairo_context (priv->print_context, cr, dpi_x, dpi_y);
1811 set_hard_margins (op);
1813 memset ( &docinfo, 0, sizeof (DOCINFOW));
1814 docinfo.cbSize = sizeof (DOCINFOW);
1815 docinfo.lpszDocName = g_utf8_to_utf16 (op->priv->job_name, -1, NULL, NULL, NULL);
1816 docinfo.lpszOutput = (LPCWSTR) NULL;
1817 docinfo.lpszDatatype = (LPCWSTR) NULL;
1820 job_id = StartDocW (printdlgex->hDC, &docinfo);
1821 g_free ((void *)docinfo.lpszDocName);
1824 result = GTK_PRINT_OPERATION_RESULT_ERROR;
1825 g_set_error_literal (&priv->error,
1827 GTK_PRINT_ERROR_GENERAL,
1828 _("Error from StartDoc"));
1830 cairo_surface_destroy (op_win32->surface);
1831 op_win32->surface = NULL;
1835 op_win32->hdc = printdlgex->hDC;
1836 op_win32->devmode = printdlgex->hDevMode;
1837 op_win32->devnames = printdlgex->hDevNames;
1838 op_win32->job_id = job_id;
1840 op->priv->print_pages = gtk_print_settings_get_print_pages (op->priv->print_settings);
1841 op->priv->num_page_ranges = 0;
1842 if (op->priv->print_pages == GTK_PRINT_PAGES_RANGES)
1843 op->priv->page_ranges = gtk_print_settings_get_page_ranges (op->priv->print_settings,
1844 &op->priv->num_page_ranges);
1845 op->priv->manual_num_copies = printdlgex->nCopies;
1846 op->priv->manual_collation = (printdlgex->Flags & PD_COLLATE) != 0;
1847 op->priv->manual_reverse = FALSE;
1848 op->priv->manual_orientation = FALSE;
1849 op->priv->manual_scale = 1.0;
1850 op->priv->manual_page_set = GTK_PAGE_SET_ALL;
1851 op->priv->manual_number_up = 1;
1852 op->priv->manual_number_up_layout = GTK_NUMBER_UP_LAYOUT_LEFT_TO_RIGHT_TOP_TO_BOTTOM;
1855 op->priv->start_page = win32_start_page;
1856 op->priv->end_page = win32_end_page;
1857 op->priv->end_run = win32_end_run;
1860 if (!*do_print && printdlgex && printdlgex->hDC != NULL)
1861 DeleteDC (printdlgex->hDC);
1863 if (!*do_print && printdlgex && printdlgex->hDevMode != NULL)
1864 GlobalFree (printdlgex->hDevMode);
1866 if (!*do_print && printdlgex && printdlgex->hDevNames != NULL)
1867 GlobalFree (printdlgex->hDevNames);
1870 GlobalFree (page_ranges);
1873 GlobalFree (printdlgex);
1876 gtk_widget_destroy (invisible);
1881 GtkPrintOperationResult
1882 _gtk_print_operation_platform_backend_run_dialog (GtkPrintOperation *op,
1883 gboolean show_dialog,
1888 return gtk_print_operation_run_with_dialog (op, parent, do_print);
1890 return gtk_print_operation_run_without_dialog (op, do_print);
1894 _gtk_print_operation_platform_backend_launch_preview (GtkPrintOperation *op,
1895 cairo_surface_t *surface,
1897 const gchar *filename)
1900 HENHMETAFILE metafile;
1902 dc = cairo_win32_surface_get_dc (surface);
1903 cairo_surface_destroy (surface);
1904 metafile = CloseEnhMetaFile (dc);
1905 DeleteEnhMetaFile (metafile);
1907 ShellExecuteW (NULL, L"open", (gunichar2 *)filename, NULL, NULL, SW_SHOW);
1911 _gtk_print_operation_platform_backend_preview_start_page (GtkPrintOperation *op,
1912 cairo_surface_t *surface,
1915 HDC dc = cairo_win32_surface_get_dc (surface);
1920 _gtk_print_operation_platform_backend_preview_end_page (GtkPrintOperation *op,
1921 cairo_surface_t *surface,
1926 cairo_surface_show_page (surface);
1928 /* TODO: Enhanced metafiles don't support multiple pages.
1930 dc = cairo_win32_surface_get_dc (surface);
1935 _gtk_print_operation_platform_backend_create_preview_surface (GtkPrintOperation *op,
1936 GtkPageSetup *page_setup,
1941 GtkPaperSize *paper_size;
1946 gunichar2 *filename_utf16;
1949 template = g_build_filename (g_get_tmp_dir (), "prXXXXXX", NULL);
1950 fd = g_mkstemp (template);
1953 filename = g_strconcat (template, ".emf", NULL);
1956 filename_utf16 = g_utf8_to_utf16 (filename, -1, NULL, NULL, NULL);
1959 paper_size = gtk_page_setup_get_paper_size (page_setup);
1961 /* The rectangle dimensions are given in hundredths of a millimeter */
1963 rect.right = 100.0 * gtk_paper_size_get_width (paper_size, GTK_UNIT_MM);
1965 rect.bottom = 100.0 * gtk_paper_size_get_height (paper_size, GTK_UNIT_MM);
1967 metafile_dc = CreateEnhMetaFileW (NULL, filename_utf16,
1968 &rect, L"Gtk+\0Print Preview\0\0");
1969 if (metafile_dc == NULL)
1971 g_warning ("Can't create metafile");
1975 *target = (char *)filename_utf16;
1977 *dpi_x = (double)GetDeviceCaps (metafile_dc, LOGPIXELSX);
1978 *dpi_y = (double)GetDeviceCaps (metafile_dc, LOGPIXELSY);
1980 return cairo_win32_printing_surface_create (metafile_dc);
1984 _gtk_print_operation_platform_backend_resize_preview_surface (GtkPrintOperation *op,
1985 GtkPageSetup *page_setup,
1986 cairo_surface_t *surface)
1988 /* TODO: Implement */
1992 gtk_print_run_page_setup_dialog (GtkWindow *parent,
1993 GtkPageSetup *page_setup,
1994 GtkPrintSettings *settings)
1996 LPPAGESETUPDLGW pagesetupdlg = NULL;
1998 gboolean free_settings;
1999 const char *printer;
2000 GtkPaperSize *paper_size;
2001 DWORD measure_system;
2005 pagesetupdlg = (LPPAGESETUPDLGW)GlobalAlloc (GPTR, sizeof (PAGESETUPDLGW));
2009 free_settings = FALSE;
2010 if (settings == NULL)
2012 settings = gtk_print_settings_new ();
2013 free_settings = TRUE;
2016 memset (pagesetupdlg, 0, sizeof (PAGESETUPDLGW));
2018 pagesetupdlg->lStructSize = sizeof (PAGESETUPDLGW);
2021 pagesetupdlg->hwndOwner = get_parent_hwnd (GTK_WIDGET (parent));
2023 pagesetupdlg->hwndOwner = NULL;
2025 pagesetupdlg->Flags = PSD_DEFAULTMINMARGINS;
2026 pagesetupdlg->hDevMode = devmode_from_settings (settings, page_setup);
2027 pagesetupdlg->hDevNames = NULL;
2028 printer = gtk_print_settings_get_printer (settings);
2030 pagesetupdlg->hDevNames = gtk_print_win32_devnames_to_win32_from_printer_name (printer);
2032 GetLocaleInfoW (LOCALE_USER_DEFAULT, LOCALE_IMEASURE|LOCALE_RETURN_NUMBER,
2033 (LPWSTR)&measure_system, sizeof (DWORD));
2035 if (measure_system == 0)
2037 pagesetupdlg->Flags |= PSD_INHUNDREDTHSOFMILLIMETERS;
2043 pagesetupdlg->Flags |= PSD_INTHOUSANDTHSOFINCHES;
2044 unit = GTK_UNIT_INCH;
2048 /* This is the object we return, we allocate it here so that
2049 * we can use the default page margins */
2051 page_setup = gtk_page_setup_copy (page_setup);
2053 page_setup = gtk_page_setup_new ();
2055 pagesetupdlg->Flags |= PSD_MARGINS;
2056 pagesetupdlg->rtMargin.left =
2057 floor (gtk_page_setup_get_left_margin (page_setup, unit) * scale + 0.5);
2058 pagesetupdlg->rtMargin.right =
2059 floor (gtk_page_setup_get_right_margin (page_setup, unit) * scale + 0.5);
2060 pagesetupdlg->rtMargin.top =
2061 floor (gtk_page_setup_get_top_margin (page_setup, unit) * scale + 0.5);
2062 pagesetupdlg->rtMargin.bottom =
2063 floor (gtk_page_setup_get_bottom_margin (page_setup, unit) * scale + 0.5);
2065 pagesetupdlg->Flags |= PSD_ENABLEPAGESETUPHOOK;
2066 pagesetupdlg->lpfnPageSetupHook = run_mainloop_hook;
2067 got_gdk_events_message = RegisterWindowMessage ("GDK_WIN32_GOT_EVENTS");
2069 res = PageSetupDlgW (pagesetupdlg);
2070 gdk_win32_set_modal_dialog_libgtk_only (NULL);
2074 if (pagesetupdlg->hDevNames != NULL)
2075 devnames_to_settings (settings, pagesetupdlg->hDevNames);
2077 if (pagesetupdlg->hDevMode != NULL)
2078 devmode_to_settings (settings, pagesetupdlg->hDevMode);
2083 gtk_page_setup_set_orientation (page_setup,
2084 gtk_print_settings_get_orientation (settings));
2085 paper_size = gtk_print_settings_get_paper_size (settings);
2088 gtk_page_setup_set_paper_size (page_setup, paper_size);
2089 gtk_paper_size_free (paper_size);
2092 if (pagesetupdlg->Flags & PSD_INHUNDREDTHSOFMILLIMETERS)
2099 unit = GTK_UNIT_INCH;
2103 gtk_page_setup_set_left_margin (page_setup,
2104 pagesetupdlg->rtMargin.left / scale,
2106 gtk_page_setup_set_right_margin (page_setup,
2107 pagesetupdlg->rtMargin.right / scale,
2109 gtk_page_setup_set_top_margin (page_setup,
2110 pagesetupdlg->rtMargin.top / scale,
2112 gtk_page_setup_set_bottom_margin (page_setup,
2113 pagesetupdlg->rtMargin.bottom / scale,
2118 g_object_unref (settings);
2124 gtk_print_run_page_setup_dialog_async (GtkWindow *parent,
2125 GtkPageSetup *page_setup,
2126 GtkPrintSettings *settings,
2127 GtkPageSetupDoneFunc done_cb,
2130 GtkPageSetup *new_page_setup;
2132 new_page_setup = gtk_print_run_page_setup_dialog (parent, page_setup, settings);
2133 done_cb (new_page_setup, data);
2134 g_object_unref (new_page_setup);