1 ///////////////////////////////////////////////////////////////////////////////
2 // Name:        src/gtk/aboutdlg.cpp
3 // Purpose:     native GTK+ wxAboutBox() implementation
4 // Author:      Vadim Zeitlin
5 // Created:     2006-10-08
6 // Copyright:   (c) 2006 Vadim Zeitlin <vadim@wxwindows.org>
7 // Licence:     wxWindows licence
8 ///////////////////////////////////////////////////////////////////////////////
9 
10 // ============================================================================
11 // declarations
12 // ============================================================================
13 
14 // ----------------------------------------------------------------------------
15 // headers
16 // ----------------------------------------------------------------------------
17 
18 // for compilers that support precompilation, includes "wx.h".
19 #include "wx/wxprec.h"
20 
21 #if wxUSE_ABOUTDLG
22 
23 #include "wx/aboutdlg.h"
24 
25 #ifndef WX_PRECOMP
26     #include "wx/window.h"
27 #endif //WX_PRECOMP
28 
29 #include <gtk/gtk.h>
30 #include "wx/gtk/private.h"
31 #include "wx/gtk/private/gtk2-compat.h"
32 
33 // ----------------------------------------------------------------------------
34 // GtkArray: temporary array of GTK strings
35 // ----------------------------------------------------------------------------
36 
37 namespace
38 {
39 
40 class GtkArray
41 {
42 public:
43     // Create empty GtkArray
GtkArray()44     GtkArray() : m_strings(0), m_count(0)
45     {
46     }
47 
48     // Create GtkArray from wxArrayString. Note that the created object is
49     // only valid as long as 'a' is!
GtkArray(const wxArrayString & a)50     GtkArray(const wxArrayString& a)
51     {
52         m_count = a.size();
53         m_strings = new const gchar *[m_count + 1];
54 
55         for ( size_t n = 0; n < m_count; n++ )
56         {
57 #if wxUSE_UNICODE
58             // notice that there is no need to copy the string pointer here
59             // because this class is used only as a temporary and during its
60             // existence the pointer persists in wxString which uses it either
61             // for internal representation (in wxUSE_UNICODE_UTF8 case) or as
62             // cached m_convertedToChar (in wxUSE_UNICODE_WCHAR case)
63             m_strings[n] = wxGTK_CONV_SYS(a[n]);
64 #else // !wxUSE_UNICODE
65             // and in ANSI build we can simply borrow the pointer from
66             // wxCharBuffer (which owns it in this case) instead of copying it
67             // but we then become responsible for freeing it
68             m_strings[n] = wxGTK_CONV_SYS(a[n]).release();
69 #endif // wxUSE_UNICODE/!wxUSE_UNICODE
70         }
71 
72         // array must be NULL-terminated
73         m_strings[m_count] = NULL;
74     }
75 
operator const gchar**() const76     operator const gchar **() const { return m_strings; }
77 
~GtkArray()78     ~GtkArray()
79     {
80 #if !wxUSE_UNICODE
81         for ( size_t n = 0; n < m_count; n++ )
82             free(const_cast<gchar *>(m_strings[n]));
83 #endif
84 
85         delete [] m_strings;
86     }
87 
88 private:
89     const gchar **m_strings;
90     size_t m_count;
91 
92     wxDECLARE_NO_COPY_CLASS(GtkArray);
93 };
94 
95 } // anonymous namespace
96 
97 // ============================================================================
98 // implementation
99 // ============================================================================
100 
101 // GTK+ about dialog is modeless, keep track of it in this variable
102 static GtkAboutDialog *gs_aboutDialog = NULL;
103 
104 extern "C" {
wxGtkAboutDialogOnClose(GtkAboutDialog * about)105 static void wxGtkAboutDialogOnClose(GtkAboutDialog *about)
106 {
107     gtk_widget_destroy(GTK_WIDGET(about));
108     if ( about == gs_aboutDialog )
109         gs_aboutDialog = NULL;
110 }
111 }
112 
113 #ifdef __WXGTK3__
114 extern "C" {
activate_link(GtkAboutDialog *,const char * link,void * dontIgnore)115 static gboolean activate_link(GtkAboutDialog*, const char* link, void* dontIgnore)
116 {
117     if (dontIgnore)
118     {
119         wxLaunchDefaultBrowser(wxGTK_CONV_BACK_SYS(link));
120         return true;
121     }
122     return false;
123 }
124 }
125 #else
126 extern "C" {
wxGtkAboutDialogOnLink(GtkAboutDialog *,const char * link,void *)127 static void wxGtkAboutDialogOnLink(GtkAboutDialog*, const char* link, void*)
128 {
129     wxLaunchDefaultBrowser(wxGTK_CONV_BACK_SYS(link));
130 }
131 }
132 #endif
133 
wxAboutBox(const wxAboutDialogInfo & info,wxWindow * parent)134 void wxAboutBox(const wxAboutDialogInfo& info, wxWindow* parent)
135 {
136     // don't create another dialog if one is already present
137     if ( !gs_aboutDialog )
138         gs_aboutDialog = GTK_ABOUT_DIALOG(gtk_about_dialog_new());
139 
140     GtkAboutDialog * const dlg = gs_aboutDialog;
141     gtk_about_dialog_set_program_name(dlg, wxGTK_CONV_SYS(info.GetName()));
142     if ( info.HasVersion() )
143         gtk_about_dialog_set_version(dlg, wxGTK_CONV_SYS(info.GetVersion()));
144     else
145         gtk_about_dialog_set_version(dlg, NULL);
146     if ( info.HasCopyright() )
147         gtk_about_dialog_set_copyright(dlg, wxGTK_CONV_SYS(info.GetCopyrightToDisplay()));
148     else
149         gtk_about_dialog_set_copyright(dlg, NULL);
150     if ( info.HasDescription() )
151         gtk_about_dialog_set_comments(dlg, wxGTK_CONV_SYS(info.GetDescription()));
152     else
153         gtk_about_dialog_set_comments(dlg, NULL);
154     if ( info.HasLicence() )
155         gtk_about_dialog_set_license(dlg, wxGTK_CONV_SYS(info.GetLicence()));
156     else
157         gtk_about_dialog_set_license(dlg, NULL);
158 
159     wxIcon icon = info.GetIcon();
160     if ( icon.IsOk() )
161         gtk_about_dialog_set_logo(dlg, info.GetIcon().GetPixbuf());
162 
163     if ( info.HasWebSite() )
164     {
165 #ifdef __WXGTK3__
166         g_signal_connect(dlg, "activate-link", G_CALLBACK(activate_link), dlg);
167 #else
168         // NB: must be called before gtk_about_dialog_set_website() as
169         //     otherwise it has no effect (although GTK+ docs don't mention
170         //     this...)
171         gtk_about_dialog_set_url_hook(wxGtkAboutDialogOnLink, NULL, NULL);
172 #endif
173 
174         gtk_about_dialog_set_website(dlg, wxGTK_CONV_SYS(info.GetWebSiteURL()));
175         gtk_about_dialog_set_website_label
176         (
177             dlg,
178             wxGTK_CONV_SYS(info.GetWebSiteDescription())
179         );
180     }
181     else
182     {
183         gtk_about_dialog_set_website(dlg, NULL);
184         gtk_about_dialog_set_website_label(dlg, NULL);
185 #ifdef __WXGTK3__
186         g_signal_connect(dlg, "activate-link", G_CALLBACK(activate_link), NULL);
187 #else
188         gtk_about_dialog_set_url_hook(NULL, NULL, NULL);
189 #endif
190     }
191 
192     if ( info.HasDevelopers() )
193         gtk_about_dialog_set_authors(dlg, GtkArray(info.GetDevelopers()));
194     else
195         gtk_about_dialog_set_authors(dlg, GtkArray());
196     if ( info.HasDocWriters() )
197         gtk_about_dialog_set_documenters(dlg, GtkArray(info.GetDocWriters()));
198     else
199         gtk_about_dialog_set_documenters(dlg, GtkArray());
200     if ( info.HasArtists() )
201         gtk_about_dialog_set_artists(dlg, GtkArray(info.GetArtists()));
202     else
203         gtk_about_dialog_set_artists(dlg, GtkArray());
204 
205     wxString transCredits;
206     if ( info.HasTranslators() )
207     {
208         const wxArrayString& translators = info.GetTranslators();
209         const size_t count = translators.size();
210         for ( size_t n = 0; n < count; n++ )
211         {
212             transCredits << translators[n] << wxT('\n');
213         }
214     }
215     else // no translators explicitly specified
216     {
217         // maybe we have translator credits in the message catalog?
218         wxString translator = _("translator-credits");
219 
220         // gtk_about_dialog_set_translator_credits() is smart enough to
221         // detect if "translator-credits" is untranslated and hide the
222         // translators tab in that case, however it will still show the
223         // "credits" button, (at least GTK 2.10.6) even if there are no
224         // credits informations at all, so we still need to do the check
225         // ourselves
226         if ( translator != wxT("translator-credits") ) // untranslated!
227             transCredits = translator;
228     }
229 
230     if ( !transCredits.empty() )
231         gtk_about_dialog_set_translator_credits(dlg, wxGTK_CONV_SYS(transCredits));
232     else
233         gtk_about_dialog_set_translator_credits(dlg, NULL);
234 
235     g_signal_connect(dlg, "response",
236                         G_CALLBACK(wxGtkAboutDialogOnClose), NULL);
237 
238     GtkWindow* gtkParent = NULL;
239     if (parent && parent->m_widget)
240         gtkParent = (GtkWindow*)gtk_widget_get_ancestor(parent->m_widget, GTK_TYPE_WINDOW);
241     gtk_window_set_transient_for(GTK_WINDOW(dlg), gtkParent);
242 
243     gtk_window_present(GTK_WINDOW(dlg));
244 }
245 
246 #endif // wxUSE_ABOUTDLG
247