root/branches/gajim_0.8.2/src/chat.py

Revision 3303, 46.1 kB (checked in by nk, 3 years ago)

[stephan k.] fix a typo

Line 
1##      plugins/tabbed_chat_window.py
2##
3## Gajim Team:
4##      - Yann Le Boulanger <asterix@lagaule.org>
5##      - Vincent Hanquez <tab@snarc.org>
6##      - Nikos Kouremenos <kourem@gmail.com>
7##
8##      Copyright (C) 2003-2005 Gajim Team
9##
10## This program is free software; you can redistribute it and/or modify
11## it under the terms of the GNU General Public License as published
12## by the Free Software Foundation; version 2 only.
13##
14## This program is distributed in the hope that it will be useful,
15## but WITHOUT ANY WARRANTY; without even the implied warranty of
16## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17## GNU General Public License for more details.
18##
19
20import gtk
21import gtk.glade
22import pango
23import gobject
24import time
25import dialogs
26import history_window
27
28try:
29        import gtkspell
30except:
31        pass
32
33from common import gajim
34from common import helpers
35from common import i18n
36
37_ = i18n._
38APP = i18n.APP
39gtk.glade.bindtextdomain(APP, i18n.DIR)
40gtk.glade.textdomain(APP)
41
42GTKGUI_GLADE = 'gtkgui.glade'
43
44class Chat:
45        """Class for chat/groupchat windows"""
46        def __init__(self, plugin, account, widget_name):
47                self.xml = gtk.glade.XML(GTKGUI_GLADE, widget_name, APP)
48                self.window = self.xml.get_widget(widget_name)
49
50                self.widget_name = widget_name
51
52                self.plugin = plugin
53                self.account = account
54                self.change_cursor = None
55                self.xmls = {}
56                self.tagIn = {} # holds tag for nick that talks to us
57                self.tagOut = {} # holds tag for our nick
58                self.tagStatus = {} # holds status messages
59                self.nb_unread = {}
60                self.last_time_printout = {}
61                self.print_time_timeout_id = {}
62                self.names = {} # what is printed in the tab (eg. user.name)
63                self.childs = {} # holds the contents for every tab (VBox)
64                self.popup_is_shown = False # is a context menu shown or not?
65
66                # the following vars are used to keep history of user's messages
67                self.sent_history = {}
68                self.sent_history_pos = {}
69                self.typing_new = {}
70                self.orig_msg = {}
71               
72                # we check that on opening new windows
73                self.always_compact_view = gajim.config.get('always_compact_view')
74
75                # notebook customizations
76                self.notebook = self.xml.get_widget('chat_notebook')
77                self.notebook.remove_page(0)
78                pref_pos = gajim.config.get('tabs_position')
79                if pref_pos != 'top':
80                        if pref_pos == 'bottom':
81                                nb_pos = gtk.POS_BOTTOM
82                        elif pref_pos == 'left':
83                                nb_pos = gtk.POS_LEFT
84                        elif pref_pos == 'right':
85                                nb_pos = gtk.POS_RIGHT
86                        else:
87                                nb_pos = gtk.POS_TOP
88                else:
89                        nb_pos = gtk.POS_TOP
90                self.notebook.set_tab_pos(nb_pos)
91                self.notebook.set_show_tabs(gajim.config.get('tabs_always_visible'))
92                self.notebook.set_show_border(gajim.config.get('tabs_border'))
93
94        def update_font(self):
95                font = pango.FontDescription(gajim.config.get('conversation_font'))
96                for jid in self.tagIn:
97                        conversation_textview = self.xmls[jid].get_widget(
98                                'conversation_textview')
99                        conversation_textview.modify_font(font)
100                        message_textview = self.xmls[jid].get_widget('message_textview')
101                        message_textview.modify_font(font)
102
103        def update_tags(self):
104                for jid in self.tagIn:
105                        self.tagIn[jid].set_property('foreground',
106                                        gajim.config.get('inmsgcolor'))
107                        self.tagOut[jid].set_property('foreground',
108                                        gajim.config.get('outmsgcolor'))
109                        self.tagStatus[jid].set_property('foreground',
110                                        gajim.config.get('statusmsgcolor'))
111
112        def update_print_time(self):
113                if gajim.config.get('print_time') != 'sometimes':
114                        list_jid = self.print_time_timeout_id.keys()
115                        for jid in list_jid:
116                                gobject.source_remove(self.print_time_timeout_id[jid])
117                                del self.print_time_timeout_id[jid]
118                else:
119                        for jid in self.xmls:
120                                if self.print_time_timeout_id.has_key(jid):
121                                        continue
122                                self.print_time_timeout(jid)
123                                self.print_time_timeout_id[jid] = \
124                                                gobject.timeout_add(300000,
125                                                        self.print_time_timeout,
126                                                        jid)
127
128        def show_title(self):
129                """redraw the window's title"""
130                unread = 0
131                for jid in self.nb_unread:
132                        unread += self.nb_unread[jid]
133                start = ""
134                if unread > 1:
135                        start = '[' + unicode(unread) + '] '
136                elif unread == 1:
137                        start = '* '
138                chat = self.names[jid]
139                if len(self.xmls) > 1: # if more than one tab in the same window
140                        if self.widget_name == 'tabbed_chat_window':
141                                add = _('Chat')
142                        elif self.widget_name == 'groupchat_window':
143                                add = _('Group Chat')
144                elif len(self.xmls) == 1: # just one tab
145                        if self.widget_name == 'tabbed_chat_window':
146                                c = gajim.get_first_contact_instance_from_jid(self.account, jid)
147                                if c is None: # FIXME: I don't know why but c can be None!
148                                        add = ''
149                                else:
150                                        add = c.name
151                        elif self.widget_name == 'groupchat_window':
152                                name = gajim.get_nick_from_jid(jid)
153                                add = name
154
155                title = start + add
156                if len(gajim.connections) >= 2: # if we have 2 or more accounts
157                        title += ' (' + _('account: ') + self.account + ')'
158
159                self.window.set_title(title)
160
161        def redraw_tab(self, jid):
162                """redraw the label of the tab"""
163                start = ''
164                if self.nb_unread[jid] > 1:
165                        start = '[' + unicode(self.nb_unread[jid]) + '] '
166                elif self.nb_unread[jid] == 1:
167                        start = '* '
168                       
169                child = self.childs[jid]
170                hb = self.notebook.get_tab_label(child).get_children()[0]
171                if self.widget_name == 'tabbed_chat_window':
172                        nickname = hb.get_children()[1]
173                        close_button = hb.get_children()[2]
174                elif self.widget_name == 'groupchat_window':
175                        nickname = hb.get_children()[0]
176                        close_button = hb.get_children()[1]
177
178                if gajim.config.get('tabs_close_button'):
179                        close_button.show()
180                else:
181                        close_button.hide()
182
183                #FIXME: when gtk2.4 is OOOOLD do it via glade2.10+
184                if gtk.pygtk_version >= (2, 6, 0) and gtk.gtk_version >= (2, 6, 0):
185                        nickname.set_max_width_chars(10)
186
187                nickname.set_text(start + self.names[jid])
188
189
190        def on_window_destroy(self, widget, kind): #kind is 'chats' or 'gc'
191                '''clean self.plugin.windows[self.account][kind]'''
192                for jid in self.xmls:
193                        windows = self.plugin.windows[self.account][kind]
194                        if kind == 'chats':
195                                # send 'gone' chatstate to every tabbed chat tab
196                                windows[jid].send_chatstate('gone', jid)
197                                gobject.source_remove(self.possible_paused_timeout_id[jid])
198                                gobject.source_remove(self.possible_inactive_timeout_id[jid])
199                        if self.plugin.systray_enabled and self.nb_unread[jid] > 0:
200                                self.plugin.systray.remove_jid(jid, self.account)
201                        del windows[jid]
202                        if self.print_time_timeout_id.has_key(jid):
203                                gobject.source_remove(self.print_time_timeout_id[jid])
204                if windows.has_key('tabbed'):
205                        del windows['tabbed']
206
207        def get_active_jid(self):
208                notebook = self.notebook
209                active_child = notebook.get_nth_page(notebook.get_current_page())
210                active_jid = ''
211                for jid in self.xmls:
212                        if self.childs[jid] == active_child:
213                                active_jid = jid
214                                break
215                return active_jid
216
217        def on_close_button_clicked(self, button, jid):
218                """When close button is pressed: close a tab"""
219                self.remove_tab(jid)
220
221        def on_history_menuitem_clicked(self, widget = None, jid = None):
222                """When history menuitem is pressed: call history window"""
223                if jid is None:
224                        jid = self.get_active_jid()
225                if self.plugin.windows['logs'].has_key(jid):
226                        self.plugin.windows['logs'][jid].window.present()
227                else:
228                        self.plugin.windows['logs'][jid] = history_window.HistoryWindow(
229                                self.plugin, jid, self.account)
230
231        def on_chat_window_focus_in_event(self, widget, event):
232                """When window gets focus"""
233                jid = self.get_active_jid()
234                textview = self.xmls[jid].get_widget('conversation_textview')
235                buffer = textview.get_buffer()
236                end_iter = buffer.get_end_iter()
237                end_rect = textview.get_iter_location(end_iter)
238                visible_rect = textview.get_visible_rect()
239                if end_rect.y <= (visible_rect.y + visible_rect.height):
240                        #we are at the end
241                        if self.nb_unread[jid] > 0:
242                                self.nb_unread[jid] = 0
243                                self.redraw_tab(jid)
244                                self.show_title()
245                                if self.plugin.systray_enabled:
246                                        self.plugin.systray.remove_jid(jid, self.account)
247       
248        def on_compact_view_menuitem_activate(self, widget):
249                isactive = widget.get_active()
250                self.set_compact_view(isactive)
251
252        def on_actions_button_clicked(self, widget):
253                '''popup action menu'''
254                menu = self.prepare_context_menu()
255                self.popup_is_shown = True
256                menu.connect('deactivate', self.on_popup_deactivate)
257                menu.popup(None, None, None, 1, 0)
258                menu.show_all()
259
260        def remove_possible_switch_to_menuitems(self, menu):
261                ''' remove duplicate 'Switch to' if they exist and return clean menu'''
262                childs = menu.get_children()
263
264                if self.widget_name == 'tabbed_chat_window':
265                        jid = self.get_active_jid()
266                        c = gajim.get_first_contact_instance_from_jid(self.account, jid)
267                        if _('not in the roster') in c.groups: # for add_to_roster_menuitem
268                                childs[5].show()
269                                childs[5].set_no_show_all(False)
270                        else:
271                                childs[5].hide()
272                                childs[5].set_no_show_all(True)
273                       
274                        start_removing_from = 6 # this is from the seperator and after
275                       
276                else:
277                        start_removing_from = 7 # # this is from the seperator and after
278                               
279                for child in childs[start_removing_from:]:
280                        menu.remove(child)
281
282                return menu
283       
284        def prepare_context_menu(self):
285                '''sets compact view menuitem active state
286                sets active and sensitivity state for toggle_gpg_menuitem
287                and remove possible 'Switch to' menuitems'''
288                if self.widget_name == 'groupchat_window':
289                        menu = self.gc_popup_menu
290                        childs = menu.get_children()
291                        # compact_view_menuitem
292                        childs[5].set_active(self.compact_view_current_state)
293                elif self.widget_name == 'tabbed_chat_window':
294                        menu = self.tabbed_chat_popup_menu
295                        childs = menu.get_children()
296                        # check if gpg capabitlies or else make gpg toggle insensitive
297                        jid = self.get_active_jid()
298                        gpg_btn = self.xmls[jid].get_widget('gpg_togglebutton')
299                        isactive = gpg_btn.get_active()
300                        issensitive = gpg_btn.get_property('sensitive')
301                        childs[3].set_active(isactive)
302                        childs[3].set_property('sensitive', issensitive)
303                        # compact_view_menuitem
304                        childs[4].set_active(self.compact_view_current_state)
305                menu = self.remove_possible_switch_to_menuitems(menu)
306               
307                return menu
308
309        def popup_menu(self, event):
310                self.popup_is_shown = True
311                menu = self.prepare_context_menu()
312                menu.connect('deactivate', self.on_popup_deactivate)
313                # common menuitems (tab switches)
314                if len(self.xmls) > 1: # if there is more than one tab
315                        menu.append(gtk.SeparatorMenuItem()) # seperator
316                        for jid in self.xmls:
317                                if jid != self.get_active_jid():
318                                        item = gtk.ImageMenuItem(_('Switch to %s') % self.names[jid])
319                                        img = gtk.image_new_from_stock(gtk.STOCK_JUMP_TO,
320                                                gtk.ICON_SIZE_MENU)
321                                        item.set_image(img)
322                                        item.connect('activate', lambda obj, jid:self.set_active_tab(
323                                                jid), jid)
324                                        menu.append(item)
325
326                # show the menu
327                menu.popup(None, None, None, event.button, event.time)
328                menu.show_all()
329
330        def on_banner_eventbox_button_press_event(self, widget, event):
331                '''If right-clicked, show popup'''
332                if event.button == 3: # right click
333                        self.popup_menu(event)
334
335        def on_popup_deactivate(self, widget):
336                self.popup_is_shown = False
337
338        def on_chat_notebook_switch_page(self, notebook, page, page_num):
339                # get the index of the page and then the page that we're leaving
340                old_no = notebook.get_current_page()
341                old_child = notebook.get_nth_page(old_no)
342               
343                new_child = notebook.get_nth_page(page_num)
344               
345                old_jid = ''
346                new_jid = ''
347                for jid in self.xmls:
348                        if self.childs[jid] == new_child:
349                                new_jid = jid
350                        elif self.childs[jid] == old_child:
351                                old_jid = jid
352                       
353                        if old_jid != '' and new_jid != '': # we found both jids
354                                break # so stop looping
355               
356                if self.widget_name == 'tabbed_chat_window':
357                        # send chatstate inactive to the one we're leaving
358                        # and active to the one we visit
359                        if old_jid != '':
360                                self.send_chatstate('inactive', old_jid)
361                        self.send_chatstate('active', new_jid)
362
363                conversation_textview = self.xmls[new_jid].get_widget(
364                        'conversation_textview')
365                conversation_buffer = conversation_textview.get_buffer()
366                end_iter = conversation_buffer.get_end_iter()
367                end_rect = conversation_textview.get_iter_location(end_iter)
368                visible_rect = conversation_textview.get_visible_rect()
369                if end_rect.y <= (visible_rect.y + visible_rect.height):
370                        #we are at the end
371                        if self.nb_unread[new_jid] > 0:
372                                self.nb_unread[new_jid] = 0
373                                self.redraw_tab(new_jid)
374                                self.show_title()
375                                if self.plugin.systray_enabled:
376                                        self.plugin.systray.remove_jid(new_jid, self.account)
377               
378                conversation_textview.grab_focus()
379
380        def set_active_tab(self, jid):
381                self.notebook.set_current_page(self.notebook.page_num(self.childs[jid]))
382
383        def remove_tab(self, jid, kind): #kind is 'chats' or 'gc'
384                if len(self.xmls) == 1: # only one tab when we asked to remove
385                        # so destroy window
386               
387                        # we check and possibly save positions here, because Ctrl+W, Escape
388                        # etc.. call remove_tab so similar code in delete_event callbacks
389                        # is not enough
390                        if gajim.config.get('saveposition'):
391                                if kind == 'chats':
392                                        x, y = self.window.get_position()
393                                        gajim.config.set('chat-x-position', x)
394                                        gajim.config.set('chat-y-position', y)
395                                        width, height = self.window.get_size()
396                                        gajim.config.set('chat-width', width)
397                                        gajim.config.set('chat-height', height)
398                                elif kind == 'gc':
399                                        gajim.config.set('gc-hpaned-position', self.hpaned_position)
400                                        x, y = self.window.get_position()
401                                        gajim.config.set('gc-x-position', x)
402                                        gajim.config.set('gc-y-position', y)
403                                        width, height = self.window.get_size()
404                                        gajim.config.set('gc-width', width)
405                                        gajim.config.set('gc-height', height)
406
407                        self.window.destroy()
408                else:
409                        if self.nb_unread[jid] > 0:
410                                self.nb_unread[jid] = 0
411                                if self.plugin.systray_enabled:
412                                        self.plugin.systray.remove_jid(jid, self.account)
413                        if self.print_time_timeout_id.has_key(jid):
414                                gobject.source_remove(self.print_time_timeout_id[jid])
415                                del self.print_time_timeout_id[jid]
416
417                        self.notebook.remove_page(self.notebook.page_num(self.childs[jid]))
418                               
419                       
420
421                if self.plugin.windows[self.account][kind].has_key(jid):
422                        del self.plugin.windows[self.account][kind][jid]
423                del self.nb_unread[jid]
424                del gajim.last_message_time[self.account][jid]
425                del self.last_time_printout[jid]
426                del self.xmls[jid]
427                del self.childs[jid]
428                del self.tagIn[jid]
429                del self.tagOut[jid]
430                del self.tagStatus[jid]
431               
432                if len(self.xmls) == 1: # we now have only one tab
433                        show_tabs_if_one_tab = gajim.config.get('tabs_always_visible')
434                        self.notebook.set_show_tabs(show_tabs_if_one_tab)
435                        self.show_title()
436       
437        def bring_scroll_to_end(self, textview, diff_y = 0):
438                ''' scrolls to the end of textview if end is not visible '''
439                buffer = textview.get_buffer()
440                buffer.begin_user_action()
441                at_the_end = False
442                end_iter = buffer.get_end_iter()
443                end_rect = textview.get_iter_location(end_iter)
444                visible_rect = textview.get_visible_rect()
445                # scroll only if expected end is not visible
446                if end_rect.y >= (visible_rect.y + visible_rect.height + diff_y):
447                        gobject.idle_add(self.scroll_to_end_iter, textview)
448                       
449        def scroll_to_end_iter(self, textview):
450                buffer = textview.get_buffer()
451                end_iter = buffer.get_end_iter()
452                textview.scroll_to_iter(end_iter, 0, False, 1, 1)
453                return False
454               
455        def size_request(self, message_textview , requisition, xml_top):
456                ''' When message_textview changes its size. If the new height
457                will enlarge the window, enable the scrollbar automatic policy'''
458                if message_textview.window is None:
459                        return
460                message_scrolledwindow = xml_top.get_widget('message_scrolledwindow')
461                conversation_scrolledwindow = \
462                        xml_top.get_widget('conversation_scrolledwindow')
463                conversation_textview = \
464                        xml_top.get_widget('conversation_textview')
465
466                min_height = conversation_scrolledwindow.get_property('height-request')
467                conversation_height = conversation_textview.window.get_size()[1]
468                message_height = message_textview.window.get_size()[1]
469                # new tab is not exposed yet
470                if conversation_height < 2:
471                        return
472               
473                if conversation_height < min_height:
474                        min_height = conversation_height
475                       
476                diff_y =  message_height - requisition.height
477                if diff_y is not 0:
478                        if  conversation_height + diff_y < min_height:
479                                if message_height + conversation_height - min_height > min_height:
480                                        message_scrolledwindow.set_property('vscrollbar-policy', 
481                                                gtk.POLICY_AUTOMATIC)
482                                        message_scrolledwindow.set_property('hscrollbar-policy', 
483                                                gtk.POLICY_AUTOMATIC)
484                                        message_scrolledwindow.set_property('height-request', 
485                                                message_height + conversation_height - min_height)
486                                        self.bring_scroll_to_end(message_textview)
487                        else:
488                                message_scrolledwindow.set_property('vscrollbar-policy', 
489                                        gtk.POLICY_NEVER)
490                                message_scrolledwindow.set_property('hscrollbar-policy', 
491                                        gtk.POLICY_NEVER)
492                                message_scrolledwindow.set_property('height-request', -1)
493                self.bring_scroll_to_end(conversation_textview, diff_y - 18)
494                return True
495
496        def on_tab_eventbox_button_press_event(self, widget, event, child):
497                if event.button == 3:
498                        n = self.notebook.page_num(child)
499                        self.notebook.set_current_page(n)
500                        self.popup_menu(event)
501
502        def new_tab(self, jid):
503                #FIXME: text formating buttons will be hidden in 0.8 release
504                for w in ['bold_togglebutton', 'italic_togglebutton', 'underline_togglebutton']:
505                        self.xmls[jid].get_widget(w).set_no_show_all(True)
506
507                self.set_compact_view(self.always_compact_view)
508                self.nb_unread[jid] = 0
509                gajim.last_message_time[self.account][jid] = 0
510                self.last_time_printout[jid] = 0.
511                font = pango.FontDescription(gajim.config.get('conversation_font'))
512               
513                if gajim.config.get('use_speller') and 'gtkspell' in globals():
514                        message_textview = self.xmls[jid].get_widget('message_textview')
515                        try:
516                                gtkspell.Spell(message_textview)
517                        except gobject.GError, msg:
518                                #FIXME: add a ui for this use spell.set_language()
519                                dialogs.ErrorDialog(unicode(msg), _('If that is not your language for which you want to highlight misspelled words, then please set your $LANG as appropriate. Eg. for French do export LANG=fr_FR or export LANG=fr_FR.UTF-8 in ~/.bash_profile or to make it global in /etc/profile.\n\nHighlighting misspelled words feature will not be used')).get_response()
520                                gajim.config.set('use_speller', False)
521               
522                conversation_textview = self.xmls[jid].get_widget(
523                        'conversation_textview')
524                conversation_textview.modify_font(font)
525                conversation_buffer = conversation_textview.get_buffer()
526                end_iter = conversation_buffer.get_end_iter()
527               
528                conversation_buffer.create_mark('end', end_iter, False)
529               
530                self.tagIn[jid] = conversation_buffer.create_tag('incoming')
531                color = gajim.config.get('inmsgcolor')
532                self.tagIn[jid].set_property('foreground', color)
533                self.tagOut[jid] = conversation_buffer.create_tag('outgoing')
534                color = gajim.config.get('outmsgcolor')
535                self.tagOut[jid].set_property('foreground', color)
536                self.tagStatus[jid] = conversation_buffer.create_tag('status')
537                color = gajim.config.get('statusmsgcolor')
538                self.tagStatus[jid].set_property('foreground', color)
539
540                tag = conversation_buffer.create_tag('marked')
541                color = gajim.config.get('markedmsgcolor')
542                tag.set_property('foreground', color)
543                tag.set_property('weight', pango.WEIGHT_BOLD)
544
545                tag = conversation_buffer.create_tag('time_sometimes')
546                tag.set_property('foreground', '#9e9e9e')
547                tag.set_property('scale', pango.SCALE_SMALL)
548                tag.set_property('justification', gtk.JUSTIFY_CENTER)
549               
550                tag = conversation_buffer.create_tag('small')
551                tag.set_property('scale', pango.SCALE_SMALL)
552               
553                tag = conversation_buffer.create_tag('grey')
554                tag.set_property('foreground', '#9e9e9e')
555               
556                tag = conversation_buffer.create_tag('url')
557                tag.set_property('foreground', '#0000ff')
558                tag.set_property('underline', pango.UNDERLINE_SINGLE)
559                tag.connect('event', self.hyperlink_handler, 'url')
560
561                tag = conversation_buffer.create_tag('mail')
562                tag.set_property('foreground', '#0000ff')
563                tag.set_property('underline', pango.UNDERLINE_SINGLE)
564                tag.connect('event', self.hyperlink_handler, 'mail')
565               
566                tag = conversation_buffer.create_tag('bold')
567                tag.set_property('weight', pango.WEIGHT_BOLD)
568               
569                tag = conversation_buffer.create_tag('italic')
570                tag.set_property('style', pango.STYLE_ITALIC)
571               
572                tag = conversation_buffer.create_tag('underline')
573                tag.set_property('underline', pango.UNDERLINE_SINGLE)
574               
575                self.xmls<