Plugin's configuration is now saved to disk (currently: using UserDict and shelve modules).

Length Notifier Plugin has configuration dialog (added entry with JIDs to be included when plugin is working) - fully usable.

Default values of configuration key has been added to GajimPlugin.

Some other minor changes/fixes.
This commit is contained in:
Mateusz Biliński 2008-07-29 19:09:28 +00:00
parent d8075a23e6
commit 8aa9cad2e0
10 changed files with 415 additions and 39 deletions

View File

@ -51,14 +51,15 @@ class AcronymsExpanderPlugin(GajimPlugin):
self.disconnect_from_chat_control_base) self.disconnect_from_chat_control_base)
} }
self.INVOKER = ' ' self.config_default_values = {'INVOKER' : (' ', _('')),
self.ACRONYMS = {'RTFM' : 'Read The Friendly Manual', 'ACRONYMS' : ({'RTFM' : 'Read The Friendly Manual',
'/slap' : '/me slaps', '/slap' : '/me slaps',
'PS-' : 'plug-in system', 'PS-' : 'plug-in system',
'G-' : 'Gajim', 'G-' : 'Gajim',
'GNT-' : 'http://trac.gajim.org/newticket', 'GNT-' : 'http://trac.gajim.org/newticket',
'GW-' : 'http://trac.gajim.org/', 'GW-' : 'http://trac.gajim.org/',
'GTS-' : 'http://trac.gajim.org/report' 'GTS-' : 'http://trac.gajim.org/report'
}, _('')),
} }
@log_calls('AcronymsExpanderPlugin') @log_calls('AcronymsExpanderPlugin')
@ -66,17 +67,19 @@ class AcronymsExpanderPlugin(GajimPlugin):
""" """
@param tb gtk.TextBuffer @param tb gtk.TextBuffer
""" """
assert isinstance(tb,gtk.TextBuffer) #assert isinstance(tb,gtk.TextBuffer)
ACRONYMS = self.config['ACRONYMS']
INVOKER = self.config['INVOKER']
t = tb.get_text(tb.get_start_iter(), tb.get_end_iter()) t = tb.get_text(tb.get_start_iter(), tb.get_end_iter())
log.debug('%s %d'%(t, len(t))) log.debug('%s %d'%(t, len(t)))
if t and t[-1] == self.INVOKER: if t and t[-1] == INVOKER:
log.debug("changing msg text") log.debug("changing msg text")
base,sep,head=t[:-1].rpartition(self.INVOKER) base,sep,head=t[:-1].rpartition(INVOKER)
log.debug('%s | %s | %s'%(base, sep, head)) log.debug('%s | %s | %s'%(base, sep, head))
if head in self.ACRONYMS: if head in ACRONYMS:
head = self.ACRONYMS[head] head = ACRONYMS[head]
log.debug("head: %s"%(head)) log.debug("head: %s"%(head))
t = "".join((base, sep, head, self.INVOKER)) t = "".join((base, sep, head, INVOKER))
log.debug("turning off notify") log.debug("turning off notify")
tb.freeze_notify() tb.freeze_notify()
log.debug("setting text: '%s'"%(t)) log.debug("setting text: '%s'"%(t))
@ -102,4 +105,3 @@ class AcronymsExpanderPlugin(GajimPlugin):
tv = chat_control.msg_textview tv = chat_control.msg_textview
tv.get_buffer().disconnect(d['h_id']) tv.get_buffer().disconnect(d['h_id'])

View File

@ -0,0 +1,2 @@
from length_notifier import LengthNotifierPlugin

View File

