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"
48 #define MAX_PAGE_RANGES 20
49 #define STATUS_POLLING_TIME 2000
51 #ifndef JOB_STATUS_RESTART
52 #define JOB_STATUS_RESTART 0x800
55 #ifndef JOB_STATUS_COMPLETE
56 #define JOB_STATUS_COMPLETE 0x1000
67 cairo_surface_t *surface;
68 GtkWidget *embed_widget;
69 } GtkPrintOperationWin32;
71 static void win32_poll_status (GtkPrintOperation *op);
73 static const GUID myIID_IPrintDialogCallback = {0x5852a2c3,0x6530,0x11d1,{0xb6,0xa3,0x0,0x0,0xf8,0x75,0x7b,0xf9}};
77 #define INTERFACE IPrintDialogCallback
78 DECLARE_INTERFACE_ (IPrintDialogCallback, IUnknown)
80 STDMETHOD (QueryInterface)(THIS_ REFIID,LPVOID*) PURE;
81 STDMETHOD_ (ULONG, AddRef)(THIS) PURE;
82 STDMETHOD_ (ULONG, Release)(THIS) PURE;
83 STDMETHOD (InitDone)(THIS) PURE;
84 STDMETHOD (SelectionChange)(THIS) PURE;
85 STDMETHOD (HandleMessage)(THIS_ HWND,UINT,WPARAM,LPARAM,LRESULT*) PURE;
89 static UINT got_gdk_events_message;
92 run_mainloop_hook (HWND hdlg, UINT uiMsg, WPARAM wParam, LPARAM lParam)
94 if (uiMsg == WM_INITDIALOG)
96 gdk_win32_set_modal_dialog_libgtk_only (hdlg);
97 while (gtk_events_pending ())
98 gtk_main_iteration ();
100 else if (uiMsg == got_gdk_events_message)
102 while (gtk_events_pending ())
103 gtk_main_iteration ();
109 static GtkPageOrientation
110 orientation_from_win32 (short orientation)
112 if (orientation == DMORIENT_LANDSCAPE)
113 return GTK_PAGE_ORIENTATION_LANDSCAPE;
114 return GTK_PAGE_ORIENTATION_PORTRAIT;
118 orientation_to_win32 (GtkPageOrientation orientation)
120 if (orientation == GTK_PAGE_ORIENTATION_LANDSCAPE ||
121 orientation == GTK_PAGE_ORIENTATION_REVERSE_LANDSCAPE)
122 return DMORIENT_LANDSCAPE;
123 return DMORIENT_PORTRAIT;
126 static GtkPaperSize *
127 paper_size_from_win32 (short size)
133 case DMPAPER_LETTER_TRANSVERSE:
135 case DMPAPER_LETTERSMALL:
138 case DMPAPER_TABLOID:
145 case DMPAPER_STATEMENT:
148 case DMPAPER_EXECUTIVE:
149 name = "na_executive";
152 case DMPAPER_A3_TRANSVERSE:
156 case DMPAPER_A4SMALL:
157 case DMPAPER_A4_TRANSVERSE:
161 case DMPAPER_A5_TRANSVERSE:
168 case DMPAPER_B5_TRANSVERSE:
184 name = "na_number-9";
187 name = "na_number-10";
190 name = "na_number-11";
193 name = "na_number-12";
196 name = "na_number-14";
222 case DMPAPER_ENV_C65:
234 case DMPAPER_ENV_ITALY:
237 case DMPAPER_ENV_MONARCH:
240 case DMPAPER_ENV_PERSONAL:
241 name = "na_personal";
243 case DMPAPER_FANFOLD_US:
244 name = "na_fanfold-us";
246 case DMPAPER_FANFOLD_STD_GERMAN:
247 name = "na_fanfold-eur";
249 case DMPAPER_FANFOLD_LGL_GERMAN:
250 name = "na_foolscap";
255 case DMPAPER_JAPANESE_POSTCARD:
264 case DMPAPER_ENV_INVITE:
267 case DMPAPER_LETTER_EXTRA:
268 case DMPAPER_LETTER_EXTRA_TRANSVERSE:
269 name = "na_letter-extra";
271 case DMPAPER_LEGAL_EXTRA:
272 name = "na_legal-extra";
274 case DMPAPER_TABLOID_EXTRA:
277 case DMPAPER_A4_EXTRA:
278 name = "iso_a4-extra";
283 case DMPAPER_LETTER_PLUS:
284 name = "na_letter-plus";
286 case DMPAPER_A3_EXTRA:
287 case DMPAPER_A3_EXTRA_TRANSVERSE:
288 name = "iso_a3-extra";
290 case DMPAPER_A5_EXTRA:
291 name = "iso_a5-extra";
293 case DMPAPER_B5_EXTRA:
294 name = "iso_b5-extra";
306 return gtk_paper_size_new (name);
312 paper_size_to_win32 (GtkPaperSize *paper_size)
316 if (gtk_paper_size_is_custom (paper_size))
319 format = gtk_paper_size_get_name (paper_size);
321 if (strcmp (format, "na_letter") == 0)
322 return DMPAPER_LETTER;
323 if (strcmp (format, "na_ledger") == 0)
324 return DMPAPER_LEDGER;
325 if (strcmp (format, "na_legal") == 0)
326 return DMPAPER_LEGAL;
327 if (strcmp (format, "na_invoice") == 0)
328 return DMPAPER_STATEMENT;
329 if (strcmp (format, "na_executive") == 0)
330 return DMPAPER_EXECUTIVE;
331 if (strcmp (format, "iso_a2") == 0)
333 if (strcmp (format, "iso_a3") == 0)
335 if (strcmp (format, "iso_a4") == 0)
337 if (strcmp (format, "iso_a5") == 0)
339 if (strcmp (format, "iso_b4") == 0)
341 if (strcmp (format, "iso_b5") == 0)
343 if (strcmp (format, "na_quarto") == 0)
344 return DMPAPER_QUARTO;
345 if (strcmp (format, "na_10x14") == 0)
346 return DMPAPER_10X14;
347 if (strcmp (format, "na_number-9") == 0)
348 return DMPAPER_ENV_9;
349 if (strcmp (format, "na_number-10") == 0)
350 return DMPAPER_ENV_10;
351 if (strcmp (format, "na_number-11") == 0)
352 return DMPAPER_ENV_11;
353 if (strcmp (format, "na_number-12") == 0)
354 return DMPAPER_ENV_12;
355 if (strcmp (format, "na_number-14") == 0)
356 return DMPAPER_ENV_14;
357 if (strcmp (format, "na_c") == 0)
358 return DMPAPER_CSHEET;
359 if (strcmp (format, "na_d") == 0)
360 return DMPAPER_DSHEET;
361 if (strcmp (format, "na_e") == 0)
362 return DMPAPER_ESHEET;
363 if (strcmp (format, "iso_dl") == 0)
364 return DMPAPER_ENV_DL;
365 if (strcmp (format, "iso_c3") == 0)
366 return DMPAPER_ENV_C3;
367 if (strcmp (format, "iso_c4") == 0)
368 return DMPAPER_ENV_C4;
369 if (strcmp (format, "iso_c5") == 0)
370 return DMPAPER_ENV_C5;
371 if (strcmp (format, "iso_c6") == 0)
372 return DMPAPER_ENV_C6;
373 if (strcmp (format, "iso_c5c6") == 0)
374 return DMPAPER_ENV_C65;
375 if (strcmp (format, "iso_b6") == 0)
376 return DMPAPER_ENV_B6;
377 if (strcmp (format, "om_italian") == 0)
378 return DMPAPER_ENV_ITALY;
379 if (strcmp (format, "na_monarch") == 0)
380 return DMPAPER_ENV_MONARCH;
381 if (strcmp (format, "na_personal") == 0)
382 return DMPAPER_ENV_PERSONAL;
383 if (strcmp (format, "na_fanfold-us") == 0)
384 return DMPAPER_FANFOLD_US;
385 if (strcmp (format, "na_fanfold-eur") == 0)
386 return DMPAPER_FANFOLD_STD_GERMAN;
387 if (strcmp (format, "na_foolscap") == 0)
388 return DMPAPER_FANFOLD_LGL_GERMAN;
389 if (strcmp (format, "jpn_hagaki") == 0)
390 return DMPAPER_JAPANESE_POSTCARD;
391 if (strcmp (format, "na_9x11") == 0)
393 if (strcmp (format, "na_10x11") == 0)
394 return DMPAPER_10X11;
395 if (strcmp (format, "om_invite") == 0)
396 return DMPAPER_ENV_INVITE;
397 if (strcmp (format, "na_letter-extra") == 0)
398 return DMPAPER_LETTER_EXTRA;
399 if (strcmp (format, "na_legal-extra") == 0)
400 return DMPAPER_LEGAL_EXTRA;
401 if (strcmp (format, "na_arch") == 0)
402 return DMPAPER_TABLOID_EXTRA;
403 if (strcmp (format, "iso_a3-extra") == 0)
404 return DMPAPER_A3_EXTRA;
405 if (strcmp (format, "iso_a4-extra") == 0)
406 return DMPAPER_A4_EXTRA;
407 if (strcmp (format, "iso_a5-extra") == 0)
408 return DMPAPER_A5_EXTRA;
409 if (strcmp (format, "iso_b5-extra") == 0)
410 return DMPAPER_B5_EXTRA;
411 if (strcmp (format, "na_b-plus") == 0)
412 return DMPAPER_B_PLUS;
413 if (strcmp (format, "na_letter-plus") == 0)
414 return DMPAPER_LETTER_PLUS;
420 get_default_printer (void)
422 wchar_t *win32_printer_name = NULL;
423 gchar *printer_name = NULL;
426 GetDefaultPrinterW (NULL, &needed);
427 win32_printer_name = g_malloc ((gsize) needed * sizeof (wchar_t));
428 if (!GetDefaultPrinterW (win32_printer_name, &needed))
430 g_free (win32_printer_name);
433 printer_name = g_utf16_to_utf8 (win32_printer_name, -1, NULL, NULL, NULL);
434 g_free (win32_printer_name);
440 win32_start_page (GtkPrintOperation *op,
441 GtkPrintContext *print_context,
442 GtkPageSetup *page_setup)
444 GtkPrintOperationWin32 *op_win32 = op->priv->platform_data;
446 GtkPaperSize *paper_size;
448 devmode = GlobalLock (op_win32->devmode);
450 devmode->dmFields |= DM_ORIENTATION;
451 devmode->dmOrientation =
452 orientation_to_win32 (gtk_page_setup_get_orientation (page_setup));
454 paper_size = gtk_page_setup_get_paper_size (page_setup);
455 devmode->dmFields |= DM_PAPERSIZE;
456 devmode->dmFields &= ~(DM_PAPERWIDTH | DM_PAPERLENGTH);
457 devmode->dmPaperSize = paper_size_to_win32 (paper_size);
458 if (devmode->dmPaperSize == 0)
460 devmode->dmPaperSize = DMPAPER_USER;
461 devmode->dmFields |= DM_PAPERWIDTH | DM_PAPERLENGTH;
463 /* Lengths in DEVMODE are in tenths of a millimeter */
464 devmode->dmPaperWidth = gtk_paper_size_get_width (paper_size, GTK_UNIT_MM) * 10.0;
465 devmode->dmPaperLength = gtk_paper_size_get_height (paper_size, GTK_UNIT_MM) * 10.0;
468 ResetDCW (op_win32->hdc, devmode);
470 GlobalUnlock (op_win32->devmode);
472 StartPage (op_win32->hdc);
476 win32_end_page (GtkPrintOperation *op,
477 GtkPrintContext *print_context)
479 GtkPrintOperationWin32 *op_win32 = op->priv->platform_data;
481 cairo_surface_show_page (op_win32->surface);
483 EndPage (op_win32->hdc);
487 win32_poll_status_timeout (GtkPrintOperation *op)
489 GtkPrintOperationWin32 *op_win32 = op->priv->platform_data;
491 op_win32->timeout_id = 0;
492 /* We need to ref this, as setting the status to finished
493 might unref the object */
495 win32_poll_status (op);
497 if (!gtk_print_operation_is_finished (op))
498 op_win32->timeout_id = gdk_threads_add_timeout (STATUS_POLLING_TIME,
499 (GSourceFunc)win32_poll_status_timeout,
507 win32_end_run (GtkPrintOperation *op,
511 GtkPrintOperationWin32 *op_win32 = op->priv->platform_data;
513 HANDLE printerHandle = 0;
515 cairo_surface_finish (op_win32->surface);
517 EndDoc (op_win32->hdc);
519 if (op->priv->track_print_status)
521 devnames = GlobalLock (op_win32->devnames);
522 if (!OpenPrinterW (((gunichar2 *)devnames) + devnames->wDeviceOffset,
523 &printerHandle, NULL))
525 GlobalUnlock (op_win32->devnames);
528 GlobalFree (op_win32->devmode);
529 GlobalFree (op_win32->devnames);
531 cairo_surface_destroy (op_win32->surface);
532 op_win32->surface = NULL;
534 DeleteDC (op_win32->hdc);
536 if (printerHandle != 0)
538 op_win32->printerHandle = printerHandle;
539 win32_poll_status (op);
540 op_win32->timeout_id = gdk_threads_add_timeout (STATUS_POLLING_TIME,
541 (GSourceFunc)win32_poll_status_timeout,
545 /* Dunno what happened, pretend its finished */
546 _gtk_print_operation_set_status (op, GTK_PRINT_STATUS_FINISHED, NULL);
550 win32_poll_status (GtkPrintOperation *op)
552 GtkPrintOperationWin32 *op_win32 = op->priv->platform_data;
555 JOB_INFO_1W *job_info;
556 GtkPrintStatus status;
560 GetJobW (op_win32->printerHandle, op_win32->job_id,
561 1,(LPBYTE)NULL, 0, &needed);
562 data = g_malloc (needed);
563 ret = GetJobW (op_win32->printerHandle, op_win32->job_id,
564 1, (LPBYTE)data, needed, &needed);
570 job_info = (JOB_INFO_1W *)data;
571 win32_status = job_info->Status;
573 if (job_info->pStatus)
574 status_str = g_utf16_to_utf8 (job_info->pStatus,
575 -1, NULL, NULL, NULL);
578 (JOB_STATUS_COMPLETE | JOB_STATUS_PRINTED))
579 status = GTK_PRINT_STATUS_FINISHED;
580 else if (win32_status &
581 (JOB_STATUS_OFFLINE |
582 JOB_STATUS_PAPEROUT |
584 JOB_STATUS_USER_INTERVENTION))
586 status = GTK_PRINT_STATUS_PENDING_ISSUE;
587 if (status_str == NULL)
589 if (win32_status & JOB_STATUS_OFFLINE)
590 status_str = g_strdup (_("Printer offline"));
591 else if (win32_status & JOB_STATUS_PAPEROUT)
592 status_str = g_strdup (_("Out of paper"));
593 else if (win32_status & JOB_STATUS_PAUSED)
594 status_str = g_strdup (_("Paused"));
595 else if (win32_status & JOB_STATUS_USER_INTERVENTION)
596 status_str = g_strdup (_("Need user intervention"));
599 else if (win32_status &
600 (JOB_STATUS_BLOCKED_DEVQ |
603 status = GTK_PRINT_STATUS_FINISHED_ABORTED;
604 else if (win32_status &
605 (JOB_STATUS_SPOOLING |
606 JOB_STATUS_DELETING))
607 status = GTK_PRINT_STATUS_PENDING;
608 else if (win32_status & JOB_STATUS_PRINTING)
609 status = GTK_PRINT_STATUS_PRINTING;
611 status = GTK_PRINT_STATUS_FINISHED;
614 status = GTK_PRINT_STATUS_FINISHED;
618 _gtk_print_operation_set_status (op, status, status_str);
624 op_win32_free (GtkPrintOperationWin32 *op_win32)
626 if (op_win32->printerHandle)
627 ClosePrinter (op_win32->printerHandle);
628 if (op_win32->timeout_id != 0)
629 g_source_remove (op_win32->timeout_id);
634 get_parent_hwnd (GtkWidget *widget)
636 gtk_widget_realize (widget);
637 return gdk_win32_drawable_get_handle (widget->window);
641 devnames_to_settings (GtkPrintSettings *settings,
644 GtkPrintWin32Devnames *devnames = gtk_print_win32_devnames_from_win32 (hDevNames);
645 gtk_print_settings_set_printer (settings, devnames->device);
646 gtk_print_win32_devnames_free (devnames);
650 devmode_to_settings (GtkPrintSettings *settings,
655 devmode = GlobalLock (hDevMode);
657 gtk_print_settings_set_int (settings, GTK_PRINT_SETTINGS_WIN32_DRIVER_VERSION,
658 devmode->dmDriverVersion);
659 if (devmode->dmDriverExtra != 0)
661 char *extra = g_base64_encode (((char *)devmode) + sizeof (DEVMODEW),
662 devmode->dmDriverExtra);
663 gtk_print_settings_set (settings,
664 GTK_PRINT_SETTINGS_WIN32_DRIVER_EXTRA,
669 if (devmode->dmFields & DM_ORIENTATION)
670 gtk_print_settings_set_orientation (settings,
671 orientation_from_win32 (devmode->dmOrientation));
674 if (devmode->dmFields & DM_PAPERSIZE &&
675 devmode->dmPaperSize != 0)
677 GtkPaperSize *paper_size = paper_size_from_win32 (devmode->dmPaperSize);
680 gtk_print_settings_set_paper_size (settings, paper_size);
681 gtk_paper_size_free (paper_size);
683 gtk_print_settings_set_int (settings, "win32-paper-size", (int)devmode->dmPaperSize);
685 else if ((devmode->dmFields & DM_PAPERSIZE &&
686 devmode->dmPaperSize == 0) ||
687 ((devmode->dmFields & DM_PAPERWIDTH) &&
688 (devmode->dmFields & DM_PAPERLENGTH)))
690 GtkPaperSize *paper_size;
691 char *form_name = NULL;
692 if (devmode->dmFields & DM_FORMNAME)
693 form_name = g_utf16_to_utf8 (devmode->dmFormName,
694 -1, NULL, NULL, NULL);
695 if (form_name == NULL || form_name[0] == 0)
696 form_name = g_strdup (_("Custom size"));
698 /* Lengths in DEVMODE are in tenths of a millimeter */
699 paper_size = gtk_paper_size_new_custom (form_name,
701 devmode->dmPaperWidth / 10.0,
702 devmode->dmPaperLength / 10.0,
704 gtk_print_settings_set_paper_size (settings, paper_size);
705 gtk_paper_size_free (paper_size);
708 if (devmode->dmFields & DM_SCALE)
709 gtk_print_settings_set_scale (settings, devmode->dmScale);
711 if (devmode->dmFields & DM_COPIES)
712 gtk_print_settings_set_n_copies (settings,
715 if (devmode->dmFields & DM_DEFAULTSOURCE)
718 switch (devmode->dmDefaultSource)
730 case DMBIN_ENVMANUAL:
731 source = "envelope-manual";
745 case DMBIN_FORMSOURCE:
746 source = "form-source";
748 case DMBIN_LARGECAPACITY:
749 source = "large-capacity";
752 source = "large-format";
758 source = "small-format";
761 gtk_print_settings_set_default_source (settings, source);
762 gtk_print_settings_set_int (settings, "win32-default-source", devmode->dmDefaultSource);
765 if (devmode->dmFields & DM_PRINTQUALITY)
767 GtkPrintQuality quality;
768 switch (devmode->dmPrintQuality)
771 quality = GTK_PRINT_QUALITY_LOW;
774 quality = GTK_PRINT_QUALITY_NORMAL;
778 quality = GTK_PRINT_QUALITY_HIGH;
781 quality = GTK_PRINT_QUALITY_DRAFT;
784 gtk_print_settings_set_quality (settings, quality);
785 gtk_print_settings_set_int (settings, "win32-print-quality", devmode->dmPrintQuality);
788 if (devmode->dmFields & DM_COLOR)
789 gtk_print_settings_set_use_color (settings, devmode->dmColor == DMCOLOR_COLOR);
791 if (devmode->dmFields & DM_DUPLEX)
793 GtkPrintDuplex duplex;
794 switch (devmode->dmDuplex)
798 duplex = GTK_PRINT_DUPLEX_SIMPLEX;
800 case DMDUP_HORIZONTAL:
801 duplex = GTK_PRINT_DUPLEX_HORIZONTAL;
804 duplex = GTK_PRINT_DUPLEX_VERTICAL;
808 gtk_print_settings_set_duplex (settings, duplex);
811 if (devmode->dmFields & DM_COLLATE)
812 gtk_print_settings_set_collate (settings,
813 devmode->dmCollate == DMCOLLATE_TRUE);
815 if (devmode->dmFields & DM_MEDIATYPE)
818 switch (devmode->dmMediaType)
821 case DMMEDIA_STANDARD:
822 media_type = "stationery";
824 case DMMEDIA_TRANSPARENCY:
825 media_type = "transparency";
828 media_type = "photographic-glossy";
831 gtk_print_settings_set_media_type (settings, media_type);
832 gtk_print_settings_set_int (settings, "win32-media-type", devmode->dmMediaType);
835 if (devmode->dmFields & DM_DITHERTYPE)
838 switch (devmode->dmDitherType)
847 case DMDITHER_COARSE:
850 case DMDITHER_LINEART:
853 case DMDITHER_GRAYSCALE:
854 dither = "grayscale";
856 case DMDITHER_ERRORDIFFUSION:
857 dither = "error-diffusion";
860 gtk_print_settings_set_dither (settings, dither);
861 gtk_print_settings_set_int (settings, "win32-dither-type", devmode->dmDitherType);
864 GlobalUnlock (hDevMode);
868 dialog_to_print_settings (GtkPrintOperation *op,
869 LPPRINTDLGEXW printdlgex)
872 GtkPrintSettings *settings;
874 settings = gtk_print_settings_new ();
876 gtk_print_settings_set_print_pages (settings,
877 GTK_PRINT_PAGES_ALL);
878 if (printdlgex->Flags & PD_CURRENTPAGE)
879 gtk_print_settings_set_print_pages (settings,
880 GTK_PRINT_PAGES_CURRENT);
881 else if (printdlgex->Flags & PD_PAGENUMS)
882 gtk_print_settings_set_print_pages (settings,
883 GTK_PRINT_PAGES_RANGES);
885 if (printdlgex->nPageRanges > 0)
887 GtkPageRange *ranges;
888 ranges = g_new (GtkPageRange, printdlgex->nPageRanges);
890 for (i = 0; i < printdlgex->nPageRanges; i++)
892 ranges[i].start = printdlgex->lpPageRanges[i].nFromPage - 1;
893 ranges[i].end = printdlgex->lpPageRanges[i].nToPage - 1;
896 gtk_print_settings_set_page_ranges (settings, ranges,
897 printdlgex->nPageRanges);
901 if (printdlgex->hDevNames != NULL)
902 devnames_to_settings (settings, printdlgex->hDevNames);
904 if (printdlgex->hDevMode != NULL)
905 devmode_to_settings (settings, printdlgex->hDevMode);
907 gtk_print_operation_set_print_settings (op, settings);
911 devmode_from_settings (GtkPrintSettings *settings,
912 GtkPageSetup *page_setup)
917 GtkPaperSize *paper_size;
918 const char *extras_base64;
924 extras_base64 = gtk_print_settings_get (settings, GTK_PRINT_SETTINGS_WIN32_DRIVER_EXTRA);
926 extras = g_base64_decode (extras_base64, &extras_len);
928 hDevMode = GlobalAlloc (GMEM_MOVEABLE,
929 sizeof (DEVMODEW) + extras_len);
931 devmode = GlobalLock (hDevMode);
933 memset (devmode, 0, sizeof (DEVMODEW));
935 devmode->dmSpecVersion = DM_SPECVERSION;
936 devmode->dmSize = sizeof (DEVMODEW);
938 devmode->dmDriverExtra = 0;
939 if (extras && extras_len > 0)
941 devmode->dmDriverExtra = extras_len;
942 memcpy (((char *)devmode) + sizeof (DEVMODEW), extras, extras_len);
945 if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_WIN32_DRIVER_VERSION))
946 devmode->dmDriverVersion = gtk_print_settings_get_int (settings, GTK_PRINT_SETTINGS_WIN32_DRIVER_VERSION);
949 gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_ORIENTATION))
951 GtkPageOrientation orientation = gtk_print_settings_get_orientation (settings);
953 orientation = gtk_page_setup_get_orientation (page_setup);
954 devmode->dmFields |= DM_ORIENTATION;
955 devmode->dmOrientation = orientation_to_win32 (orientation);
959 paper_size = gtk_paper_size_copy (gtk_page_setup_get_paper_size (page_setup));
963 if (gtk_print_settings_has_key (settings, "win32-paper-size") &&
964 (size = gtk_print_settings_get_int (settings, "win32-paper-size")) != 0)
966 devmode->dmFields |= DM_PAPERSIZE;
967 devmode->dmPaperSize = size;
971 paper_size = gtk_print_settings_get_paper_size (settings);
975 devmode->dmFields |= DM_PAPERSIZE;
976 devmode->dmPaperSize = paper_size_to_win32 (paper_size);
977 if (devmode->dmPaperSize == 0)
979 devmode->dmPaperSize = DMPAPER_USER;
980 devmode->dmFields |= DM_PAPERWIDTH | DM_PAPERLENGTH;
982 /* Lengths in DEVMODE are in tenths of a millimeter */
983 devmode->dmPaperWidth = gtk_paper_size_get_width (paper_size, GTK_UNIT_MM) * 10.0;
984 devmode->dmPaperLength = gtk_paper_size_get_height (paper_size, GTK_UNIT_MM) * 10.0;
986 gtk_paper_size_free (paper_size);
989 if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_SCALE))
991 devmode->dmFields |= DM_SCALE;
992 devmode->dmScale = gtk_print_settings_get_scale (settings);
995 if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_N_COPIES))
997 devmode->dmFields |= DM_COPIES;
998 devmode->dmCopies = gtk_print_settings_get_n_copies (settings);
1001 if (gtk_print_settings_has_key (settings, "win32-default-source"))
1003 devmode->dmFields |= DM_DEFAULTSOURCE;
1004 devmode->dmDefaultSource = gtk_print_settings_get_int (settings, "win32-default-source");
1006 else if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_DEFAULT_SOURCE))
1008 devmode->dmFields |= DM_DEFAULTSOURCE;
1009 devmode->dmDefaultSource = DMBIN_AUTO;
1011 val = gtk_print_settings_get_default_source (settings);
1012 if (strcmp (val, "auto") == 0)
1013 devmode->dmDefaultSource = DMBIN_AUTO;
1014 if (strcmp (val, "cassette") == 0)
1015 devmode->dmDefaultSource = DMBIN_CASSETTE;
1016 if (strcmp (val, "envelope") == 0)
1017 devmode->dmDefaultSource = DMBIN_ENVELOPE;
1018 if (strcmp (val, "envelope-manual") == 0)
1019 devmode->dmDefaultSource = DMBIN_ENVMANUAL;
1020 if (strcmp (val, "lower") == 0)
1021 devmode->dmDefaultSource = DMBIN_LOWER;
1022 if (strcmp (val, "manual") == 0)
1023 devmode->dmDefaultSource = DMBIN_MANUAL;
1024 if (strcmp (val, "middle") == 0)
1025 devmode->dmDefaultSource = DMBIN_MIDDLE;
1026 if (strcmp (val, "only-one") == 0)
1027 devmode->dmDefaultSource = DMBIN_ONLYONE;
1028 if (strcmp (val, "form-source") == 0)
1029 devmode->dmDefaultSource = DMBIN_FORMSOURCE;
1030 if (strcmp (val, "large-capacity") == 0)
1031 devmode->dmDefaultSource = DMBIN_LARGECAPACITY;
1032 if (strcmp (val, "large-format") == 0)
1033 devmode->dmDefaultSource = DMBIN_LARGEFMT;
1034 if (strcmp (val, "tractor") == 0)
1035 devmode->dmDefaultSource = DMBIN_TRACTOR;
1036 if (strcmp (val, "small-format") == 0)
1037 devmode->dmDefaultSource = DMBIN_SMALLFMT;
1040 if (gtk_print_settings_has_key (settings, "win32-print-quality"))
1042 devmode->dmFields |= DM_PRINTQUALITY;
1043 devmode->dmPrintQuality = gtk_print_settings_get_int (settings, "win32-print-quality");
1045 else if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_RESOLUTION))
1047 devmode->dmFields |= DM_PRINTQUALITY;
1048 devmode->dmPrintQuality = gtk_print_settings_get_resolution (settings);
1050 else if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_QUALITY))
1052 devmode->dmFields |= DM_PRINTQUALITY;
1053 switch (gtk_print_settings_get_quality (settings))
1055 case GTK_PRINT_QUALITY_LOW:
1056 devmode->dmPrintQuality = DMRES_LOW;
1058 case GTK_PRINT_QUALITY_DRAFT:
1059 devmode->dmPrintQuality = DMRES_DRAFT;
1062 case GTK_PRINT_QUALITY_NORMAL:
1063 devmode->dmPrintQuality = DMRES_MEDIUM;
1065 case GTK_PRINT_QUALITY_HIGH:
1066 devmode->dmPrintQuality = DMRES_HIGH;
1071 if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_USE_COLOR))
1073 devmode->dmFields |= DM_COLOR;
1074 if (gtk_print_settings_get_use_color (settings))
1075 devmode->dmColor = DMCOLOR_COLOR;
1077 devmode->dmColor = DMCOLOR_MONOCHROME;
1080 if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_DUPLEX))
1082 devmode->dmFields |= DM_DUPLEX;
1083 switch (gtk_print_settings_get_duplex (settings))
1086 case GTK_PRINT_DUPLEX_SIMPLEX:
1087 devmode->dmDuplex = DMDUP_SIMPLEX;
1089 case GTK_PRINT_DUPLEX_HORIZONTAL:
1090 devmode->dmDuplex = DMDUP_HORIZONTAL;
1092 case GTK_PRINT_DUPLEX_VERTICAL:
1093 devmode->dmDuplex = DMDUP_VERTICAL;
1098 if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_COLLATE))
1100 devmode->dmFields |= DM_COLLATE;
1101 if (gtk_print_settings_get_collate (settings))
1102 devmode->dmCollate = DMCOLLATE_TRUE;
1104 devmode->dmCollate = DMCOLLATE_FALSE;
1107 if (gtk_print_settings_has_key (settings, "win32-media-type"))
1109 devmode->dmFields |= DM_MEDIATYPE;
1110 devmode->dmMediaType = gtk_print_settings_get_int (settings, "win32-media-type");
1112 else if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_MEDIA_TYPE))
1114 devmode->dmFields |= DM_MEDIATYPE;
1115 devmode->dmMediaType = DMMEDIA_STANDARD;
1117 val = gtk_print_settings_get_media_type (settings);
1118 if (strcmp (val, "transparency") == 0)
1119 devmode->dmMediaType = DMMEDIA_TRANSPARENCY;
1120 if (strcmp (val, "photographic-glossy") == 0)
1121 devmode->dmMediaType = DMMEDIA_GLOSSY;
1124 if (gtk_print_settings_has_key (settings, "win32-dither-type"))
1126 devmode->dmFields |= DM_DITHERTYPE;
1127 devmode->dmDitherType = gtk_print_settings_get_int (settings, "win32-dither-type");
1129 else if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_DITHER))
1131 devmode->dmFields |= DM_DITHERTYPE;
1132 devmode->dmDitherType = DMDITHER_FINE;
1134 val = gtk_print_settings_get_dither (settings);
1135 if (strcmp (val, "none") == 0)
1136 devmode->dmDitherType = DMDITHER_NONE;
1137 if (strcmp (val, "coarse") == 0)
1138 devmode->dmDitherType = DMDITHER_COARSE;
1139 if (strcmp (val, "fine") == 0)
1140 devmode->dmDitherType = DMDITHER_FINE;
1141 if (strcmp (val, "lineart") == 0)
1142 devmode->dmDitherType = DMDITHER_LINEART;
1143 if (strcmp (val, "grayscale") == 0)
1144 devmode->dmDitherType = DMDITHER_GRAYSCALE;
1145 if (strcmp (val, "error-diffusion") == 0)
1146 devmode->dmDitherType = DMDITHER_ERRORDIFFUSION;
1149 GlobalUnlock (hDevMode);
1155 dialog_from_print_settings (GtkPrintOperation *op,
1156 LPPRINTDLGEXW printdlgex)
1158 GtkPrintSettings *settings = op->priv->print_settings;
1159 const char *printer;
1161 if (settings == NULL)
1164 if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_PRINT_PAGES))
1166 GtkPrintPages print_pages = gtk_print_settings_get_print_pages (settings);
1168 switch (print_pages)
1171 case GTK_PRINT_PAGES_ALL:
1172 printdlgex->Flags |= PD_ALLPAGES;
1174 case GTK_PRINT_PAGES_CURRENT:
1175 printdlgex->Flags |= PD_CURRENTPAGE;
1177 case GTK_PRINT_PAGES_RANGES:
1178 printdlgex->Flags |= PD_PAGENUMS;
1182 if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_PAGE_RANGES))
1184 GtkPageRange *ranges;
1187 ranges = gtk_print_settings_get_page_ranges (settings, &num_ranges);
1189 if (num_ranges > MAX_PAGE_RANGES)
1190 num_ranges = MAX_PAGE_RANGES;
1192 printdlgex->nPageRanges = num_ranges;
1193 for (i = 0; i < num_ranges; i++)
1195 printdlgex->lpPageRanges[i].nFromPage = ranges[i].start + 1;
1196 printdlgex->lpPageRanges[i].nToPage = ranges[i].end + 1;
1200 printer = gtk_print_settings_get_printer (settings);
1202 printdlgex->hDevNames = gtk_print_win32_devnames_to_win32_from_printer_name (printer);
1204 printdlgex->hDevMode = devmode_from_settings (settings,
1205 op->priv->default_page_setup);
1209 IPrintDialogCallback iPrintDialogCallback;
1212 } PrintDialogCallback;
1215 static ULONG STDMETHODCALLTYPE
1216 iprintdialogcallback_addref (IPrintDialogCallback *This)
1218 PrintDialogCallback *callback = (PrintDialogCallback *)This;
1219 return ++callback->ref_count;
1222 static ULONG STDMETHODCALLTYPE
1223 iprintdialogcallback_release (IPrintDialogCallback *This)
1225 PrintDialogCallback *callback = (PrintDialogCallback *)This;
1226 int ref_count = --callback->ref_count;
1234 static HRESULT STDMETHODCALLTYPE
1235 iprintdialogcallback_queryinterface (IPrintDialogCallback *This,
1239 if (IsEqualIID (riid, &IID_IUnknown) ||
1240 IsEqualIID (riid, &myIID_IPrintDialogCallback))
1243 IUnknown_AddRef ((IUnknown *)This);
1249 return E_NOINTERFACE;
1253 static HRESULT STDMETHODCALLTYPE
1254 iprintdialogcallback_initdone (IPrintDialogCallback *This)
1259 static HRESULT STDMETHODCALLTYPE
1260 iprintdialogcallback_selectionchange (IPrintDialogCallback *This)
1265 static HRESULT STDMETHODCALLTYPE
1266 iprintdialogcallback_handlemessage (IPrintDialogCallback *This,
1273 PrintDialogCallback *callback = (PrintDialogCallback *)This;
1275 if (!callback->set_hwnd)
1277 gdk_win32_set_modal_dialog_libgtk_only (hDlg);
1278 callback->set_hwnd = TRUE;
1279 while (gtk_events_pending ())
1280 gtk_main_iteration ();
1282 else if (uMsg == got_gdk_events_message)
1284 while (gtk_events_pending ())
1285 gtk_main_iteration ();
1294 static IPrintDialogCallbackVtbl ipdc_vtbl = {
1295 iprintdialogcallback_queryinterface,
1296 iprintdialogcallback_addref,
1297 iprintdialogcallback_release,
1298 iprintdialogcallback_initdone,
1299 iprintdialogcallback_selectionchange,
1300 iprintdialogcallback_handlemessage
1303 static IPrintDialogCallback *
1304 print_callback_new (void)
1306 PrintDialogCallback *callback;
1308 callback = g_new0 (PrintDialogCallback, 1);
1309 callback->iPrintDialogCallback.lpVtbl = &ipdc_vtbl;
1310 callback->ref_count = 1;
1311 callback->set_hwnd = FALSE;
1313 return &callback->iPrintDialogCallback;
1317 plug_grab_notify (GtkWidget *widget,
1318 gboolean was_grabbed,
1319 GtkPrintOperation *op)
1321 EnableWindow (GetAncestor (GDK_WINDOW_HWND (widget->window), GA_ROOT),
1326 static BOOL CALLBACK
1327 pageDlgProc (HWND wnd, UINT message, WPARAM wparam, LPARAM lparam)
1329 GtkPrintOperation *op;
1330 GtkPrintOperationWin32 *op_win32;
1332 if (message == WM_INITDIALOG)
1334 PROPSHEETPAGEW *page = (PROPSHEETPAGEW *)lparam;
1337 op = GTK_PRINT_OPERATION ((gpointer)page->lParam);
1338 op_win32 = op->priv->platform_data;
1340 SetWindowLongPtrW (wnd, GWLP_USERDATA, (LONG_PTR)op);
1342 plug = _gtk_win32_embed_widget_new ((GdkNativeWindow) wnd);
1343 gtk_window_set_modal (GTK_WINDOW (plug), TRUE);
1344 op_win32->embed_widget = plug;
1345 gtk_container_add (GTK_CONTAINER (plug), op->priv->custom_widget);
1346 gtk_widget_show (op->priv->custom_widget);
1347 gtk_widget_show (plug);
1348 gdk_window_focus (plug->window, GDK_CURRENT_TIME);
1350 /* This dialog is modal, so we grab the embed widget */
1351 gtk_grab_add (plug);
1353 /* When we lose the grab we need to disable the print dialog */
1354 g_signal_connect (plug, "grab-notify", G_CALLBACK (plug_grab_notify), op);
1357 else if (message == WM_DESTROY)
1359 op = GTK_PRINT_OPERATION (GetWindowLongPtrW (wnd, GWLP_USERDATA));
1360 op_win32 = op->priv->platform_data;
1362 g_signal_emit_by_name (op, "custom-widget-apply", op->priv->custom_widget);
1363 gtk_widget_destroy (op_win32->embed_widget);
1364 op_win32->embed_widget = NULL;
1365 op->priv->custom_widget = NULL;
1369 op = GTK_PRINT_OPERATION (GetWindowLongPtrW (wnd, GWLP_USERDATA));
1370 op_win32 = op->priv->platform_data;
1372 return _gtk_win32_embed_widget_dialog_procedure (GTK_WIN32_EMBED_WIDGET (op_win32->embed_widget),
1373 wnd, message, wparam, lparam);
1379 static HPROPSHEETPAGE
1380 create_application_page (GtkPrintOperation *op)
1382 HPROPSHEETPAGE hpage;
1383 PROPSHEETPAGEW page;
1384 DLGTEMPLATE *template;
1387 WORD baseunitX, baseunitY;
1389 GtkRequisition requisition;
1390 const char *tab_label;
1392 /* Make the template the size of the custom widget size request */
1393 gtk_widget_size_request (op->priv->custom_widget, &requisition);
1395 base_units = GetDialogBaseUnits ();
1396 baseunitX = LOWORD (base_units);
1397 baseunitY = HIWORD (base_units);
1399 htemplate = GlobalAlloc (GMEM_MOVEABLE,
1400 sizeof (DLGTEMPLATE) + sizeof (WORD) * 3);
1401 template = GlobalLock (htemplate);
1402 template->style = WS_CHILDWINDOW | DS_CONTROL;
1403 template->dwExtendedStyle = WS_EX_CONTROLPARENT;
1405 template->x = MulDiv (0, 4, baseunitX);
1406 template->y = MulDiv (0, 8, baseunitY);
1407 template->cx = MulDiv (requisition.width, 4, baseunitX);
1408 template->cy = MulDiv (requisition.height, 8, baseunitY);
1410 array = (WORD *) (template+1);
1411 *array++ = 0; /* menu */
1412 *array++ = 0; /* class */
1413 *array++ = 0; /* title */
1415 memset (&page, 0, sizeof (page));
1416 page.dwSize = sizeof (page);
1417 page.dwFlags = PSP_DLGINDIRECT | PSP_USETITLE | PSP_PREMATURE;
1418 page.hInstance = GetModuleHandle (NULL);
1419 page.pResource = template;
1421 tab_label = op->priv->custom_tab_label;
1422 if (tab_label == NULL)
1423 tab_label = g_get_application_name ();
1424 if (tab_label == NULL)
1425 tab_label = _("Application");
1426 page.pszTitle = g_utf8_to_utf16 (tab_label,
1427 -1, NULL, NULL, NULL);
1428 page.pfnDlgProc = pageDlgProc;
1429 page.pfnCallback = NULL;
1430 page.lParam = (LPARAM) op;
1431 hpage = CreatePropertySheetPageW (&page);
1433 GlobalUnlock (htemplate);
1435 /* TODO: We're leaking htemplate here... */
1440 static GtkPageSetup *
1441 create_page_setup (GtkPrintOperation *op)
1443 GtkPrintOperationPrivate *priv = op->priv;
1444 GtkPageSetup *page_setup;
1445 GtkPrintSettings *settings;
1447 if (priv->default_page_setup)
1448 page_setup = gtk_page_setup_copy (priv->default_page_setup);
1450 page_setup = gtk_page_setup_new ();
1452 settings = priv->print_settings;
1455 GtkPaperSize *paper_size;
1457 if (gtk_print_settings_has_key (settings, GTK_PRINT_SETTINGS_ORIENTATION))
1458 gtk_page_setup_set_orientation (page_setup,
1459 gtk_print_settings_get_orientation (settings));
1462 paper_size = gtk_print_settings_get_paper_size (settings);
1465 gtk_page_setup_set_paper_size (page_setup, paper_size);
1466 gtk_paper_size_free (paper_size);
1469 /* TODO: Margins? */
1475 GtkPrintOperationResult
1476 gtk_print_operation_run_without_dialog (GtkPrintOperation *op,
1479 GtkPrintOperationResult result;
1480 GtkPrintOperationWin32 *op_win32;
1481 GtkPrintOperationPrivate *priv;
1482 GtkPrintSettings *settings;
1483 GtkPageSetup *page_setup;
1485 HGLOBAL hDevMode = NULL;
1486 HGLOBAL hDevNames = NULL;
1488 const char *printer = NULL;
1489 double dpi_x, dpi_y;
1498 settings = priv->print_settings;
1500 op_win32 = g_new0 (GtkPrintOperationWin32, 1);
1501 priv->platform_data = op_win32;
1502 priv->free_platform_data = (GDestroyNotify) op_win32_free;
1503 printer = gtk_print_settings_get_printer (settings);
1507 /* No printer selected. Get the system default printer and store
1510 gchar *tmp_printer = get_default_printer ();
1513 result = GTK_PRINT_OPERATION_RESULT_ERROR;
1514 g_set_error_literal (&priv->error,
1516 GTK_PRINT_ERROR_INTERNAL_ERROR,
1517 _("No printer found"));
1520 gtk_print_settings_set_printer (settings, tmp_printer);
1521 printer = gtk_print_settings_get_printer (settings);
1522 g_free (tmp_printer);
1525 hDevNames = gtk_print_win32_devnames_to_win32_from_printer_name (printer);
1526 hDevMode = devmode_from_settings (settings, op->priv->default_page_setup);
1528 /* Create a printer DC for the print settings and page setup provided. */
1529 pdn = GlobalLock (hDevNames);
1530 pdm = GlobalLock (hDevMode);
1531 hDC = CreateDCW ((wchar_t*)pdn + pdn->wDriverOffset,
1532 (wchar_t*)pdn + pdn->wDeviceOffset,
1533 (wchar_t*)pdn + pdn->wOutputOffset,
1535 GlobalUnlock (hDevNames);
1536 GlobalUnlock (hDevMode);
1540 result = GTK_PRINT_OPERATION_RESULT_ERROR;
1541 g_set_error_literal (&priv->error,
1543 GTK_PRINT_ERROR_INTERNAL_ERROR,
1544 _("Invalid argument to CreateDC"));
1548 priv->print_context = _gtk_print_context_new (op);
1549 page_setup = create_page_setup (op);
1550 _gtk_print_context_set_page_setup (priv->print_context, page_setup);
1551 g_object_unref (page_setup);
1555 op_win32->surface = cairo_win32_printing_surface_create (hDC);
1556 dpi_x = (double) GetDeviceCaps (hDC, LOGPIXELSX);
1557 dpi_y = (double) GetDeviceCaps (hDC, LOGPIXELSY);
1559 cr = cairo_create (op_win32->surface);
1560 gtk_print_context_set_cairo_context (priv->print_context, cr, dpi_x, dpi_y);
1563 memset (&docinfo, 0, sizeof (DOCINFOW));
1564 docinfo.cbSize = sizeof (DOCINFOW);
1565 docinfo.lpszDocName = g_utf8_to_utf16 (op->priv->job_name, -1, NULL, NULL, NULL);
1566 docinfo.lpszOutput = NULL;
1567 docinfo.lpszDatatype = NULL;
1570 job_id = StartDocW (hDC, &docinfo);
1571 g_free ((void *)docinfo.lpszDocName);
1574 result = GTK_PRINT_OPERATION_RESULT_ERROR;
1575 g_set_error_literal (&priv->error,
1577 GTK_PRINT_ERROR_GENERAL,
1578 _("Error from StartDoc"));
1580 cairo_surface_destroy (op_win32->surface);
1581 op_win32->surface = NULL;
1585 result = GTK_PRINT_OPERATION_RESULT_APPLY;
1586 op_win32->hdc = hDC;
1587 op_win32->devmode = hDevMode;
1588 op_win32->devnames = hDevNames;
1589 op_win32->job_id = job_id;
1590 op->priv->print_pages = gtk_print_settings_get_print_pages (op->priv->print_settings);
1591 op->priv->num_page_ranges = 0;
1592 if (op->priv->print_pages == GTK_PRINT_PAGES_RANGES)
1593 op->priv->page_ranges = gtk_print_settings_get_page_ranges (op->priv->print_settings,
1594 &op->priv->num_page_ranges);
1595 op->priv->manual_num_copies = 1;
1596 op->priv->manual_collation = FALSE;
1597 op->priv->manual_reverse = FALSE;
1598 op->priv->manual_orientation = FALSE;
1599 op->priv->manual_scale = 1.0;
1600 op->priv->manual_page_set = GTK_PAGE_SET_ALL;
1602 op->priv->start_page = win32_start_page;
1603 op->priv->end_page = win32_end_page;
1604 op->priv->end_run = win32_end_run;
1607 if (!*do_print && hDC != NULL)
1610 if (!*do_print && hDevMode != NULL)
1611 GlobalFree (hDevMode);
1613 if (!*do_print && hDevNames != NULL)
1614 GlobalFree (hDevNames);
1619 GtkPrintOperationResult
1620 gtk_print_operation_run_with_dialog (GtkPrintOperation *op,
1625 LPPRINTDLGEXW printdlgex = NULL;
1626 LPPRINTPAGERANGE page_ranges = NULL;
1628 GtkWidget *invisible = NULL;
1629 GtkPrintOperationResult result;
1630 GtkPrintOperationWin32 *op_win32;
1631 GtkPrintOperationPrivate *priv;
1632 IPrintDialogCallback *callback;
1633 HPROPSHEETPAGE prop_page;
1639 op_win32 = g_new0 (GtkPrintOperationWin32, 1);
1640 priv->platform_data = op_win32;
1641 priv->free_platform_data = (GDestroyNotify) op_win32_free;
1645 invisible = gtk_invisible_new ();
1646 parentHWnd = get_parent_hwnd (invisible);
1649 parentHWnd = get_parent_hwnd (GTK_WIDGET (parent));
1651 printdlgex = (LPPRINTDLGEXW)GlobalAlloc (GPTR, sizeof (PRINTDLGEXW));
1654 result = GTK_PRINT_OPERATION_RESULT_ERROR;
1655 g_set_error_literal (&priv->error,
1657 GTK_PRINT_ERROR_NOMEM,
1658 _("Not enough free memory"));
1662 printdlgex->lStructSize = sizeof (PRINTDLGEXW);
1663 printdlgex->hwndOwner = parentHWnd;
1664 printdlgex->hDevMode = NULL;
1665 printdlgex->hDevNames = NULL;
1666 printdlgex->hDC = NULL;
1667 printdlgex->Flags = PD_RETURNDC | PD_NOSELECTION;
1668 if (op->priv->current_page == -1)
1669 printdlgex->Flags |= PD_NOCURRENTPAGE;
1670 printdlgex->Flags2 = 0;
1671 printdlgex->ExclusionFlags = 0;
1673 page_ranges = (LPPRINTPAGERANGE) GlobalAlloc (GPTR,
1674 MAX_PAGE_RANGES * sizeof (PRINTPAGERANGE));
1677 result = GTK_PRINT_OPERATION_RESULT_ERROR;
1678 g_set_error_literal (&priv->error,
1680 GTK_PRINT_ERROR_NOMEM,
1681 _("Not enough free memory"));
1685 printdlgex->nPageRanges = 0;
1686 printdlgex->nMaxPageRanges = MAX_PAGE_RANGES;
1687 printdlgex->lpPageRanges = page_ranges;
1688 printdlgex->nMinPage = 1;
1689 if (op->priv->nr_of_pages != -1)
1690 printdlgex->nMaxPage = op->priv->nr_of_pages;
1692 printdlgex->nMaxPage = 10000;
1693 printdlgex->nCopies = 1;
1694 printdlgex->hInstance = 0;
1695 printdlgex->lpPrintTemplateName = NULL;
1696 printdlgex->lpCallback = NULL;
1698 g_signal_emit_by_name (op, "create-custom-widget",
1699 &op->priv->custom_widget);
1700 if (op->priv->custom_widget) {
1701 prop_page = create_application_page (op);
1702 printdlgex->nPropertyPages = 1;
1703 printdlgex->lphPropertyPages = &prop_page;
1705 printdlgex->nPropertyPages = 0;
1706 printdlgex->lphPropertyPages = NULL;
1709 printdlgex->nStartPage = START_PAGE_GENERAL;
1710 printdlgex->dwResultAction = 0;
1712 dialog_from_print_settings (op, printdlgex);
1714 callback = print_callback_new ();
1715 printdlgex->lpCallback = (IUnknown *)callback;
1716 got_gdk_events_message = RegisterWindowMessage ("GDK_WIN32_GOT_EVENTS");
1718 hResult = PrintDlgExW (printdlgex);
1719 IUnknown_Release ((IUnknown *)callback);
1720 gdk_win32_set_modal_dialog_libgtk_only (NULL);
1722 if (hResult != S_OK)
1724 result = GTK_PRINT_OPERATION_RESULT_ERROR;
1725 if (hResult == E_OUTOFMEMORY)
1726 g_set_error_literal (&priv->error,
1728 GTK_PRINT_ERROR_NOMEM,
1729 _("Not enough free memory"));
1730 else if (hResult == E_INVALIDARG)
1731 g_set_error_literal (&priv->error,
1733 GTK_PRINT_ERROR_INTERNAL_ERROR,
1734 _("Invalid argument to PrintDlgEx"));
1735 else if (hResult == E_POINTER)
1736 g_set_error_literal (&priv->error,
1738 GTK_PRINT_ERROR_INTERNAL_ERROR,
1739 _("Invalid pointer to PrintDlgEx"));
1740 else if (hResult == E_HANDLE)
1741 g_set_error_literal (&priv->error,
1743 GTK_PRINT_ERROR_INTERNAL_ERROR,
1744 _("Invalid handle to PrintDlgEx"));
1746 g_set_error_literal (&priv->error,
1748 GTK_PRINT_ERROR_GENERAL,
1749 _("Unspecified error"));
1753 if (printdlgex->dwResultAction == PD_RESULT_PRINT ||
1754 printdlgex->dwResultAction == PD_RESULT_APPLY)
1756 result = GTK_PRINT_OPERATION_RESULT_APPLY;
1757 dialog_to_print_settings (op, printdlgex);
1760 result = GTK_PRINT_OPERATION_RESULT_CANCEL;
1762 if (printdlgex->dwResultAction == PD_RESULT_PRINT)
1766 double dpi_x, dpi_y;
1768 GtkPageSetup *page_setup;
1770 priv->print_context = _gtk_print_context_new (op);
1771 page_setup = create_page_setup (op);
1772 _gtk_print_context_set_page_setup (priv->print_context, page_setup);
1773 g_object_unref (page_setup);
1777 op_win32->surface = cairo_win32_printing_surface_create (printdlgex->hDC);
1779 dpi_x = (double)GetDeviceCaps (printdlgex->hDC, LOGPIXELSX);
1780 dpi_y = (double)GetDeviceCaps (printdlgex->hDC, LOGPIXELSY);
1782 cr = cairo_create (op_win32->surface);
1783 gtk_print_context_set_cairo_context (priv->print_context, cr, dpi_x, dpi_y);
1786 memset ( &docinfo, 0, sizeof (DOCINFOW));
1787 docinfo.cbSize = sizeof (DOCINFOW);
1788 docinfo.lpszDocName = g_utf8_to_utf16 (op->priv->job_name, -1, NULL, NULL, NULL);
1789 docinfo.lpszOutput = (LPCWSTR) NULL;
1790 docinfo.lpszDatatype = (LPCWSTR) NULL;
1793 job_id = StartDocW (printdlgex->hDC, &docinfo);
1794 g_free ((void *)docinfo.lpszDocName);
1797 result = GTK_PRINT_OPERATION_RESULT_ERROR;
1798 g_set_error_literal (&priv->error,
1800 GTK_PRINT_ERROR_GENERAL,
1801 _("Error from StartDoc"));
1803 cairo_surface_destroy (op_win32->surface);
1804 op_win32->surface = NULL;
1808 op_win32->hdc = printdlgex->hDC;
1809 op_win32->devmode = printdlgex->hDevMode;
1810 op_win32->devnames = printdlgex->hDevNames;
1811 op_win32->job_id = job_id;
1813 op->priv->print_pages = gtk_print_settings_get_print_pages (op->priv->print_settings);
1814 op->priv->num_page_ranges = 0;
1815 if (op->priv->print_pages == GTK_PRINT_PAGES_RANGES)
1816 op->priv->page_ranges = gtk_print_settings_get_page_ranges (op->priv->print_settings,
1817 &op->priv->num_page_ranges);
1818 op->priv->manual_num_copies = printdlgex->nCopies;
1819 op->priv->manual_collation = (printdlgex->Flags & PD_COLLATE) != 0;
1820 op->priv->manual_reverse = FALSE;
1821 op->priv->manual_orientation = FALSE;
1822 op->priv->manual_scale = 1.0;
1823 op->priv->manual_page_set = GTK_PAGE_SET_ALL;
1826 op->priv->start_page = win32_start_page;
1827 op->priv->end_page = win32_end_page;
1828 op->priv->end_run = win32_end_run;
1831 if (!*do_print && printdlgex && printdlgex->hDC != NULL)
1832 DeleteDC (printdlgex->hDC);
1834 if (!*do_print && printdlgex && printdlgex->hDevMode != NULL)
1835 GlobalFree (printdlgex->hDevMode);
1837 if (!*do_print && printdlgex && printdlgex->hDevNames != NULL)
1838 GlobalFree (printdlgex->hDevNames);
1841 GlobalFree (page_ranges);
1844 GlobalFree (printdlgex);
1847 gtk_widget_destroy (invisible);
1852 GtkPrintOperationResult
1853 _gtk_print_operation_platform_backend_run_dialog (GtkPrintOperation *op,
1854 gboolean show_dialog,
1859 return gtk_print_operation_run_with_dialog (op, parent, do_print);
1861 return gtk_print_operation_run_without_dialog (op, do_print);
1865 _gtk_print_operation_platform_backend_launch_preview (GtkPrintOperation *op,
1866 cairo_surface_t *surface,
1868 const gchar *filename)
1871 HENHMETAFILE metafile;
1873 dc = cairo_win32_surface_get_dc (surface);
1874 cairo_surface_destroy (surface);
1875 metafile = CloseEnhMetaFile (dc);
1876 DeleteEnhMetaFile (metafile);
1878 ShellExecuteW (NULL, L"open", (gunichar2 *)filename, NULL, NULL, SW_SHOW);
1882 _gtk_print_operation_platform_backend_preview_start_page (GtkPrintOperation *op,
1883 cairo_surface_t *surface,
1886 HDC dc = cairo_win32_surface_get_dc (surface);
1891 _gtk_print_operation_platform_backend_preview_end_page (GtkPrintOperation *op,
1892 cairo_surface_t *surface,
1897 cairo_surface_show_page (surface);
1899 /* TODO: Enhanced metafiles don't support multiple pages.
1901 dc = cairo_win32_surface_get_dc (surface);
1906 _gtk_print_operation_platform_backend_create_preview_surface (GtkPrintOperation *op,
1907 GtkPageSetup *page_setup,
1912 GtkPaperSize *paper_size;
1917 gunichar2 *filename_utf16;
1920 template = g_build_filename (g_get_tmp_dir (), "prXXXXXX", NULL);
1921 fd = g_mkstemp (template);
1924 filename = g_strconcat (template, ".emf", NULL);
1927 filename_utf16 = g_utf8_to_utf16 (filename, -1, NULL, NULL, NULL);
1930 paper_size = gtk_page_setup_get_paper_size (page_setup);
1932 /* The rectangle dimensions are given in hundredths of a millimeter */
1934 rect.right = 100.0 * gtk_paper_size_get_width (paper_size, GTK_UNIT_MM);
1936 rect.bottom = 100.0 * gtk_paper_size_get_height (paper_size, GTK_UNIT_MM);
1938 metafile_dc = CreateEnhMetaFileW (NULL, filename_utf16,
1939 &rect, L"Gtk+\0Print Preview\0\0");
1940 if (metafile_dc == NULL)
1942 g_warning ("Can't create metafile");
1946 *target = (char *)filename_utf16;
1948 *dpi_x = (double)GetDeviceCaps (metafile_dc, LOGPIXELSX);
1949 *dpi_y = (double)GetDeviceCaps (metafile_dc, LOGPIXELSY);
1951 return cairo_win32_printing_surface_create (metafile_dc);
1955 _gtk_print_operation_platform_backend_resize_preview_surface (GtkPrintOperation *op,
1956 GtkPageSetup *page_setup,
1957 cairo_surface_t *surface)
1959 /* TODO: Implement */
1963 gtk_print_run_page_setup_dialog (GtkWindow *parent,
1964 GtkPageSetup *page_setup,
1965 GtkPrintSettings *settings)
1967 LPPAGESETUPDLGW pagesetupdlg = NULL;
1969 gboolean free_settings;
1970 const char *printer;
1971 GtkPaperSize *paper_size;
1972 DWORD measure_system;
1976 pagesetupdlg = (LPPAGESETUPDLGW)GlobalAlloc (GPTR, sizeof (PAGESETUPDLGW));
1980 free_settings = FALSE;
1981 if (settings == NULL)
1983 settings = gtk_print_settings_new ();
1984 free_settings = TRUE;
1987 memset (pagesetupdlg, 0, sizeof (PAGESETUPDLGW));
1989 pagesetupdlg->lStructSize = sizeof (PAGESETUPDLGW);
1992 pagesetupdlg->hwndOwner = get_parent_hwnd (GTK_WIDGET (parent));
1994 pagesetupdlg->hwndOwner = NULL;
1996 pagesetupdlg->Flags = PSD_DEFAULTMINMARGINS;
1997 pagesetupdlg->hDevMode = devmode_from_settings (settings, page_setup);
1998 pagesetupdlg->hDevNames = NULL;
1999 printer = gtk_print_settings_get_printer (settings);
2001 pagesetupdlg->hDevNames = gtk_print_win32_devnames_to_win32_from_printer_name (printer);
2003 GetLocaleInfoW (LOCALE_USER_DEFAULT, LOCALE_IMEASURE|LOCALE_RETURN_NUMBER,
2004 (LPWSTR)&measure_system, sizeof (DWORD));
2006 if (measure_system == 0)
2008 pagesetupdlg->Flags |= PSD_INHUNDREDTHSOFMILLIMETERS;
2014 pagesetupdlg->Flags |= PSD_INTHOUSANDTHSOFINCHES;
2015 unit = GTK_UNIT_INCH;
2019 /* This is the object we return, we allocate it here so that
2020 * we can use the default page margins */
2022 page_setup = gtk_page_setup_copy (page_setup);
2024 page_setup = gtk_page_setup_new ();
2026 pagesetupdlg->Flags |= PSD_MARGINS;
2027 pagesetupdlg->rtMargin.left =
2028 floor (gtk_page_setup_get_left_margin (page_setup, unit) * scale + 0.5);
2029 pagesetupdlg->rtMargin.right =
2030 floor (gtk_page_setup_get_right_margin (page_setup, unit) * scale + 0.5);
2031 pagesetupdlg->rtMargin.top =
2032 floor (gtk_page_setup_get_top_margin (page_setup, unit) * scale + 0.5);
2033 pagesetupdlg->rtMargin.bottom =
2034 floor (gtk_page_setup_get_bottom_margin (page_setup, unit) * scale + 0.5);
2036 pagesetupdlg->Flags |= PSD_ENABLEPAGESETUPHOOK;
2037 pagesetupdlg->lpfnPageSetupHook = run_mainloop_hook;
2038 got_gdk_events_message = RegisterWindowMessage ("GDK_WIN32_GOT_EVENTS");
2040 res = PageSetupDlgW (pagesetupdlg);
2041 gdk_win32_set_modal_dialog_libgtk_only (NULL);
2045 if (pagesetupdlg->hDevNames != NULL)
2046 devnames_to_settings (settings, pagesetupdlg->hDevNames);
2048 if (pagesetupdlg->hDevMode != NULL)
2049 devmode_to_settings (settings, pagesetupdlg->hDevMode);
2054 gtk_page_setup_set_orientation (page_setup,
2055 gtk_print_settings_get_orientation (settings));
2056 paper_size = gtk_print_settings_get_paper_size (settings);
2059 gtk_page_setup_set_paper_size (page_setup, paper_size);
2060 gtk_paper_size_free (paper_size);
2063 if (pagesetupdlg->Flags & PSD_INHUNDREDTHSOFMILLIMETERS)
2070 unit = GTK_UNIT_INCH;
2074 gtk_page_setup_set_left_margin (page_setup,
2075 pagesetupdlg->rtMargin.left / scale,
2077 gtk_page_setup_set_right_margin (page_setup,
2078 pagesetupdlg->rtMargin.right / scale,
2080 gtk_page_setup_set_top_margin (page_setup,
2081 pagesetupdlg->rtMargin.top / scale,
2083 gtk_page_setup_set_bottom_margin (page_setup,
2084 pagesetupdlg->rtMargin.bottom / scale,
2089 g_object_unref (settings);
2095 gtk_print_run_page_setup_dialog_async (GtkWindow *parent,
2096 GtkPageSetup *page_setup,
2097 GtkPrintSettings *settings,
2098 GtkPageSetupDoneFunc done_cb,
2101 GtkPageSetup *new_page_setup;
2103 new_page_setup = gtk_print_run_page_setup_dialog (parent, page_setup, settings);
2104 done_cb (new_page_setup, data);
2105 g_object_unref (new_page_setup);
2108 #define __GTK_PRINT_OPERATION_WIN32_C__
2109 #include "gtkaliasdef.c"