/* * Copyright (C) 2009-2015 Canonical Ltd. * Author: Robert Ancell <robert.ancell@canonical.com>, * Eduard Gotwig <g@ox.io> * * This program is free software: you can redistribute it and/or modify it under * the terms of the GNU General Public License as published by the Free Software * Foundation, either version 3 of the License, or (at your option) any later * version. See http://www.gnu.org/copyleft/gpl.html the full text of the * license. */ private const int DEFAULT_TEXT_DPI = 150; private const int DEFAULT_PHOTO_DPI = 300; [GtkTemplate (ui = "/org/gnome/SimpleScan/ui/app-window.ui")] public class AppWindow : Hdy.ApplicationWindow { private const GLib.ActionEntry[] action_entries = { { "new_document", new_document_cb }, { "scan_single", scan_single_cb }, { "scan_adf", scan_adf_cb }, { "scan_batch", scan_batch_cb }, { "scan_stop", scan_stop_cb }, { "rotate_left", rotate_left_cb }, { "rotate_right", rotate_right_cb }, { "move_left", move_left_cb }, { "move_right", move_right_cb }, { "copy_page", copy_page_cb }, { "delete_page", delete_page_cb }, { "reorder", reorder_document_cb }, { "save", save_document_activate_cb }, { "email", email_document_cb }, { "print", print_document_cb }, { "preferences", preferences_cb }, { "help", help_cb }, { "about", about_cb }, { "quit", quit_cb } }; private Settings settings; private ScanType scan_type = ScanType.SINGLE; private PreferencesDialog preferences_dialog; private bool setting_devices; private bool user_selected_device; [GtkChild] private unowned Hdy.HeaderBar header_bar; [GtkChild] private unowned Gtk.Menu page_menu; [GtkChild] private unowned Gtk.Stack stack; [GtkChild] private unowned Hdy.StatusPage status_page; [GtkChild] private unowned Gtk.Label status_secondary_label; [GtkChild] private unowned Gtk.ListStore device_model; [GtkChild] private unowned Gtk.Box device_buttons_box; [GtkChild] private unowned Gtk.ComboBox device_combo; [GtkChild] private unowned Gtk.Box main_vbox; [GtkChild] private unowned Gtk.RadioMenuItem custom_crop_menuitem; [GtkChild] private unowned Gtk.RadioMenuItem a3_menuitem; [GtkChild] private unowned Gtk.RadioMenuItem a4_menuitem; [GtkChild] private unowned Gtk.RadioMenuItem a5_menuitem; [GtkChild] private unowned Gtk.RadioMenuItem a6_menuitem; [GtkChild] private unowned Gtk.RadioMenuItem letter_menuitem; [GtkChild] private unowned Gtk.RadioMenuItem legal_menuitem; [GtkChild] private unowned Gtk.RadioMenuItem four_by_six_menuitem; [GtkChild] private unowned Gtk.RadioMenuItem no_crop_menuitem; [GtkChild] private unowned Gtk.MenuItem page_move_left_menuitem; [GtkChild] private unowned Gtk.MenuItem page_move_right_menuitem; [GtkChild] private unowned Gtk.MenuItem page_delete_menuitem; [GtkChild] private unowned Gtk.MenuItem crop_rotate_menuitem; [GtkChild] private unowned Gtk.MenuItem copy_to_clipboard_menuitem; [GtkChild] private unowned Gtk.Button save_button; [GtkChild] private unowned Gtk.Button stop_button; [GtkChild] private unowned Gtk.Button scan_button; [GtkChild] private unowned Gtk.ActionBar action_bar; private Gtk.ToggleButton crop_button; private Gtk.Button delete_button; [GtkChild] private unowned Gtk.Image scan_options_image; [GtkChild] private unowned Gtk.Image scan_hint_image; [GtkChild] private unowned Gtk.RadioButton scan_single_radio; [GtkChild] private unowned Gtk.RadioButton scan_adf_radio; [GtkChild] private unowned Gtk.RadioButton scan_batch_radio; [GtkChild] private unowned Gtk.RadioButton text_radio; [GtkChild] private unowned Gtk.RadioButton photo_radio; [GtkChild] private unowned Gtk.MenuButton menu_button; private bool have_devices = false; private string? missing_driver = null; public Book book { get; private set; } private bool book_needs_saving; private string? book_uri = null; public Page selected_page { get { return book_view.selected_page; } set { book_view.selected_page = value; } } private AutosaveManager autosave_manager; private BookView book_view; private bool updating_page_menu; private string document_hint = "photo"; private bool scanning_ = false; public bool scanning { get { return scanning_; } set { scanning_ = value; stack.set_visible_child_name ("document"); page_delete_menuitem.sensitive = !value; delete_button.sensitive = !value; scan_button.visible = !value; stop_button.visible = value; } } private int window_width; private int window_height; private bool window_is_maximized; private bool window_is_fullscreen; private uint save_state_timeout; public int brightness { get { return preferences_dialog.get_brightness (); } set { preferences_dialog.set_brightness (value); } } public int contrast { get { return preferences_dialog.get_contrast (); } set { preferences_dialog.set_contrast (value); } } public int page_delay { get { return preferences_dialog.get_page_delay (); } set { preferences_dialog.set_page_delay (value); } } public signal void start_scan (string? device, ScanOptions options); public signal void stop_scan (); public signal void redetect (); public AppWindow () { settings = new Settings ("org.gnome.SimpleScan"); var renderer = new Gtk.CellRendererText (); renderer.set_property ("xalign", 0.5); device_combo.pack_start (renderer, true); device_combo.add_attribute (renderer, "text", 1); book = new Book (); book.page_added.connect (page_added_cb); book.reordered.connect (reordered_cb); book.page_removed.connect (page_removed_cb); book.changed.connect (book_changed_cb); load (); clear_document (); } ~AppWindow () { book.page_added.disconnect (page_added_cb); book.reordered.disconnect (reordered_cb); book.page_removed.disconnect (page_removed_cb); } public void show_error_dialog (string error_title, string error_text) { var dialog = new Gtk.MessageDialog (this, Gtk.DialogFlags.MODAL, Gtk.MessageType.WARNING, Gtk.ButtonsType.NONE, "%s", error_title); dialog.add_button (_("_Close"), 0); dialog.format_secondary_markup ("%s", error_text); dialog.run (); dialog.destroy (); } public void authorize (string resource, out string username, out string password) { /* Label in authorization dialog. “%s” is replaced with the name of the resource requesting authorization */ var description = _("Username and password required to access “%s”").printf (resource); var authorize_dialog = new AuthorizeDialog (description); authorize_dialog.visible = true; authorize_dialog.transient_for = this; authorize_dialog.run (); authorize_dialog.destroy (); username = authorize_dialog.get_username (); password = authorize_dialog.get_password (); } private void update_scan_status () { scan_button.sensitive = false; if (!have_devices) { status_page.set_title (/* Label shown when searching for scanners */ _("Searching for Scanners…")); status_secondary_label.visible = false; device_buttons_box.visible = false; } else if (get_selected_device () != null) { scan_button.sensitive = true; status_page.set_title (/* Label shown when detected a scanner */ _("Ready to Scan")); status_secondary_label.set_text (get_selected_device_label ()); status_secondary_label.visible = false; device_buttons_box.visible = true; device_buttons_box.sensitive = true; device_combo.sensitive = true; } else if (this.missing_driver != null) { status_page.set_title (/* Warning displayed when no drivers are installed but a compatible scanner is detected */ _("Additional Software Needed")); /* Instructions to install driver software */ status_secondary_label.set_markup (_("You need to <a href=\"install-firmware\">install driver software</a> for your scanner.")); status_secondary_label.visible = true; device_buttons_box.visible = false; } else { /* Warning displayed when no scanners are detected */ status_page.set_title (_("No Scanners Detected")); /* Hint to user on why there are no scanners detected */ status_secondary_label.set_text (_("Please check your scanner is connected and powered on.")); status_secondary_label.visible = true; device_buttons_box.visible = true; device_buttons_box.sensitive = true; device_combo.sensitive = false; // We would like to be refresh button to be active } } public void set_scan_devices (List<ScanDevice> devices, string? missing_driver = null) { have_devices = true; this.missing_driver = missing_driver; setting_devices = true; /* If the user hasn't chosen a scanner choose the best available one */ var have_selection = false; if (user_selected_device) have_selection = device_combo.active >= 0; /* Add new devices */ int index = 0; Gtk.TreeIter iter; foreach (var device in devices) { int n_delete = -1; /* Find if already exists */ if (device_model.iter_nth_child (out iter, null, index)) { int i = 0; do { string name; bool matched; device_model.get (iter, 0, out name, -1); matched = name == device.name; if (matched) { n_delete = i; break; } i++; } while (device_model.iter_next (ref iter)); } /* If exists, remove elements up to this one */ if (n_delete >= 0) { int i; /* Update label */ device_model.set (iter, 1, device.label, -1); for (i = 0; i < n_delete; i++) { device_model.iter_nth_child (out iter, null, index); #if VALA_0_36 device_model.remove (ref iter); #else device_model.remove (iter); #endif } } else { device_model.insert (out iter, index); device_model.set (iter, 0, device.name, 1, device.label, -1); } index++; } /* Remove any remaining devices */ while (device_model.iter_nth_child (out iter, null, index)) #if VALA_0_36 device_model.remove (ref iter); #else device_model.remove (iter); #endif /* Select the previously selected device or the first available device */ if (!have_selection) { var device = settings.get_string ("selected-device"); if (device != null && find_scan_device (device, out iter)) device_combo.set_active_iter (iter); else device_combo.set_active (0); } setting_devices = false; update_scan_status (); } private bool prompt_to_load_autosaved_book () { var dialog = new Gtk.MessageDialog (this, Gtk.DialogFlags.MODAL, Gtk.MessageType.QUESTION, Gtk.ButtonsType.YES_NO, /* Contents of dialog that shows if autosaved book should be loaded. */ _("An autosaved book exists. Do you want to open it?")); dialog.set_default_response(Gtk.ResponseType.YES); var response = dialog.run (); dialog.destroy (); return response == Gtk.ResponseType.YES; } private string? get_selected_device () { Gtk.TreeIter iter; if (device_combo.get_active_iter (out iter)) { string device; device_model.get (iter, 0, out device, -1); return device; } return null; } private string? get_selected_device_label () { Gtk.TreeIter iter; if (device_combo.get_active_iter (out iter)) { string label; device_model.get (iter, 1, out label, -1); return label; } return null; } public void set_selected_device (string device) { user_selected_device = true; Gtk.TreeIter iter; if (!find_scan_device (device, out iter)) return; device_combo.set_active_iter (iter); } private bool find_scan_device (string device, out Gtk.TreeIter iter) { bool have_iter = false; if (device_model.get_iter_first (out iter)) { do { string d; device_model.get (iter, 0, out d, -1); if (d == device) have_iter = true; } while (!have_iter && device_model.iter_next (ref iter)); } return have_iter; } private string? choose_file_location () { /* Get directory to save to */ string? directory = null; directory = settings.get_string ("save-directory"); if (directory == null || directory == "") directory = GLib.Filename.to_uri(Environment.get_user_special_dir (UserDirectory.DOCUMENTS)); var save_dialog = new Gtk.FileChooserNative (/* Save dialog: Dialog title */ _("Save As…"), this, Gtk.FileChooserAction.SAVE, _("_Save"), _("_Cancel")); save_dialog.local_only = false; var save_format = settings.get_string ("save-format"); if (book_uri != null) save_dialog.set_uri (book_uri); else { save_dialog.set_current_folder_uri (directory); /* Default filename to use when saving document. */ /* To that filename the extension will be added, eg. "Scanned Document.pdf" */ save_dialog.set_current_name (_("Scanned Document") + "." + mime_type_to_extension (save_format)); } /* Filter to only show images by default */ var filter = new Gtk.FileFilter (); filter.set_filter_name (/* Save dialog: Filter name to show only supported image files */ _("Image Files")); filter.add_mime_type ("image/jpeg"); filter.add_mime_type ("image/png"); #if HAVE_WEBP filter.add_mime_type ("image/webp"); #endif filter.add_mime_type ("application/pdf"); save_dialog.add_filter (filter); filter = new Gtk.FileFilter (); filter.set_filter_name (/* Save dialog: Filter name to show all files */ _("All Files")); filter.add_pattern ("*"); save_dialog.add_filter (filter); var file_type_store = new Gtk.ListStore (2, typeof (string), typeof (string)); Gtk.TreeIter iter; file_type_store.append (out iter); file_type_store.set (iter, /* Save dialog: Label for saving in PDF format */ 0, _("PDF (multi-page document)"), 1, "application/pdf", -1); file_type_store.append (out iter); file_type_store.set (iter, /* Save dialog: Label for saving in JPEG format */ 0, _("JPEG (compressed)"), 1, "image/jpeg", -1); file_type_store.append (out iter); file_type_store.set (iter, /* Save dialog: Label for saving in PNG format */ 0, _("PNG (lossless)"), 1, "image/png", -1); #if HAVE_WEBP file_type_store.append (out iter); file_type_store.set (iter, /* Save dialog: Label for sabing in WEBP format */ 0, _("WebP (compressed)"), 1, "image/webp", -1); #endif var box = new Gtk.Box (Gtk.Orientation.HORIZONTAL, 6); box.visible = true; box.spacing = 10; save_dialog.set_extra_widget (box); /* Label in save dialog beside combo box to choose file format (PDF, JPEG, PNG, WEBP) */ var label = new Gtk.Label (_("File format:")); label.visible = true; box.add (label); var file_type_combo = new Gtk.ComboBox.with_model (file_type_store); file_type_combo.visible = true; var renderer = new Gtk.CellRendererText (); file_type_combo.pack_start (renderer, true); file_type_combo.add_attribute (renderer, "text", 0); box.add (file_type_combo); if (file_type_store.get_iter_first (out iter)) { do { string mime_type; file_type_store.get (iter, 1, out mime_type, -1); if (mime_type == save_format) file_type_combo.set_active_iter (iter); } while (file_type_store.iter_next (ref iter)); } /* Label in save dialog beside compression slider */ var quality_label = new Gtk.Label (_("Compression:")); box.add (quality_label); var quality_adjustment = new Gtk.Adjustment (75, 0, 100, 1, 10, 0); var quality_scale = new Gtk.Scale (Gtk.Orientation.HORIZONTAL, quality_adjustment); quality_scale.width_request = 250; quality_scale.draw_value = false; var minimum_size_label = "<small>%s</small>".printf (_("Minimum size")); quality_scale.add_mark (quality_adjustment.lower, Gtk.PositionType.BOTTOM, minimum_size_label); quality_scale.add_mark (75, Gtk.PositionType.BOTTOM, null); quality_scale.add_mark (90, Gtk.PositionType.BOTTOM, null); var full_detail_label = "<small>%s</small>".printf (_("Full detail")); quality_scale.add_mark (quality_adjustment.upper, Gtk.PositionType.BOTTOM, full_detail_label); quality_adjustment.value = settings.get_int ("jpeg-quality"); quality_adjustment.value_changed.connect (() => { settings.set_int ("jpeg-quality", (int) quality_adjustment.value); }); box.add (quality_scale); /* Quality not applicable to PNG */ quality_scale.visible = quality_label.visible = (save_format != "image/png"); file_type_combo.changed.connect (() => { var mime_type = ""; Gtk.TreeIter i; if (file_type_combo.get_active_iter (out i)) { file_type_store.get (i, 1, out mime_type, -1); settings.set_string ("save-format", mime_type); } var filename = save_dialog.get_current_name (); /* Replace extension */ var extension_index = filename.last_index_of_char ('.'); if (extension_index >= 0) filename = filename.slice (0, extension_index); filename = filename + "." + mime_type_to_extension (mime_type); save_dialog.set_current_name (filename); /* Quality not applicable to PNG */ quality_scale.visible = quality_label.visible = (mime_type != "image/png"); }); while (true) { var response = save_dialog.run (); if (response != Gtk.ResponseType.ACCEPT) { save_dialog.destroy (); return null; } var mime_type = ""; Gtk.TreeIter i; if (file_type_combo.get_active_iter (out i)) file_type_store.get (i, 1, out mime_type, -1); var uri = save_dialog.get_uri (); var extension_index = uri.last_index_of_char ('.'); if (extension_index < 0) uri += "." + mime_type_to_extension (mime_type); /* Check the file(s) don't already exist */ var files = new List<File> (); if (mime_type == "image/jpeg" || mime_type == "image/png" || mime_type == "image/webp") { for (var j = 0; j < book.n_pages; j++) files.append (make_indexed_file (uri, j, book.n_pages)); } else files.append (File.new_for_uri (uri)); if (check_overwrite (save_dialog.transient_for, files)) { var directory_uri = uri.substring (0, uri.last_index_of ("/") + 1); settings.set_string ("save-directory", directory_uri); save_dialog.destroy (); return uri; } } } private bool check_overwrite (Gtk.Window parent, List<File> files) { foreach (var file in files) { if (!file.query_exists ()) continue; var dialog = new Gtk.MessageDialog (parent, Gtk.DialogFlags.MODAL | Gtk.DialogFlags.DESTROY_WITH_PARENT, Gtk.MessageType.QUESTION, Gtk.ButtonsType.NONE, /* Contents of dialog that shows if saving would overwrite and existing file. %s is replaced with the name of the file. */ _("A file named “%s” already exists. Do you want to replace it?"), file.get_basename ()); dialog.add_button (_("_Cancel"), Gtk.ResponseType.CANCEL); dialog.add_button (/* Button in dialog that shows if saving would overwrite and existing file. Clicking the button allows simple-scan to overwrite the file. */ _("_Replace"), Gtk.ResponseType.ACCEPT); var response = dialog.run (); dialog.destroy (); if (response != Gtk.ResponseType.ACCEPT) return false; } return true; } private string? mime_type_to_extension (string mime_type) { if (mime_type == "application/pdf") return "pdf"; else if (mime_type == "image/jpeg") return "jpg"; else if (mime_type == "image/png") return "png"; else if (mime_type == "image/webp") return "webp"; else return null; } private string? extension_to_mime_type (string extension) { var extension_lower = extension.down (); if (extension_lower == "pdf") return "application/pdf"; else if (extension_lower == "jpg") return "image/jpeg"; else if (extension_lower == "png") return "image/png"; else if (extension_lower == "webp") return "image/webp"; else return null; } private string uri_to_mime_type (string uri) { var extension_index = uri.last_index_of_char ('.'); if (extension_index < 0) return "image/jpeg"; var extension = uri.substring (extension_index + 1); var mime_type = extension_to_mime_type (extension); if (mime_type == null) return "image/jpeg"; return mime_type; } private async bool save_document_async () { var uri = choose_file_location (); if (uri == null) return false; var file = File.new_for_uri (uri); debug ("Saving to '%s'", uri); var mime_type = uri_to_mime_type (uri); var cancellable = new Cancellable (); var progress_bar = new CancellableProgressBar (_("Saving"), cancellable); action_bar.pack_end (progress_bar); progress_bar.visible = true; save_button.sensitive = false; try { yield book.save_async (mime_type, settings.get_int ("jpeg-quality"), file, settings.get_boolean ("postproc-enabled"), settings.get_string ("postproc-script"), settings.get_string ("postproc-arguments"), settings.get_boolean ("postproc-keep-original"), (fraction) => { progress_bar.set_fraction (fraction); }, cancellable); } catch (Error e) { save_button.sensitive = true; progress_bar.destroy (); warning ("Error saving file: %s", e.message); show_error_dialog (/* Title of error dialog when save failed */ _("Failed to save file"), e.message); return false; } save_button.sensitive = true; progress_bar.destroy_with_delay (500); book_needs_saving = false; book_uri = uri; return true; } private async bool prompt_to_save_async (string title, string discard_label) { if (!book_needs_saving || (book.n_pages == 0)) return true; var dialog = new Gtk.MessageDialog (this, Gtk.DialogFlags.MODAL, Gtk.MessageType.WARNING, Gtk.ButtonsType.NONE, "%s", title); dialog.format_secondary_text ("%s", /* Text in dialog warning when a document is about to be lost*/ _("If you don’t save, changes will be permanently lost.")); dialog.add_button (discard_label, Gtk.ResponseType.NO); dialog.add_button (_("_Cancel"), Gtk.ResponseType.CANCEL); dialog.add_button (_("_Save"), Gtk.ResponseType.YES); var response = dialog.run (); dialog.destroy (); switch (response) { case Gtk.ResponseType.YES: if (yield save_document_async ()) return true; else return false; case Gtk.ResponseType.NO: return true; default: return false; } } private void clear_document () { book.clear (); book_needs_saving = false; book_uri = null; save_button.sensitive = false; copy_to_clipboard_menuitem.sensitive = false; update_scan_status (); stack.set_visible_child_name ("startup"); } private void new_document () { prompt_to_save_async.begin (/* Text in dialog warning when a document is about to be lost */ _("Save current document?"), /* Button in dialog to create new document and discard unsaved document */ _("_Discard Changes"), (obj, res) => { if (!prompt_to_save_async.end(res)) return; if (scanning) stop_scan (); clear_document (); autosave_manager.cleanup (); }); } [GtkCallback] private bool status_label_activate_link_cb (Gtk.Label label, string uri) { if (uri == "install-firmware") { install_drivers (); return true; } return false; } private void new_document_cb () { new_document (); } [GtkCallback] private void redetect_button_clicked_cb (Gtk.Button button) { have_devices = false; update_scan_status (); redetect (); } private void scan (ScanOptions options) { status_page.set_title (/* Label shown when scan started */ _("Contacting Scanner…")); device_buttons_box.visible = true; device_buttons_box.sensitive = false; start_scan (get_selected_device (), options); } private void scan_single_cb () { var options = make_scan_options (); options.type = ScanType.SINGLE; scan (options); } private void scan_adf_cb () { var options = make_scan_options (); options.type = ScanType.ADF; scan (options); } private void scan_batch_cb () { var options = make_scan_options (); options.type = ScanType.BATCH; scan (options); } private void scan_stop_cb () { stop_scan (); } private void rotate_left_cb () { rotate_left_button_clicked_cb (); } private void rotate_right_cb () { rotate_right_button_clicked_cb (); } private void move_left_cb () { page_move_left_menuitem_activate_cb (); } private void move_right_cb () { page_move_right_menuitem_activate_cb (); } private void copy_page_cb () { copy_to_clipboard_button_clicked_cb (); } private void delete_page_cb () { page_delete_menuitem_activate_cb (); } private void set_scan_type (ScanType scan_type) { this.scan_type = scan_type; switch (scan_type) { case ScanType.SINGLE: scan_single_radio.active = true; scan_options_image.icon_name = "scanner-symbolic"; scan_button.tooltip_text = _("Scan a single page from the scanner"); break; case ScanType.ADF: scan_adf_radio.active = true; scan_options_image.icon_name = "scan-type-adf-symbolic"; scan_button.tooltip_text = _("Scan multiple pages from the scanner"); break; case ScanType.BATCH: scan_batch_radio.active = true; scan_options_image.icon_name = "scan-type-batch-symbolic"; scan_button.tooltip_text = _("Scan multiple pages from the scanner"); break; } } [GtkCallback] private void scan_single_radio_toggled_cb (Gtk.ToggleButton button) { if (button.active) set_scan_type (ScanType.SINGLE); } [GtkCallback] private void scan_adf_radio_toggled_cb (Gtk.ToggleButton button) { if (button.active) set_scan_type (ScanType.ADF); } [GtkCallback] private void scan_batch_radio_toggled_cb (Gtk.ToggleButton button) { if (button.active) set_scan_type (ScanType.BATCH); } private void set_document_hint (string document_hint, bool save = false) { this.document_hint = document_hint; if (document_hint == "text") { text_radio.active = true; scan_hint_image.icon_name = "x-office-document-symbolic"; } else if (document_hint == "photo") { photo_radio.active = true; scan_hint_image.icon_name = "image-x-generic-symbolic"; } if (save) settings.set_string ("document-type", document_hint); } [GtkCallback] private void text_radio_toggled_cb (Gtk.ToggleButton button) { if (button.active) set_document_hint ("text", true); } [GtkCallback] private void photo_radio_toggled_cb (Gtk.ToggleButton button) { if (button.active) set_document_hint ("photo", true); } [GtkCallback] private void preferences_button_clicked_cb (Gtk.Button button) { preferences_dialog.present (); } private ScanOptions make_scan_options () { var options = new ScanOptions (); if (document_hint == "text") { options.scan_mode = ScanMode.GRAY; options.dpi = preferences_dialog.get_text_dpi (); options.depth = 2; } else { options.scan_mode = ScanMode.COLOR; options.dpi = preferences_dialog.get_photo_dpi (); options.depth = 8; } preferences_dialog.get_paper_size (out options.paper_width, out options.paper_height); options.brightness = brightness; options.contrast = contrast; options.page_delay = page_delay; options.side = preferences_dialog.get_page_side (); return options; } [GtkCallback] private void device_combo_changed_cb (Gtk.Widget widget) { if (setting_devices) return; user_selected_device = true; if (get_selected_device () != null) settings.set_string ("selected-device", get_selected_device ()); } [GtkCallback] private void scan_button_clicked_cb (Gtk.Widget widget) { scan_button.visible = false; stop_button.visible = true; var options = make_scan_options (); options.type = scan_type; scan (options); } [GtkCallback] private void stop_scan_button_clicked_cb (Gtk.Widget widget) { scan_button.visible = true; stop_button.visible = false; stop_scan (); } private void preferences_cb () { preferences_dialog.present (); } private void update_page_menu () { var page = book_view.selected_page; if (page == null) { page_move_left_menuitem.sensitive = false; page_move_right_menuitem.sensitive = false; } else { var index = book.get_page_index (page); page_move_left_menuitem.sensitive = index > 0; page_move_right_menuitem.sensitive = index < book.n_pages - 1; } } private void page_selected_cb (BookView view, Page? page) { if (page == null) return; updating_page_menu = true; update_page_menu (); var menuitem = no_crop_menuitem; if (page.has_crop) { var crop_name = page.crop_name; if (crop_name != null) { if (crop_name == "A3") menuitem = a3_menuitem; else if (crop_name == "A4") menuitem = a4_menuitem; else if (crop_name == "A5") menuitem = a5_menuitem; else if (crop_name == "A6") menuitem = a6_menuitem; else if (crop_name == "letter") menuitem = letter_menuitem; else if (crop_name == "legal") menuitem = legal_menuitem; else if (crop_name == "4x6") menuitem = four_by_six_menuitem; } else menuitem = custom_crop_menuitem; } menuitem.active = true; crop_button.active = page.has_crop; updating_page_menu = false; } private void show_page_cb (BookView view, Page page) { File file; try { var dir = DirUtils.make_tmp ("simple-scan-XXXXXX"); file = File.new_for_path (Path.build_filename (dir, "scan.png")); page.save_png (file); } catch (Error e) { show_error_dialog (/* Error message display when unable to save image for preview */ _("Unable to save image for preview"), e.message); return; } try { Gtk.show_uri (screen, file.get_uri (), Gtk.get_current_event_time ()); } catch (Error e) { show_error_dialog (/* Error message display when unable to preview image */ _("Unable to open image preview application"), e.message); } } private void show_page_menu_cb (BookView view, Gdk.Event event) { page_menu.popup_at_pointer (event); } [GtkCallback] private void rotate_left_button_clicked_cb () { if (updating_page_menu) return; var page = book_view.selected_page; if (page != null) page.rotate_left (); } [GtkCallback] private void rotate_right_button_clicked_cb () { if (updating_page_menu) return; var page = book_view.selected_page; if (page != null) page.rotate_right (); } private void set_crop (string? crop_name) { crop_rotate_menuitem.sensitive = crop_name != null; if (updating_page_menu) return; var page = book_view.selected_page; if (page == null) { warning ("Trying to set crop but no selected page"); return; } if (crop_name == null) page.set_no_crop (); else if (crop_name == "custom") { var width = page.width; var height = page.height; var crop_width = (int) (width * 0.8 + 0.5); var crop_height = (int) (height * 0.8 + 0.5); page.set_custom_crop (crop_width, crop_height); page.move_crop ((width - crop_width) / 2, (height - crop_height) / 2); } else page.set_named_crop (crop_name); } [GtkCallback] private void no_crop_menuitem_toggled_cb (Gtk.CheckMenuItem widget) { if (widget.active) set_crop (null); } [GtkCallback] private void custom_crop_menuitem_toggled_cb (Gtk.CheckMenuItem widget) { if (widget.active) set_crop ("custom"); } [GtkCallback] private void four_by_six_menuitem_toggled_cb (Gtk.CheckMenuItem widget) { if (widget.active) set_crop ("4x6"); } [GtkCallback] private void legal_menuitem_toggled_cb (Gtk.CheckMenuItem widget) { if (widget.active) set_crop ("legal"); } [GtkCallback] private void letter_menuitem_toggled_cb (Gtk.CheckMenuItem widget) { if (widget.active) set_crop ("letter"); } [GtkCallback] private void a6_menuitem_toggled_cb (Gtk.CheckMenuItem widget) { if (widget.active) set_crop ("A6"); } [GtkCallback] private void a5_menuitem_toggled_cb (Gtk.CheckMenuItem widget) { if (widget.active) set_crop ("A5"); } [GtkCallback] private void a4_menuitem_toggled_cb (Gtk.CheckMenuItem widget) { if (widget.active) set_crop ("A4"); } [GtkCallback] private void a3_menuitem_toggled_cb (Gtk.CheckMenuItem widget) { if (widget.active) set_crop ("A3"); } [GtkCallback] private void crop_rotate_menuitem_activate_cb () { var page = book_view.selected_page; if (page == null) return; page.rotate_crop (); } [GtkCallback] private void page_move_left_menuitem_activate_cb () { var page = book_view.selected_page; var index = book.get_page_index (page); if (index > 0) book.move_page (page, index - 1); } [GtkCallback] private void page_move_right_menuitem_activate_cb () { var page = book_view.selected_page; var index = book.get_page_index (page); if (index < book.n_pages - 1) book.move_page (page, book.get_page_index (page) + 1); } [GtkCallback] private void page_delete_menuitem_activate_cb () { book_view.book.delete_page (book_view.selected_page); } private void reorder_document () { var dialog = new Gtk.Window (); dialog.type_hint = Gdk.WindowTypeHint.DIALOG; dialog.modal = true; dialog.border_width = 12; /* Title of dialog to reorder pages */ dialog.title = C_("dialog title", "Reorder Pages"); dialog.set_transient_for (this); dialog.key_press_event.connect ((e) => { if (e.state == 0 && e.keyval == Gdk.Key.Escape) { dialog.destroy (); return true; } return false; }); dialog.visible = true; var g = new Gtk.Grid (); g.row_homogeneous = true; g.row_spacing = 6; g.column_homogeneous = true; g.column_spacing = 6; g.visible = true; dialog.add (g); /* Label on button for combining sides in reordering dialog */ var b = make_reorder_button (_("Combine sides"), "F1F2F3B1B2B3-F1B1F2B2F3B3"); b.clicked.connect (() => { book.combine_sides (); dialog.destroy (); }); b.visible = true; g.attach (b, 0, 0, 1, 1); /* Label on button for combining sides in reverse order in reordering dialog */ b = make_reorder_button (_("Combine sides (reverse)"), "F1F2F3B3B2B1-F1B1F2B2F3B3"); b.clicked.connect (() => { book.combine_sides_reverse (); dialog.destroy (); }); b.visible = true; g.attach (b, 1, 0, 1, 1); /* Label on button for reversing in reordering dialog */ b = make_reorder_button (_("Reverse"), "C1C2C3C4C5C6-C6C5C4C3C2C1"); b.clicked.connect (() => { book.reverse (); dialog.destroy (); }); b.visible = true; g.attach (b, 0, 2, 1, 1); /* Label on button for cancelling page reordering dialog */ b = make_reorder_button (_("Keep unchanged"), "C1C2C3C4C5C6-C1C2C3C4C5C6"); b.clicked.connect (() => { dialog.destroy (); }); b.visible = true; g.attach (b, 1, 2, 1, 1); dialog.present (); } private void reorder_document_cb () { reorder_document (); } private Gtk.Button make_reorder_button (string text, string items) { var b = new Gtk.Button (); var vbox = new Gtk.Box (Gtk.Orientation.VERTICAL, 6); vbox.visible = true; b.add (vbox); var label = new Gtk.Label (text); label.visible = true; label.vexpand = true; vbox.add (label); var rb = make_reorder_box (items); rb.visible = true; rb.vexpand = true; vbox.add (rb); return b; } private Gtk.Box make_reorder_box (string items) { var box = new Gtk.Box (Gtk.Orientation.HORIZONTAL, 6); box.visible = true; Gtk.Box? page_box = null; for (var i = 0; items[i] != '\0'; i++) { if (items[i] == '-') { var a = new Gtk.Label ("➤"); a.visible = true; box.add (a); page_box = null; continue; } /* First character describes side */ var side = items[i]; i++; if (items[i] == '\0') break; if (page_box == null) { page_box = new Gtk.Box (Gtk.Orientation.HORIZONTAL, 3); page_box.visible = true; box.add (page_box); } var icon = new PageIcon (side, items[i] - '1'); icon.visible = true; page_box.add (icon); } return box; } [GtkCallback] private void save_file_button_clicked_cb (Gtk.Widget widget) { save_document_async.begin (); } public void save_document_activate_cb () { save_document_async.begin (); } [GtkCallback] private void copy_to_clipboard_button_clicked_cb () { var page = book_view.selected_page; if (page != null) page.copy_to_clipboard (this); } private void draw_page (Gtk.PrintOperation operation, Gtk.PrintContext print_context, int page_number) { var context = print_context.get_cairo_context (); var page = book.get_page (page_number); /* Rotate to same aspect */ bool is_landscape = false; if (print_context.get_width () > print_context.get_height ()) is_landscape = true; if (page.is_landscape != is_landscape) { context.translate (print_context.get_width (), 0); context.rotate (Math.PI_2); } context.scale (print_context.get_dpi_x () / page.dpi, print_context.get_dpi_y () / page.dpi); var image = page.get_image (true); Gdk.cairo_set_source_pixbuf (context, image, 0, 0); context.paint (); } private void email_document_cb () { email_document_async.begin (); } private async void email_document_async () { try { var dir = DirUtils.make_tmp ("simple-scan-XXXXXX"); string mime_type, filename; if (document_hint == "text") { mime_type = "application/pdf"; filename = "scan.pdf"; } else { mime_type = "image/jpeg"; filename = "scan.jpg"; } var file = File.new_for_path (Path.build_filename (dir, filename)); yield book.save_async (mime_type, settings.get_int ("jpeg-quality"), file, settings.get_boolean ("postproc-enabled"), settings.get_string ("postproc-script"), settings.get_string ("postproc-arguments"), settings.get_boolean ("postproc-keep-original"), null, null); var command_line = "xdg-email"; if (mime_type == "application/pdf") command_line += " --attach %s".printf (file.get_path ()); else { for (var i = 0; i < book.n_pages; i++) { var indexed_file = make_indexed_file (file.get_uri (), i, book.n_pages); command_line += " --attach %s".printf (indexed_file.get_path ()); } } Process.spawn_command_line_async (command_line); } catch (Error e) { warning ("Unable to email document: %s", e.message); } } private void print_document () { var print = new Gtk.PrintOperation (); print.n_pages = (int) book.n_pages; print.draw_page.connect (draw_page); try { print.run (Gtk.PrintOperationAction.PRINT_DIALOG, this); } catch (Error e) { warning ("Error printing: %s", e.message); } print.draw_page.disconnect (draw_page); } private void print_document_cb () { print_document (); } private void launch_help () { try { Gtk.show_uri (screen, "help:simple-scan", Gtk.get_current_event_time ()); } catch (Error e) { show_error_dialog (/* Error message displayed when unable to launch help browser */ _("Unable to open help file"), e.message); } } private void help_cb () { launch_help (); } private void show_about () { string[] authors = { "Robert Ancell <robert.ancell@canonical.com>" }; string title = _("About Document Scanner"); string description = _("Simple document scanning tool"); Gtk.show_about_dialog (this, "title", title, "authors", authors, "translator-credits", _("translator-credits"), "comments", description, "copyright", "Copyright © 2009-2018 Canonical Ltd.", "license-type", Gtk.License.GPL_3_0, "program-name", _("Document Scanner"), "logo-icon-name", "org.gnome.SimpleScan", "version", VERSION, "website", "https://gitlab.gnome.org/GNOME/simple-scan", "wrap-license", true); } private void about_cb () { show_about (); } private void on_quit () { prompt_to_save_async.begin (/* Text in dialog warning when a document is about to be lost */ _("Save document before quitting?"), /* Text in dialog warning when a document is about to be lost */ _("_Quit without Saving"), (obj, res) => { if (!prompt_to_save_async.end(res)) return; destroy (); if (save_state_timeout != 0) save_state (true); autosave_manager.cleanup (); }); } private void quit_cb () { on_quit (); } public override void size_allocate (Gtk.Allocation allocation) { base.size_allocate (allocation); if (!window_is_maximized && !window_is_fullscreen) { get_size (out window_width, out window_height); save_state (); } } private void install_drivers () { var message = "", instructions = ""; string[] packages_to_install = {}; switch (missing_driver) { case "brscan": case "brscan2": case "brscan3": case "brscan4": /* Message to indicate a Brother scanner has been detected */ message = _("You appear to have a Brother scanner."); /* Instructions on how to install Brother scanner drivers */ instructions = _("Drivers for this are available on the <a href=\"http://support.brother.com\">Brother website</a>."); break; case "pixma": /* Message to indicate a Canon Pixma scanner has been detected */ message = _("You appear to have a Canon scanner, which is supported by the <a href=\"http://www.sane-project.org/man/sane-pixma.5.html\">Pixma SANE backend</a>."); /* Instructions on how to resolve issue with SANE scanner drivers */ instructions = _("Please check if your <a href=\"http://www.sane-project.org/sane-supported-devices.html\">scanner is supported by SANE</a>, otherwise report the issue to the <a href=\"https://alioth-lists.debian.net/cgi-bin/mailman/listinfo/sane-devel\">SANE mailing list</a>."); break; case "samsung": /* Message to indicate a Samsung scanner has been detected */ message = _("You appear to have a Samsung scanner."); /* Instructions on how to install Samsung scanner drivers. Because HP acquired Samsung's global printing business in 2017, the support is made on HP site. */ instructions = _("Drivers for this are available on the <a href=\"https://support.hp.com\">HP website</a> (HP acquired Samsung's printing business)."); break; case "hpaio": case "smfp": /* Message to indicate a HP scanner has been detected */ message = _("You appear to have an HP scanner."); if (missing_driver == "hpaio") packages_to_install = { "libsane-hpaio" }; else /* Instructions on how to install HP scanner drivers. smfp is rebranded and slightly modified Samsung devices, for example: HP Laser MFP 135a is rebranded Samsung Xpress SL-M2070. It require custom drivers, not available in hpaio package */ instructions = _("Drivers for this are available on the <a href=\"https://support.hp.com\">HP website</a>."); break; case "epkowa": /* Message to indicate an Epson scanner has been detected */ message = _("You appear to have an Epson scanner."); /* Instructions on how to install Epson scanner drivers */ instructions = _("Drivers for this are available on the <a href=\"http://support.epson.com\">Epson website</a>."); break; case "lexmark_nscan": /* Message to indicate an Lexmark scanner has been detected */ message = _("You appear to have an Lexmark scanner."); /* Instructions on how to install Lexmark scanner drivers */ instructions = _("Drivers for this are available on the <a href=\"http://support.lexmark.com\">Lexmark website</a>."); break; } var dialog = new Gtk.Dialog.with_buttons (/* Title of dialog giving instructions on how to install drivers */ _("Install drivers"), this, Gtk.DialogFlags.MODAL, _("_Close"), Gtk.ResponseType.CLOSE); dialog.get_content_area ().border_width = 12; dialog.get_content_area ().spacing = 6; var label = new Gtk.Label (message); label.visible = true; label.xalign = 0f; label.vexpand = true; label.use_markup = true; dialog.get_content_area ().add (label); var instructions_box = new Gtk.Box (Gtk.Orientation.HORIZONTAL, 6); instructions_box.visible = true; instructions_box.vexpand = true; dialog.get_content_area ().add (instructions_box); var stack = new Gtk.Stack (); instructions_box.add (stack); var spinner = new Gtk.Spinner (); spinner.visible = true; stack.add (spinner); var status_label = new Gtk.Label (""); status_label.visible = true; stack.add (status_label); var instructions_label = new Gtk.Label (instructions); instructions_label.visible = true; instructions_label.xalign = 0f; instructions_label.use_markup = true; instructions_box.add (instructions_label); label = new Gtk.Label (/* Message in driver install dialog */ _("Once installed you will need to restart this app.")); label.visible = true; label.xalign = 0f; label.vexpand = true; dialog.get_content_area ().border_width = 12; dialog.get_content_area ().add (label); if (packages_to_install.length > 0) { #if HAVE_PACKAGEKIT stack.visible = true; spinner.active = true; instructions_label.set_text (/* Label shown while installing drivers */ _("Installing drivers…")); install_packages.begin (packages_to_install, () => {}, (object, result) => { status_label.visible = true; spinner.active = false; status_label.set_text ("☒"); stack.visible_child = status_label; /* Label shown once drivers successfully installed */ var result_text = _("Drivers installed successfully!"); try { var results = install_packages.end (result); if (results.get_error_code () == null) status_label.set_text ("☑"); else { var e = results.get_error_code (); /* Label shown if failed to install drivers */ result_text = _("Failed to install drivers (error code %d).").printf (e.code); } } catch (Error e) { /* Label shown if failed to install drivers */ result_text = _("Failed to install drivers."); warning ("Failed to install drivers: %s", e.message); } instructions_label.set_text (result_text); }); #else instructions_label.set_text (/* Label shown to prompt user to install packages (when PackageKit not available) */ ngettext ("You need to install the %s package.", "You need to install the %s packages.", packages_to_install.length).printf (string.joinv (", ", packages_to_install))); #endif } dialog.run (); dialog.destroy (); } #if HAVE_PACKAGEKIT private async Pk.Results? install_packages (string[] packages, Pk.ProgressCallback progress_callback) throws GLib.Error { var task = new Pk.Task (); Pk.Results results; results = yield task.resolve_async (Pk.Filter.NOT_INSTALLED, packages, null, progress_callback); if (results == null || results.get_error_code () != null) return results; var package_array = results.get_package_array (); var package_ids = new string[package_array.length + 1]; package_ids[package_array.length] = null; for (var i = 0; i < package_array.length; i++) package_ids[i] = package_array.data[i].get_id (); return yield task.install_packages_async (package_ids, null, progress_callback); } #endif public override bool window_state_event (Gdk.EventWindowState event) { var result = Gdk.EVENT_PROPAGATE; if (base.window_state_event != null) result = base.window_state_event (event); if ((event.changed_mask & Gdk.WindowState.MAXIMIZED) != 0) { window_is_maximized = (event.new_window_state & Gdk.WindowState.MAXIMIZED) != 0; save_state (); } if ((event.changed_mask & Gdk.WindowState.FULLSCREEN) != 0) { window_is_fullscreen = (event.new_window_state & Gdk.WindowState.FULLSCREEN) != 0; save_state (); } return result; } [GtkCallback] private bool window_delete_event_cb (Gtk.Widget widget, Gdk.EventAny event) { on_quit (); return true; /* Let us quit on our own terms */ } private void page_added_cb (Book book, Page page) { update_page_menu (); } private void reordered_cb (Book book) { update_page_menu (); } private void page_removed_cb (Book book, Page page) { update_page_menu (); } private void book_changed_cb (Book book) { save_button.sensitive = true; book_needs_saving = true; copy_to_clipboard_menuitem.sensitive = true; } private void load () { preferences_dialog = new PreferencesDialog (settings); preferences_dialog.delete_event.connect (() => { preferences_dialog.visible = false; return true; }); preferences_dialog.transient_for = this; preferences_dialog.modal = true; Gtk.Window.set_default_icon_name ("org.gnome.SimpleScan"); var app = Application.get_default () as Gtk.Application; /* Set HeaderBar title here because Glade doesn't keep it translated */ /* https://bugzilla.gnome.org/show_bug.cgi?id=782753 */ /* Title of scan window */ header_bar.title = _("Document Scanner"); app.add_action_entries (action_entries, this); app.set_accels_for_action ("app.new_document", { "<Ctrl>N" }); app.set_accels_for_action ("app.scan_single", { "<Ctrl>1" }); app.set_accels_for_action ("app.scan_adf", { "<Ctrl>F" }); app.set_accels_for_action ("app.scan_batch", { "<Ctrl>M" }); app.set_accels_for_action ("app.scan_stop", { "Escape" }); app.set_accels_for_action ("app.rotate_left", { "bracketleft" }); app.set_accels_for_action ("app.rotate_right", { "bracketright" }); app.set_accels_for_action ("app.move_left", { "less" }); app.set_accels_for_action ("app.move_right", { "greater" }); app.set_accels_for_action ("app.copy_page", { "<Ctrl>C" }); app.set_accels_for_action ("app.delete_page", { "Delete" }); app.set_accels_for_action ("app.save", { "<Ctrl>S" }); app.set_accels_for_action ("app.email", { "<Ctrl>E" }); app.set_accels_for_action ("app.print", { "<Ctrl>P" }); app.set_accels_for_action ("app.help", { "F1" }); app.set_accels_for_action ("app.quit", { "<Ctrl>Q" }); app.set_accels_for_action ("win.show-help-overlay", { "<Ctrl>question" }); var gear_menu = new Menu (); var section = new Menu (); gear_menu.append_section (null, section); section.append (_("_Email"), "app.email"); section.append (_("Pri_nt"), "app.print"); section.append (C_("menu", "_Reorder Pages"), "app.reorder"); section = new Menu (); gear_menu.append_section (null, section); section.append (_("_Preferences"), "app.preferences"); section.append (_("_Keyboard Shortcuts"), "win.show-help-overlay"); section.append (_("_Help"), "app.help"); section.append (_("_About Document Scanner"), "app.about"); menu_button.set_menu_model (gear_menu); app.add_window (this); /* Populate ActionBar (not supported in Glade) */ /* https://bugzilla.gnome.org/show_bug.cgi?id=769966 */ var button = new Gtk.Button.with_mnemonic (/* Label on new document button */ _("_New Document")); button.visible = true; button.clicked.connect (new_document_cb); action_bar.pack_start (button); var box = new Gtk.Box (Gtk.Orientation.HORIZONTAL, 10); box.visible = true; action_bar.set_center_widget (box); var rotate_box = new Gtk.Box (Gtk.Orientation.HORIZONTAL, 0); rotate_box.get_style_context ().add_class (Gtk.STYLE_CLASS_LINKED); rotate_box.visible = true; box.add (rotate_box); button = new Gtk.Button.from_icon_name ("object-rotate-left-symbolic"); button.visible = true; button.image.margin_start = 18; button.image.margin_end = 18; /* Tooltip for rotate left (counter-clockwise) button */ button.tooltip_text = _("Rotate the page to the left (counter-clockwise)"); button.clicked.connect (rotate_left_button_clicked_cb); rotate_box.add (button); button = new Gtk.Button.from_icon_name ("object-rotate-right-symbolic"); button.visible = true; button.image.margin_start = 18; button.image.margin_end = 18; /* Tooltip for rotate right (clockwise) button */ button.tooltip_text = _("Rotate the page to the right (clockwise)"); button.clicked.connect (rotate_right_button_clicked_cb); rotate_box.add (button); crop_button = new Gtk.ToggleButton (); crop_button.visible = true; var image = new Gtk.Image.from_icon_name ("crop-symbolic", Gtk.IconSize.BUTTON); image.visible = true; image.margin_start = 18; image.margin_end = 18; crop_button.add (image); /* Tooltip for crop button */ crop_button.tooltip_text = _("Crop the selected page"); crop_button.toggled.connect ((widget) => { if (updating_page_menu) return; if (widget.active) custom_crop_menuitem.active = true; else no_crop_menuitem.active = true; }); box.add (crop_button); delete_button = new Gtk.Button.from_icon_name ("user-trash-symbolic"); delete_button.visible = true; delete_button.image.margin_start = 18; delete_button.image.margin_end = 18; /* Tooltip for delete button */ delete_button.tooltip_text = _("Delete the selected page"); delete_button.clicked.connect (() => { book_view.book.delete_page (book_view.selected_page); }); box.add (delete_button); var document_type = settings.get_string ("document-type"); if (document_type != null) set_document_hint (document_type); book_view = new BookView (book); book_view.border_width = 18; book_view.vexpand = true; main_vbox.add (book_view); book_view.page_selected.connect (page_selected_cb); book_view.show_page.connect (show_page_cb); book_view.show_menu.connect (show_page_menu_cb); book_view.visible = true; preferences_dialog.transient_for = this; /* Load previous state */ load_state (); /* Restore window size */ debug ("Restoring window to %dx%d pixels", window_width, window_height); set_default_size (window_width, window_height); if (window_is_maximized) { debug ("Restoring window to maximized"); maximize (); } if (window_is_fullscreen) { debug ("Restoring window to fullscreen"); fullscreen (); } } private string state_filename { owned get { return Path.build_filename (Environment.get_user_config_dir (), "simple-scan", "state"); } } private void load_state () { debug ("Loading state from %s", state_filename); var f = new KeyFile (); try { f.load_from_file (state_filename, KeyFileFlags.NONE); } catch (Error e) { if (!(e is FileError.NOENT)) warning ("Failed to load state: %s", e.message); } window_width = state_get_integer (f, "window", "width", 600); if (window_width <= 0) window_width = 600; window_height = state_get_integer (f, "window", "height", 400); if (window_height <= 0) window_height = 400; window_is_maximized = state_get_boolean (f, "window", "is-maximized"); window_is_fullscreen = state_get_boolean (f, "window", "is-fullscreen"); scan_type = Scanner.type_from_string(state_get_string (f, "scanner", "scan-type")); set_scan_type (scan_type); } private string state_get_string (KeyFile f, string group_name, string key, string default = "") { try { return f.get_string (group_name, key); } catch { return default; } } private int state_get_integer (KeyFile f, string group_name, string key, int default = 0) { try { return f.get_integer (group_name, key); } catch { return default; } } private bool state_get_boolean (KeyFile f, string group_name, string key, bool default = false) { try { return f.get_boolean (group_name, key); } catch { return default; } } private static string STATE_DIR = Path.build_filename (Environment.get_user_config_dir (), "simple-scan", null); private void save_state (bool force = false) { if (!force) { if (save_state_timeout != 0) Source.remove (save_state_timeout); save_state_timeout = Timeout.add (100, () => { save_state (true); save_state_timeout = 0; return false; }); return; } debug ("Saving state to %s", state_filename); var f = new KeyFile (); f.set_integer ("window", "width", window_width); f.set_integer ("window", "height", window_height); f.set_boolean ("window", "is-maximized", window_is_maximized); f.set_boolean ("window", "is-fullscreen", window_is_fullscreen); f.set_string ("scanner", "scan-type", Scanner.type_to_string(scan_type)); try { DirUtils.create_with_parents (STATE_DIR, 0700); FileUtils.set_contents (state_filename, f.to_data ()); } catch (Error e) { warning ("Failed to write state: %s", e.message); } } public void start () { visible = true; autosave_manager = new AutosaveManager (); autosave_manager.book = book; if (autosave_manager.exists () && prompt_to_load_autosaved_book ()) autosave_manager.load (); if (book.n_pages == 0) book_needs_saving = false; else { stack.set_visible_child_name ("document"); book_view.selected_page = book.get_page (0); book_needs_saving = true; book_changed_cb (book); } } } private class CancellableProgressBar : Gtk.HBox { private Gtk.ProgressBar bar; private Gtk.Button? button; public CancellableProgressBar (string? text, Cancellable? cancellable) { bar = new Gtk.ProgressBar (); bar.visible = true; bar.set_text (text); bar.set_show_text (true); pack_start (bar); if (cancellable != null) { button = new Gtk.Button.with_label (/* Text of button for cancelling save */ _("Cancel")); button.visible = true; button.clicked.connect (() => { set_visible (false); cancellable.cancel (); }); pack_start (button); } } public void set_fraction (double fraction) { bar.set_fraction (fraction); } public void destroy_with_delay (uint delay) { button.set_sensitive (false); Timeout.add (delay, () => { this.destroy (); return false; }); } }