imageglom.cc 26.2 KB
Newer Older
1
2
/* Glom
 *
3
 * Copyright (C) 2001-2011 Murray Cumming
4
5
6
7
8
9
10
11
12
13
14
15
16
 *
 * 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 2 of the
 * License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful, but
 * WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * General Public License for more details.
 *
 * You should have received a copy of the GNU General Public
 * License along with this program; if not, write to the
17
18
 * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
 * Boston, MA 02110-1301 USA.
19
20
21
22
 */

#include "imageglom.h"
#include <glibmm/i18n.h>
23
#include <glom/appwindow.h>
24
#include <glom/utils_ui.h>
25
#include <glom/glade_utils.h>
Murray Cumming's avatar
Murray Cumming committed
26
#include <libglom/algorithms_utils.h>
27
#include <libglom/data_structure/glomconversions.h>
28
29
#include <glom/utility_widgets/dialog_image_load_progress.h>
#include <glom/utility_widgets/dialog_image_save_progress.h>
Murray Cumming's avatar
Murray Cumming committed
30
#include <libglom/utils.h>
31
#include <libglom/file_utils.h>
32
33
#include <gtkmm/appchooserdialog.h>
#include <gtkmm/filechooserdialog.h>
34
#include <giomm/contenttype.h>
35
#include <giomm/menu.h>
36
#include <libgda/gda-blob-op.h>
37
#include <glibmm/convert.h>
38

39
40
41
42
#ifdef G_OS_WIN32
#include <windows.h>
#endif

43
44
#include <iostream>   // for cout, endl

