hexchat/src/fe-gtk/textgui.c
TingPing 449553ffb0 Replace xtext's transparency with full window transparency
Xtext's transparency barely worked on windows, didn't work on any modern
linux wm and used fake transparency.
This uses gtk's built in window opacity that works on more systems and
is real transparency.
Text area only transparency may return with a transition to cairo, if it
works on Windows.
2013-09-27 17:40:05 -04:00

444 lines
11 KiB
C

/* X-Chat
* Copyright (C) 1998 Peter Zelezny.
*
* 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 Free Software
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include "fe-gtk.h"
#include "../common/hexchat.h"
#include "../common/hexchatc.h"
#include "../common/cfgfiles.h"
#include "../common/outbound.h"
#include "../common/fe.h"
#include "../common/text.h"
#include "gtkutil.h"
#include "xtext.h"
#include "maingui.h"
#include "palette.h"
#include "textgui.h"
extern struct text_event te[];
extern char *pntevts_text[];
extern char *pntevts[];
static GtkWidget *pevent_dialog = NULL, *pevent_dialog_twid,
*pevent_dialog_entry,
*pevent_dialog_list, *pevent_dialog_hlist;
enum
{
COL_EVENT_NAME,
COL_EVENT_TEXT,
COL_ROW,
N_COLUMNS
};
/* this is only used in xtext.c for indented timestamping */
int
xtext_get_stamp_str (time_t tim, char **ret)
{
return get_stamp_str (prefs.hex_stamp_text_format, tim, ret);
}
static void
PrintTextLine (xtext_buffer *xtbuf, unsigned char *text, int len, int indent, time_t timet)
{
unsigned char *tab, *new_text;
int leftlen;
if (len == 0)
len = 1;
if (!indent)
{
if (prefs.hex_stamp_text)
{
int stamp_size;
char *stamp;
if (timet == 0)
timet = time (0);
stamp_size = get_stamp_str (prefs.hex_stamp_text_format, timet, &stamp);
new_text = malloc (len + stamp_size + 1);
memcpy (new_text, stamp, stamp_size);
g_free (stamp);
memcpy (new_text + stamp_size, text, len);
gtk_xtext_append (xtbuf, new_text, len + stamp_size);
free (new_text);
} else
gtk_xtext_append (xtbuf, text, len);
return;
}
tab = strchr (text, '\t');
if (tab && tab < (text + len))
{
leftlen = tab - text;
gtk_xtext_append_indent (xtbuf,
text, leftlen, tab + 1, len - (leftlen + 1), timet);
} else
gtk_xtext_append_indent (xtbuf, 0, 0, text, len, timet);
}
void
PrintTextRaw (void *xtbuf, unsigned char *text, int indent, time_t stamp)
{
char *last_text = text;
int len = 0;
int beep_done = FALSE;
/* split the text into separate lines */
while (1)
{
switch (*text)
{
case 0:
PrintTextLine (xtbuf, last_text, len, indent, stamp);
return;
case '\n':
PrintTextLine (xtbuf, last_text, len, indent, stamp);
text++;
if (*text == 0)
return;
last_text = text;
len = 0;
break;
case ATTR_BEEP:
*text = ' ';
if (!beep_done) /* beeps may be slow, so only do 1 per line */
{
beep_done = TRUE;
if (!prefs.hex_input_filter_beep)
gdk_beep ();
}
default:
text++;
len++;
}
}
}
static void
pevent_dialog_close (GtkWidget *wid, gpointer arg)
{
pevent_dialog = NULL;
pevent_save (NULL);
}
static void
pevent_dialog_update (GtkWidget * wid, GtkWidget * twid)
{
int len, m;
const char *text;
char *out;
int sig;
GtkTreeIter iter;
GtkListStore *store;
if (!gtkutil_treeview_get_selected (GTK_TREE_VIEW (pevent_dialog_list),
&iter, COL_ROW, &sig, -1))
return;
text = gtk_entry_get_text (GTK_ENTRY (wid));
len = strlen (text);
if (pevt_build_string (text, &out, &m) != 0)
{
fe_message (_("There was an error parsing the string"), FE_MSG_ERROR);
return;
}
if (m > (te[sig].num_args & 0x7f))
{
free (out);
out = malloc (4096);
snprintf (out, 4096,
_("This signal is only passed %d args, $%d is invalid"),
te[sig].num_args & 0x7f, m);
fe_message (out, FE_MSG_WARN);
free (out);
return;
}
store = (GtkListStore *)gtk_tree_view_get_model (GTK_TREE_VIEW (pevent_dialog_list));
gtk_list_store_set (store, &iter, COL_EVENT_TEXT, text, -1);
if (pntevts_text[sig])
free (pntevts_text[sig]);
if (pntevts[sig])
free (pntevts[sig]);
pntevts_text[sig] = malloc (len + 1);
memcpy (pntevts_text[sig], text, len + 1);
pntevts[sig] = out;
out = malloc (len + 2);
memcpy (out, text, len + 1);
out[len] = '\n';
out[len + 1] = 0;
check_special_chars (out, TRUE);
PrintTextRaw (GTK_XTEXT (twid)->buffer, out, 0, 0);
free (out);
/* save this when we exit */
prefs.save_pevents = 1;
}
static void
pevent_dialog_hfill (GtkWidget * list, int e)
{
int i = 0;
char *text;
GtkTreeIter iter;
GtkListStore *store;
store = (GtkListStore *)gtk_tree_view_get_model (GTK_TREE_VIEW (pevent_dialog_hlist));
gtk_list_store_clear (store);
while (i < (te[e].num_args & 0x7f))
{
text = _(te[e].help[i]);
i++;
if (text[0] == '\001')
text++;
gtk_list_store_insert_with_values (store, &iter, -1,
0, i,
1, text, -1);
}
}
static void
pevent_dialog_unselect (void)
{
gtk_entry_set_text (GTK_ENTRY (pevent_dialog_entry), "");
gtk_list_store_clear ((GtkListStore *)gtk_tree_view_get_model (GTK_TREE_VIEW (pevent_dialog_hlist)));
}
static void
pevent_dialog_select (GtkTreeSelection *sel, gpointer store)
{
char *text;
int sig;
GtkTreeIter iter;
if (!gtkutil_treeview_get_selected (GTK_TREE_VIEW (pevent_dialog_list),
&iter, COL_ROW, &sig, -1))
{
pevent_dialog_unselect ();
}
else
{
gtk_tree_model_get (store, &iter, COL_EVENT_TEXT, &text, -1);
gtk_entry_set_text (GTK_ENTRY (pevent_dialog_entry), text);
g_free (text);
pevent_dialog_hfill (pevent_dialog_hlist, sig);
}
}
static void
pevent_dialog_fill (GtkWidget * list)
{
int i;
GtkListStore *store;
GtkTreeIter iter;
store = (GtkListStore *)gtk_tree_view_get_model (GTK_TREE_VIEW (list));
gtk_list_store_clear (store);
i = NUM_XP;
do
{
i--;
gtk_list_store_insert_with_values (store, &iter, 0,
COL_EVENT_NAME, te[i].name,
COL_EVENT_TEXT, pntevts_text[i],
COL_ROW, i, -1);
}
while (i != 0);
}
static void
pevent_save_req_cb (void *arg1, char *file)
{
if (file)
pevent_save (file);
}
static void
pevent_save_cb (GtkWidget * wid, void *data)
{
if (data)
{
gtkutil_file_req (_("Print Texts File"), pevent_save_req_cb, NULL,
NULL, NULL, FRF_WRITE);
return;
}
pevent_save (NULL);
}
static void
pevent_load_req_cb (void *arg1, char *file)
{
if (file)
{
pevent_load (file);
pevent_make_pntevts ();
pevent_dialog_fill (pevent_dialog_list);
pevent_dialog_unselect ();
prefs.save_pevents = 1;
}
}
static void
pevent_load_cb (GtkWidget * wid, void *data)
{
gtkutil_file_req (_("Print Texts File"), pevent_load_req_cb, NULL, NULL, NULL, 0);
}
static void
pevent_ok_cb (GtkWidget * wid, void *data)
{
gtk_widget_destroy (pevent_dialog);
}
static void
pevent_test_cb (GtkWidget * wid, GtkWidget * twid)
{
int len, n;
char *out, *text;
for (n = 0; n < NUM_XP; n++)
{
text = _(pntevts_text[n]);
len = strlen (text);
out = malloc (len + 2);
memcpy (out, text, len + 1);
out[len] = '\n';
out[len + 1] = 0;
check_special_chars (out, TRUE);
PrintTextRaw (GTK_XTEXT (twid)->buffer, out, 0, 0);
free (out);
}
}
void
pevent_dialog_show ()
{
GtkWidget *vbox, *hbox, *tbox, *wid, *bh, *th;
GtkListStore *store, *hstore;
GtkTreeSelection *sel;
if (pevent_dialog)
{
mg_bring_tofront (pevent_dialog);
return;
}
pevent_dialog =
mg_create_generic_tab ("edit events", _("Edit Events"),
TRUE, FALSE, pevent_dialog_close, NULL,
600, 455, &vbox, 0);
wid = gtk_vpaned_new ();
th = gtk_vbox_new (0, 2);
bh = gtk_vbox_new (0, 2);
gtk_widget_show (th);
gtk_widget_show (bh);
gtk_paned_pack1 (GTK_PANED (wid), th, 1, 1);
gtk_paned_pack2 (GTK_PANED (wid), bh, 0, 1);
gtk_box_pack_start (GTK_BOX (vbox), wid, 1, 1, 0);
gtk_widget_show (wid);
store = gtk_list_store_new (N_COLUMNS, G_TYPE_STRING,
G_TYPE_STRING, G_TYPE_INT);
pevent_dialog_list = gtkutil_treeview_new (th, GTK_TREE_MODEL (store), NULL,
COL_EVENT_NAME, _("Event"),
COL_EVENT_TEXT, _("Text"), -1);
sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (pevent_dialog_list));
g_signal_connect (G_OBJECT (sel), "changed",
G_CALLBACK (pevent_dialog_select), store);
pevent_dialog_twid = gtk_xtext_new (colors, 0);
pevent_dialog_entry = gtk_entry_new_with_max_length (255);
g_signal_connect (G_OBJECT (pevent_dialog_entry), "activate",
G_CALLBACK (pevent_dialog_update), pevent_dialog_twid);
gtk_box_pack_start (GTK_BOX (bh), pevent_dialog_entry, 0, 0, 0);
gtk_widget_show (pevent_dialog_entry);
tbox = gtk_hbox_new (0, 0);
gtk_container_add (GTK_CONTAINER (bh), tbox);
gtk_widget_show (tbox);
gtk_widget_set_usize (pevent_dialog_twid, 150, 20);
gtk_container_add (GTK_CONTAINER (tbox), pevent_dialog_twid);
gtk_xtext_set_font (GTK_XTEXT (pevent_dialog_twid), prefs.hex_text_font);
wid = gtk_vscrollbar_new (GTK_XTEXT (pevent_dialog_twid)->adj);
gtk_box_pack_start (GTK_BOX (tbox), wid, FALSE, FALSE, 0);
show_and_unfocus (wid);
gtk_widget_show (pevent_dialog_twid);
hstore = gtk_list_store_new (2, G_TYPE_INT, G_TYPE_STRING);
pevent_dialog_hlist = gtkutil_treeview_new (bh, GTK_TREE_MODEL (hstore),
NULL,
0, _("$ Number"),
1, _("Description"), -1);
gtk_widget_show (pevent_dialog_hlist);
pevent_dialog_fill (pevent_dialog_list);
gtk_widget_show (pevent_dialog_list);
hbox = gtk_hbutton_box_new ();
gtk_box_pack_end (GTK_BOX (vbox), hbox, 0, 0, 2);
/*wid = gtk_button_new_with_label (_("Save"));
gtk_box_pack_end (GTK_BOX (hbox), wid, 0, 0, 0);
gtk_signal_connect (GTK_OBJECT (wid), "clicked",
GTK_SIGNAL_FUNC (pevent_save_cb), NULL);
gtk_widget_show (wid);*/
gtkutil_button (hbox, GTK_STOCK_SAVE_AS, NULL, pevent_save_cb,
(void *) 1, _("Save As..."));
gtkutil_button (hbox, GTK_STOCK_OPEN, NULL, pevent_load_cb,
(void *) 0, _("Load From..."));
wid = gtk_button_new_with_label (_("Test All"));
gtk_box_pack_end (GTK_BOX (hbox), wid, 0, 0, 0);
g_signal_connect (G_OBJECT (wid), "clicked",
G_CALLBACK (pevent_test_cb), pevent_dialog_twid);
gtk_widget_show (wid);
wid = gtk_button_new_from_stock (GTK_STOCK_OK);
gtk_box_pack_start (GTK_BOX (hbox), wid, 0, 0, 0);
g_signal_connect (G_OBJECT (wid), "clicked",
G_CALLBACK (pevent_ok_cb), NULL);
gtk_widget_show (wid);
gtk_widget_show (hbox);
gtk_widget_show (pevent_dialog);
}