@ -0,0 +1,152 @@
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<!DOCTYPE glade-interface SYSTEM "glade-2.0.dtd">
<!--Generated with glade3 3.4.5 on Tue Jul 29 19:58:18 2008 -->
<glade-interface>
<widget class="GtkWindow" id="window1">
<child>
<widget class="GtkTable" id="length_notifier_config_table">
<property name="visible">True</property>
<property name="border_width">9</property>
<property name="n_rows">3</property>
<property name="n_columns">2</property>
<property name="column_spacing">7</property>
<property name="row_spacing">5</property>
<child>
<widget class="GtkHBox" id="hbox2">
<property name="visible">True</property>
<child>
<widget class="GtkSpinButton" id="message_length_spinbutton">
<property name="visible">True</property>
<property name="can_focus">True</property>
<property name="tooltip" translatable="yes">Message length at which notification is invoked.</property>
<property name="width_chars">6</property>
<property name="adjustment">0 0 999999 1 10 10</property>
<property name="snap_to_ticks">True</property>
<property name="numeric">True</property>
<signal name="value_changed" handler="on_message_length_spinbutton_value_changed"/>
</widget>
<packing>
<property name="expand">False</property>
<property name="fill">False</property>
</packing>
</child>
<child>
<widget class="GtkAlignment" id="alignment2">
<property name="visible">True</property>
<child>
<placeholder/>
</child>
</widget>
<packing>
<property name="position">1</property>
</packing>
</child>
</widget>
<packing>
<property name="left_attach">1</property>
<property name="right_attach">2</property>
<property name="x_options">GTK_FILL</property>
<property name="y_options">GTK_FILL</property>
</packing>
</child>
<child>
<widget class="GtkHBox" id="hbox1">
<property name="visible">True</property>
<child>
<widget class="GtkColorButton" id="notification_colorbutton">
<property name="visible">True</property>
<property name="can_focus">True</property>
<property name="receives_default">True</property>
<property name="tooltip" translatable="yes">Background color of text entry field in chat window when notification is invoked.</property>
<property name="xalign">0</property>
<property name="response_id">0</property>
<property name="title" translatable="yes">Pick a color for notification</property>
<property name="color">#000000000000</property>
<signal name="color_set" handler="on_notification_colorbutton_color_set"/>
</widget>
<packing>
<property name="expand">False</property>
<property name="fill">False</property>
</packing>
</child>
<child>
<widget class="GtkAlignment" id="alignment1">
<property name="visible">True</property>
<child>
<placeholder/>
</child>
</widget>
<packing>
<property name="position">1</property>
</packing>
</child>
</widget>
<packing>
<property name="left_attach">1</property>
<property name="right_attach">2</property>
<property name="top_attach">1</property>
<property name="bottom_attach">2</property>
<property name="x_options">GTK_FILL</property>
<property name="y_options"></property>
</packing>
</child>
<child>
<widget class="GtkEntry" id="jids_entry">
<property name="visible">True</property>
<property name="can_focus">True</property>
<property name="tooltip" translatable="yes">JabberIDs that plugin should be used with (eg. restrict only to one microblogging bot). If empty plugin is used with every JID. [not implemented]</property>
<signal name="changed" handler="on_jids_entry_changed"/>
<signal name="editing_done" handler="on_jids_entry_editing_done"/>
</widget>
<packing>
<property name="left_attach">1</property>
<property name="right_attach">2</property>
<property name="top_attach">2</property>
<property name="bottom_attach">3</property>
<property name="y_options">GTK_FILL</property>
</packing>
</child>
<child>
<widget class="GtkLabel" id="label3">
<property name="visible">True</property>
<property name="tooltip" translatable="yes">JabberIDs that plugin should be used with (eg. restrict only to one microblogging bot). If empty plugin is used with every JID. [not implemented]</property>
<property name="xalign">0</property>
<property name="label" translatable="yes">JabberIDs to include:</property>
</widget>
<packing>
<property name="top_attach">2</property>
<property name="bottom_attach">3</property>
<property name="x_options">GTK_FILL</property>
<property name="y_options">GTK_FILL</property>
</packing>
</child>
<child>
<widget class="GtkLabel" id="label2">
<property name="visible">True</property>
<property name="tooltip" translatable="yes">Background color of text entry field in chat window when notification is invoked.</property>
<property name="xalign">0</property>
<property name="label" translatable="yes">Notification color:</property>
</widget>
<packing>
<property name="top_attach">1</property>
<property name="bottom_attach">2</property>
<property name="x_options">GTK_FILL</property>
<property name="y_options">GTK_FILL</property>
</packing>
</child>
<child>
<widget class="GtkLabel" id="label1">
<property name="visible">True</property>
<property name="tooltip" translatable="yes">Message length at which notification is invoked.</property>
<property name="xalign">0</property>
<property name="label" translatable="yes">Message length:</property>
</widget>
<packing>
<property name="x_options">GTK_FILL</property>
<property name="y_options">GTK_FILL</property>
</packing>
</child>
</widget>
</child>
</widget>
</glade-interface>

View File

