dockapps/wmtrash/wmtrash.c
2016-12-27 20:53:45 +00:00

453 lines
13 KiB
C

/*
* this file is part of the "wmtrash" project
* Copyright (C) 2004 by Jean Philippe GUILLEMIN <jp.guillemin@free.fr>
* license: This software is under GPL license
* rev: 0.2
*/
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <getopt.h>
#include <sysexits.h>
#include <gtk/gtk.h>
#include <gdk/gdk.h>
#include <gdk/gdkx.h>
#include "wmtrash.h"
// Start main *******************************************************************
int main( int argc, char *argv[] ) {
// Here we parse command line args and configfile *******************************************************************
GtkWidget *dockapp;
static char *defaultcf;
static char *configfile;
static char *homecf;
static char *homedir;
homecf = malloc(MEDIUM_STRING);
configfile = malloc(MEDIUM_STRING);
defaultcf = malloc(MEDIUM_STRING);
strncpy(defaultcf, __CONFPATH, MEDIUM_STRING);
strcat(defaultcf, "/");
strcat(defaultcf, __CONFFILE);
homedir = malloc(SHORT_STRING);
homedir = getenv("HOME");
strncpy(homecf, homedir, SHORT_STRING);
strcat(homecf, "/");
strcat(homecf, ".wmtrash.cf");
int test1, test2;
if ((test1 = access(homecf, F_OK)) == -1){
if ((test2 = fcopy(defaultcf, homecf)) == EXIT_FAILURE){
fprintf (stderr,"Error creating config file %s !\n",homecf);
}
}
int i;
if (argc < 2){
configfile = homecf;
}else{
while ((i = getopt(argc, argv, "hc:")) != EOF){
switch (i){
case 'c': /* config file */
strncpy(configfile, optarg, MEDIUM_STRING);
break;
case 'h': usage(homecf, defaultcf); exit (EXIT_SUCCESS);
}
}
}
gtk_init(&argc, &argv);
dockapp = (GtkWidget *) build_dockapp(configfile);
gtk_widget_show_all (dockapp);
gtk_main ();
return(0);
} // end main
GtkWidget * build_dockapp(char *configfile) {
static GtkWidget *mainwin;
static GtkWidget *mainbox;
static GtkWidget *box;
static GtkWidget *pixmap;
static GdkBitmap *mask;
static GtkWidget *pixmap_widget;
static struct wmtrash *wmtrash;
wmtrash = malloc(sizeof(struct wmtrash));
memset(wmtrash, 0, sizeof(struct wmtrash));
char *image_path_file;
strncpy (wmtrash->param_img, "trashicon", MEDIUM_STRING);
strncpy (wmtrash->param_fm, "filemanager", MEDIUM_STRING);
strncpy (wmtrash->param_path, "trashpath", MEDIUM_STRING);
strncpy (wmtrash->configfile, configfile, MEDIUM_STRING);
image_path_file = malloc(LONG_STRING);
parse_from_config(wmtrash->configfile, wmtrash->param_img, wmtrash->imagefile);
sprintf(image_path_file, "%s/%s", __ICONPATH, wmtrash->imagefile);
parse_from_config(wmtrash->configfile, wmtrash->param_fm, wmtrash->fm);
parse_from_config(wmtrash->configfile, wmtrash->param_path, wmtrash->path);
// GTK stuff *******************************************************************
gtk_widget_destroy(mainwin);
mainwin = gtk_window_new(GTK_WINDOW_TOPLEVEL);
gtk_widget_realize(mainwin);
mainbox = create_main_dockapp_icon_window(mainwin, 52);
box = gtk_event_box_new();
pixmap = (gpointer) gdk_pixmap_create_from_xpm (mainwin->window, &mask,
NULL, image_path_file);
pixmap_widget = gtk_pixmap_new((gpointer) pixmap, mask);
gtk_container_add(GTK_CONTAINER(box), pixmap_widget);
gtk_container_add (GTK_CONTAINER (mainbox), box);
gtk_signal_connect (GTK_OBJECT(box), "button_press_event", GTK_SIGNAL_FUNC(launchonclick), wmtrash);
free(image_path_file);
return mainwin;
}
// Function create_main_dockapp_icon_window (c) 2000, Alexey Vyskubov <alexey@pepper.spb.ru>
GtkWidget *create_main_dockapp_icon_window
(GtkWidget *mw,
unsigned int s)
{
GtkWidget *dockappbox; // This will become icon box
Window xmw;
XWMHints *wm_hints;
xmw = GDK_WINDOW_XWINDOW(mw->window);
dockappbox = gtk_event_box_new();
gtk_widget_set_usize(dockappbox, s, s);
gtk_container_add (GTK_CONTAINER (mw), dockappbox);
gtk_widget_realize(dockappbox);
// Time for game with Xlib
wm_hints = XAllocWMHints();
wm_hints->window_group = xmw;
wm_hints->icon_window = GDK_WINDOW_XWINDOW(dockappbox->window);
wm_hints->icon_x = 0;
wm_hints->icon_y = 0;
wm_hints->initial_state = WithdrawnState;
wm_hints->flags = StateHint |
IconPositionHint |
WindowGroupHint |
IconWindowHint;
XSetWMHints(GDK_DISPLAY(), xmw, wm_hints);
XFree(wm_hints);
return dockappbox;
} // end create_main_dockapp_icon_window
/* the function to execute command when left button is clicked *************************************
or change the command and pixmap icon when right button is 2-clicked */
void launchonclick
(GtkWidget *event_box,
GdkEventButton *event,
struct wmtrash *data)
{
char *cmd;
cmd=malloc(BIG_STRING);
memset(cmd, 0, BIG_STRING);
/* left button of mouse is clicked */
if(event->button == 1){
strcat (cmd, data->fm);
strcat (cmd, " ");
strcat (cmd, data->path);
strcat (cmd, "&");
//fprintf(stderr, cmd); //debug
int output = system(cmd);
if (output){
fprintf(stderr, "error launching command in function \"launchonclick\"\n");
}
return;
}
if((event->button == 3)&&(event->type == GDK_2BUTTON_PRESS)) {
strcat (cmd, "rm -rf");
strcat (cmd, " ");
strcat (cmd, data->path);
strcat (cmd, "/* ");
strcat (cmd, data->path);
strcat (cmd, "/.*");
//fprintf(stderr, cmd); //debug
int output = system(cmd);
if (output){
fprintf(stderr, "error launching command in function \"launchonclick\"\n");
}
return;
}
/* right button of mouse is 2-clicked */
if((event->button == 2)&&(event->type == GDK_2BUTTON_PRESS)){
/* create a new text entry window for entering the new command */
GtkWidget *dialog, *table, *vbox;
GtkWidget *entry1, *entry2, *button;
// window parameters
dialog = gtk_window_new (GTK_WINDOW_TOPLEVEL);
gtk_window_set_title (GTK_WINDOW (dialog), "Change command & pixmap");
gtk_container_set_border_width (GTK_CONTAINER (dialog), 5);
table = gtk_table_new(2,2, TRUE);
gtk_table_set_col_spacings(GTK_TABLE (table), 5);
gtk_table_set_row_spacings(GTK_TABLE (table), 5);
vbox = gtk_vbox_new (FALSE, 0);
// entry box parameters
entry1 = gtk_entry_new ();
entry2 = gtk_entry_new ();
gtk_entry_set_max_length (GTK_ENTRY (entry1), 120);
gtk_entry_set_max_length (GTK_ENTRY (entry2), 120);
gtk_signal_connect (GTK_OBJECT (entry1), "changed", GTK_SIGNAL_FUNC (enter_callback1), data);
gtk_signal_connect (GTK_OBJECT (entry2), "changed", GTK_SIGNAL_FUNC (enter_callback2), data);
gtk_entry_set_text (GTK_ENTRY (entry1), data->fm);
gtk_entry_set_text (GTK_ENTRY (entry2), data->path);
gtk_editable_set_editable (GTK_EDITABLE (entry1), TRUE);
gtk_entry_set_visibility (GTK_ENTRY (entry1), TRUE);
gtk_editable_set_editable (GTK_EDITABLE (entry2), TRUE);
gtk_entry_set_visibility (GTK_ENTRY (entry2), TRUE);
// button parameters
button = gtk_button_new_with_label("OK");
//label = gtk_label_new ("OK");
gtk_signal_connect (GTK_OBJECT (button), "clicked", GTK_SIGNAL_FUNC (close_callback), data);
gtk_container_add (GTK_CONTAINER (vbox), entry1);
gtk_container_add (GTK_CONTAINER (vbox), entry2);
gtk_table_attach_defaults(GTK_TABLE(table), vbox, 0,1, 0,2);
gtk_table_attach_defaults(GTK_TABLE(table), button, 1,2, 0,2);
gtk_container_add (GTK_CONTAINER (dialog), table);
gtk_widget_show_all(dialog);
return;
}
free(cmd);
} // end launchonclick
/* some callback to handle change of values in entry boxes************************* */
void enter_callback1(
GtkWidget *entry,
struct wmtrash *data){
char *entry_text;
entry_text = malloc(BIG_STRING);
memset(entry_text, 0, BIG_STRING);
strncpy (entry_text, gtk_entry_get_text (GTK_ENTRY (entry)), BIG_STRING);
memset(data->fm, 0, BIG_STRING);
strncpy (data->fm, entry_text, BIG_STRING);
}
void enter_callback2(
GtkWidget *entry,
struct wmtrash *data){
char *entry_text;
entry_text = malloc(BIG_STRING);
memset(entry_text, 0, BIG_STRING);
strncpy (entry_text, gtk_entry_get_text (GTK_ENTRY (entry)), BIG_STRING);
memset(data->path, 0, BIG_STRING);
strncpy (data->path, entry_text, BIG_STRING);
}
void close_callback(
GtkWidget *entry,
struct wmtrash *data){
GtkWidget *parent;
GtkWidget *dockapp;
replace_value_in_config(data->configfile, data->param_fm, data->fm);
replace_value_in_config(data->configfile, data->param_path, data->path);
parent = gtk_widget_get_ancestor(entry, GTK_TYPE_WINDOW);
gtk_widget_destroy(parent); // close the dialog box
dockapp = (GtkWidget *) build_dockapp(data->configfile); // will redraw icons
gtk_widget_show_all (dockapp);
}
// the function to parse the config file for parameters *********************************************
int parse_from_config
(char *filename,
char *param,
char *value)
{
int c = 0;
FILE *file;
char *sep,
*buffer,
*line,
*val;
line = malloc(LONG_STRING);
buffer = malloc(MEDIUM_STRING);
sep = malloc(SHORT_STRING);
val = malloc(MEDIUM_STRING);
if((file=fopen(filename, "rb")) == 0) {
fprintf (stderr,"Error opening file %s !\n",filename);
exit(EXIT_FAILURE);
}
while (fgets (line, LONG_STRING, file) !=NULL) {
if (strstr (line, "[") !=0) continue;
if (strstr (line, "#") !=0) continue;
memset(val, 0, MEDIUM_STRING);
c = sscanf (line, "%s %s %s", buffer, sep, val);
if (strncmp (param, buffer, MEDIUM_STRING) == 0) {
memset(value, 0, BIG_STRING);
strncpy(value,line + strlen(buffer) + strlen(sep) +2, BIG_STRING);
//fprintf(stderr,"%s\n", value);
char *p = strchr (value, '\n');
if (p){
*p = 0;
}
fclose (file);
return(EXIT_SUCCESS);
}
memset(line, 0, LONG_STRING);
}
fclose (file);
free(sep);
free(buffer);
free(line);
free(val);
return(EXIT_FAILURE);
} // end parse_from_config
// the function to parse the config file to replace the value of a given parameter ****************************
int replace_value_in_config
(char *filename,
char *param,
char *new_value)
{
int c = 0;
FILE *file;
char *bigbuffer,
*sep,
*buffer,
*line,
*old_value;
bigbuffer = malloc(EXTRALONG_STRING);
line = malloc(LONG_STRING);
buffer = malloc(MEDIUM_STRING);
sep = malloc(MEDIUM_STRING);
old_value = malloc(BIG_STRING);
if((file=fopen(filename, "rb")) == 0) {
fprintf (stderr,"Error opening file %s !\n",filename);
exit(EXIT_FAILURE);
}
memset(bigbuffer, 0, EXTRALONG_STRING);
while (fgets (line, LONG_STRING, file) !=NULL) {
memset(buffer, 0, MEDIUM_STRING);
c = sscanf (line, "%s %s %s\n", buffer, sep, old_value);
if (strncmp (param, buffer, MEDIUM_STRING) == 0) {
memset(line, 0, LONG_STRING);
sprintf(line, "%s %s %s\n", buffer, sep, new_value);
}
strncat (bigbuffer, line, LONG_STRING);
memset(line, 0, LONG_STRING);
}
fclose (file);
if((file=fopen(filename, "w+")) == 0) {
fprintf (stderr,"Error opening file %s !\n",filename);
exit(EXIT_FAILURE);
}
fprintf(file,bigbuffer);
fclose (file);
memset(bigbuffer, 0, EXTRALONG_STRING);
free(bigbuffer);
free(sep);
free(buffer);
free(line);
free(old_value);
return(EXIT_SUCCESS);
} // end replace_value_in_config *************************************************************************
int usage(char *homecf, char *defaultcf) {
fprintf(stdout,"\nwmtrash v0.2 : Desktop trash dockapp\n");
fprintf(stdout, "Usage : \n");
fprintf(stdout,"- single LEFT click : browse trash\n");
fprintf(stdout,"- double MIDDLE click : change location & filemanager\n");
fprintf(stdout,"- double RIGHT click : empty trash\n");
fprintf(stdout, "---------------\n");
fprintf(stdout, "How to start it : \n"
"\t wmtrash [-c path_to_configfile] use a custom config\n"
"\t wmtrash [-h] display this help\n");
fprintf(stdout, "---------------\n");
fprintf(stdout,"home config file is %s\n", homecf);
fprintf(stdout,"default config file is %s\n",defaultcf);
return(EXIT_SUCCESS);
} // end usage
//*************************************************************************
// this one just copy sfile into dfile :)
int fcopy( char *sfile,
char *dfile){
int c; /* Character read/written between files */
FILE *IPFile; /* Pointer to the I/P file. FILE is a
structure defined in <stdio.h> */
FILE *OPFile;
/* Open the file - */
if ((IPFile = fopen(sfile,"r")) == 0) {
fprintf (stderr,"Error opening file %s !\n",sfile);
return (EXIT_FAILURE);
}
if ((OPFile = fopen(dfile,"w")) == 0) {
fprintf (stderr,"Error opening file %s !\n",dfile);
return (EXIT_FAILURE);
}
/* Read one character at a time, checking
for the End of File. EOF is defined
in <stdio.h> as -1 */
while ((c = fgetc(IPFile)) != EOF){
fputc(c, OPFile); /* O/P the character */
}
fclose(IPFile); /* Close the files. */
fclose(OPFile); /* Close the files. */
return (EXIT_SUCCESS);
} // end fcopy