# HG changeset patch # User Herman Bloggs # Date 1060304703 0 # Node ID 752d0600b514b426f26d449556a8e75b252fe20b # Parent 90fc2199c1563197162119fd3b2cbbde7606fe35 [gaim-migrate @ 6915] Blist can now be docked, using Windows appbar features committer: Tailor Script diff -r 90fc2199c156 -r 752d0600b514 plugins/win32/winprefs/Makefile.mingw --- a/plugins/win32/winprefs/Makefile.mingw Fri Aug 08 01:02:33 2003 +0000 +++ b/plugins/win32/winprefs/Makefile.mingw Fri Aug 08 01:05:03 2003 +0000 @@ -56,7 +56,8 @@ ## SOURCES, OBJECTS ## -C_SRC = winprefs.c +C_SRC = winprefs.c \ + gtkappbar.c OBJECTS = $(C_SRC:%.c=%.o) diff -r 90fc2199c156 -r 752d0600b514 plugins/win32/winprefs/gtkappbar.c --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/plugins/win32/winprefs/gtkappbar.c Fri Aug 08 01:05:03 2003 +0000 @@ -0,0 +1,456 @@ +/* + * gaim - WinGaim Options Plugin + * + * File: gtkappbar.c + * Date: August 2, 2003 + * Description: Appbar functionality for Windows GTK+ applications + * + * Copyright (C) 2003, Herman Bloggs + * + * 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + * + */ +/* + * TODO: + * - Move 'App on top' feature from Trans plugin to here + * - Bug: Multiple Show/Hide Desktop calls causes client area to disapear + */ +#include +#include +#include +#include +#include +#include "gtkappbar.h" +#include "debug.h" + +#define APPBAR_CALLBACK WM_USER + 1010 + +static void get_window_normal_rc(HWND hwnd, RECT *rc) { + WINDOWPLACEMENT wplc; + GetWindowPlacement(hwnd, &wplc); + CopyRect(rc, &wplc.rcNormalPosition); +} + +static void print_rect(RECT *rc) { + gaim_debug(GAIM_DEBUG_INFO, "gtkappbar", "RECT: L:%ld R:%ld T:%ld B:%ld\n", + rc->left, rc->right, rc->top, rc->bottom); +} + +static void set_toolbar(HWND hwnd, gboolean val) { + LONG style=0; + + style = GetWindowLong(hwnd, GWL_EXSTYLE); + if(val && !(style & WS_EX_TOOLWINDOW)) + style |= WS_EX_TOOLWINDOW; + else if(!val && style & WS_EX_TOOLWINDOW) + style &= ~WS_EX_TOOLWINDOW; + else + return; + SetWindowLong(hwnd, GWL_EXSTYLE, style); + SetWindowPos(hwnd, 0, 0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE | SWP_NOZORDER | SWP_FRAMECHANGED); +} + +static gboolean gtk_appbar_register(GtkAppBar *ab, HWND hwnd) { + APPBARDATA abd; + + abd.cbSize = sizeof(APPBARDATA); + abd.hWnd = hwnd; + abd.uCallbackMessage = APPBAR_CALLBACK; + + ab->registered = SHAppBarMessage(ABM_NEW, &abd); + + return ab->registered; +} + +static gboolean gtk_appbar_unregister(GtkAppBar *ab, HWND hwnd) { + APPBARDATA abd; + + if(!ab->registered) + return TRUE; + + abd.cbSize = sizeof(APPBARDATA); + abd.hWnd = hwnd; + + ab->registered = !SHAppBarMessage(ABM_REMOVE, &abd); + + if(!ab->registered) { + ab->docked = FALSE; + ab->docking = FALSE; + } + return !ab->registered; +} + +static void gtk_appbar_querypos(GtkAppBar *ab, HWND hwnd, RECT *rc) { + APPBARDATA abd; + int iWidth = 0; + + if(!ab->registered) + gtk_appbar_register(ab, hwnd); + + abd.hWnd = hwnd; + abd.cbSize = sizeof(APPBARDATA); + CopyRect(&abd.rc, rc); + abd.uEdge = ab->side; + + iWidth = abd.rc.right - abd.rc.left; + + abd.rc.top = 0; + abd.rc.bottom = GetSystemMetrics(SM_CYSCREEN); + switch (abd.uEdge) + { + case ABE_LEFT: + abd.rc.left = 0; + abd.rc.right = iWidth; + break; + + case ABE_RIGHT: + abd.rc.right = GetSystemMetrics(SM_CXSCREEN); + abd.rc.left = abd.rc.right - iWidth; + break; + } + + /* Ask the system for the screen space */ + SHAppBarMessage(ABM_QUERYPOS, &abd); + + switch (abd.uEdge) + { + case ABE_LEFT: + abd.rc.right = abd.rc.left + iWidth; + break; + + case ABE_RIGHT: + abd.rc.left = abd.rc.right - iWidth; + break; + } + + CopyRect(rc, &abd.rc); +} + +static void gtk_appbar_setpos(GtkAppBar *ab, HWND hwnd) { + APPBARDATA abd; + + if(!ab->registered) + gtk_appbar_register(ab, hwnd); + + abd.hWnd = hwnd; + abd.cbSize = sizeof(APPBARDATA); + CopyRect(&abd.rc, &(ab->docked_rect)); + abd.uEdge = ab->side; + + SHAppBarMessage(ABM_SETPOS, &abd); +} + +static GdkFilterReturn wnd_moving(GtkAppBar *ab, GdkXEvent *xevent) { + MSG *msg = (MSG*)xevent; + POINT cp; + LONG cxScreen; + RECT *rc = (RECT*)msg->lParam; + int side = -1; + + gaim_debug(GAIM_DEBUG_INFO, "gtkappbar", "wnd_moving\n"); + + cxScreen = GetSystemMetrics(SM_CXSCREEN); + GetCursorPos(&cp); + + /* Which part of the screen are we in ? */ + if( cp.x > (cxScreen - (cxScreen / 10)) ) + side = ABE_RIGHT; + else if( cp.x < (cxScreen / 10) ) + side = ABE_LEFT; + + if(!ab->docked) { + if( (side == ABE_RIGHT || side == ABE_LEFT) ) { + if( !ab->docking ) { + ab->side = side; + GetWindowRect(msg->hwnd, &(ab->docked_rect)); + gtk_appbar_querypos(ab, msg->hwnd, &(ab->docked_rect)); + + /* save pre-docking height */ + ab->undocked_height = rc->bottom - rc->top; + ab->docking = TRUE; + } + } + else + ab->docking = FALSE; + } + else if(side < 0) { + gtk_appbar_unregister(ab, msg->hwnd); + rc->bottom = rc->top + ab->undocked_height; + } + + /* Switch to toolbar/regular caption*/ + if(ab->docking) + set_toolbar(msg->hwnd, TRUE); + else if(!ab->docked) + set_toolbar(msg->hwnd, FALSE); + + return GDK_FILTER_CONTINUE; +} + +static GdkFilterReturn wnd_sizing(GtkAppBar *ab, GdkXEvent *xevent) { + MSG *msg = (MSG*)xevent; + + gaim_debug(GAIM_DEBUG_INFO, "gtkappbar", "wnd_sizing\n"); + if(ab->docked) { + RECT *rc = (RECT*)msg->lParam; + if(ab->side == ABE_LEFT && msg->wParam == WMSZ_RIGHT) { + ab->docked_rect.right = rc->right; + gtk_appbar_setpos(ab, msg->hwnd); + } + else if(ab->side == ABE_RIGHT && msg->wParam == WMSZ_LEFT) { + ab->docked_rect.left = rc->left; + gtk_appbar_setpos(ab, msg->hwnd); + } + return GDK_FILTER_REMOVE; + } + return GDK_FILTER_CONTINUE; +} + +static GdkFilterReturn wnd_poschanging(GtkAppBar *ab, GdkXEvent *xevent) { + MSG *msg = (MSG*)xevent; + WINDOWPOS *wpos = (WINDOWPOS*)msg->lParam; + + gaim_debug(GAIM_DEBUG_INFO, "gtkappbar", "wnd_poschanging\n"); + + if(ab->docked || ab->docking) { + wpos->x = ab->docked_rect.left; + wpos->y = ab->docked_rect.top; + wpos->cx = ab->docked_rect.right - ab->docked_rect.left; + wpos->cy = ab->docked_rect.bottom - ab->docked_rect.top; + if(IsIconic(msg->hwnd)) + set_toolbar(msg->hwnd, FALSE); + /*return GDK_FILTER_REMOVE;*/ + } + return GDK_FILTER_CONTINUE; +} + +static GdkFilterReturn wnd_exitsizemove(GtkAppBar *ab, GdkXEvent *xevent) { + MSG *msg = (MSG*)xevent; + + if(ab->docking) { + gtk_appbar_setpos(ab, msg->hwnd); + ab->docking = FALSE; + ab->docked = TRUE; + } + else if(!ab->docked) { + gtk_appbar_unregister(ab, msg->hwnd); + } + + return GDK_FILTER_CONTINUE; +} + +static GdkFilterReturn wnd_showwindow(GtkAppBar *ab, GdkXEvent *xevent) { + MSG *msg = (MSG*)xevent; + + gaim_debug(GAIM_DEBUG_INFO, "gtkappbar", "wnd_showwindow\n"); + if(msg->wParam && ab->docked) { + gaim_debug(GAIM_DEBUG_INFO, "gtkappbar", "shown\n"); + ab->docked = FALSE; + gtk_appbar_dock(ab, ab->side); + + } + else if(!msg->wParam && ab->docked) { + gaim_debug(GAIM_DEBUG_INFO, "gtkappbar", "hidden\n"); + gtk_appbar_unregister(ab, GDK_WINDOW_HWND(ab->win->window)); + set_toolbar(GDK_WINDOW_HWND(ab->win->window), FALSE); + ab->docked = TRUE; + } + return GDK_FILTER_CONTINUE; +} + +static GdkFilterReturn wnd_size(GtkAppBar *ab, GdkXEvent *xevent) { + MSG *msg = (MSG*)xevent; + + gaim_debug(GAIM_DEBUG_INFO, "gtkappbar", "wnd_size\n"); + + if(msg->wParam == SIZE_MINIMIZED) { + gaim_debug(GAIM_DEBUG_INFO, "gtkappbar", "Minimize\n"); + if(ab->docked) { + gtk_appbar_unregister(ab, GDK_WINDOW_HWND(ab->win->window)); + ab->docked = TRUE; + } + } + else if(msg->wParam == SIZE_RESTORED) { + gaim_debug(GAIM_DEBUG_INFO, "gtkappbar", "Restore\n"); + if(ab->docked) { + gtk_appbar_dock(ab, ab->side); + } + } + return GDK_FILTER_CONTINUE; +} + +static GdkFilterReturn wnd_nchittest(GtkAppBar *ab, GdkXEvent *xevent) { + MSG *msg = (MSG*)xevent; + + if(ab->docked) { + UINT ret = DefWindowProc(msg->hwnd, msg->message, msg->wParam, msg->lParam); + + switch(ret) { + case HTBOTTOM: + case HTBOTTOMLEFT: + case HTBOTTOMRIGHT: + case HTTOP: + case HTTOPLEFT: + case HTTOPRIGHT: + return GDK_FILTER_REMOVE; + case HTLEFT: + if(ab->side == ABE_LEFT) + return GDK_FILTER_REMOVE; + break; + case HTRIGHT: + if(ab->side == ABE_RIGHT) + return GDK_FILTER_REMOVE; + break; + } + } + return GDK_FILTER_CONTINUE; +} + +#if 0 +static GdkFilterReturn wnd_initmenupopup(GtkAppBar *ab, GdkXEvent *xevent) { + MSG *msg = (MSG*)xevent; + + if(ab->docked && HIWORD(msg->lParam)) { + HMENU sysmenu = GetSystemMenu(msg->hwnd, FALSE); + gaim_debug(GAIM_DEBUG_INFO, "gtkappbar", "wnd_initpopupmenu: docked: %d ismenu: %d\n", ab->docked, IsMenu(sysmenu)); + if(EnableMenuItem(sysmenu, SC_MAXIMIZE, MF_BYCOMMAND|MF_GRAYED)<0) + gaim_debug(GAIM_DEBUG_INFO, "gtkappbar", "SC_MAXIMIZE Menu item does not exist\n"); + if(EnableMenuItem(sysmenu, SC_MOVE, MF_BYCOMMAND|MF_GRAYED)<0) + gaim_debug(GAIM_DEBUG_INFO, "gtkappbar", "SC_MOVE Menu item does not exist\n"); + return GDK_FILTER_CONTINUE; + } + else + GetSystemMenu(msg->hwnd, TRUE); + return GDK_FILTER_CONTINUE; +} +#endif + +static GdkFilterReturn gtk_appbar_callback(GtkAppBar *ab, GdkXEvent *xevent) { + MSG *msg = (MSG*)xevent; + + gaim_debug(GAIM_DEBUG_INFO, "gtkappbar", "gtk_appbar_callback\n"); + switch (msg->wParam) { + case ABN_STATECHANGE: + gaim_debug(GAIM_DEBUG_INFO, "gtkappbar", "gtk_appbar_callback: ABN_STATECHANGE\n"); + break; + + case ABN_FULLSCREENAPP: + gaim_debug(GAIM_DEBUG_INFO, "gtkappbar", "gtk_appbar_callback: ABN_FULLSCREENAPP: %d\n", (BOOL)msg->lParam); + break; + + case ABN_POSCHANGED: + gaim_debug(GAIM_DEBUG_INFO, "gtkappbar", "gtk_appbar_callback: ABN_POSCHANGED\n"); + gtk_appbar_querypos(ab, msg->hwnd, &(ab->docked_rect)); + MoveWindow(msg->hwnd, ab->docked_rect.left, ab->docked_rect.top, + ab->docked_rect.right - ab->docked_rect.left, + ab->docked_rect.bottom - ab->docked_rect.top, TRUE); + gtk_appbar_setpos(ab, msg->hwnd); + break; + } + return GDK_FILTER_CONTINUE; +} + +static GdkFilterReturn gtk_appbar_event_filter(GdkXEvent *xevent, GdkEvent *event, gpointer data) { + MSG *msg = (MSG*)xevent; + + /*printf("MSG: %s\n", message_to_string (msg->message));*/ + switch(msg->message) { + case WM_EXITSIZEMOVE: + return wnd_exitsizemove(data, xevent); + case WM_WINDOWPOSCHANGING: + return wnd_poschanging(data, xevent); + case WM_SIZING: + return wnd_sizing(data, xevent); + case WM_MOVING: + return wnd_moving(data, xevent); + case WM_SHOWWINDOW: + return wnd_showwindow(data, xevent); + case WM_NCHITTEST: + return wnd_nchittest(data, xevent); +#if 0 + case WM_INITMENUPOPUP: + return wnd_initmenupopup(data, xevent); +#endif + case WM_SIZE: + return wnd_size(data, xevent); + case APPBAR_CALLBACK: + return gtk_appbar_callback(data, xevent); + default: + } + return GDK_FILTER_CONTINUE; +} + +void gtk_appbar_dock(GtkAppBar *ab, UINT side) { + RECT orig; + + gaim_debug(GAIM_DEBUG_INFO, "gtkappbar", "gtk_appbar_dock\n"); + + if(!ab || !IsWindow(GDK_WINDOW_HWND(ab->win->window))) + return; + + ab->side = side; + get_window_normal_rc(GDK_WINDOW_HWND(ab->win->window), &(ab->docked_rect)); + CopyRect(&orig, &(ab->docked_rect)); + print_rect(&(ab->docked_rect)); + gtk_appbar_querypos(ab, GDK_WINDOW_HWND(ab->win->window), &(ab->docked_rect)); + if(EqualRect(&orig, &(ab->docked_rect)) == 0) + MoveWindow(GDK_WINDOW_HWND(ab->win->window), + ab->docked_rect.left, + ab->docked_rect.top, + ab->docked_rect.right - ab->docked_rect.left, + ab->docked_rect.bottom - ab->docked_rect.top, TRUE); + gtk_appbar_setpos(ab, GDK_WINDOW_HWND(ab->win->window)); + set_toolbar(GDK_WINDOW_HWND(ab->win->window), TRUE); + ab->docked = TRUE; +} + +GtkAppBar *gtk_appbar_add(GtkWidget *win) { + GtkAppBar *ab; + + gaim_debug(GAIM_DEBUG_INFO, "gtkappbar", "gtk_appbar_add\n"); + + if(!win) + return NULL; + ab = g_new0(GtkAppBar, 1); + ab->win = win; + + /* init docking coords */ + get_window_normal_rc(GDK_WINDOW_HWND(win->window), &(ab->docked_rect)); + + /* Add main window filter */ + gdk_window_add_filter(win->window, + gtk_appbar_event_filter, + ab); + return ab; +} + +void gtk_appbar_remove(GtkAppBar *ab) { + gaim_debug(GAIM_DEBUG_INFO, "gtkappbar", "gtk_appbar_remove\n"); + + if(!ab) + return; + gdk_window_remove_filter(ab->win->window, + gtk_appbar_event_filter, + ab); + if(ab->docked) { + gtk_window_resize(GTK_WINDOW(ab->win), + ab->docked_rect.right - ab->docked_rect.left, + ab->undocked_height); + set_toolbar(GDK_WINDOW_HWND(ab->win->window), FALSE); + } + gtk_appbar_unregister(ab, GDK_WINDOW_HWND(ab->win->window)); + + g_free(ab); +} diff -r 90fc2199c156 -r 752d0600b514 plugins/win32/winprefs/gtkappbar.h --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/plugins/win32/winprefs/gtkappbar.h Fri Aug 08 01:05:03 2003 +0000 @@ -0,0 +1,42 @@ +/* + * gaim - WinGaim Options Plugin + * + * File: gtkappbar.h + * Date: August 2, 2003 + * Description: Appbar functionality for Windows GTK+ applications + * + * Copyright (C) 2003, Herman Bloggs + * + * 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + * + */ +#ifndef _GTKAPPBAR_H_ +#define _GTKAPPBAR_H_ + +typedef struct { + GtkWidget *win; + RECT docked_rect; + UINT undocked_height; + UINT side; + gboolean docked; + gboolean docking; + gboolean registered; +} GtkAppBar; + +GtkAppBar *gtk_appbar_add(GtkWidget *win); +void gtk_appbar_remove(GtkAppBar *ab); +void gtk_appbar_dock(GtkAppBar *ab, UINT side); + +#endif /* _GTKAPPBAR_H_ */ diff -r 90fc2199c156 -r 752d0600b514 plugins/win32/winprefs/winprefs.c --- a/plugins/win32/winprefs/winprefs.c Fri Aug 08 01:02:33 2003 +0000 +++ b/plugins/win32/winprefs/winprefs.c Fri Aug 08 01:05:03 2003 +0000 @@ -1,7 +1,11 @@ /* - * winprefs.c + * gaim - WinGaim Options Plugin * - * copyright (c) 1998-2002, Herman Bloggs + * File: winprefs.c + * Date: December 12, 2002 + * Description: Gaim Plugin interface + * + * copyright (c) 2002-2003, Herman Bloggs * * 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 @@ -21,43 +25,110 @@ #include #include #include +#include #include "internal.h" +#include "gtkinternal.h" #include "prefs.h" #include "debug.h" #include "gtkplugin.h" #include "gtkutils.h" +#include "gtkblist.h" +#include "gtkappbar.h" /* * MACROS & DEFINES */ #define WINPREFS_PLUGIN_ID "gtk-win-prefs" -#define WINPREFS_VERSION 1 /* * LOCALS */ -static const char *OPT_WINPREFS_AUTOSTART="/plugins/gtk/win32/winprefs/auto_start"; +static const char *OPT_WINPREFS_DBLIST_DOCKABLE = "/plugins/gtk/win32/winprefs/dblist_dockable"; +static const char *OPT_WINPREFS_DBLIST_DOCKED = "/plugins/gtk/win32/winprefs/dblist_docked"; +static const char *OPT_WINPREFS_DBLIST_HEIGHT = "/plugins/gtk/win32/winprefs/dblist_height"; +static const char *OPT_WINPREFS_DBLIST_SIDE = "/plugins/gtk/win32/winprefs/dblist_side"; +static GaimPlugin *plugin_id = NULL; +static GtkAppBar *blist_ab = NULL; +static GtkWidget *blist = NULL; /* * PROTOS */ +static void blist_create_cb(); /* * CODE */ -static GtkWidget *wgaim_button(const char *text, const char *pref, GtkWidget *page) { +/* UTIL */ + +static GtkWidget *wgaim_button(const char *text, GtkWidget *page) { GtkWidget *button; button = gtk_check_button_new_with_mnemonic(text); - gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(button), gaim_prefs_get_bool(pref)); gtk_box_pack_start(GTK_BOX(page), button, FALSE, FALSE, 0); gtk_widget_show(button); return button; } +/* BLIST DOCKING */ + +static void blist_save_state() { + if(blist_ab && gaim_prefs_get_bool(OPT_WINPREFS_DBLIST_DOCKABLE)) { + if(blist_ab->docked) { + gaim_prefs_set_int(OPT_WINPREFS_DBLIST_HEIGHT, blist_ab->undocked_height); + gaim_prefs_set_int(OPT_WINPREFS_DBLIST_SIDE, blist_ab->side); + } + gaim_prefs_set_bool(OPT_WINPREFS_DBLIST_DOCKED, blist_ab->docked); + } + else + gaim_prefs_set_bool(OPT_WINPREFS_DBLIST_DOCKED, FALSE); +} + +static void blist_set_dockable(gboolean val) { + if(val) { + if(!blist_ab && blist) + blist_ab = gtk_appbar_add(blist); + } + else { + gtk_appbar_remove(blist_ab); + blist_ab = NULL; + } +} + +static void gaim_quit_cb() { + gaim_debug(GAIM_DEBUG_INFO, WINPREFS_PLUGIN_ID, "gaim_quit_cb: removing appbar\n"); + blist_save_state(); + blist_set_dockable(FALSE); +} + +static gboolean blist_create_cb_remove(gpointer data) { + gaim_signal_disconnect(plugin_id, event_signon, blist_create_cb); + return FALSE; +} + +static void blist_create_cb() { + gaim_debug(GAIM_DEBUG_INFO, WINPREFS_PLUGIN_ID, "event_signon\n"); + + blist = GAIM_GTK_BLIST(gaim_get_blist())->window; + if(gaim_prefs_get_bool(OPT_WINPREFS_DBLIST_DOCKABLE)) { + blist_set_dockable(TRUE); + if(gaim_prefs_get_bool(OPT_WINPREFS_DBLIST_DOCKED)) { + blist_ab->undocked_height = gaim_prefs_get_int(OPT_WINPREFS_DBLIST_HEIGHT); + gtk_appbar_dock(blist_ab, + gaim_prefs_get_int(OPT_WINPREFS_DBLIST_SIDE)); + } + } + /* removing here will cause a crash when going to next cb + in the gaim signal cb loop.. so process delayed. */ + g_idle_add(blist_create_cb_remove, NULL); +} + + +/* AUTOSTART */ + static int open_run_key(PHKEY phKey, REGSAM samDesired) { /* First try current user key (for WinNT & Win2k +), fall back to local machine */ if(ERROR_SUCCESS == RegOpenKeyEx(HKEY_CURRENT_USER, @@ -73,44 +144,72 @@ return 1; } -static void set_winprefs_option(GtkWidget *w, const char *key) { - gaim_prefs_set_bool(key, gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(w))); - if(key == OPT_WINPREFS_AUTOSTART) { - HKEY hKey; +/* WIN PREFS GENERAL */ - if(!open_run_key(&hKey, KEY_SET_VALUE)) - return; - if(gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(w))) { - char buffer[1024]; - DWORD size; +static void winprefs_set_autostart(GtkWidget *w) { + HKEY hKey; - if((size = GetModuleFileName(wgaim_hinstance(), - (LPBYTE)buffer, - sizeof(buffer)))==0) { - gaim_debug(GAIM_DEBUG_ERROR, WINPREFS_PLUGIN_ID, "GetModuleFileName Error.. Could not set Gaim autostart.\n"); - RegCloseKey(hKey); - return; - } - /* Now set value of new key */ - if(ERROR_SUCCESS != RegSetValueEx(hKey, - "Gaim", - 0, - REG_SZ, - buffer, - size)) - gaim_debug(GAIM_DEBUG_ERROR, WINPREFS_PLUGIN_ID, "Could not set registry key value\n"); - } - else { - if(ERROR_SUCCESS != RegDeleteValue(hKey, "Gaim")) - gaim_debug(GAIM_DEBUG_ERROR, WINPREFS_PLUGIN_ID, "Could not delete registry key value\n"); - } - RegCloseKey(hKey); - } + if(!open_run_key(&hKey, KEY_SET_VALUE)) + return; + if(gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(w))) { + char buffer[1024]; + DWORD size; + + if((size = GetModuleFileName(wgaim_hinstance(), + (LPBYTE)buffer, + sizeof(buffer)))==0) { + gaim_debug(GAIM_DEBUG_ERROR, WINPREFS_PLUGIN_ID, "GetModuleFileName Error.. Could not set Gaim autostart.\n"); + RegCloseKey(hKey); + return; + } + /* Now set value of new key */ + if(ERROR_SUCCESS != RegSetValueEx(hKey, + "Gaim", + 0, + REG_SZ, + buffer, + size)) + gaim_debug(GAIM_DEBUG_ERROR, WINPREFS_PLUGIN_ID, "Could not set registry key value\n"); + } + else { + if(ERROR_SUCCESS != RegDeleteValue(hKey, "Gaim")) + gaim_debug(GAIM_DEBUG_ERROR, WINPREFS_PLUGIN_ID, "Could not delete registry key value\n"); + } + RegCloseKey(hKey); } +static void winprefs_set_blist_dockable(GtkWidget *w) { + gaim_prefs_set_bool(OPT_WINPREFS_DBLIST_DOCKABLE, gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(w))); + gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(w)) ? blist_set_dockable(TRUE) : blist_set_dockable(FALSE); +} + + /* * EXPORTED FUNCTIONS */ + +gboolean plugin_load(GaimPlugin *plugin) { + plugin_id = plugin; + + /* blist docking init */ + if(!blist && gaim_get_blist() && GAIM_GTK_BLIST(gaim_get_blist())) { + blist = GAIM_GTK_BLIST(gaim_get_blist())->window; + if(gaim_prefs_get_bool(OPT_WINPREFS_DBLIST_DOCKABLE)) + blist_set_dockable(TRUE); + } + else + gaim_signal_connect(plugin, event_signon, blist_create_cb, NULL); + + gaim_signal_connect(plugin, event_quit, gaim_quit_cb, NULL); + + return TRUE; +} + +gboolean plugin_unload(GaimPlugin *plugin) { + blist_set_dockable(FALSE); + return TRUE; +} + static GtkWidget* get_config_frame(GaimPlugin *plugin) { GtkWidget *ret; GtkWidget *button; @@ -120,16 +219,21 @@ ret = gtk_vbox_new(FALSE, 18); gtk_container_set_border_width (GTK_CONTAINER (ret), 12); - /* IM Convo trans options */ vbox = gaim_gtk_make_frame (ret, _("Startup")); - button = wgaim_button(_("_Start Gaim on Windows startup"), OPT_WINPREFS_AUTOSTART, vbox); - /* Set initial value */ + + /* Autostart */ + button = wgaim_button(_("_Start Gaim on Windows startup"), vbox); if(open_run_key(&hKey, KEY_QUERY_VALUE)) { if(ERROR_SUCCESS == RegQueryValueEx(hKey, "Gaim", 0, NULL, NULL, NULL)) { gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(button), TRUE); } } - gtk_signal_connect(GTK_OBJECT(button), "clicked", GTK_SIGNAL_FUNC(set_winprefs_option), (void *)OPT_WINPREFS_AUTOSTART); + gtk_signal_connect(GTK_OBJECT(button), "clicked", GTK_SIGNAL_FUNC(winprefs_set_autostart), NULL); + + /* Dockable Blist */ + button = wgaim_button(_("_Dockable Buddy List"), vbox); + gtk_signal_connect(GTK_OBJECT(button), "clicked", GTK_SIGNAL_FUNC(winprefs_set_blist_dockable), NULL); + gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(button), gaim_prefs_get_bool(OPT_WINPREFS_DBLIST_DOCKABLE)); gtk_widget_show_all(ret); return ret; @@ -154,9 +258,9 @@ N_("Options specific to Windows Gaim."), N_("Options specific to Windows Gaim."), "Herman Bloggs ", - WEBSITE, - NULL, - NULL, + GAIM_WEBSITE, + plugin_load, + plugin_unload, NULL, &ui_info, NULL @@ -165,9 +269,12 @@ static void init_plugin(GaimPlugin *plugin) { - gaim_prefs_add_none("/plugins/gtk/win32"); - gaim_prefs_add_none("/plugins/gtk/win32/winprefs"); - gaim_prefs_add_bool("/plugins/gtk/win32/winprefs/auto_start", FALSE); + gaim_prefs_add_none("/plugins/gtk/win32"); + gaim_prefs_add_none("/plugins/gtk/win32/winprefs"); + gaim_prefs_add_bool(OPT_WINPREFS_DBLIST_DOCKABLE, FALSE); + gaim_prefs_add_bool(OPT_WINPREFS_DBLIST_DOCKED, FALSE); + gaim_prefs_add_int(OPT_WINPREFS_DBLIST_HEIGHT, 0); + gaim_prefs_add_int(OPT_WINPREFS_DBLIST_SIDE, 0); } GAIM_INIT_PLUGIN(winprefs, init_plugin, info)