45
46
47
namespace Glom
{

48
ImageGlom::type_vec_ustrings ImageGlom::m_evince_supported_mime_types;
49
ImageGlom::type_vec_ustrings ImageGlom::m_gdkpixbuf_supported_mime_types;
50

51
ImageGlom::ImageGlom()
52
53
: m_ev_view(nullptr),
  m_ev_document_model(nullptr)
54
{
55
56
57
  init();
}

58
ImageGlom::ImageGlom(BaseObjectType* cobject, const Glib::RefPtr<Gtk::Builder>& /* builder */)
59
: Gtk::EventBox(cobject),
Murray Cumming's avatar
Murray Cumming committed
60
  m_ev_view(nullptr),
61
  m_ev_document_model(nullptr)
62
63
64
65
{
  init();
}

66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
void ImageGlom::clear_image_from_widgets()
{
  if(m_image)
  {
    m_image->set(Glib::RefPtr<Gdk::Pixbuf>()); //TODO: Add an unset() to gtkmm.
  }

  if(m_ev_document_model)
  {
    g_object_unref(m_ev_document_model);
    m_ev_document_model = nullptr;
  }
}

void ImageGlom::init_widgets(bool use_evince)
81
{
82
  clear_image_from_widgets();
83

84
  m_frame.remove();
85

86
  Gtk::Widget* widget = nullptr;
87

88
89
90
91
92
93
94
  if(use_evince)
  {
    if(!m_ev_view)
    {
      m_ev_view = EV_VIEW(ev_view_new());
      gtk_widget_show(GTK_WIDGET(m_ev_view));

95
      m_ev_scrolled_window = std::make_unique<Gtk::ScrolledWindow>();
96
97
98
99
100
101
102
103
104
105
106
107
      gtk_container_add(GTK_CONTAINER(m_ev_scrolled_window->gobj()), GTK_WIDGET(m_ev_view));

      //gtk_widget_add_events(GTK_WIDGET(m_ev_view), GDK_BUTTON_PRESS_MASK);

      //Connect the the EvView's button-press-event signal, 
      //because we don't get it otherwise.
      //For some reason this is not necessary with the GtkImage.
      auto cppEvView = Glib::wrap(GTK_WIDGET(m_ev_view));
      cppEvView->signal_button_press_event().connect(
        sigc::mem_fun(*this, &ImageGlom::on_button_press_event), false);
    }

108
    m_image.reset();
109

110
    widget = m_ev_scrolled_window.get();
111
112
113
  }
  else
  {
114
    m_image = std::make_unique<Gtk::Image>();
115
116
117
118
    if(m_ev_view)
    {
      gtk_widget_destroy(GTK_WIDGET(m_ev_view));
      m_ev_view = nullptr;
119
      m_ev_scrolled_window.reset();
120
121
    }

122
    widget = m_image.get();
123
124
125
126
127
128
129
130
  }

  widget->show();
  m_frame.add(*widget);
}

void ImageGlom::init()
{
131
132
  m_read_only = false;

133
#ifndef GLOM_ENABLE_CLIENT_ONLY
134
  setup_menu(this);
135
136
#endif // !GLOM_ENABLE_CLIENT_ONLY

137
  setup_menu_usermode();
138

139
140
  m_frame.set_shadow_type(Gtk::SHADOW_ETCHED_IN); //Without this, the image widget has no borders and is completely invisible when empty.
  m_frame.show();
141

142
  add(m_frame);
143
144
}

145
146


147
148
149
150
ImageGlom::~ImageGlom()
{
}

151
void ImageGlom::set_layout_item(const std::shared_ptr<LayoutItem>& layout_item, const Glib::ustring& table_name)
152
153
{
  LayoutWidgetField::set_layout_item(layout_item, table_name);
154
#ifdef GTKMM_ATKMM_ENABLED
155
  get_accessible()->set_name(layout_item->get_name());
156
#endif  
157
}
158

159
bool ImageGlom::on_button_press_event(GdkEventButton *button_event)
160
161
{
  GdkModifierType mods;
162
  gdk_window_get_device_position( gtk_widget_get_window (Gtk::Widget::gobj()), button_event->device, 0, 0, &mods );
163

164
  //Enable/Disable items.
165
  //We did this earlier, but get_appwindow is more likely to work now:
Murray Cumming's avatar
Murray Cumming committed
166
  auto pApp = get_appwindow();
167
168
  if(pApp)
  {
169
#ifndef GLOM_ENABLE_CLIENT_ONLY
170
171
172
173
174
175
    pApp->add_developer_action(m_refContextLayout); //So that it can be disabled when not in developer mode.
    pApp->add_developer_action(m_refContextAddField);
    pApp->add_developer_action(m_refContextAddRelatedRecords);
    pApp->add_developer_action(m_refContextAddGroup);

    pApp->update_userlevel_ui(); //Update our action's sensitivity.
176
#endif // !GLOM_ENABLE_CLIENT_ONLY
177
178
179

    //Only show this popup in developer mode, so operators still see the default GtkEntry context menu.
    //TODO: It would be better to add it somehow to the standard context menu.
180
#ifndef GLOM_ENABLE_CLIENT_ONLY
181
    if(pApp->get_userlevel() == AppState::userlevels::DEVELOPER)
182
183
184
185
    {
      if(mods & GDK_BUTTON3_MASK)
      {
        //Give user choices of actions on this item:
186
        popup_menu(button_event->button, button_event->time);
187
       
188
189
190
        return true; //We handled this event.
      }
    }
191
    else
192
#endif // !GLOM_ENABLE_CLIENT_ONLY
193
    {
194
      // We cannot be in developer mode in client only mode.
195
196
197
      if(mods & GDK_BUTTON3_MASK)
      {
        //Give user choices of actions on this item:
198
        popup_menu(button_event->button, button_event->time);
199

200
201
202
        return true; //We handled this event.
      }
    }
203

204
205
    //Single-click to select file:
    if(mods & GDK_BUTTON1_MASK)
206
    {
207
      on_menupopup_activate_select_file();
208
209
      return true; //We handled this event.

210
211
212
    }
  }

213
  return Gtk::EventBox::on_button_press_event(button_event);
214
215
}

216
AppWindow* ImageGlom::get_appwindow() const
217
{
Murray Cumming's avatar
Murray Cumming committed
218
  auto pWindow = const_cast<Gtk::Container*>(get_toplevel());
219
220
  //TODO: This only works when the child widget is already in its parent.

221
  return dynamic_cast<AppWindow*>(pWindow);
222
223
}

224
225
226
227
bool ImageGlom::get_has_original_data() const
{
  return true; //TODO.
}
228

229
/*
230
231
232
void ImageGlom::set_pixbuf(const Glib::RefPtr<Gdk::Pixbuf>& pixbuf)
{
  m_pixbuf_original = pixbuf;
233
  show_image_data();
234
}
235
*/
236

237
void ImageGlom::set_value(const Gnome::Gda::Value& value)
238
{
Armin Burgmeier's avatar
Armin Burgmeier committed
239
  // Remember original data 
240
  clear_original_data();
Armin Burgmeier's avatar
Armin Burgmeier committed
241
  m_original_data = value;
242
  show_image_data();
243
244
245
246
}

Gnome::Gda::Value ImageGlom::get_value() const
{
247
  return m_original_data;
248
}
249

250
void ImageGlom::on_size_allocate(Gtk::Allocation& allocation)
251
{
252
  Gtk::EventBox::on_size_allocate(allocation);
253

254
  //Resize the GtkImage if necessary:
Murray Cumming's avatar
Murray Cumming committed
255
  if(m_pixbuf_original)
256
  {
Murray Cumming's avatar
Murray Cumming committed
257
    const auto pixbuf_scaled = get_scaled_image();
258
    m_image->set(pixbuf_scaled);
259
  }
260
261
}

262
static void image_glom_ev_job_finished(EvJob* job, void* user_data)
263
{
264
265
  g_assert(job);
  
Murray Cumming's avatar
Murray Cumming committed
266
  auto self = static_cast<ImageGlom*>(user_data);
267
268
269
270
271
272
273
  g_assert(self);
  
  self->on_ev_job_finished(job);
}
  
void ImageGlom::on_ev_job_finished(EvJob* job)
{
274
  if(ev_job_is_failed (job)) {
Murray Cumming's avatar
Murray Cumming committed
275
276
277
278
279
280
281
282
283
284
285
286
    g_warning ("%s", job->error->message);
    g_object_unref (job);

    return;
  }

  ev_document_model_set_document(m_ev_document_model, job->document);
  ev_document_model_set_page(m_ev_document_model, 1);
  g_object_unref (job);
  
  //TODO: Show that we are no longer loading.
  //ev_view_set_loading(m_ev_view, FALSE);
287
}
288

289
290
const GdaBinary* ImageGlom::get_binary() const
{
Murray Cumming's avatar
Murray Cumming committed
291
  const GdaBinary* gda_binary = nullptr;
292
293
  if(m_original_data.get_value_type() == GDA_TYPE_BINARY)
    gda_binary = gda_value_get_binary(m_original_data.gobj());
294
295
  else if(m_original_data.get_value_type() == GDA_TYPE_BLOB)
  {
Murray Cumming's avatar
Murray Cumming committed
296
    const auto gda_blob = gda_value_get_blob(m_original_data.gobj());
297
    if(gda_blob && gda_blob_op_read_all(gda_blob->op, const_cast<GdaBlob*>(gda_blob)))
298
299
      gda_binary = &(gda_blob->data);
  }
300
301
302
303
  
  return gda_binary;
}

304
305
Glib::ustring ImageGlom::get_mime_type() const
{
Murray Cumming's avatar
Murray Cumming committed
306
  const auto gda_binary = get_binary();
307

308
309
310
311
312
313
314
  if(!gda_binary)
    return Glib::ustring();
    
  if(!gda_binary->data)
    return Glib::ustring();

  bool uncertain = false;
Murray Cumming's avatar
Murray Cumming committed
315
  const auto result = Gio::content_type_guess(std::string(),
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
    gda_binary->data, gda_binary->binary_length,
    uncertain);

  //std::cout << G_STRFUNC << ": mime_type=" << result << ", uncertain=" << uncertain << std::endl;
  return result;  
}

void ImageGlom::fill_evince_supported_mime_types()
{
  //Fill the static list if it has not already been filled:
  if(!m_evince_supported_mime_types.empty())
    return;

  //Discover what mime types libevview can support.
  //Older versions supported image types too, via GdkPixbuf,
  //but that support was then removed.  
Murray Cumming's avatar
Murray Cumming committed
332
  auto types_list = ev_backends_manager_get_all_types_info();
333
334
335
336
337
338
339
  if(!types_list)
  {
    return;
  }
  
  for(GList* l = types_list; l; l = g_list_next(l))
  {
Murray Cumming's avatar
Murray Cumming committed
340
341
342
343
    EvTypeInfo *info = (EvTypeInfo *)l->data;
    if(!info)
      continue;

Murray Cumming's avatar
Murray Cumming committed
344
    const char* mime_type = nullptr;
Murray Cumming's avatar
Murray Cumming committed
345
346
347
348
    int i = 0;
    while((mime_type = info->mime_types[i++]))
    {
      if(mime_type)
349
        m_evince_supported_mime_types.emplace_back(mime_type);
Murray Cumming's avatar
Murray Cumming committed
350
351
352
      //std::cout << "evince supported mime_type=" << mime_type << std::endl; 
    }
  }  
353
354
}

355
356
357
358
359
void ImageGlom::fill_gdkpixbuf_supported_mime_types()
{
  //Fill the static list if it has not already been filled:
  if(!m_gdkpixbuf_supported_mime_types.empty())
    return;
Murray Cumming's avatar
Murray Cumming committed
360

361
  for(const auto& format : Gdk::Pixbuf::get_formats())
362
  {
Murray Cumming's avatar
Murray Cumming committed
363
    const auto mime_types = format.get_mime_types();
364
365
366
367
368
369
    m_gdkpixbuf_supported_mime_types.insert(
      m_gdkpixbuf_supported_mime_types.end(),
      mime_types.begin(), mime_types.end());
  }
}

370
371
372
373
void ImageGlom::show_image_data()
{
  bool use_evince = false;
  
Murray Cumming's avatar
Murray Cumming committed
374
  const auto mime_type = get_mime_type();
375
376

  //std::cout << "mime_type=" << mime_type << std::endl; 
377
  
378
  fill_evince_supported_mime_types();
Murray Cumming's avatar
Murray Cumming committed
379
  if(Utils::find_exists(m_evince_supported_mime_types, mime_type))
380
  {
381
    use_evince = true;
382
  }
383
  
384
385
  init_widgets(use_evince);

386
387
388
389
390
  //Clear all possible display widgets:
  m_pixbuf_original.reset();

  if(use_evince)
  {
391
392
393
    // Try loading from data in memory:
    // TODO: Uncomment this if this API is added: https://bugzilla.gnome.org/show_bug.cgi?id=654832
    /*
Murray Cumming's avatar
Murray Cumming committed
394
    const auto gda_binary = get_binary();
395
396
    if(!gda_binary || !gda_binary->data || !gda_binary->binary_length)
    {
397
       std::cerr << G_STRFUNC << "Data was null or empty.\n";
398
399
400
401
402
403
404
405
      return;
    }
    
    EvJob *job = ev_job_load_new_with_data(
      (char*)gda_binary->data, gda_binary->binary_length);
    */
    //TODO: Test failure asynchronously.
    
Murray Cumming's avatar
Murray Cumming committed
406
    const auto uri = save_to_temp_file(false /* don't show progress */);
407
408
    if(uri.empty())
    {
409
      std::cerr << G_STRFUNC << "Could not save temp file to show in the EvView.\n";
410
411
412
413
414
415
    }
  
    EvJob *job = ev_job_load_new(uri.c_str());
  
    m_ev_document_model = ev_document_model_new();
    ev_view_set_model(m_ev_view, m_ev_document_model);
416
417
418
419
    ev_document_model_set_continuous(m_ev_document_model, FALSE); //Show only one page.

    //TODO: Show that we are loading.
    //ev_view_set_loading(m_ev_view, TRUE);
Murray Cumming's avatar
Murray Cumming committed
420

421
    g_signal_connect (job, "finished",
422
423
      G_CALLBACK (image_glom_ev_job_finished), this);
    ev_job_scheduler_push_job (job, EV_JOB_PRIORITY_NONE);
424
425
426
427
  }
  else
  {
    //Use GtkImage instead:
428
    Glib::RefPtr<const Gio::Icon> icon;
429
430
431
      
    bool use_gdkpixbuf = false;
    fill_gdkpixbuf_supported_mime_types();
Murray Cumming's avatar
Murray Cumming committed
432
    if(Utils::find_exists(m_gdkpixbuf_supported_mime_types, mime_type))
433
434
435
436
437
438
439
    {
      use_gdkpixbuf = true;
    }
    
    if(use_gdkpixbuf)
    {
      //Try to use GdkPixbuf's loader:
440
      m_pixbuf_original = UiUtils::get_pixbuf_for_gda_value(m_original_data);
441
442
443
    }
    else
    {
444
445
      //Get an icon for the file type;
      icon = Gio::content_type_get_icon(mime_type);
446
    }
447
    
448
    if(m_pixbuf_original)
449
    {
450
      auto pixbuf_scaled = get_scaled_image();
451
      m_image->set(pixbuf_scaled);
452
    }
453
454
    else if(icon)
    {
455
      m_image->set(icon, Gtk::ICON_SIZE_DIALOG);
456
    }
457
    else
458
    {
459
      m_image->set_from_icon_name("image-missing", Gtk::ICON_SIZE_DIALOG);
460
    }
461
462
463
  }
}

464
Glib::RefPtr<Gdk::Pixbuf> ImageGlom::get_scaled_image()
465
{
466
  auto pixbuf = m_pixbuf_original;
467
468

  if(!pixbuf)
469
    return pixbuf;
470
 
471
  const auto allocation = m_image->get_allocation();
Murray Cumming's avatar
Murray Cumming committed
472
473
  const auto pixbuf_height = pixbuf->get_height();
  const auto pixbuf_width = pixbuf->get_width();
474
    
Murray Cumming's avatar
Murray Cumming committed
475
476
  const auto allocation_height = allocation.get_height();
  const auto allocation_width = allocation.get_width();
477
      
Murray Cumming's avatar
Murray Cumming committed
478
479
  //std::cout << "pixbuf_height=" << pixbuf_height << ", pixbuf_width=" << pixbuf_width << std::endl;
  //std::cout << "allocation_height=" << allocation.get_height() << ", allocation_width=" << allocation.get_width() << std::endl;
480

481
  if( (pixbuf_height > allocation_height) ||
482
483
      (pixbuf_width > allocation_width) )
  {
484
    if(true) //allocation_height > 10 || allocation_width > 10)
485
    {
486
      auto pixbuf_scaled = UiUtils::image_scale_keeping_ratio(pixbuf, allocation_height, allocation_width);
487
488
      
      //Don't set a new pixbuf if the dimensions have not changed:
489
      Glib::RefPtr<Gdk::Pixbuf> pixbuf_in_image;
490

491
492
      if(m_image->get_storage_type() == Gtk::IMAGE_PIXBUF) //Prevent warning.
        pixbuf_in_image = m_image->get_pixbuf();
493
494

      if( !pixbuf_in_image || !pixbuf_scaled || (pixbuf_in_image->get_height() != pixbuf_scaled->get_height()) || (pixbuf_in_image->get_width() != pixbuf_scaled->get_width()) )
495
      {
Murray Cumming's avatar
Murray Cumming committed
496
        /*
497
        std::cout << "get_scale(): returning scaled\n";
498
499
500
501
        if(pixbuf_scaled)
        {
          std::cout << "scaled height=" << pixbuf_scaled->get_height() << ", scaled width=" << pixbuf_scaled->get_width() << std::endl;
        }
Murray Cumming's avatar
Murray Cumming committed
502
        */
503
504
        
        return pixbuf_scaled;
505
      }
506
      else
507
      {
508
509
510
511
        //Return the existing one, 
        //instead of a new one with the same contents,
        //so no unnecessary changes will be triggered.
        return pixbuf_in_image;
512
      }
513
    }
514
  }
515
  
516
  //std::cout << "get_scaled(): returning original\n";
517
  return pixbuf;
518
519
}

520
void ImageGlom::on_menupopup_activate_open_file()
521
{
522
523
524
  open_with();
}

525
void ImageGlom::on_menupopup_activate_open_file_with()
526
{
Murray Cumming's avatar
Murray Cumming committed
527
  auto pApp = get_appwindow();
528
529

  //Offer the user a choice of suitable applications:
Murray Cumming's avatar
Murray Cumming committed
530
  const auto mime_type = get_mime_type();
531
532
  if(mime_type.empty())
  {
533
    std::cerr << G_STRFUNC << ": mime_type is empty.\n";
534
535
536
  }
  
  Gtk::AppChooserDialog dialog(mime_type);
537
538
539
540
  if(pApp)
    dialog.set_transient_for(*pApp);

  if(dialog.run() != Gtk::RESPONSE_OK)
541
    return;
542
  
543
  auto app_info = dialog.get_app_info();
544
545
  if(!app_info)
  {
546
    std::cerr << G_STRFUNC << ": app_info was null.\n";
547
548
549
550
  }
  
  open_with(app_info);
}
551

552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
static void make_file_read_only(const Glib::ustring& uri)
{
  std::string filepath;

  try
  {
    filepath = Glib::filename_from_uri(uri);
  }
  catch(const Glib::Error& ex)
  {
    std::cerr << G_STRFUNC << "Exception: " << ex.what() << std::endl;
    return;
  }
  
  if(filepath.empty())
  {
568
    std::cerr << G_STRFUNC << ": filepath is empty.\n";
569
570
  }
  
Murray Cumming's avatar
Murray Cumming committed
571
  const auto result = chmod(filepath.c_str(), S_IRUSR);
572
573
  if(result != 0)
  {
574
    std::cerr << G_STRFUNC << ": chmod() failed.\n";
575
576
577
578
579
  }
  
  //Setting the attribute via gio gives us this exception:
  //"Setting attribute access::can-write not supported"
  /*
580
  auto file = Gio::File::create_for_uri(uri);
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595

  Glib::RefPtr<Gio::FileInfo> file_info;

  try
  {
    file_info = file->query_info(G_FILE_ATTRIBUTE_ACCESS_CAN_WRITE);
  }
  catch(const Glib::Error& ex)
  {
    std::cerr << G_STRFUNC << ": query_info() failed: " << ex.what() << std::endl;
    return;
  }
  
  if(!file_info)
  {
596
    std::cerr << G_STRFUNC << ": : file_info is null\n";
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
    return;
  }
  
  const bool can_write =
    file_info->get_attribute_boolean(G_FILE_ATTRIBUTE_ACCESS_CAN_WRITE);
  if(!can_write)
    return;
    
  file_info->set_attribute_boolean(G_FILE_ATTRIBUTE_ACCESS_CAN_WRITE, false);
  
  try
  {
    file->set_attributes_from_info(file_info);
  }
  catch(const Glib::Error& ex)
  {
    std::cerr << G_STRFUNC << ": set_attributes_from_info() failed: " << ex.what() << std::endl;
  }
  */
}

618
Glib::ustring ImageGlom::save_to_temp_file(bool show_progress)
619
{
620
  Glib::ustring uri = FileUtils::get_temp_file_uri("glom_image");
621
  if(uri.empty())
622
  {
623
    std::cerr << G_STRFUNC << ": : uri is empty.\n";
624
  }
625
  
626
627
628
629
630
  bool saved = false;
  if(show_progress)
    saved = save_file(uri);
  else
    saved = save_file_sync(uri);
631
  
632
633
634
  if(!saved)
  {
    uri = Glib::ustring();
635
    std::cerr << G_STRFUNC << ": save_file() failed.\n";
636
  }
637
638
639
640
641
642
643
  else
  {
    //Don't let people easily edit the saved file,
    //because they would lose data when it is automatically deleted later.
    //Also they might think that editing it will change it in the database.
    make_file_read_only(uri);
  }
644
645
646
647
648
649

  return uri;
}

void ImageGlom::open_with(const Glib::RefPtr<Gio::AppInfo>& app_info)
{
Murray Cumming's avatar
Murray Cumming committed
650
  const auto uri = save_to_temp_file();
651
  if(uri.empty())
652
653
654
655
    return;

  if(app_info)
  {
Murray Cumming's avatar
Murray Cumming committed
656
    app_info->launch_uri(uri); //TODO: Get a GdkAppLaunchContext?
657
658
659
660
661
662
663
664
  }
  else
  {
    //TODO: Avoid duplication in xsl_utils.cc, by moving this into a utility function:  
#ifdef G_OS_WIN32
    // gtk_show_uri doesn't seem to work on Win32, at least not for local files
    // We use Windows API instead.
    // TODO: Check it again and file a bug if necessary.
665
666
    // TODO: and this might not be necessary with Gio::AppInfo::launch_default_for_uri().
    //   Previously we used gtk_show_uri().
667
668
    ShellExecute(0, "open", uri.c_str(), 0, 0, SW_SHOW);
#else
669
    Gio::AppInfo::launch_default_for_uri(uri);
670
671
672
#endif //G_OS_WIN32
  }
}
673

674

675
static void set_file_filter_images(Gtk::FileChooser& file_chooser)
676
{
677
  //Get image formats only:
678
  auto filter = Gtk::FileFilter::create();
679
680
  filter->set_name(_("Images"));
  filter->add_pixbuf_formats();
681
  file_chooser.add_filter(filter);
682
  
683
684
  ev_document_factory_add_filters(GTK_WIDGET(file_chooser.gobj()), 0);
  
685
686
687
  //Make Images the currently-selected one:
  file_chooser.set_filter(filter);
  
688
  /*  ev_document_factory_add_filters() add this already:
689
690
691
692
  filter = Gtk::FileFilter::create();
  filter->set_name(_("All Files"));
  filter->add_pattern("*");
  file_chooser.add_filter(filter);
693
  */
694
695
}

696
void ImageGlom::on_menupopup_activate_save_file()
697
{
Murray Cumming's avatar
Murray Cumming committed
698
  auto pApp = get_appwindow();
699
700
701
702
703

  Gtk::FileChooserDialog dialog(_("Save Image"), Gtk::FILE_CHOOSER_ACTION_SAVE);
  if(pApp)
    dialog.set_transient_for(*pApp);
          
704
  set_file_filter_images(dialog);
705

706
707
  dialog.add_button(_("_Cancel"), Gtk::RESPONSE_CANCEL);
  dialog.add_button(_("_Save"), Gtk::RESPONSE_OK);
Murray Cumming's avatar
Murray Cumming committed
708
  const auto response = dialog.run();
709
710
711
712
  dialog.hide();
  if(response != Gtk::RESPONSE_OK)
    return;
    
Murray Cumming's avatar
Murray Cumming committed
713
  const auto uri = dialog.get_uri();
714
715
716
717
718
719
  if(uri.empty())
    return;
    
  save_file(uri);
}

720
721
722
723
724
725
726
bool ImageGlom::save_file_sync(const Glib::ustring& uri)
{
  //TODO: We should still do this asynchronously, 
  //even when we don't use the dialog's run() to do that
  //because we don't want to offer feedback.
  //Ideally, EvView would just load from data anyway.
  
Murray Cumming's avatar
Murray Cumming committed
727
  const auto gda_binary = get_binary();
728
729
  if(!gda_binary)
  {
730
    std::cerr << G_STRFUNC << ": GdaBinary is null\n";
731
732
733
734
735
    return false;
  }
    
  if(!gda_binary->data)
  {
736
    std::cerr << G_STRFUNC << ": GdaBinary::data is null\n";
737
738
739
740
741
    return false;
  }

  try
  {
Murray Cumming's avatar
Murray Cumming committed
742
    const auto filepath = Glib::filename_from_uri(uri);
743
744
745
746
747
748
749
750
751
752
753
    Glib::file_set_contents(filepath, (const char*)gda_binary->data, gda_binary->binary_length);
  }
  catch(const Glib::Error& ex)
  {
    std::cerr << G_STRFUNC << "Exception: " << ex.what() << std::endl;
    return false;
  }
  
  return true;
}

754
755
bool ImageGlom::save_file(const Glib::ustring& uri)
{
Murray Cumming's avatar
Murray Cumming committed
756
  DialogImageSaveProgress* dialog_save = nullptr;
757
758
759
760
761
  Utils::get_glade_widget_derived_with_warning(dialog_save);
  if(!dialog_save)
    return false;
    
  // Automatically delete the dialog when we no longer need it:
762
  std::shared_ptr<Gtk::Dialog> dialog_keeper(dialog_save);
763

Murray Cumming's avatar
Murray Cumming committed
764
  auto pApp = get_appwindow();
765
766
767
  if(pApp)
    dialog_save->set_transient_for(*pApp);

Murray Cumming's avatar
Murray Cumming committed
768
  const auto gda_binary = get_binary();
769
770
771
772
  if(!gda_binary)
    return false;

  dialog_save->set_image_data(*gda_binary);
773
774
  dialog_save->save(uri);

775
  dialog_save->run();
776

777
778
779
  return true;
}

780
void ImageGlom::on_menupopup_activate_select_file()
781
782
783
784
{
  if(m_read_only)
    return;
    
Murray Cumming's avatar
Murray Cumming committed
785
  auto pApp = get_appwindow();
786
787
788
789
790

  Gtk::FileChooserDialog dialog(_("Choose Image"), Gtk::FILE_CHOOSER_ACTION_OPEN);
  if(pApp)
    dialog.set_transient_for(*pApp);
          
791
  set_file_filter_images(dialog);
792

793
  dialog.add_button(_("_Cancel"), Gtk::RESPONSE_CANCEL);
794
795
796
  dialog.add_button(_("Select"), Gtk::RESPONSE_OK);
  int response = dialog.run();
  dialog.hide();
797

798
  if((response != Gtk::RESPONSE_CANCEL) && (response != Gtk::RESPONSE_DELETE_EVENT))
799
  {
Murray Cumming's avatar
Murray Cumming committed
800
    const auto uri = dialog.get_uri();
801
    if(!uri.empty())
802
    {
Murray Cumming's avatar
Murray Cumming committed
803
804
805
      DialogImageLoadProgress* dialog_progress = nullptr;
      Utils::get_glade_widget_derived_with_warning(dialog_progress);
      if(dialog_progress)
806
      {
807
        // Automatically delete the dialog when we no longer need it:
Murray Cumming's avatar
Murray Cumming committed
808
        std::shared_ptr<Gtk::Dialog> dialog_keeper(dialog_progress);
Armin Burgmeier's avatar
Armin Burgmeier committed
809

810
        if(pApp)
Murray Cumming's avatar
Murray Cumming committed
811
          dialog_progress->set_transient_for(*pApp);
812

Murray Cumming's avatar
Murray Cumming committed
813
        dialog_progress->load(uri);
Armin Burgmeier's avatar
Armin Burgmeier committed
814

Murray Cumming's avatar
Murray Cumming committed
815
        if(dialog_progress->run() == Gtk::RESPONSE_ACCEPT)
Armin Burgmeier's avatar
Armin Burgmeier committed
816
        {
817
          GdaBinary* bin = g_new(GdaBinary, 1);
818
          auto image_data = dialog_progress->get_image_data();
819
820
821
          bin->data = image_data->data;
          bin->binary_length = image_data->binary_length;

822
          clear_original_data();
823
824
825
          
          g_value_unset(m_original_data.gobj());
          g_value_init(m_original_data.gobj(), GDA_TYPE_BINARY);
826
827
          gda_value_take_binary(m_original_data.gobj(), bin);

828
          show_image_data();
Armin Burgmeier's avatar
Armin Burgmeier committed
829
830
          signal_edited().emit();
        }
831
832
      }
    }
833
  }
834
835
}

836
837
838
839
840
void ImageGlom::clear_original_data()
{
  m_original_data = Gnome::Gda::Value();
}

841
842
843
844
845
void ImageGlom::on_clipboard_get(Gtk::SelectionData& selection_data, guint /* info */)
{
  //info is meant to indicate the target, but it seems to be always 0,
  //so we use the selection_data's target instead.

Murray Cumming's avatar
Murray Cumming committed
846
  const auto target = selection_data.get_target(); 
847

Murray Cumming's avatar
Murray Cumming committed
848
  const auto mime_type = get_mime_type();
849
850
  if(mime_type.empty())
  {
851
    std::cerr << G_STRFUNC << ": mime_type is empty.\n";
852
853
854
  }
  
  if(target == mime_type)
855
  {
Murray Cumming's avatar
Murray Cumming committed
856
    const auto gda_binary = get_binary();
857
858
859
860
861
862
863
864
    if(!gda_binary)
      return;
    
    if(!gda_binary->data)
      return;
    
    selection_data.set(mime_type, 8, gda_binary->data, gda_binary->binary_length);

865
    // This set() override uses an 8-bit text format for the data.
866
    //selection_data.set_pixbuf(m_pixbuf_clipboard);
867
868
869
  }
  else
  {
870
    std::cout << "ExampleWindow::on_clipboard_get(): Unexpected clipboard target format. expected: " << mime_type << std::endl;
871
  }
872
873
874
875
}

void ImageGlom::on_clipboard_clear()
{
876
877
878
  if(m_read_only)
    return;

879
  m_pixbuf_clipboard.reset();
880
881
}

882
void ImageGlom::on_menupopup_activate_copy()
883
{
884
885
886
887
888
889
  if(m_pixbuf_original)
  {
    //When copy is used, store it here until it is pasted.
    m_pixbuf_clipboard = m_pixbuf_original->copy(); //TODO: Get it from the DB, when we stop storing the original here instead of just the preview.
  }
  else
890
    m_pixbuf_clipboard.reset();
891

892
  auto refClipboard = Gtk::Clipboard::get();
893
894

  //Targets:
Murray Cumming's avatar
Murray Cumming committed
895
  const auto mime_type = get_mime_type();
896
897
  if(mime_type.empty())
  {
898
    std::cerr << G_STRFUNC << ": mime_type is empty.\n";
899
900
  }
  
901
  std::vector<Gtk::TargetEntry> listTargets;
902
  listTargets.emplace_back( Gtk::TargetEntry(mime_type) );
903

904
905
  refClipboard->set( listTargets, sigc::mem_fun(*this, &ImageGlom::on_clipboard_get), sigc::mem_fun(*this, &ImageGlom::on_clipboard_clear) );
}
906

907
908
void ImageGlom::on_clipboard_received_image(const Glib::RefPtr<Gdk::Pixbuf>& pixbuf)
{
909
910
911
  if(m_read_only)
    return;

912
913
  if(pixbuf)
  {
914
    clear_original_data();
Armin Burgmeier's avatar
Armin Burgmeier committed
915

916
    m_pixbuf_original = pixbuf;
917
    show_image_data();
918

919
920
    signal_edited().emit();
  }
921
922
}

923

924
void ImageGlom::on_menupopup_activate_paste()
925
{
926
927
928
  if(m_read_only)
    return;

929
  //Tell the clipboard to call our method when it is ready:
930
  auto refClipboard = Gtk::Clipboard::get();
931

932
933
  if(refClipboard)
    refClipboard->request_image( sigc::mem_fun(*this, &ImageGlom::on_clipboard_received_image) );
934
935
}

936
void ImageGlom::on_menupopup_activate_clear()
Murray Cumming's avatar
0.8.34:    
Murray Cumming committed
937
{
938
939
940
  if(m_read_only)
    return;

941
  clear_original_data();
942
  show_image_data();
943
  signal_edited().emit();
Murray Cumming's avatar
0.8.34:    
Murray Cumming committed
944
945
}

946
947
void ImageGlom::setup_menu_usermode()
{
948
949
  //Create the Gio::ActionGroup and associate it with this widget:
  m_refActionGroup_UserModePopup = Gio::SimpleActionGroup::create();
950

951
  m_refActionOpenFile = m_refActionGroup_UserModePopup->add_action("open-file",
952
953
    sigc::mem_fun(*this, &ImageGlom::on_menupopup_activate_open_file) );

954
  m_refActionOpenFileWith = m_refActionGroup_UserModePopup->add_action("open-fil-ewith",
955
956
    sigc::mem_fun(*this, &ImageGlom::on_menupopup_activate_open_file_with) );
    
957
  m_refActionSaveFile = m_refActionGroup_UserModePopup->add_action("save-file",
958
959
    sigc::mem_fun(*this, &ImageGlom::on_menupopup_activate_save_file) );
    
960
  m_refActionSelectFile = m_refActionGroup_UserModePopup->add_action("select-file",
961
962
    sigc::mem_fun(*this, &ImageGlom::on_menupopup_activate_select_file) );

963
  m_refActionCopy = m_refActionGroup_UserModePopup->add_action("copy",
964
965
    sigc::mem_fun(*this, &ImageGlom::on_menupopup_activate_copy) );

966
  m_refActionPaste = m_refActionGroup_UserModePopup->add_action("paste",