@ -0,0 +1,157 @@
# -*- coding: utf-8 -*-
## This file is part of Gajim.
##
## Gajim 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 3 only.
##
## Gajim 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 Gajim. If not, see <http://www.gnu.org/licenses/>.
##
'''
Message length notifier plugin.
:author: Mateusz Biliński <mateusz@bilinski.it>
:since: 1st June 2008
:copyright: Copyright (2008) Mateusz Biliński <mateusz@bilinski.it>
:license: GPL
'''
import sys
import gtk
from common import i18n
from plugins import GajimPlugin
from plugins.helpers import log, log_calls
from plugins.gui import GajimPluginConfigDialog
class LengthNotifierPlugin(GajimPlugin):
name = u'Message Length Notifier'
short_name = u'length_notifier'
version = u'0.1'
description = u'''Highlights message entry field in chat window when given length of message is exceeded.'''
authors = [u'Mateusz Biliński <mateusz@bilinski.it>']
homepage = u'http://blog.bilinski.it'
#@log_calls('LengthNotifierPlugin')
#def __init__(self):
#super(LengthNotifierPlugin, self).__init__()
@log_calls('LengthNotifierPlugin')
def init(self):
self.config_dialog = LengthNotifierPluginConfigDialog(self)
self.gui_extension_points = {
'chat_control' : (self.connect_with_chat_control,
self.disconnect_from_chat_control)
}
self.config_default_values = {'MESSAGE_WARNING_LENGTH' : (140, _('Message length at which notification is invoked.')),
'WARNING_COLOR' : ('#F0DB3E', _('Background color of text entry field in chat window when notification is invoked.')),
'JIDS' : ([], _('JabberIDs that plugin should be used with (eg. restrict only to one microblogging bot). If empty plugin is used with every JID. [not implemented]'))
}
@log_calls('LengthNotifierPlugin')
def textview_length_warning(self, tb, chat_control):
tv = chat_control.msg_textview
d = chat_control.length_notifier_plugin_data
t = tb.get_text(tb.get_start_iter(), tb.get_end_iter())
if t:
len_t = len(t)
#print("len_t: %d"%(len_t))
if len_t>self.config['MESSAGE_WARNING_LENGTH']:
if not d['prev_color']:
d['prev_color'] = tv.style.copy().base[gtk.STATE_NORMAL]
tv.modify_base(gtk.STATE_NORMAL, gtk.gdk.color_parse(self.config['WARNING_COLOR']))
elif d['prev_color']:
tv.modify_base(gtk.STATE_NORMAL, d['prev_color'])
d['prev_color'] = None
@log_calls('LengthNotifierPlugin')
def connect_with_chat_control(self, chat_control):
jid = chat_control.contact.jid
if self.jid_is_ok(jid):
d = {'prev_color' : None}
tv = chat_control.msg_textview
tb = tv.get_buffer()
h_id = tb.connect('changed', self.textview_length_warning, chat_control)
d['h_id'] = h_id
t = tb.get_text(tb.get_start_iter(), tb.get_end_iter())
if t:
len_t = len(t)
if len_t>self.config['MESSAGE_WARNING_LENGTH']:
d['prev_color'] = tv.style.copy().base[gtk.STATE_NORMAL]
tv.modify_base(gtk.STATE_NORMAL, gtk.gdk.color_parse(self.config['WARNING_COLOR']))
chat_control.length_notifier_plugin_data = d
return True
return False
@log_calls('LengthNotifierPlugin')
def disconnect_from_chat_control(self, chat_control):
try:
d = chat_control.length_notifier_plugin_data
tv = chat_control.msg_textview
tv.get_buffer().disconnect(d['h_id'])
if d['prev_color']:
tv.modify_base(gtk.STATE_NORMAL, d['prev_color'])
except AttributeError, error:
log.debug('Length Notifier Plugin was (probably) never connected with this chat window.\n Error: %s' % (error))
@log_calls('LengthNotifierPlugin')
def jid_is_ok(self, jid):
if jid in self.config['JIDS'] or not self.config['JIDS']:
return True
return False
class LengthNotifierPluginConfigDialog(GajimPluginConfigDialog):
def init(self):
self.GLADE_FILE_PATH = self.plugin.local_file_path('config_dialog.glade')
self.xml = gtk.glade.XML(self.GLADE_FILE_PATH, root='length_notifier_config_table', domain=i18n.APP)
self.config_table = self.xml.get_widget('length_notifier_config_table')
self.child.pack_start(self.config_table)
self.message_length_spinbutton = self.xml.get_widget('message_length_spinbutton')
self.notification_colorbutton = self.xml.get_widget('notification_colorbutton')
self.jids_entry = self.xml.get_widget('jids_entry')
self.xml.signal_autoconnect(self)
def on_run(self):
self.message_length_spinbutton.set_value(self.plugin.config['MESSAGE_WARNING_LENGTH'])
self.notification_colorbutton.set_color(gtk.gdk.color_parse(self.plugin.config['WARNING_COLOR']))
#self.jids_entry.set_text(self.plugin.config['JIDS'])
self.jids_entry.set_text(','.join(self.plugin.config['JIDS']))
@log_calls('LengthNotifierPluginConfigDialog')
def on_message_length_spinbutton_value_changed(self, spinbutton):
self.plugin.config['MESSAGE_WARNING_LENGTH'] = spinbutton.get_value()
@log_calls('LengthNotifierPluginConfigDialog')
def on_notification_colorbutton_color_set(self, colorbutton):
self.plugin.config['WARNING_COLOR'] = colorbutton.get_color().to_string()
@log_calls('LengthNotifierPluginConfigDialog')
def on_jids_entry_changed(self, entry):
text = entry.get_text()
if len(text)>0:
self.plugin.config['JIDS'] = entry.get_text().split(',')
else:
self.plugin.config['JIDS'] = []
@log_calls('LengthNotifierPluginConfigDialog')
def on_jids_entry_editing_done(self, entry):
pass

