/* $Id: view.c,v 1.30 2011/03/03 06:16:10 khorben Exp $ */ static char const _copyright[] = "Copyright (c) 2011 Pierre Pronchery "; /* This file is part of DeforaOS Desktop Browser */ static char const _license[] = "view is free software; you can redistribute it and/or modify it under the\n" "terms of the GNU General Public License version 3 as published by the Free\n" "Software Foundation.\n" "\n" "view is distributed in the hope that it will be useful, but WITHOUT ANY\n" "WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS\n" "FOR A PARTICULAR PURPOSE. See the GNU General Public License for more\n" "details.\n" "\n" "You should have received a copy of the GNU General Public License along with\n" "view; if not, see .\n"; #include #include #include #include #include #include #include #include #include #include #include #include "mime.h" #include "../config.h" #define _(string) gettext(string) #define N_(string) (string) #include "common.c" /* constants */ #ifndef PREFIX # define PREFIX "/usr/local" #endif #ifndef DATADIR # define DATADIR PREFIX "/share" #endif #ifndef LOCALEDIR # define LOCALEDIR DATADIR "/locale" #endif /* View */ /* private */ /* types */ typedef struct _View { char * pathname; /* widgets */ GtkWidget * window; GtkWidget * ab_window; } View; /* constants */ #ifndef EMBEDDED static char const * _view_authors[] = { "Pierre Pronchery ", NULL }; #endif /* prototypes */ static View * _view_new(char const * path); static View * _view_new_open(void); static void _view_delete(View * view); /* useful */ static int _view_error(View * view, char const * message, int ret); /* callbacks */ static gboolean _on_closex(gpointer data); #ifndef EMBEDDED static void _on_file_edit(gpointer data); static void _on_file_open_with(gpointer data); static void _on_file_close(gpointer data); static void _on_help_about(gpointer data); #else static void _on_close(gpointer data); static void _on_edit(gpointer data); static void _on_open_with(gpointer data); #endif /* constants */ #ifndef EMBEDDED static DesktopMenu _view_menu_file[] = { { N_("Open _with..."), G_CALLBACK(_on_file_open_with), NULL, 0, 0 }, { "", NULL, NULL, 0, 0 }, { N_("_Close"), G_CALLBACK(_on_file_close), GTK_STOCK_CLOSE, GDK_CONTROL_MASK, GDK_W }, { NULL, NULL, NULL, 0, 0 } }; static DesktopMenu _view_menu_file_edit[] = { { N_("_Edit"), G_CALLBACK(_on_file_edit), GTK_STOCK_EDIT, GDK_CONTROL_MASK, GDK_E }, { N_("Open _with..."), G_CALLBACK(_on_file_open_with), NULL, 0, 0 }, { "", NULL, NULL, 0, 0 }, { N_("_Close"), G_CALLBACK(_on_file_close), GTK_STOCK_CLOSE, GDK_CONTROL_MASK, GDK_W }, { NULL, NULL, NULL, 0, 0 } }; static DesktopMenu _view_menu_help[] = { # if GTK_CHECK_VERSION(2, 6, 0) { N_("_About"), G_CALLBACK(_on_help_about), GTK_STOCK_ABOUT, 0, 0 }, # else { N_("_About"), G_CALLBACK(_on_help_about), NULL, 0, 0 }, # endif { NULL, NULL, NULL, 0, 0 } }; static DesktopMenubar _view_menubar[] = { { N_("_File"), _view_menu_file }, { N_("_Help"), _view_menu_help }, { NULL, NULL } }; static DesktopMenubar _view_menubar_edit[] = { { N_("_File"), _view_menu_file_edit }, { N_("_Help"), _view_menu_help }, { NULL, NULL } }; #else static DesktopToolbar _view_toolbar[] = { { N_("Open with..."), G_CALLBACK(_on_open_with), GTK_STOCK_OPEN, GDK_CONTROL_MASK, GDK_O, NULL }, { N_("Edit"), G_CALLBACK(_on_edit), GTK_STOCK_EDIT, GDK_CONTROL_MASK, GDK_E, NULL }, { "", NULL, NULL, 0, 0, NULL }, { N_("Close"), G_CALLBACK(_on_close), GTK_STOCK_CLOSE, GDK_CONTROL_MASK, GDK_W, NULL }, { NULL, NULL, NULL, 0, 0, NULL } }; #endif /* EMBEDDED */ /* variables */ static Mime * _mime = NULL; static unsigned int _view_cnt = 0; /* functions */ /* view_new */ static GtkWidget * _new_image(View * view, char const * path); static GtkWidget * _new_text(View * view, char const * path); static View * _view_new(char const * pathname) { View * view; struct stat st; char const * type; char buf[256]; GtkAccelGroup * group; GtkWidget * vbox; GtkWidget * widget; if((view = malloc(sizeof(*view))) == NULL) return NULL; /* FIXME handle error */ view->window = NULL; view->ab_window = NULL; _view_cnt++; if((view->pathname = strdup(pathname)) == NULL || lstat(pathname, &st) != 0) { _view_error(view, strerror(errno), 2); return NULL; } if(_mime == NULL) _mime = mime_new(NULL); if((type = mime_type(_mime, pathname)) == NULL) { _view_error(view, _("Unknown file type"), 2); return NULL; } group = gtk_accel_group_new(); view->window = gtk_window_new(GTK_WINDOW_TOPLEVEL); gtk_window_add_accel_group(GTK_WINDOW(view->window), group); snprintf(buf, sizeof(buf), "%s%s", _("View - "), pathname); gtk_window_set_title(GTK_WINDOW(view->window), buf); g_signal_connect_swapped(view->window, "delete-event", G_CALLBACK( _on_closex), view); vbox = gtk_vbox_new(FALSE, 0); #ifndef EMBEDDED widget = desktop_menubar_create(mime_get_handler(_mime, type, "edit") != NULL ? _view_menubar_edit : _view_menubar, view, group); #else widget = desktop_toolbar_create(_view_toolbar, view, group); if(mime_get_handler(_mime, type, "edit") == NULL) gtk_widget_set_sensitive(GTK_WIDGET(_view_toolbar[1].widget), FALSE); #endif gtk_box_pack_start(GTK_BOX(vbox), widget, FALSE, FALSE, 0); if(strncmp(type, "image/", 6) == 0) { if((widget = _new_image(view, pathname)) == NULL) return NULL; } else if(strncmp(type, "text/", 5) == 0) { widget = _new_text(view, pathname); gtk_window_set_default_size(GTK_WINDOW(view->window), 600, 400); } else { _view_error(view, _("Unable to view file type"), 2); return NULL; } gtk_box_pack_start(GTK_BOX(vbox), widget, TRUE, TRUE, 0); gtk_container_add(GTK_CONTAINER(view->window), vbox); gtk_widget_show_all(view->window); return view; } static GtkWidget * _new_image(View * view, char const * path) { GtkWidget * window; GError * error = NULL; GdkPixbuf * pixbuf; GtkWidget * widget; int pw; int ph; GdkScreen * screen; gint monitor; GdkRectangle rect; window = gtk_scrolled_window_new(NULL, NULL); gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(window), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); if((pixbuf = gdk_pixbuf_new_from_file_at_size(path, -1, -1, &error)) == NULL) { _view_error(view, error->message, 1); return NULL; } widget = gtk_image_new_from_pixbuf(pixbuf); gtk_scrolled_window_add_with_viewport(GTK_SCROLLED_WINDOW(window), widget); pw = gdk_pixbuf_get_width(pixbuf) + 4; ph = gdk_pixbuf_get_height(pixbuf) + 4; /* get the current monitor size */ screen = gdk_screen_get_default(); #if GTK_CHECK_VERSION(2, 14, 0) gtk_widget_realize(view->window); monitor = gdk_screen_get_monitor_at_window(screen, gtk_widget_get_window(view->window)); #else monitor = 0; /* XXX hard-coded */ #endif gdk_screen_get_monitor_geometry(screen, monitor, &rect); /* set an upper bound to the size of the window */ gtk_widget_set_size_request(window, min(pw, rect.width), min(ph, rect.height)); return window; } static GtkWidget * _new_text(View * view, char const * path) { GtkWidget * widget; GtkWidget * text; PangoFontDescription * desc; FILE * fp; GtkTextBuffer * tbuf; GtkTextIter iter; char buf[BUFSIZ]; size_t len; widget = gtk_scrolled_window_new(NULL, NULL); gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(widget), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); text = gtk_text_view_new(); gtk_text_view_set_cursor_visible(GTK_TEXT_VIEW(text), FALSE); gtk_text_view_set_editable(GTK_TEXT_VIEW(text), FALSE); gtk_text_view_set_wrap_mode(GTK_TEXT_VIEW(text), GTK_WRAP_WORD_CHAR); desc = pango_font_description_new(); pango_font_description_set_family(desc, "monospace"); gtk_widget_modify_font(text, desc); pango_font_description_free(desc); gtk_container_add(GTK_CONTAINER(widget), text); if((fp = fopen(path, "r")) == NULL) { _view_error(view, strerror(errno), 0); return widget; } tbuf = gtk_text_view_get_buffer(GTK_TEXT_VIEW(text)); while((len = fread(buf, sizeof(char), sizeof(buf), fp)) > 0) { gtk_text_buffer_get_end_iter(tbuf, &iter); gtk_text_buffer_insert(tbuf, &iter, buf, len); } fclose(fp); return widget; } /* view_new_open */ static View * _view_new_open(void) { View * ret; GtkWidget * dialog; char * pathname = NULL; dialog = gtk_file_chooser_dialog_new(_("View file..."), NULL, GTK_FILE_CHOOSER_ACTION_OPEN, GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL, GTK_STOCK_OPEN, GTK_RESPONSE_ACCEPT, NULL); if(gtk_dialog_run(GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT) pathname = gtk_file_chooser_get_filename(GTK_FILE_CHOOSER( dialog)); gtk_widget_destroy(dialog); if(pathname == NULL) return NULL; ret = _view_new(pathname); free(pathname); return ret; } /* view_delete */ static void _view_delete(View * view) { free(view->pathname); if(view->ab_window != NULL) gtk_widget_destroy(view->ab_window); if(view->window != NULL) gtk_widget_destroy(view->window); free(view); if(--_view_cnt == 0) gtk_main_quit(); } /* useful */ /* view_error * POST view is deleted if ret != 0 */ static void _error_response(GtkWidget * widget, gint arg, gpointer data); static int _view_error(View * view, char const * message, int ret) { GtkWidget * dialog; dialog = gtk_message_dialog_new(view != NULL && view->window != NULL ? GTK_WINDOW(view->window) : NULL, GTK_DIALOG_DESTROY_WITH_PARENT, GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE, "%s", #if GTK_CHECK_VERSION(2, 6, 0) _("Error")); gtk_message_dialog_format_secondary_text(GTK_MESSAGE_DIALOG(dialog), "%s", #endif message); gtk_window_set_title(GTK_WINDOW(dialog), _("Error")); g_signal_connect(G_OBJECT(dialog), "response", G_CALLBACK( _error_response), (ret != 0) ? view : NULL); gtk_widget_show(dialog); return ret; } static void _error_response(GtkWidget * widget, gint arg, gpointer data) { View * view = data; if(view != NULL) _view_delete(view); gtk_widget_destroy(widget); } /* callbacks */ static gboolean _on_closex(gpointer data) { View * view = data; _view_delete(view); if(_view_cnt == 0) gtk_main_quit(); return FALSE; } #ifndef EMBEDDED /* on_file_edit */ static void _on_file_edit(gpointer data) { View * view = data; if(mime_action(_mime, "edit", view->pathname) != 0) _view_error(view, _("Could not edit file"), 0); } /* on_file_open_with */ static void _on_file_open_with(gpointer data) { View * view = data; GtkWidget * dialog; char * filename = NULL; pid_t pid; dialog = gtk_file_chooser_dialog_new(_("Open with..."), GTK_WINDOW(view->window), GTK_FILE_CHOOSER_ACTION_OPEN, GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL, GTK_STOCK_OPEN, GTK_RESPONSE_ACCEPT, NULL); if(gtk_dialog_run(GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT) filename = gtk_file_chooser_get_filename(GTK_FILE_CHOOSER( dialog)); gtk_widget_destroy(dialog); if(filename == NULL) return; if((pid = fork()) == -1) _view_error(view, "fork", 0); else if(pid == 0) { if(close(0) != 0) _view_error(NULL, "stdin", 0); execlp(filename, filename, view->pathname, NULL); _view_error(NULL, filename, 0); exit(2); } g_free(filename); } /* on_file_close */ static void _on_file_close(gpointer data) { _on_closex(data); } /* on_help_about */ static gboolean _about_on_closex(gpointer data); static void _on_help_about(gpointer data) { View * view = data; if(view->ab_window != NULL) { gtk_widget_show(view->ab_window); return; } view->ab_window = desktop_about_dialog_new(); gtk_window_set_transient_for(GTK_WINDOW(view->ab_window), GTK_WINDOW( view->window)); desktop_about_dialog_set_authors(view->ab_window, _view_authors); desktop_about_dialog_set_copyright(view->ab_window, _copyright); desktop_about_dialog_set_logo_icon_name(view->ab_window, "system-file-manager"); desktop_about_dialog_set_license(view->ab_window, _license); desktop_about_dialog_set_name(view->ab_window, PACKAGE); desktop_about_dialog_set_version(view->ab_window, VERSION); g_signal_connect_swapped(G_OBJECT(view->ab_window), "delete-event", G_CALLBACK(_about_on_closex), view); gtk_widget_show(view->ab_window); } static gboolean _about_on_closex(gpointer data) { View * view = data; gtk_widget_hide(view->ab_window); return TRUE; } #else /* on_close */ static void _on_close(gpointer data) { _on_closex(data); } static void _on_edit(gpointer data) { View * view = data; if(mime_action(_mime, "edit", view->pathname) != 0) _view_error(view, _("Could not edit file"), 0); } static void _on_open_with(gpointer data) { /* FIXME implement */ } #endif /* EMBEDDED */ /* usage */ static int _usage(void) { fputs(_("Usage: view file...\n"), stderr); return 1; } /* main */ int main(int argc, char * argv[]) { int o; int i; setlocale(LC_ALL, ""); bindtextdomain(PACKAGE, LOCALEDIR); textdomain(PACKAGE); gtk_init(&argc, &argv); while((o = getopt(argc, argv, "")) != -1) switch(o) { default: return _usage(); } if(optind == argc) _view_new_open(); else for(i = optind; i < argc; i++) _view_new(argv[i]); if(_view_cnt) gtk_main(); return 0; }