2005-12-28 00:55:34 +01:00
|
|
|
## message_window.py
|
|
|
|
##
|
|
|
|
## Copyright (C) 2005-2006 Travis Shirk <travis@pobox.com>
|
|
|
|
##
|
|
|
|
## 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; version 2 only.
|
|
|
|
##
|
|
|
|
## 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.
|
|
|
|
##
|
|
|
|
|
|
|
|
import gtk
|
|
|
|
import gtk.glade
|
|
|
|
import pango
|
|
|
|
import gobject
|
|
|
|
|
2005-12-29 04:20:06 +01:00
|
|
|
import common
|
2005-12-31 07:27:22 +01:00
|
|
|
import gtkgui_helpers
|
2006-01-02 03:12:34 +01:00
|
|
|
import message_control
|
2006-01-03 04:34:32 +01:00
|
|
|
from chat_control import ChatControlBase
|
2005-12-31 07:27:22 +01:00
|
|
|
|
2005-12-28 00:55:34 +01:00
|
|
|
from common import gajim
|
|
|
|
|
|
|
|
####################
|
|
|
|
# FIXME: Can't this stuff happen once?
|
|
|
|
from common import i18n
|
|
|
|
_ = i18n._
|
|
|
|
APP = i18n.APP
|
|
|
|
|
|
|
|
GTKGUI_GLADE = 'gtkgui.glade'
|
|
|
|
####################
|
|
|
|
|
|
|
|
class MessageWindow:
|
|
|
|
'''Class for windows which contain message like things; chats,
|
|
|
|
groupchats, etc.'''
|
|
|
|
|
2006-01-12 04:09:33 +01:00
|
|
|
# DND_TARGETS is the targets needed by drag_source_set and drag_dest_set
|
|
|
|
DND_TARGETS = [('GAJIM_TAB', 0, 81)]
|
|
|
|
hid = 0 # drag_data_received handler id
|
|
|
|
|
2006-01-08 05:31:02 +01:00
|
|
|
def __init__(self, acct, type):
|
2006-01-25 06:39:07 +01:00
|
|
|
# A dictionary of dictionaries where _contacts[account][jid] == A MessageControl
|
2005-12-28 00:55:34 +01:00
|
|
|
self._controls = {}
|
2006-01-08 05:31:02 +01:00
|
|
|
# If None, the window is not tied to any specific account
|
|
|
|
self.account = acct
|
|
|
|
# If None, the window is not tied to any specific type
|
|
|
|
self.type = type
|
2005-12-28 00:55:34 +01:00
|
|
|
|
|
|
|
self.widget_name = 'message_window'
|
|
|
|
self.xml = gtk.glade.XML(GTKGUI_GLADE, self.widget_name, APP)
|
2006-01-12 03:49:41 +01:00
|
|
|
self.xml.signal_autoconnect(self)
|
2005-12-28 00:55:34 +01:00
|
|
|
self.window = self.xml.get_widget(self.widget_name)
|
2006-01-12 03:49:41 +01:00
|
|
|
# I don't really understand, but get_property('visible') returns True at this point,
|
|
|
|
# which seems way early. Anyway, hide until first tab is shown
|
|
|
|
self.window.hide()
|
2006-01-02 23:08:50 +01:00
|
|
|
# gtk+ doesn't make use of the motion notify on gtkwindow by default
|
|
|
|
# so this line adds that
|
2006-01-07 04:09:51 +01:00
|
|
|
self.window.add_events(gtk.gdk.POINTER_MOTION_MASK)
|
2005-12-28 00:55:34 +01:00
|
|
|
self.alignment = self.xml.get_widget('alignment')
|
2006-01-03 04:34:32 +01:00
|
|
|
|
2005-12-28 00:55:34 +01:00
|
|
|
self.notebook = self.xml.get_widget('notebook')
|
2006-01-03 04:34:32 +01:00
|
|
|
self.notebook.connect('switch-page',
|
|
|
|
self._on_notebook_switch_page)
|
|
|
|
self.notebook.connect('key-press-event',
|
|
|
|
self._on_notebook_key_press)
|
2005-12-28 00:55:34 +01:00
|
|
|
|
2005-12-29 04:20:06 +01:00
|
|
|
# Remove the glade pages
|
|
|
|
while self.notebook.get_n_pages():
|
|
|
|
self.notebook.remove_page(0)
|
|
|
|
# Tab customizations
|
|
|
|
pref_pos = gajim.config.get('tabs_position')
|
2006-01-12 07:59:59 +01:00
|
|
|
if pref_pos == 'bottom':
|
|
|
|
nb_pos = gtk.POS_BOTTOM
|
|
|
|
elif pref_pos == 'left':
|
|
|
|
nb_pos = gtk.POS_LEFT
|
|
|
|
elif pref_pos == 'right':
|
|
|
|
nb_pos = gtk.POS_RIGHT
|
2005-12-29 04:20:06 +01:00
|
|
|
else:
|
|
|
|
nb_pos = gtk.POS_TOP
|
|
|
|
self.notebook.set_tab_pos(nb_pos)
|
|
|
|
if gajim.config.get('tabs_always_visible'):
|
|
|
|
self.notebook.set_show_tabs(True)
|
|
|
|
self.alignment.set_property('top-padding', 2)
|
|
|
|
else:
|
|
|
|
self.notebook.set_show_tabs(False)
|
|
|
|
self.notebook.set_show_border(gajim.config.get('tabs_border'))
|
|
|
|
|
2006-01-12 04:09:33 +01:00
|
|
|
# set up DnD
|
|
|
|
self.hid = self.notebook.connect('drag_data_received',
|
|
|
|
self.on_tab_label_drag_data_received_cb)
|
|
|
|
self.notebook.drag_dest_set(gtk.DEST_DEFAULT_ALL, self.DND_TARGETS,
|
2006-01-20 16:51:33 +01:00
|
|
|
gtk.gdk.ACTION_MOVE)
|
2006-01-12 04:09:33 +01:00
|
|
|
|
2006-01-25 06:39:07 +01:00
|
|
|
def get_num_controls(self):
|
|
|
|
n = 0
|
|
|
|
for dict in self._controls.values():
|
|
|
|
n += len(dict)
|
|
|
|
return n
|
|
|
|
|
2006-01-02 03:12:34 +01:00
|
|
|
def _on_window_focus(self, widget, event):
|
|
|
|
# window received focus, so if we had urgency REMOVE IT
|
|
|
|
# NOTE: we do not have to read the message (it maybe in a bg tab)
|
|
|
|
# to remove urgency hint so this functions does that
|
2006-01-23 01:03:28 +01:00
|
|
|
gtkgui_helpers.set_unset_urgency_hint(self.window, False)
|
2006-01-02 03:12:34 +01:00
|
|
|
|
2006-01-11 23:30:49 +01:00
|
|
|
ctrl = self.get_active_control()
|
|
|
|
if ctrl:
|
|
|
|
ctrl.set_control_active(True)
|
2006-01-02 10:04:30 +01:00
|
|
|
# Undo "unread" state display, etc.
|
2006-01-11 23:30:49 +01:00
|
|
|
if ctrl.type_id == message_control.TYPE_GC:
|
2006-01-25 06:39:07 +01:00
|
|
|
self.redraw_tab(ctrl, 'active')
|
2006-01-02 10:04:30 +01:00
|
|
|
else:
|
|
|
|
# NOTE: we do not send any chatstate to preserve
|
|
|
|
# inactive, gone, etc.
|
2006-01-25 06:39:07 +01:00
|
|
|
self.redraw_tab(ctrl)
|
2006-01-02 03:12:34 +01:00
|
|
|
|
2005-12-30 21:47:59 +01:00
|
|
|
def _on_window_delete(self, win, event):
|
2006-01-01 20:40:05 +01:00
|
|
|
# Make sure all controls are okay with being deleted
|
2006-01-24 03:57:26 +01:00
|
|
|
for ctrl in self.controls():
|
2006-01-11 23:30:49 +01:00
|
|
|
if not ctrl.allow_shutdown():
|
2006-01-01 20:40:05 +01:00
|
|
|
return True # halt the delete
|
2005-12-31 07:27:22 +01:00
|
|
|
return False
|
2006-01-01 20:40:05 +01:00
|
|
|
|
2005-12-30 21:47:59 +01:00
|
|
|
def _on_window_destroy(self, win):
|
2006-01-24 03:57:26 +01:00
|
|
|
for ctrl in self.controls():
|
2006-01-11 23:30:49 +01:00
|
|
|
ctrl.shutdown()
|
2006-01-01 20:40:05 +01:00
|
|
|
self._controls.clear()
|
2005-12-30 21:47:59 +01:00
|
|
|
|
|
|
|
def new_tab(self, control):
|
2006-01-25 06:39:07 +01:00
|
|
|
if not self._controls.has_key(control.account):
|
|
|
|
self._controls[control.account] = {}
|
|
|
|
self._controls[control.account][control.contact.jid] = control
|
|
|
|
|
|
|
|
if self.get_num_controls() > 1:
|
2006-01-02 10:04:30 +01:00
|
|
|
self.notebook.set_show_tabs(True)
|
|
|
|
self.alignment.set_property('top-padding', 2)
|
2005-12-30 21:47:59 +01:00
|
|
|
|
|
|
|
# Add notebook page and connect up to the tab's close button
|
|
|
|
xml = gtk.glade.XML(GTKGUI_GLADE, 'chat_tab_ebox', APP)
|
|
|
|
tab_label_box = xml.get_widget('chat_tab_ebox')
|
2006-01-25 03:43:55 +01:00
|
|
|
xml.signal_connect('on_close_button_clicked', self._on_close_button_clicked,
|
|
|
|
control)
|
2006-01-02 10:04:30 +01:00
|
|
|
xml.signal_connect('on_tab_eventbox_button_press_event',
|
|
|
|
self.on_tab_eventbox_button_press_event, control.widget)
|
2005-12-30 21:47:59 +01:00
|
|
|
self.notebook.append_page(control.widget, tab_label_box)
|
|
|
|
|
2006-01-12 04:09:33 +01:00
|
|
|
self.setup_tab_dnd(control.widget)
|
2006-01-02 10:04:30 +01:00
|
|
|
|
2006-01-25 06:39:07 +01:00
|
|
|
self.redraw_tab(control)
|
2005-12-30 21:47:59 +01:00
|
|
|
self.window.show_all()
|
2006-01-02 10:04:30 +01:00
|
|
|
# NOTE: we do not call set_control_active(True) since we don't know whether
|
|
|
|
# the tab is the active one.
|
2006-01-03 04:34:32 +01:00
|
|
|
self.show_title()
|
2006-01-02 10:04:30 +01:00
|
|
|
|
|
|
|
def on_tab_eventbox_button_press_event(self, widget, event, child):
|
|
|
|
if event.button == 3:
|
|
|
|
n = self.notebook.page_num(child)
|
|
|
|
self.notebook.set_current_page(n)
|
|
|
|
self.popup_menu(event)
|
2005-12-31 01:50:33 +01:00
|
|
|
|
2006-01-03 04:34:32 +01:00
|
|
|
def _on_message_textview_mykeypress_event(self, widget, event_keyval,
|
2005-12-31 01:50:33 +01:00
|
|
|
event_keymod):
|
|
|
|
# NOTE: handles mykeypress which is custom signal; see message_textview.py
|
|
|
|
|
|
|
|
# construct event instance from binding
|
|
|
|
event = gtk.gdk.Event(gtk.gdk.KEY_PRESS) # it's always a key-press here
|
|
|
|
event.keyval = event_keyval
|
|
|
|
event.state = event_keymod
|
|
|
|
event.time = 0 # assign current time
|
|
|
|
|
|
|
|
if event.state & gtk.gdk.CONTROL_MASK:
|
2006-01-03 04:34:32 +01:00
|
|
|
# Tab switch bindings
|
2005-12-31 01:50:33 +01:00
|
|
|
if event.keyval == gtk.keysyms.Tab: # CTRL + TAB
|
2006-01-19 02:30:18 +01:00
|
|
|
self.move_to_next_unread_tab(True)
|
2005-12-31 01:50:33 +01:00
|
|
|
elif event.keyval == gtk.keysyms.ISO_Left_Tab: # CTRL + SHIFT + TAB
|
2006-01-19 02:30:18 +01:00
|
|
|
self.move_to_next_unread_tab(False)
|
2005-12-31 01:50:33 +01:00
|
|
|
elif event.keyval == gtk.keysyms.Page_Down: # CTRL + PAGE DOWN
|
|
|
|
self.notebook.emit('key_press_event', event)
|
|
|
|
elif event.keyval == gtk.keysyms.Page_Up: # CTRL + PAGE UP
|
|
|
|
self.notebook.emit('key_press_event', event)
|
|
|
|
|
2006-01-25 03:43:55 +01:00
|
|
|
def _on_close_button_clicked(self, button, control):
|
2005-12-30 21:47:59 +01:00
|
|
|
'''When close button is pressed: close a tab'''
|
2006-01-25 03:43:55 +01:00
|
|
|
self.remove_tab(control)
|
2005-12-31 07:27:22 +01:00
|
|
|
|
2006-01-08 00:40:37 +01:00
|
|
|
def show_title(self, urgent = True, control = None):
|
2005-12-31 07:27:22 +01:00
|
|
|
'''redraw the window's title'''
|
|
|
|
unread = 0
|
2006-01-24 03:57:26 +01:00
|
|
|
for ctrl in self.controls():
|
2006-01-20 00:05:37 +01:00
|
|
|
if ctrl.type_id == message_control.TYPE_GC and not \
|
|
|
|
gajim.config.get('notify_on_all_muc_messages') and not \
|
|
|
|
ctrl.attention_flag:
|
|
|
|
continue
|
2006-01-11 23:30:49 +01:00
|
|
|
unread += ctrl.nb_unread
|
2006-01-08 00:40:37 +01:00
|
|
|
unread_str = ''
|
2005-12-31 07:27:22 +01:00
|
|
|
if unread > 1:
|
2006-01-08 00:40:37 +01:00
|
|
|
unread_str = '[' + unicode(unread) + '] '
|
2005-12-31 07:27:22 +01:00
|
|
|
elif unread == 1:
|
2006-01-08 00:40:37 +01:00
|
|
|
unread_str = '* '
|
2006-01-20 00:05:37 +01:00
|
|
|
else:
|
|
|
|
urgent = False
|
2006-01-08 00:40:37 +01:00
|
|
|
|
|
|
|
if not control:
|
|
|
|
control = self.get_active_control()
|
|
|
|
if control.type_id == message_control.TYPE_GC:
|
2006-01-20 00:05:37 +01:00
|
|
|
title = unread_str + control.room_jid
|
2006-01-13 05:36:42 +01:00
|
|
|
urgent = control.attention_flag
|
2006-01-03 05:44:56 +01:00
|
|
|
else:
|
2006-01-12 04:19:59 +01:00
|
|
|
title = unread_str + control.contact.get_shown_name()
|
2005-12-31 07:27:22 +01:00
|
|
|
|
|
|
|
self.window.set_title(title)
|
2006-01-08 00:40:37 +01:00
|
|
|
|
2005-12-31 07:27:22 +01:00
|
|
|
if urgent:
|
|
|
|
gtkgui_helpers.set_unset_urgency_hint(self.window, unread)
|
2006-01-23 01:03:28 +01:00
|
|
|
else:
|
|
|
|
gtkgui_helpers.set_unset_urgency_hint(self.window, False)
|
2005-12-31 18:00:04 +01:00
|
|
|
|
2006-01-25 03:43:55 +01:00
|
|
|
def set_active_tab(self, jid, acct):
|
2006-01-25 06:39:07 +01:00
|
|
|
ctrl = self._controls[acct][jid]
|
2006-01-11 23:30:49 +01:00
|
|
|
ctrl_page = self.notebook.page_num(ctrl.widget)
|
|
|
|
self.notebook.set_current_page(ctrl_page)
|
2005-12-30 21:47:59 +01:00
|
|
|
|
2006-01-25 03:43:55 +01:00
|
|
|
def remove_tab(self, ctrl):
|
2006-01-02 23:08:50 +01:00
|
|
|
# Shutdown the MessageControl
|
2006-01-11 23:30:49 +01:00
|
|
|
if not ctrl.allow_shutdown():
|
2006-01-02 10:04:30 +01:00
|
|
|
return
|
2006-01-11 23:30:49 +01:00
|
|
|
ctrl.shutdown()
|
2006-01-02 10:04:30 +01:00
|
|
|
|
2006-01-02 23:08:50 +01:00
|
|
|
# Update external state
|
2006-01-02 10:04:30 +01:00
|
|
|
if gajim.interface.systray_enabled:
|
2006-01-25 03:43:55 +01:00
|
|
|
gajim.interface.systray.remove_jid(ctrl.contact.jid, ctrl.account,
|
2006-01-11 23:30:49 +01:00
|
|
|
ctrl.type_id)
|
|
|
|
del gajim.last_message_time[ctrl.account][ctrl.contact.jid]
|
2006-01-02 10:04:30 +01:00
|
|
|
|
2006-01-12 04:09:33 +01:00
|
|
|
self.disconnect_tab_dnd(ctrl.widget)
|
2006-01-11 23:30:49 +01:00
|
|
|
self.notebook.remove_page(self.notebook.page_num(ctrl.widget))
|
2006-01-02 10:04:30 +01:00
|
|
|
|
2006-01-25 06:39:07 +01:00
|
|
|
del self._controls[ctrl.account][ctrl.contact.jid]
|
|
|
|
if len(self._controls[ctrl.account]) == 0:
|
|
|
|
del self._controls[ctrl.account]
|
|
|
|
|
|
|
|
if self.get_num_controls() == 1: # we are going from two tabs to one
|
2006-01-02 10:04:30 +01:00
|
|
|
show_tabs_if_one_tab = gajim.config.get('tabs_always_visible')
|
|
|
|
self.notebook.set_show_tabs(show_tabs_if_one_tab)
|
|
|
|
if not show_tabs_if_one_tab:
|
|
|
|
self.alignment.set_property('top-padding', 0)
|
|
|
|
self.show_title()
|
2006-01-25 06:39:07 +01:00
|
|
|
elif self.get_num_controls() == 0:
|
|
|
|
# These are not called when the window is destroyed like this, fake it
|
2006-01-08 05:31:02 +01:00
|
|
|
gajim.interface.msg_win_mgr._on_window_delete(self.window, None)
|
|
|
|
gajim.interface.msg_win_mgr._on_window_destroy(self.window)
|
2006-01-12 04:09:33 +01:00
|
|
|
# dnd clean up
|
|
|
|
self.notebook.disconnect(self.hid)
|
|
|
|
self.notebook.drag_dest_unset()
|
2005-12-30 21:47:59 +01:00
|
|
|
|
2006-01-12 04:09:33 +01:00
|
|
|
self.window.destroy()
|
|
|
|
|
2006-01-25 06:39:07 +01:00
|
|
|
def redraw_tab(self, ctrl, chatstate = None):
|
2006-01-11 23:30:49 +01:00
|
|
|
ctrl.update_ui()
|
2005-12-31 01:50:33 +01:00
|
|
|
|
2006-01-11 23:30:49 +01:00
|
|
|
hbox = self.notebook.get_tab_label(ctrl.widget).get_children()[0]
|
2005-12-31 01:50:33 +01:00
|
|
|
status_img = hbox.get_children()[0]
|
|
|
|
nick_label = hbox.get_children()[1]
|
|
|
|
|
|
|
|
# Optionally hide close button
|
|
|
|
close_button = hbox.get_children()[2]
|
|
|
|
if gajim.config.get('tabs_close_button'):
|
|
|
|
close_button.show()
|
|
|
|
else:
|
|
|
|
close_button.hide()
|
|
|
|
|
|
|
|
# Update nick
|
2005-12-31 08:19:43 +01:00
|
|
|
nick_label.set_max_width_chars(10)
|
2006-01-11 23:30:49 +01:00
|
|
|
(tab_label_str, tab_label_color) = ctrl.get_tab_label(chatstate)
|
2005-12-31 07:27:22 +01:00
|
|
|
nick_label.set_markup(tab_label_str)
|
|
|
|
if tab_label_color:
|
|
|
|
nick_label.modify_fg(gtk.STATE_NORMAL, tab_label_color)
|
|
|
|
nick_label.modify_fg(gtk.STATE_ACTIVE, tab_label_color)
|
|
|
|
|
2006-01-11 23:30:49 +01:00
|
|
|
tab_img = ctrl.get_tab_image()
|
2006-01-03 08:34:18 +01:00
|
|
|
if tab_img:
|
|
|
|
if tab_img.get_storage_type() == gtk.IMAGE_ANIMATION:
|
|
|
|
status_img.set_from_animation(tab_img.get_animation())
|
|
|
|
else:
|
|
|
|
status_img.set_from_pixbuf(tab_img.get_pixbuf())
|
2005-12-31 01:50:33 +01:00
|
|
|
|
|
|
|
def repaint_themed_widgets(self):
|
|
|
|
'''Repaint controls in the window with theme color'''
|
|
|
|
# iterate through controls and repaint
|
2006-01-24 03:57:26 +01:00
|
|
|
for ctrl in self.controls():
|
2006-01-11 23:30:49 +01:00
|
|
|
ctrl.repaint_themed_widgets()
|
2005-12-28 00:55:34 +01:00
|
|
|
|
2006-01-02 10:04:30 +01:00
|
|
|
def _widget_to_control(self, widget):
|
2006-01-24 03:57:26 +01:00
|
|
|
for ctrl in self.controls():
|
2006-01-11 23:30:49 +01:00
|
|
|
if ctrl.widget == widget:
|
|
|
|
return ctrl
|
2005-12-31 04:53:48 +01:00
|
|
|
return None
|
|
|
|
|
2005-12-31 07:27:22 +01:00
|
|
|
def get_active_control(self):
|
2005-12-31 04:53:48 +01:00
|
|
|
notebook = self.notebook
|
|
|
|
active_widget = notebook.get_nth_page(notebook.get_current_page())
|
2006-01-02 10:04:30 +01:00
|
|
|
return self._widget_to_control(active_widget)
|
2005-12-31 07:27:22 +01:00
|
|
|
def get_active_contact(self):
|
2006-01-11 23:30:49 +01:00
|
|
|
ctrl = self.get_active_control()
|
|
|
|
if ctrl:
|
|
|
|
return ctrl.contact
|
2006-01-03 05:44:56 +01:00
|
|
|
return None
|
2005-12-31 04:53:48 +01:00
|
|
|
def get_active_jid(self):
|
2006-01-03 05:44:56 +01:00
|
|
|
contact = self.get_active_contact()
|
|
|
|
if contact:
|
|
|
|
return contact.jid
|
|
|
|
return None
|
2005-12-31 04:53:48 +01:00
|
|
|
|
|
|
|
def is_active(self):
|
|
|
|
return self.window.is_active()
|
2005-12-31 05:53:14 +01:00
|
|
|
def get_origin(self):
|
|
|
|
return self.window.window.get_origin()
|
2005-12-31 04:53:48 +01:00
|
|
|
|
2005-12-31 05:53:14 +01:00
|
|
|
def toggle_emoticons(self):
|
2006-01-24 03:57:26 +01:00
|
|
|
for ctrl in self.controls():
|
2006-01-11 23:30:49 +01:00
|
|
|
ctrl.toggle_emoticons()
|
2005-12-31 07:27:22 +01:00
|
|
|
def update_font(self):
|
2006-01-24 03:57:26 +01:00
|
|
|
for ctrl in self.controls():
|
2006-01-11 23:30:49 +01:00
|
|
|
ctrl.update_font()
|
2005-12-31 07:27:22 +01:00
|
|
|
def update_tags(self):
|
2006-01-24 03:57:26 +01:00
|
|
|
for ctrl in self.controls():
|
2006-01-11 23:30:49 +01:00
|
|
|
ctrl.update_tags()
|
2005-12-31 04:53:48 +01:00
|
|
|
|
2006-01-25 03:43:55 +01:00
|
|
|
def get_control(self, key, acct):
|
|
|
|
'''Return the MessageControl for jid or n, where n is a notebook page index.
|
|
|
|
When key is an int index acct may be None'''
|
2006-01-10 20:24:06 +01:00
|
|
|
if isinstance(key, str):
|
|
|
|
key = unicode(key, 'utf-8')
|
|
|
|
|
2006-01-05 06:51:28 +01:00
|
|
|
if isinstance(key, unicode):
|
|
|
|
jid = key
|
2006-01-25 06:39:07 +01:00
|
|
|
try:
|
|
|
|
return self._controls[acct][jid]
|
|
|
|
except:
|
|
|
|
return None
|
2006-01-01 20:40:05 +01:00
|
|
|
else:
|
2006-01-05 06:51:28 +01:00
|
|
|
page_num = key
|
2006-01-01 20:40:05 +01:00
|
|
|
notebook = self.notebook
|
|
|
|
if page_num == None:
|
|
|
|
page_num = notebook.get_current_page()
|
|
|
|
nth_child = notebook.get_nth_page(page_num)
|
2006-01-02 10:04:30 +01:00
|
|
|
return self._widget_to_control(nth_child)
|
2005-12-31 18:00:04 +01:00
|
|
|
|
2005-12-31 22:55:44 +01:00
|
|
|
def controls(self):
|
2006-01-25 06:39:07 +01:00
|
|
|
for ctrl_dict in self._controls.values():
|
|
|
|
for ctrl in ctrl_dict.values():
|
|
|
|
yield ctrl
|
2005-12-31 22:55:44 +01:00
|
|
|
|
|
|
|
def update_print_time(self):
|
|
|
|
if gajim.config.get('print_time') != 'sometimes':
|
2006-01-11 23:30:49 +01:00
|
|
|
for ctrl in self.controls():
|
|
|
|
if ctrl.print_time_timeout_id:
|
|
|
|
gobject.source_remove(ctrl.print_time_timeout_id)
|
|
|
|
del ctrl.print_time_timeout_id
|
2005-12-31 22:55:44 +01:00
|
|
|
else:
|
2006-01-11 23:30:49 +01:00
|
|
|
for ctrl in self.controls():
|
|
|
|
if not ctrl.print_time_timeout_id:
|
|
|
|
ctrl.print_time_timeout()
|
|
|
|
ctrl.print_time_timeout_id = gobject.timeout_add(300000,
|
|
|
|
ctrl.print_time_timeout, None)
|
2005-12-31 22:55:44 +01:00
|
|
|
|
2006-01-01 20:40:05 +01:00
|
|
|
def move_to_next_unread_tab(self, forward):
|
|
|
|
ind = self.notebook.get_current_page()
|
|
|
|
current = ind
|
|
|
|
found = False
|
|
|
|
# loop until finding an unread tab or having done a complete cycle
|
|
|
|
while True:
|
|
|
|
if forward == True: # look for the first unread tab on the right
|
|
|
|
ind = ind + 1
|
|
|
|
if ind >= self.notebook.get_n_pages():
|
|
|
|
ind = 0
|
|
|
|
else: # look for the first unread tab on the right
|
|
|
|
ind = ind - 1
|
|
|
|
if ind < 0:
|
|
|
|
ind = self.notebook.get_n_pages() - 1
|
|
|
|
if ind == current:
|
|
|
|
break # a complete cycle without finding an unread tab
|
2006-01-25 03:43:55 +01:00
|
|
|
ctrl = self.get_control(ind, None)
|
2006-01-11 23:30:49 +01:00
|
|
|
if ctrl.nb_unread > 0:
|
2006-01-01 20:40:05 +01:00
|
|
|
found = True
|
|
|
|
break # found
|
|
|
|
if found:
|
|
|
|
self.notebook.set_current_page(ind)
|
|
|
|
else: # not found
|
|
|
|
if forward: # CTRL + TAB
|
|
|
|
if current < (self.notebook.get_n_pages() - 1):
|
|
|
|
self.notebook.next_page()
|
|
|
|
else: # traverse for ever (eg. don't stop at last tab)
|
|
|
|
self.notebook.set_current_page(0)
|
|
|
|
else: # CTRL + SHIFT + TAB
|
|
|
|
if current > 0:
|
|
|
|
self.notebook.prev_page()
|
|
|
|
else: # traverse for ever (eg. don't stop at first tab)
|
|
|
|
self.notebook.set_current_page(
|
|
|
|
self.notebook.get_n_pages() - 1)
|
2006-01-02 10:04:30 +01:00
|
|
|
def popup_menu(self, event):
|
|
|
|
menu = self.get_active_control().prepare_context_menu()
|
|
|
|
# common menuitems (tab switches)
|
2006-01-25 06:39:07 +01:00
|
|
|
if self.get_num_controls() > 1: # if there is more than one tab
|
2006-01-02 10:04:30 +01:00
|
|
|
menu.append(gtk.SeparatorMenuItem()) # seperator
|
2006-01-24 03:57:26 +01:00
|
|
|
for ctrl in self.controls():
|
2006-01-11 23:30:49 +01:00
|
|
|
jid = ctrl.contact.jid
|
2006-01-02 10:04:30 +01:00
|
|
|
if jid != self.get_active_jid():
|
|
|
|
item = gtk.ImageMenuItem(_('Switch to %s') %\
|
2006-01-11 23:30:49 +01:00
|
|
|
ctrl.contact.get_shown_name())
|
2006-01-02 10:04:30 +01:00
|
|
|
img = gtk.image_new_from_stock(gtk.STOCK_JUMP_TO,
|
|
|
|
gtk.ICON_SIZE_MENU)
|
|
|
|
item.set_image(img)
|
|
|
|
item.connect('activate',
|
2006-01-25 03:43:55 +01:00
|
|
|
lambda obj, jid:self.set_active_tab(jid, ctrl.account),
|
|
|
|
jid)
|
2006-01-02 10:04:30 +01:00
|
|
|
menu.append(item)
|
|
|
|
# show the menu
|
|
|
|
menu.popup(None, None, None, event.button, event.time)
|
|
|
|
menu.show_all()
|
|
|
|
|
|
|
|
def _on_notebook_switch_page(self, notebook, page, page_num):
|
|
|
|
old_no = notebook.get_current_page()
|
2006-01-15 03:39:02 +01:00
|
|
|
if old_no >= 0:
|
|
|
|
old_ctrl = self._widget_to_control(notebook.get_nth_page(old_no))
|
|
|
|
old_ctrl.set_control_active(False)
|
2006-01-02 10:04:30 +01:00
|
|
|
|
2006-01-11 23:30:49 +01:00
|
|
|
new_ctrl = self._widget_to_control(notebook.get_nth_page(page_num))
|
|
|
|
new_ctrl.set_control_active(True)
|
|
|
|
self.show_title(control = new_ctrl)
|
2005-12-31 22:55:44 +01:00
|
|
|
|
2006-01-02 23:08:50 +01:00
|
|
|
def _on_notebook_key_press(self, widget, event):
|
|
|
|
st = '1234567890' # alt+1 means the first tab (tab 0)
|
2006-01-11 23:30:49 +01:00
|
|
|
ctrl = self.get_active_control()
|
|
|
|
contact = ctrl.contact
|
|
|
|
jid = ctrl.contact.jid
|
2006-01-03 04:34:32 +01:00
|
|
|
|
2006-01-03 04:55:11 +01:00
|
|
|
# CTRL mask
|
|
|
|
if event.state & gtk.gdk.CONTROL_MASK:
|
|
|
|
# Tab switch bindings
|
|
|
|
if event.keyval == gtk.keysyms.ISO_Left_Tab: # CTRL + SHIFT + TAB
|
|
|
|
self.move_to_next_unread_tab(False)
|
|
|
|
elif event.keyval == gtk.keysyms.Tab: # CTRL + TAB
|
|
|
|
self.move_to_next_unread_tab(True)
|
|
|
|
elif event.keyval == gtk.keysyms.F4: # CTRL + F4
|
2006-01-25 03:43:55 +01:00
|
|
|
self.remove_tab(ctrl)
|
2006-01-03 04:55:11 +01:00
|
|
|
elif event.keyval == gtk.keysyms.w: # CTRL + W
|
2006-01-25 03:43:55 +01:00
|
|
|
self.remove_tab(ctrl)
|
2006-01-03 04:55:11 +01:00
|
|
|
|
2006-01-03 04:34:32 +01:00
|
|
|
# MOD1 (ALT) mask
|
2006-01-03 04:55:11 +01:00
|
|
|
elif event.state & gtk.gdk.MOD1_MASK:
|
2006-01-03 04:34:32 +01:00
|
|
|
# Tab switch bindings
|
|
|
|
if event.keyval == gtk.keysyms.Right: # ALT + RIGHT
|
|
|
|
new = self.notebook.get_current_page() + 1
|
|
|
|
if new >= self.notebook.get_n_pages():
|
|
|
|
new = 0
|
|
|
|
self.notebook.set_current_page(new)
|
|
|
|
elif event.keyval == gtk.keysyms.Left: # ALT + LEFT
|
|
|
|
new = self.notebook.get_current_page() - 1
|
|
|
|
if new < 0:
|
|
|
|
new = self.notebook.get_n_pages() - 1
|
|
|
|
self.notebook.set_current_page(new)
|
|
|
|
elif event.string and event.string in st and \
|
|
|
|
(event.state & gtk.gdk.MOD1_MASK): # ALT + 1,2,3..
|
|
|
|
self.notebook.set_current_page(st.index(event.string))
|
2006-01-03 04:55:11 +01:00
|
|
|
elif event.keyval == gtk.keysyms.c: # ALT + C toggles compact view
|
2006-01-11 23:30:49 +01:00
|
|
|
ctrl.set_compact_view(not ctrl.compact_view_current)
|
2006-01-03 04:34:32 +01:00
|
|
|
# Close tab bindings
|
|
|
|
elif event.keyval == gtk.keysyms.Escape: # ESCAPE
|
2006-01-25 03:43:55 +01:00
|
|
|
self.remove_tab(ctrl)
|
2006-01-10 20:24:06 +01:00
|
|
|
else:
|
|
|
|
# If the active control has a message_textview pass the event to it
|
2006-01-11 23:30:49 +01:00
|
|
|
active_ctrl = self.get_active_control()
|
|
|
|
if isinstance(active_ctrl, ChatControlBase):
|
|
|
|
active_ctrl.msg_textview.emit('key_press_event', event)
|
|
|
|
active_ctrl.msg_textview.grab_focus()
|
2006-01-03 04:34:32 +01:00
|
|
|
|
2006-01-12 04:09:33 +01:00
|
|
|
def setup_tab_dnd(self, child):
|
|
|
|
'''Set tab label as drag source and connect the drag_data_get signal'''
|
|
|
|
tab_label = self.notebook.get_tab_label(child)
|
|
|
|
tab_label.dnd_handler = tab_label.connect('drag_data_get',
|
|
|
|
self.on_tab_label_drag_data_get_cb)
|
|
|
|
tab_label.drag_source_set(gtk.gdk.BUTTON1_MASK, self.DND_TARGETS,
|
2006-01-20 21:44:56 +01:00
|
|
|
gtk.gdk.ACTION_MOVE)
|
2006-01-12 04:09:33 +01:00
|
|
|
tab_label.page_num = self.notebook.page_num(child)
|
|
|
|
|
|
|
|
def on_tab_label_drag_data_get_cb(self, widget, drag_context, selection, info, time):
|
|
|
|
source_page_num = self.find_page_num_according_to_tab_label(widget)
|
|
|
|
# 8 is the data size for the string
|
|
|
|
selection.set(selection.target, 8, str(source_page_num))
|
|
|
|
|
|
|
|
def on_tab_label_drag_data_received_cb(self, widget, drag_context, x, y, selection,
|
|
|
|
type, time):
|
|
|
|
'''Reorder the tabs according to the drop position'''
|
|
|
|
source_page_num = int(selection.data)
|
|
|
|
dest_page_num, to_right = self.get_tab_at_xy(x, y)
|
|
|
|
source_child = self.notebook.get_nth_page(source_page_num)
|
|
|
|
source_tab_label = self.notebook.get_tab_label(source_child)
|
|
|
|
if dest_page_num != source_page_num:
|
|
|
|
self.notebook.reorder_child(source_child, dest_page_num)
|
|
|
|
|
|
|
|
def get_tab_at_xy(self, x, y):
|
|
|
|
'''Thanks to Gaim
|
|
|
|
Return the tab under xy and
|
|
|
|
if its nearer from left or right side of the tab
|
|
|
|
'''
|
|
|
|
page_num = -1
|
|
|
|
to_right = False
|
|
|
|
horiz = self.notebook.get_tab_pos() == gtk.POS_TOP or \
|
|
|
|
self.notebook.get_tab_pos() == gtk.POS_BOTTOM
|
|
|
|
for i in xrange(self.notebook.get_n_pages()):
|
|
|
|
page = self.notebook.get_nth_page(i)
|
|
|
|
tab = self.notebook.get_tab_label(page)
|
|
|
|
tab_alloc = tab.get_allocation()
|
|
|
|
if horiz:
|
|
|
|
if (x >= tab_alloc.x) and \
|
|
|
|
(x <= (tab_alloc.x + tab_alloc.width)):
|
|
|
|
page_num = i
|
|
|
|
if x >= tab_alloc.x + (tab_alloc.width / 2.0):
|
|
|
|
to_right = True
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
if (y >= tab_alloc.y) and \
|
|
|
|
(y <= (tab_alloc.y + tab_alloc.height)):
|
|
|
|
page_num = i
|
|
|
|
|
|
|
|
if y > tab_alloc.y + (tab_alloc.height / 2.0):
|
|
|
|
to_right = True
|
|
|
|
break
|
|
|
|
return (page_num, to_right)
|
|
|
|
|
|
|
|
def find_page_num_according_to_tab_label(self, tab_label):
|
|
|
|
'''Find the page num of the tab label'''
|
|
|
|
page_num = -1
|
|
|
|
for i in xrange(self.notebook.get_n_pages()):
|
|
|
|
page = self.notebook.get_nth_page(i)
|
|
|
|
tab = self.notebook.get_tab_label(page)
|
|
|
|
if tab == tab_label:
|
|
|
|
page_num = i
|
|
|
|
break
|
|
|
|
return page_num
|
|
|
|
|
|
|
|
def disconnect_tab_dnd(self, child):
|
|
|
|
'''Clean up DnD signals, source and dest'''
|
|
|
|
tab_label = self.notebook.get_tab_label(child)
|
|
|
|
tab_label.drag_source_unset()
|
|
|
|
tab_label.disconnect(tab_label.dnd_handler)
|
|
|
|
|
2006-01-02 03:12:34 +01:00
|
|
|
################################################################################
|
2005-12-28 00:55:34 +01:00
|
|
|
class MessageWindowMgr:
|
|
|
|
'''A manager and factory for MessageWindow objects'''
|
|
|
|
|
2005-12-29 04:20:06 +01:00
|
|
|
# These constants map to common.config.opt_one_window_types indices
|
|
|
|
CONFIG_NEVER = 0
|
|
|
|
CONFIG_ALWAYS = 1
|
|
|
|
CONFIG_PERACCT = 2
|
|
|
|
CONFIG_PERTYPE = 3
|
|
|
|
# A key constant for the main window for all messages
|
|
|
|
MAIN_WIN = 'main'
|
|
|
|
|
2005-12-28 00:55:34 +01:00
|
|
|
def __init__(self):
|
2005-12-29 04:20:06 +01:00
|
|
|
''' A dictionary of windows; the key depends on the config:
|
|
|
|
CONFIG_NEVER: The key is the contact JID
|
|
|
|
CONFIG_ALWAYS: The key is MessageWindowMgr.MAIN_WIN
|
|
|
|
CONFIG_PERACCT: The key is the account name
|
|
|
|
CONFIG_PERTYPE: The key is a message type constant'''
|
2005-12-31 22:55:44 +01:00
|
|
|
self._windows = {}
|
2005-12-29 04:20:06 +01:00
|
|
|
# Map the mode to a int constant for frequent compares
|
|
|
|
mode = gajim.config.get('one_message_window')
|
|
|
|
self.mode = common.config.opt_one_window_types.index(mode)
|
|
|
|
|
2006-01-08 05:31:02 +01:00
|
|
|
def _new_window(self, acct, type):
|
|
|
|
win = MessageWindow(acct, type)
|
2005-12-29 04:20:06 +01:00
|
|
|
# we track the lifetime of this window
|
2006-01-03 06:49:09 +01:00
|
|
|
win.window.connect('delete-event', self._on_window_delete)
|
2005-12-29 04:20:06 +01:00
|
|
|
win.window.connect('destroy', self._on_window_destroy)
|
|
|
|
return win
|
|
|
|
|
2006-01-08 05:31:02 +01:00
|
|
|
def _gtk_win_to_msg_win(self, gtk_win):
|
2006-01-24 03:57:26 +01:00
|
|
|
for w in self.windows():
|
2005-12-31 01:50:33 +01:00
|
|
|
if w.window == gtk_win:
|
2005-12-29 04:20:06 +01:00
|
|
|
return w
|
|
|
|
return None
|
|
|
|
|
2006-01-25 03:43:55 +01:00
|
|
|
def get_window(self, jid, acct):
|
2006-01-24 03:57:26 +01:00
|
|
|
for win in self.windows():
|
2006-01-25 03:43:55 +01:00
|
|
|
if win.get_control(jid, acct):
|
2005-12-31 18:00:04 +01:00
|
|
|
return win
|
|
|
|
return None
|
2006-01-12 09:28:43 +01:00
|
|
|
|
2006-01-25 03:43:55 +01:00
|
|
|
def has_window(self, jid, acct):
|
2006-01-25 06:39:07 +01:00
|
|
|
return self.get_window(jid, acct) != None
|
2005-12-31 18:00:04 +01:00
|
|
|
|
2006-01-08 05:31:02 +01:00
|
|
|
def one_window_opened(self, contact, acct, type):
|
2006-01-09 01:47:54 +01:00
|
|
|
try:
|
|
|
|
return self._windows[self._mode_to_key(contact, acct, type)] != None
|
|
|
|
except KeyError:
|
|
|
|
return False
|
2006-01-08 05:31:02 +01:00
|
|
|
|
|
|
|
def _size_window(self, win, acct, type):
|
2006-01-25 14:24:38 +01:00
|
|
|
'''Resizes window from config settings'''
|
2006-01-08 05:31:02 +01:00
|
|
|
if not gajim.config.get('saveposition'):
|
2006-01-25 14:27:23 +01:00
|
|
|
return
|
|
|
|
|
|
|
|
if self.mode == self.CONFIG_NEVER or self.mode == self.CONFIG_ALWAYS:
|
2006-01-12 07:59:59 +01:00
|
|
|
size = (gajim.config.get('msgwin-width'),
|
|
|
|
gajim.config.get('msgwin-height'))
|
|
|
|
elif self.mode == self.CONFIG_PERACCT:
|
|
|
|
size = (gajim.config.get_per('accounts', acct, 'msgwin-width'),
|
|
|
|
gajim.config.get_per('accounts', acct, 'msgwin-height'))
|
|
|
|
elif self.mode == self.CONFIG_PERTYPE:
|
|
|
|
if type == message_control.TYPE_PM:
|
|
|
|
type = message_control.TYPE_CHAT
|
|
|
|
opt_width = type + '-msgwin-width'
|
|
|
|
opt_height = type + '-msgwin-height'
|
|
|
|
size = (gajim.config.get(opt_width),
|
2006-01-08 05:31:02 +01:00
|
|
|
gajim.config.get(opt_height))
|
2006-01-25 14:27:23 +01:00
|
|
|
else:
|
|
|
|
return
|
2006-01-25 14:24:38 +01:00
|
|
|
|
2006-01-25 14:27:23 +01:00
|
|
|
gtkgui_helpers.resize_window(win.window, size[0], size[1])
|
2006-01-08 05:31:02 +01:00
|
|
|
|
|
|
|
def _position_window(self, win, acct, type):
|
|
|
|
'''Returns the position tuple: (x, y)'''
|
|
|
|
if not gajim.config.get('saveposition') or self.mode == self.CONFIG_NEVER:
|
|
|
|
return
|
|
|
|
|
|
|
|
pos = (-1, -1) # default is left up to the native window manager
|
|
|
|
if self.mode == self.CONFIG_ALWAYS:
|
|
|
|
pos = (gajim.config.get('msgwin-x-position'),
|
|
|
|
gajim.config.get('msgwin-y-position'))
|
|
|
|
elif self.mode == self.CONFIG_PERACCT:
|
2006-01-12 06:45:30 +01:00
|
|
|
pos = (gajim.config.get_per('accounts', acct, 'msgwin-x-position'),
|
|
|
|
gajim.config.get_per('accounts', acct, 'msgwin-y-position'))
|
2006-01-08 05:31:02 +01:00
|
|
|
elif self.mode == self.CONFIG_PERTYPE:
|
|
|
|
pos = (gajim.config.get(type + '-msgwin-x-position'),
|
|
|
|
gajim.config.get(type + '-msgwin-y-position'))
|
|
|
|
|
2006-01-14 21:46:20 +01:00
|
|
|
if pos[0] > 0 and pos[1] > 0:
|
2006-01-08 05:31:02 +01:00
|
|
|
gtkgui_helpers.move_window(win.window, pos[0], pos[1])
|
|
|
|
|
|
|
|
def _mode_to_key(self, contact, acct, type):
|
2005-12-29 04:20:06 +01:00
|
|
|
if self.mode == self.CONFIG_NEVER:
|
2006-01-25 06:39:07 +01:00
|
|
|
key = acct + contact.jid
|
2005-12-29 04:20:06 +01:00
|
|
|
elif self.mode == self.CONFIG_ALWAYS:
|
|
|
|
key = self.MAIN_WIN
|
|
|
|
elif self.mode == self.CONFIG_PERACCT:
|
|
|
|
key = acct
|
|
|
|
elif self.mode == self.CONFIG_PERTYPE:
|
|
|
|
key = type
|
2006-01-11 04:48:28 +01:00
|
|
|
return key
|
2005-12-29 04:20:06 +01:00
|
|
|
|
2006-01-08 05:31:02 +01:00
|
|
|
def create_window(self, contact, acct, type):
|
|
|
|
key = None
|
|
|
|
win_acct = None
|
|
|
|
win_type = None
|
|
|
|
|
|
|
|
key = self._mode_to_key(contact, acct, type)
|
|
|
|
if self.mode == self.CONFIG_PERACCT:
|
|
|
|
win_acct = acct
|
|
|
|
elif self.mode == self.CONFIG_PERTYPE:
|
|
|
|
win_type = type
|
|
|
|
|
2005-12-29 04:20:06 +01:00
|
|
|
win = None
|
|
|
|
try:
|
2005-12-31 22:55:44 +01:00
|
|
|
win = self._windows[key]
|
2005-12-29 04:20:06 +01:00
|
|
|
except KeyError:
|
2006-01-08 05:31:02 +01:00
|
|
|
win = self._new_window(win_acct, win_type)
|
2006-01-03 06:49:09 +01:00
|
|
|
|
2006-01-25 14:24:38 +01:00
|
|
|
# Position and size window based on saved state and window mode
|
2006-01-09 01:47:54 +01:00
|
|
|
if not self.one_window_opened(contact, acct, type):
|
|
|
|
self._position_window(win, acct, type)
|
|
|
|
self._size_window(win, acct, type)
|
|
|
|
|
|
|
|
self._windows[key] = win
|
2005-12-29 04:20:06 +01:00
|
|
|
return win
|
|
|
|
|
2006-01-08 05:31:02 +01:00
|
|
|
def _on_window_delete(self, win, event):
|
2006-01-12 06:45:30 +01:00
|
|
|
self.save_state(self._gtk_win_to_msg_win(win))
|
2006-01-20 04:37:41 +01:00
|
|
|
gajim.interface.save_config()
|
2006-01-12 06:45:30 +01:00
|
|
|
return False
|
|
|
|
|
|
|
|
def _on_window_destroy(self, win):
|
|
|
|
for k in self._windows.keys():
|
|
|
|
if self._windows[k].window == win:
|
|
|
|
del self._windows[k]
|
|
|
|
return
|
|
|
|
|
2006-01-25 03:43:55 +01:00
|
|
|
def get_control(self, jid, acct):
|
2006-01-12 06:45:30 +01:00
|
|
|
'''Amongst all windows, return the MessageControl for jid'''
|
2006-01-25 03:43:55 +01:00
|
|
|
win = self.get_window(jid, acct)
|
2006-01-12 06:45:30 +01:00
|
|
|
if win:
|
2006-01-25 03:43:55 +01:00
|
|
|
return win.get_control(jid, acct)
|
2006-01-12 06:45:30 +01:00
|
|
|
return None
|
|
|
|
|
|
|
|
def get_controls(self, type):
|
2006-01-25 03:43:55 +01:00
|
|
|
# FIXME: Optionally accept an account arg
|
2006-01-12 06:45:30 +01:00
|
|
|
ctrls = []
|
|
|
|
for c in self.controls():
|
|
|
|
if c.type_id == type:
|
|
|
|
ctrls.append(c)
|
|
|
|
return ctrls
|
|
|
|
|
|
|
|
def windows(self):
|
|
|
|
for w in self._windows.values():
|
|
|
|
yield w
|
|
|
|
def controls(self):
|
|
|
|
for w in self._windows.values():
|
|
|
|
for c in w.controls():
|
|
|
|
yield c
|
|
|
|
|
|
|
|
def shutdown(self):
|
|
|
|
for w in self.windows():
|
|
|
|
self.save_state(w)
|
|
|
|
w.window.hide()
|
2006-01-15 03:39:02 +01:00
|
|
|
w.window.destroy()
|
2006-01-20 04:37:41 +01:00
|
|
|
gajim.interface.save_config()
|
2006-01-12 06:45:30 +01:00
|
|
|
|
|
|
|
def save_state(self, msg_win):
|
2006-01-08 05:31:02 +01:00
|
|
|
if not gajim.config.get('saveposition'):
|
2006-01-14 21:46:20 +01:00
|
|
|
return
|
2006-01-08 05:31:02 +01:00
|
|
|
|
2006-01-10 20:24:06 +01:00
|
|
|
# Save window size and postion
|
2006-01-08 05:31:02 +01:00
|
|
|
pos_x_key = 'msgwin-x-position'
|
|
|
|
pos_y_key = 'msgwin-y-position'
|
|
|
|
size_width_key = 'msgwin-width'
|
|
|
|
size_height_key = 'msgwin-height'
|
|
|
|
|
|
|
|
acct = None
|
2006-01-12 06:45:30 +01:00
|
|
|
x, y = msg_win.window.get_position()
|
|
|
|
width, height = msg_win.window.get_size()
|
2006-01-08 05:31:02 +01:00
|
|
|
|
2006-01-14 21:46:20 +01:00
|
|
|
# If any of these values seem bogus don't update.
|
|
|
|
if x < 0 or y < 0 or width < 0 or height < 0:
|
|
|
|
return
|
|
|
|
|
2006-01-08 05:31:02 +01:00
|
|
|
if self.mode == self.CONFIG_NEVER:
|
2006-01-20 04:37:41 +01:00
|
|
|
# Use whatever is current to not overwrite the 'always' settings
|
|
|
|
# when going from never->always
|
2006-01-08 05:31:02 +01:00
|
|
|
x = y = -1
|
|
|
|
elif self.mode == self.CONFIG_PERACCT:
|
|
|
|
acct = msg_win.account
|
|
|
|
elif self.mode == self.CONFIG_PERTYPE:
|
2006-01-12 03:20:59 +01:00
|
|
|
type = msg_win.type
|
2006-01-08 05:31:02 +01:00
|
|
|
pos_x_key = type + "-msgwin-x-position"
|
|
|
|
pos_y_key = type + "-msgwin-y-position"
|
|
|
|
size_width_key = type + "-msgwin-width"
|
|
|
|
size_height_key = type + "-msgwin-height"
|
|
|
|
|
|
|
|
if acct:
|
2006-01-20 04:37:41 +01:00
|
|
|
if x >= 0 and y >= 0:
|
|
|
|
gajim.config.set_per('accounts', acct, pos_x_key, x)
|
|
|
|
gajim.config.set_per('accounts', acct, pos_y_key, y)
|
2006-01-10 20:24:06 +01:00
|
|
|
gajim.config.set_per('accounts', acct, size_width_key, width)
|
|
|
|
gajim.config.set_per('accounts', acct, size_height_key, height)
|
2006-01-08 05:31:02 +01:00
|
|
|
else:
|
2006-01-20 04:37:41 +01:00
|
|
|
if x >= 0 and y >= 0:
|
|
|
|
gajim.config.set(pos_x_key, x)
|
|
|
|
gajim.config.set(pos_y_key, y)
|
2006-01-08 05:31:02 +01:00
|
|
|
gajim.config.set(size_width_key, width)
|
|
|
|
gajim.config.set(size_height_key, height)
|
2006-01-15 03:39:02 +01:00
|
|
|
|
|
|
|
def reconfig(self):
|
2006-01-24 03:57:26 +01:00
|
|
|
for w in self.windows():
|
2006-01-15 03:39:02 +01:00
|
|
|
self.save_state(w)
|
2006-01-20 04:37:41 +01:00
|
|
|
gajim.interface.save_config()
|
2006-01-15 03:39:02 +01:00
|
|
|
# Map the mode to a int constant for frequent compares
|
|
|
|
mode = gajim.config.get('one_message_window')
|
|
|
|
if self.mode == common.config.opt_one_window_types.index(mode):
|
|
|
|
# No change
|
|
|
|
return
|
|
|
|
self.mode = common.config.opt_one_window_types.index(mode)
|
|
|
|
|
|
|
|
controls = []
|
2006-01-24 03:57:26 +01:00
|
|
|
for w in self.windows():
|
2006-01-15 03:39:02 +01:00
|
|
|
w.window.hide()
|
|
|
|
while w.notebook.get_n_pages():
|
|
|
|
page = w.notebook.get_nth_page(0)
|
|
|
|
ctrl = w._widget_to_control(page)
|
|
|
|
w.notebook.remove_page(0)
|
|
|
|
page.unparent()
|
|
|
|
controls.append(ctrl)
|
2006-01-25 06:39:07 +01:00
|
|
|
# Must clear _controls from window to prevent MessageControl.shutdown calls
|
|
|
|
w._controls = {}
|
2006-01-15 03:39:02 +01:00
|
|
|
w.window.destroy()
|
|
|
|
|
|
|
|
self._windows = {}
|
|
|
|
|
|
|
|
for ctrl in controls:
|
2006-01-25 06:39:07 +01:00
|
|
|
mw = self.get_window(ctrl.contact.jid, ctrl.account)
|
2006-01-15 03:39:02 +01:00
|
|
|
if not mw:
|
|
|
|
mw = self.create_window(ctrl.contact, ctrl.account, ctrl.type_id)
|
|
|
|
ctrl.parent_win = mw
|
|
|
|
mw.new_tab(ctrl)
|