View File

@ -101,6 +101,8 @@ def check_and_possibly_create_paths():
VCARD_PATH = gajim.VCARD_PATH VCARD_PATH = gajim.VCARD_PATH
AVATAR_PATH = gajim.AVATAR_PATH AVATAR_PATH = gajim.AVATAR_PATH
dot_gajim = os.path.dirname(VCARD_PATH) dot_gajim = os.path.dirname(VCARD_PATH)
PLUGINS_CONFIG_PATH = gajim.PLUGINS_CONFIG_DIR
if os.path.isfile(dot_gajim): if os.path.isfile(dot_gajim):
print _('%s is a file but it should be a directory') % dot_gajim print _('%s is a file but it should be a directory') % dot_gajim
print _('Gajim will now exit') print _('Gajim will now exit')
@ -132,6 +134,14 @@ def check_and_possibly_create_paths():
print _('Gajim will now exit') print _('Gajim will now exit')
sys.exit() sys.exit()
if not os.path.exists(PLUGINS_CONFIG_PATH):
create_path(PLUGINS_CONFIG_PATH)
elif os.path.isfile(PLUGINS_CONFIG_PATH):
print _('%s is a file but it should be a directory') % PLUGINS_CONFIG_PATH
print _('Gajim will now exit')
sys.exit()
else: # dot_gajim doesn't exist else: # dot_gajim doesn't exist
if dot_gajim: # is '' on win9x so avoid that if dot_gajim: # is '' on win9x so avoid that
create_path(dot_gajim) create_path(dot_gajim)

View File

@ -110,15 +110,19 @@ class ConfigPaths:
conffile = windowsify(u'config') conffile = windowsify(u'config')
pidfile = windowsify(u'gajim') pidfile = windowsify(u'gajim')
secretsfile = windowsify(u'secrets') secretsfile = windowsify(u'secrets')
pluginsconfdir = windowsify(u'pluginsconfig')
if len(profile) > 0: if len(profile) > 0:
conffile += u'.' + profile conffile += u'.' + profile
pidfile += u'.' + profile pidfile += u'.' + profile
secretsfile += u'.' + profile secretsfile += u'.' + profile
pluginsconfdir += u'.' + profile
pidfile += u'.pid' pidfile += u'.pid'
self.add_from_root('CONFIG_FILE', conffile) self.add_from_root('CONFIG_FILE', conffile)
self.add_from_root('PID_FILE', pidfile) self.add_from_root('PID_FILE', pidfile)
self.add_from_root('SECRETS_FILE', secretsfile) self.add_from_root('SECRETS_FILE', secretsfile)
self.add_from_root('PLUGINS_CONFIG_DIR', pluginsconfdir)
# for k, v in paths.iteritems(): # for k, v in paths.iteritems():
# print "%s: %s" % (repr(k), repr(v)) # print "%s: %s" % (repr(k), repr(v))

View File

@ -88,6 +88,7 @@ DATA_DIR = gajimpaths['DATA']
HOME_DIR = gajimpaths['HOME'] HOME_DIR = gajimpaths['HOME']
PLUGINS_DIRS = [gajimpaths['PLUGINS_BASE'], PLUGINS_DIRS = [gajimpaths['PLUGINS_BASE'],
gajimpaths['PLUGINS_USER']] gajimpaths['PLUGINS_USER']]
PLUGINS_CONFIG_DIR = gajimpaths['PLUGINS_CONFIG_DIR']
try: try:
LANG = locale.getdefaultlocale()[0] # en_US, fr_FR, el_GR etc.. LANG = locale.getdefaultlocale()[0] # en_US, fr_FR, el_GR etc..

View File

@ -202,15 +202,24 @@ class GajimPluginConfigDialog(gtk.Dialog):
self.child.set_spacing(3) self.child.set_spacing(3)
self.init()
#label = gtk.Label(_('<b>%s Configuration</b>') % (plugin.name)) #label = gtk.Label(_('<b>%s Configuration</b>') % (plugin.name))
#label.set_markup(label.get_label()) #label.set_markup(label.get_label())
#self.child.pack_start(label, False, False) #self.child.pack_start(label, False, False)
@log_calls('GajimPluginConfigDialog') @log_calls('GajimPluginConfigDialog')
def run(self, parent=None): def run(self, parent=None):
self.reparent(parent) self.reparent(parent)
self.on_run()
self.show_all() self.show_all()
result = super(GajimPluginConfigDialog, self).run() result = super(GajimPluginConfigDialog, self).run()
self.hide() self.hide()
return result return result
def init(self):
pass
def on_run(self):
pass

View File

@ -50,6 +50,12 @@ class log_calls(object):
Decorator class for functions to easily log when they are entered and left. Decorator class for functions to easily log when they are entered and left.
''' '''
filter_out_classes = ['PluginManager']
'''
List of classes from which no logs should be emited when methods are called,
eventhough `log_calls` decorator is used.
'''
def __init__(self, classname='', log=log): def __init__(self, classname='', log=log):
''' '''
:Keywords: :Keywords:
@ -71,10 +77,20 @@ class log_calls(object):
:type: str :type: str
''' '''
self.log_this_class = True
'''
Determines whether wrapper of given function should log calls of this
function or not.
:type: bool
'''
if classname: if classname:
self.full_func_name = classname+'.' self.full_func_name = classname+'.'
if classname in self.filter_out_classes:
self.log_this_class = False
def __call__(self, f): def __call__(self, f):
''' '''
:param f: function to be wrapped with logging statements :param f: function to be wrapped with logging statements
@ -82,15 +98,24 @@ class log_calls(object):
:return: given function wrapped by *log.debug* statements :return: given function wrapped by *log.debug* statements
:rtype: function :rtype: function
''' '''
self.full_func_name += f.func_name self.full_func_name += f.func_name
if self.log_this_class:
@functools.wraps(f) @functools.wraps(f)
def wrapper(*args, **kwargs): def wrapper(*args, **kwargs):
log.debug('%(funcname)s() <entered>'%{ log.debug('%(funcname)s() <entered>'%{
'funcname': self.full_func_name}) 'funcname': self.full_func_name})
result = f(*args, **kwargs) result = f(*args, **kwargs)
log.debug('%(funcname)s() <left>'%{ log.debug('%(funcname)s() <left>'%{
'funcname': self.full_func_name}) 'funcname': self.full_func_name})
return result return result
else:
@functools.wraps(f)
def wrapper(*args, **kwargs):
result = f(*args, **kwargs)
return result
return wrapper return wrapper
class Singleton(type): class Singleton(type):

View File

@ -134,10 +134,6 @@ class GajimPlugin(object):
def load_config(self): def load_config(self):
self.config.load() self.config.load()
@log_calls('GajimPlugin')
def __del__(self):
self.save_config()
@log_calls('GajimPlugin') @log_calls('GajimPlugin')
def local_file_path(self, file_name): def local_file_path(self, file_name):
return os.path.join(self.__path__, file_name) return os.path.join(self.__path__, file_name)
@ -155,20 +151,38 @@ class GajimPlugin(object):
pass pass
import shelve import shelve
import UserDict
class GajimPluginConfig(dict): class GajimPluginConfig(UserDict.DictMixin):
@log_calls('GajimPluginConfig') @log_calls('GajimPluginConfig')
def __init__(self, plugin): def __init__(self, plugin):
self.plugin = plugin self.plugin = plugin
self.FILE_PATH = gajim.HOME_DIR self.FILE_PATH = os.path.join(gajim.PLUGINS_CONFIG_DIR, self.plugin.short_name)
log.debug('FILE_PATH = %s'%(self.FILE_PATH)) #log.debug('FILE_PATH = %s'%(self.FILE_PATH))
#self.data = shelve.open(self.FILE_PATH) self.data = None
self.load()
@log_calls('GajimPluginConfig')
def __getitem__(self, key):
if not key in self.data:
self.data[key] = self.plugin.config_default_values[key][0]
self.save()
return self.data[key]
@log_calls('GajimPluginConfig')
def __setitem__(self, key, value):
self.data[key] = value
self.save()
def keys(self):
return self.data.keys()
@log_calls('GajimPluginConfig') @log_calls('GajimPluginConfig')
def save(self): def save(self):
pass self.data.sync()
log.debug(str(self.data))
@log_calls('GajimPluginConfig') @log_calls('GajimPluginConfig')
def load(self): def load(self):
pass self.data = shelve.open(self.FILE_PATH)