Mercurial > libervia-web
view src/browser/libervia_main.py @ 584:0a06cf833f5a
browser_side: various improvements, especially for MUC
author | souliane <souliane@mailoo.org> |
---|---|
date | Wed, 22 Oct 2014 19:03:55 +0200 |
parents | 1c1dbe03d3c6 |
children | 3eb3a2c0c011 a5019e62c3e9 |
line wrap: on
line source
#!/usr/bin/python # -*- coding: utf-8 -*- # Libervia: a Salut à Toi frontend # Copyright (C) 2011, 2012, 2013, 2014 Jérôme Poisson <goffi@goffi.org> # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU Affero General Public License as published by # the Free Software Foundation, either version 3 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 Affero General Public License for more details. # You should have received a copy of the GNU Affero General Public License # along with this program. If not, see <http://www.gnu.org/licenses/>. import pyjd # this is dummy in pyjs ### logging configuration ### from sat_browser import logging logging.configure() from sat.core.log import getLogger log = getLogger(__name__) ### from sat_frontends.tools.misc import InputHistory from sat_frontends.tools import strings from sat.core.i18n import _ from pyjamas.ui.RootPanel import RootPanel from pyjamas.ui.HTML import HTML from pyjamas.ui.KeyboardListener import KEY_ESCAPE from pyjamas.Timer import Timer from pyjamas import Window, DOM from pyjamas.JSONService import JSONProxy from sat_browser import register from sat_browser import contact from sat_browser import base_widget from sat_browser import panels from sat_browser import dialog from sat_browser import jid from sat_browser import xmlui from sat_browser import html_tools from sat_browser import notification from sat_browser.constants import Const as C try: # FIXME: import plugin dynamically from sat_browser import plugin_sec_otr except ImportError: pass MAX_MBLOG_CACHE = 500 # Max microblog entries kept in memories # Set to true to not create a new LiberviaWidget when a similar one # already exist (i.e. a chat panel with the same target). Instead # the existing widget will be eventually removed from its parent # and added to new base_widget.WidgetsPanel, or replaced to the expected # position if the previous and the new parent are the same. REUSE_EXISTING_LIBERVIA_WIDGETS = True class LiberviaJsonProxy(JSONProxy): def __init__(self, *args, **kwargs): JSONProxy.__init__(self, *args, **kwargs) self.handler = self self.cb = {} self.eb = {} def call(self, method, cb, *args): _id = self.callMethod(method, args) if cb: if isinstance(cb, tuple): if len(cb) != 2: log.error("tuple syntax for bridge.call is (callback, errback), aborting") return if cb[0] is not None: self.cb[_id] = cb[0] self.eb[_id] = cb[1] else: self.cb[_id] = cb def onRemoteResponse(self, response, request_info): if request_info.id in self.cb: _cb = self.cb[request_info.id] # if isinstance(_cb, tuple): # #we have arguments attached to the callback # #we send them after the answer # callback, args = _cb # callback(response, *args) # else: # #No additional argument, we call directly the callback _cb(response) del self.cb[request_info.id] if request_info.id in self.eb: del self.eb[request_info.id] def onRemoteError(self, code, errobj, request_info): """def dump(obj): print "\n\nDUMPING %s\n\n" % obj for i in dir(obj): print "%s: %s" % (i, getattr(obj,i))""" if request_info.id in self.eb: _eb = self.eb[request_info.id] _eb((code, errobj)) del self.cb[request_info.id] del self.eb[request_info.id] else: if code != 0: log.error("Internal server error") """for o in code, error, request_info: dump(o)""" else: if isinstance(errobj['message'], dict): log.error("Error %s: %s" % (errobj['message']['faultCode'], errobj['message']['faultString'])) else: log.error("%s" % errobj['message']) class RegisterCall(LiberviaJsonProxy): def __init__(self): LiberviaJsonProxy.__init__(self, "/register_api", ["isRegistered", "isConnected", "asyncConnect", "registerParams", "getMenus"]) class BridgeCall(LiberviaJsonProxy): def __init__(self): LiberviaJsonProxy.__init__(self, "/json_api", ["getContacts", "addContact", "sendMessage", "sendMblog", "sendMblogComment", "getLastMblogs", "getMassiveLastMblogs", "getMblogComments", "getProfileJid", "getHistory", "getPresenceStatuses", "joinMUC", "mucLeave", "getRoomsJoined", "inviteMUC", "launchTarotGame", "getTarotCardsPaths", "tarotGameReady", "tarotGamePlayCards", "launchRadioCollective", "getMblogs", "getMblogsWithComments", "getWaitingSub", "subscription", "delContact", "updateContact", "getCard", "getEntityData", "getParamsUI", "asyncGetParamA", "setParam", "launchAction", "disconnect", "chatStateComposing", "getNewAccountDomain", "confirmationAnswer", "syntaxConvert", "getAccountDialogUI", "getLastResource" ]) class BridgeSignals(LiberviaJsonProxy): RETRY_BASE_DELAY = 1000 def __init__(self, host): self.host = host self.retry_delay = self.RETRY_BASE_DELAY LiberviaJsonProxy.__init__(self, "/json_signal_api", ["getSignals"]) def onRemoteResponse(self, response, request_info): self.retry_delay = self.RETRY_BASE_DELAY LiberviaJsonProxy.onRemoteResponse(self, response, request_info) def onRemoteError(self, code, errobj, request_info): if errobj['message'] == 'Empty Response': Window.getLocation().reload() # XXX: reset page in case of session ended. # FIXME: Should be done more properly without hard reload LiberviaJsonProxy.onRemoteError(self, code, errobj, request_info) #we now try to reconnect if isinstance(errobj['message'], dict) and errobj['message']['faultCode'] == 0: Window.alert('You are not allowed to connect to server') else: def _timerCb(timer): self.host.bridge_signals.call('getSignals', self.host._getSignalsCB) Timer(notify=_timerCb).schedule(self.retry_delay) self.retry_delay *= 2 class SatWebFrontend(InputHistory): def onModuleLoad(self): log.info("============ onModuleLoad ==============") panels.ChatPanel.registerClass() panels.MicroblogPanel.registerClass() self.whoami = None self._selected_listeners = set() self.bridge = BridgeCall() self.bridge_signals = BridgeSignals(self) self.uni_box = None self.status_panel = HTML('<br />') self.contact_panel = contact.ContactPanel(self) self.panel = panels.MainPanel(self) self.discuss_panel = self.panel.discuss_panel self.tab_panel = self.panel.tab_panel self.tab_panel.addTabListener(self) self.libervia_widgets = set() # keep track of all actives LiberviaWidgets self.room_list = [] # list of rooms self.mblog_cache = [] # used to keep our own blog entries in memory, to show them in new mblog panel self.avatars_cache = {} # keep track of jid's avatar hash (key=jid, value=file) self._register_box = None RootPanel().add(self.panel) self.notification = notification.Notification() DOM.addEventPreview(self) self.importPlugins() self._register = RegisterCall() self._register.call('getMenus', self.gotMenus) self._register.call('registerParams', None) self._register.call('isRegistered', self._isRegisteredCB) self.initialised = False self.init_cache = [] # used to cache events until initialisation is done self.cached_params = {} def importPlugins(self): self.plugins = {} inhibited_menus = [] # FIXME: plugins import should be dynamic and generic like in sat try: self.plugins['otr'] = plugin_sec_otr.OTR(self) except TypeError: # plugin_sec_otr has not been imported inhibited_menus.append('OTR') class DummyPlugin(object): def inhibitMenus(self): return inhibited_menus self.plugins['dummy_plugin'] = DummyPlugin() def addSelectedListener(self, callback): self._selected_listeners.add(callback) def getSelected(self): wid = self.tab_panel.getCurrentPanel() if not isinstance(wid, base_widget.WidgetsPanel): log.error("Tab widget is not a base_widget.WidgetsPanel, can't get selected widget") return None return wid.selected def setSelected(self, widget): """Define the selected widget""" widgets_panel = self.tab_panel.getCurrentPanel() if not isinstance(widgets_panel, base_widget.WidgetsPanel): return selected = widgets_panel.selected if selected == widget: return if selected: selected.removeStyleName('selected_widget') widgets_panel.selected = widget if widget: widgets_panel.selected.addStyleName('selected_widget') for callback in self._selected_listeners: callback(widget) def resize(self): """Resize elements""" Window.onResize() def onBeforeTabSelected(self, sender, tab_index): return True def onTabSelected(self, sender, tab_index): selected = self.getSelected() for callback in self._selected_listeners: callback(selected) def onEventPreview(self, event): if event.type in ["keydown", "keypress", "keyup"] and event.keyCode == KEY_ESCAPE: #needed to prevent request cancellation in Firefox event.preventDefault() return True def getAvatar(self, jid_str): """Return avatar of a jid if in cache, else ask for it. @param jid_str (str): JID of the contact @return: the URL to the avatar (str) """ def dataReceived(result): if 'avatar' in result: self._entityDataUpdatedCb(jid_str, 'avatar', result['avatar']) else: self.bridge.call("getCard", None, jid_str) def avatarError(error_data): # The jid is maybe not in our roster, we ask for the VCard self.bridge.call("getCard", None, jid_str) if jid_str not in self.avatars_cache: self.bridge.call('getEntityData', (dataReceived, avatarError), jid_str, ['avatar']) self.avatars_cache[jid_str] = C.DEFAULT_AVATAR return self.avatars_cache[jid_str] def registerWidget(self, wid): log.debug("Registering %s" % wid.getDebugName()) self.libervia_widgets.add(wid) def unregisterWidget(self, wid): try: self.libervia_widgets.remove(wid) except KeyError: log.warning('trying to remove a non registered Widget: %s' % wid.getDebugName()) def refresh(self): """Refresh the general display.""" self.panel.refresh() if self.getCachedParam(C.COMPOSITION_KEY, C.ENABLE_UNIBOX_PARAM) == 'true': self.uni_box = self.panel.unibox_panel.unibox else: self.uni_box = None for lib_wid in self.libervia_widgets: lib_wid.refresh() self.resize() def addTab(self, label, wid, select=True): """Create a new tab and eventually add a widget in @param label: label of the tab @param wid: LiberviaWidget to add @param select: True to select the added tab """ widgets_panel = base_widget.WidgetsPanel(self) self.tab_panel.add(widgets_panel, label) widgets_panel.addWidget(wid) if select: self.tab_panel.selectTab(self.tab_panel.getWidgetCount() - 1) return widgets_panel def addWidget(self, wid, tab_index=None): """ Add a widget at the bottom of the current or specified tab @param wid: LiberviaWidget to add @param tab_index: index of the tab to add the widget to""" if tab_index is None or tab_index < 0 or tab_index >= self.tab_panel.getWidgetCount(): panel = self.tab_panel.getCurrentPanel() else: panel = self.tab_panel.tabBar.getTabWidget(tab_index) panel.addWidget(wid) def displayNotification(self, title, body): self.notification.notify(title, body) def gotMenus(self, menus): """Put the menus data in cache and build the main menu bar @param menus (list[tuple]): menu data """ def process(menus, inhibited=None): for raw_menu in menus: id_, type_, path, path_i18n = raw_menu if inhibited and path[0] in inhibited: continue menus_data = self.menus.setdefault(type_, []) menus_data.append((id_, path, path_i18n)) self.menus = {} inhibited = set() extras = [] for plugin in self.plugins.values(): if hasattr(plugin, "inhibitMenus"): inhibited.update(plugin.inhibitMenus()) if hasattr(plugin, "extraMenus"): extras.extend(plugin.extraMenus()) process(menus, inhibited) process(extras) self.panel.menu.createMenus() def _isRegisteredCB(self, result): registered, warning = result if not registered: self._register_box = register.RegisterBox(self.logged) self._register_box.centerBox() self._register_box.show() if warning: dialog.InfoDialog(_('Security warning'), warning).show() self._tryAutoConnect(skip_validation=not not warning) else: self._register.call('isConnected', self._isConnectedCB) def _isConnectedCB(self, connected): if not connected: self._register.call('asyncConnect', lambda x: self.logged()) else: self.logged() def logged(self): if self._register_box: self._register_box.hide() del self._register_box # don't work if self._register_box is None # display the real presence status panel self.panel.header.remove(self.status_panel) self.status_panel = panels.PresenceStatusPanel(self) self.panel.header.add(self.status_panel) #it's time to fill the page self.bridge.call('getContacts', self._getContactsCB) self.bridge.call('getParamsUI', self._getParamsUICB) self.bridge_signals.call('getSignals', self._getSignalsCB) #We want to know our own jid self.bridge.call('getProfileJid', self._getProfileJidCB) def domain_cb(value): self._defaultDomain = value log.info("new account domain: %s" % value) def domain_eb(value): self._defaultDomain = "libervia.org" self.bridge.call("getNewAccountDomain", (domain_cb, domain_eb)) self.discuss_panel.addWidget(panels.MicroblogPanel(self, [])) # get cached params and refresh the display def param_cb(cat, name, count): count[0] += 1 refresh = count[0] == len(C.CACHED_PARAMS) return lambda value: self._paramUpdate(name, value, cat, refresh) count = [0] # used to do something similar to DeferredList for cat, name in C.CACHED_PARAMS: self.bridge.call('asyncGetParamA', param_cb(cat, name, count), name, cat) def _tryAutoConnect(self, skip_validation=False): """This method retrieve the eventual URL parameters to auto-connect the user. @param skip_validation: if True, set the form values but do not validate it """ params = strings.getURLParams(Window.getLocation().getSearch()) if "login" in params: self._register_box._form.right_side.selectTab(0) self._register_box._form.login_box.setText(params["login"]) self._register_box._form.login_pass_box.setFocus(True) if "passwd" in params: # try to connect self._register_box._form.login_pass_box.setText(params["passwd"]) if not skip_validation: self._register_box._form.onLogin(None) return True else: # this would eventually set the browser saved password Timer(5, lambda: self._register_box._form.login_pass_box.setFocus(True)) def _actionCb(self, data): if not data: # action was a one shot, nothing to do pass elif "xmlui" in data: ui = xmlui.create(self, xml_data=data['xmlui']) ui.show() elif "public_blog" in data: # TODO: use the bare instead of node when all blogs can be retrieved node = jid.JID(data['public_blog']).node self.addTab("%s's blog" % node, panels.WebPanel(self, "/blog/%s" % node)) else: dialog.InfoDialog("Error", "Unmanaged action result", Width="400px").center() def _actionEb(self, err_data): err_code, err_obj = err_data dialog.InfoDialog("Error", str(err_obj), Width="400px").center() def launchAction(self, callback_id, data): """ Launch a dynamic action @param callback_id: id of the action to launch @param data: data needed only for certain actions """ if data is None: data = {} self.bridge.call('launchAction', (self._actionCb, self._actionEb), callback_id, data) def _getContactsCB(self, contacts_data): for contact_ in contacts_data: jid, attributes, groups = contact_ self._newContactCb(jid, attributes, groups) def _getSignalsCB(self, signal_data): self.bridge_signals.call('getSignals', self._getSignalsCB) if len(signal_data) == 1: signal_data.append([]) log.debug("Got signal ==> name: %s, params: %s" % (signal_data[0], signal_data[1])) name, args = signal_data if name == 'personalEvent': self._personalEventCb(*args) elif name == 'newMessage': self._newMessageCb(*args) elif name == 'presenceUpdate': self._presenceUpdateCb(*args) elif name == 'paramUpdate': self._paramUpdate(*args) elif name == 'roomJoined': self._roomJoinedCb(*args) elif name == 'roomLeft': self._roomLeftCb(*args) elif name == 'roomUserJoined': self._roomUserJoinedCb(*args) elif name == 'roomUserLeft': self._roomUserLeftCb(*args) elif name == 'roomUserChangedNick': self._roomUserChangedNickCb(*args) elif name == 'askConfirmation': self._askConfirmation(*args) elif name == 'newAlert': self._newAlert(*args) elif name == 'tarotGamePlayers': self._tarotGameStartedCb(True, *args) elif name == 'tarotGameStarted': self._tarotGameStartedCb(False, *args) elif name == 'tarotGameNew' or \ name == 'tarotGameChooseContrat' or \ name == 'tarotGameShowCards' or \ name == 'tarotGameInvalidCards' or \ name == 'tarotGameCardsPlayed' or \ name == 'tarotGameYourTurn' or \ name == 'tarotGameScore': self._tarotGameGenericCb(name, args[0], args[1:]) elif name == 'radiocolPlayers': self._radioColStartedCb(True, *args) elif name == 'radiocolStarted': self._radioColStartedCb(False, *args) elif name == 'radiocolPreload': self._radioColGenericCb(name, args[0], args[1:]) elif name == 'radiocolPlay': self._radioColGenericCb(name, args[0], args[1:]) elif name == 'radiocolNoUpload': self._radioColGenericCb(name, args[0], args[1:]) elif name == 'radiocolUploadOk': self._radioColGenericCb(name, args[0], args[1:]) elif name == 'radiocolSongRejected': self._radioColGenericCb(name, args[0], args[1:]) elif name == 'subscribe': self._subscribeCb(*args) elif name == 'contactDeleted': self._contactDeletedCb(*args) elif name == 'newContact': self._newContactCb(*args) elif name == 'entityDataUpdated': self._entityDataUpdatedCb(*args) elif name == 'chatStateReceived': self._chatStateReceivedCb(*args) def _getParamsUICB(self, xml_ui): """Hide the parameters item if there's nothing to display""" if not xml_ui: self.panel.menu.removeItemParams() def _ownBlogsFills(self, mblogs): #put our own microblogs in cache, then fill all panels with them for publisher in mblogs: for mblog in mblogs[publisher]: if 'content' not in mblog: log.warning("No content found in microblog [%s]" % mblog) continue if 'groups' in mblog: _groups = set(mblog['groups'].split() if mblog['groups'] else []) else: _groups = None mblog_entry = panels.MicroblogItem(mblog) self.mblog_cache.append((_groups, mblog_entry)) if len(self.mblog_cache) > MAX_MBLOG_CACHE: del self.mblog_cache[0:len(self.mblog_cache - MAX_MBLOG_CACHE)] for lib_wid in self.libervia_widgets: if isinstance(lib_wid, panels.MicroblogPanel): self.FillMicroblogPanel(lib_wid) self.initialised = True # initialisation phase is finished here for event_data in self.init_cache: # so we have to send all the cached events self._personalEventCb(*event_data) del self.init_cache def _getProfileJidCB(self, jid_s): self.whoami = jid.JID(jid_s) #we can now ask our status self.bridge.call('getPresenceStatuses', self._getPresenceStatusesCb) #the rooms where we are self.bridge.call('getRoomsJoined', self._getRoomsJoinedCb) #and if there is any subscription request waiting for us self.bridge.call('getWaitingSub', self._getWaitingSubCb) #we fill the panels already here for lib_wid in self.libervia_widgets: if isinstance(lib_wid, panels.MicroblogPanel): if lib_wid.accept_all(): self.bridge.call('getMassiveLastMblogs', lib_wid.massiveInsert, 'ALL', [], 10) else: self.bridge.call('getMassiveLastMblogs', lib_wid.massiveInsert, 'GROUP', lib_wid.accepted_groups, 10) #we ask for our own microblogs: self.bridge.call('getMassiveLastMblogs', self._ownBlogsFills, 'JID', [self.whoami.bare], 10) # initialize plugins which waited for the connection to be done for plugin in self.plugins.values(): if hasattr(plugin, 'profileConnected'): plugin.profileConnected() ## Signals callbacks ## def _personalEventCb(self, sender, event_type, data): if not self.initialised: self.init_cache.append((sender, event_type, data)) return sender = jid.JID(sender).bare if event_type == "MICROBLOG": if not 'content' in data: log.warning("No content found in microblog data") return if 'groups' in data: _groups = set(data['groups'].split() if data['groups'] else []) else: _groups = None mblog_entry = panels.MicroblogItem(data) for lib_wid in self.libervia_widgets: if isinstance(lib_wid, panels.MicroblogPanel): self.addBlogEntry(lib_wid, sender, _groups, mblog_entry) if sender == self.whoami.bare: found = False for index in xrange(0, len(self.mblog_cache)): entry = self.mblog_cache[index] if entry[1].id == mblog_entry.id: # replace existing entry self.mblog_cache.remove(entry) self.mblog_cache.insert(index, (_groups, mblog_entry)) found = True break if not found: self.mblog_cache.append((_groups, mblog_entry)) if len(self.mblog_cache) > MAX_MBLOG_CACHE: del self.mblog_cache[0:len(self.mblog_cache - MAX_MBLOG_CACHE)] elif event_type == 'MICROBLOG_DELETE': for lib_wid in self.libervia_widgets: if isinstance(lib_wid, panels.MicroblogPanel): lib_wid.removeEntry(data['type'], data['id']) log.debug("%s %s %s" % (self.whoami.bare, sender, data['type'])) if sender == self.whoami.bare and data['type'] == 'main_item': for index in xrange(0, len(self.mblog_cache)): entry = self.mblog_cache[index] if entry[1].id == data['id']: self.mblog_cache.remove(entry) break def addBlogEntry(self, mblog_panel, sender, _groups, mblog_entry): """Check if an entry can go in MicroblogPanel and add to it @param mblog_panel: MicroblogPanel instance @param sender: jid of the entry sender @param _groups: groups which can receive this entry @param mblog_entry: panels.MicroblogItem instance""" if mblog_entry.type == "comment" or mblog_panel.isJidAccepted(sender) or (_groups == None and self.whoami and sender == self.whoami.bare) \ or (_groups and _groups.intersection(mblog_panel.accepted_groups)): mblog_panel.addEntry(mblog_entry) def FillMicroblogPanel(self, mblog_panel): """Fill a microblog panel with entries in cache @param mblog_panel: MicroblogPanel instance """ #XXX: only our own entries are cached for cache_entry in self.mblog_cache: _groups, mblog_entry = cache_entry self.addBlogEntry(mblog_panel, self.whoami.bare, *cache_entry) def getEntityMBlog(self, entity): log.info("geting mblog for entity [%s]" % (entity,)) for lib_wid in self.libervia_widgets: if isinstance(lib_wid, panels.MicroblogPanel): if lib_wid.isJidAccepted(entity): self.bridge.call('getMassiveLastMblogs', lib_wid.massiveInsert, 'JID', [entity], 10) def getLiberviaWidget(self, class_, entity, ignoreOtherTabs=True): """Get the corresponding panel if it exists. @param class_ (class): class of the panel (ChatPanel, MicroblogPanel...) @param entity (dict): dictionnary to define the entity. @param ignoreOtherTabs (bool): if True, the widgets that are not contained by the currently selected tab will be ignored @return: the existing widget that has been found or None.""" selected_tab = self.tab_panel.getCurrentPanel() for lib_wid in self.libervia_widgets: parent = lib_wid.getWidgetsPanel(expect=False) if parent is None or (ignoreOtherTabs and parent != selected_tab): # do not return a widget that is not in the currently selected tab continue if isinstance(lib_wid, class_): try: if lib_wid.matchEntity(*(entity.values())): # XXX: passing **entity bugs! log.debug("existing widget found: %s" % lib_wid.getDebugName()) return lib_wid except AttributeError as e: e.stack_list() return None return None def getOrCreateLiberviaWidget(self, class_, entity, select=True, new_tab=None): """Get the matching LiberviaWidget if it exists, or create a new one. @param class_ (class): class of the panel (ChatPanel, MicroblogPanel...) @param entity (dict): dictionnary to define the entity. @param select (bool): if True, select the widget that has been found or created @param new_tab (str): if not None, a widget which is created is created in a new tab. In that case new_tab is a unicode to label that new tab. If new_tab is not None and a widget is found, no tab is created. @return: the newly created wigdet if REUSE_EXISTING_LIBERVIA_WIDGETS is set to False or if the widget has not been found, the existing widget that has been found otherwise.""" lib_wid = None tab = None if REUSE_EXISTING_LIBERVIA_WIDGETS: lib_wid = self.getLiberviaWidget(class_, entity, new_tab is None) if lib_wid is None: # create a new widget lib_wid = class_.createPanel(self, *(entity.values())) # XXX: passing **entity bugs! if new_tab is None: self.addWidget(lib_wid) else: tab = self.addTab(new_tab, lib_wid, False) else: # reuse existing widget tab = lib_wid.getWidgetsPanel(expect=False) if new_tab is None: if tab is not None: tab.removeWidget(lib_wid) self.addWidget(lib_wid) if select: if new_tab is not None: self.tab_panel.selectTab(tab) # must be done after the widget is added, # for example to scroll to the bottom self.setSelected(lib_wid) lib_wid.refresh() return lib_wid def getRoomWidget(self, target): """Get the MUC widget for the given target. @param target (jid.JID): BARE jid of the MUC @return: panels.ChatPanel instance or None """ entity = {'item': target, 'type_': 'group'} if target.full() in self.room_list or target in self.room_list: # as JID is a string-based class, we don't know what will please Pyjamas... return self.getLiberviaWidget(panels.ChatPanel, entity, ignoreOtherTabs=False) return None def getOrCreateRoomWidget(self, target): """Get the MUC widget for the given target, create it if necessary. @param target (jid.JID): BARE jid of the MUC @return: panels.ChatPanel instance """ lib_wid = self.getRoomWidget(target) if lib_wid: return lib_wid # XXX: target.node.startwith(...) raises an error "startswith is not a function" # This happens when node a is property defined in the JID class # FIXME: pyjamas doesn't handle the properties well node = target.node # XXX: it's not really beautiful, but it works :) if node.startswith('sat_tarot_'): tab_name = "Tarot" elif node.startswith('sat_radiocol_'): tab_name = "Radio collective" else: tab_name = target.node self.room_list.append(target) entity = {'item': target, 'type_': 'group'} return self.getOrCreateLiberviaWidget(panels.ChatPanel, entity, new_tab=tab_name) def _newMessageCb(self, from_jid_s, msg, msg_type, to_jid_s, extra): from_jid = jid.JID(from_jid_s) to_jid = jid.JID(to_jid_s) for plugin in self.plugins.values(): if hasattr(plugin, 'messageReceivedTrigger'): if not plugin.messageReceivedTrigger(from_jid, msg, msg_type, to_jid, extra): return # plugin returned False to interrupt the process self.newMessageCb(from_jid, msg, msg_type, to_jid, extra) def newMessageCb(self, from_jid, msg, msg_type, to_jid, extra): other = to_jid if from_jid.bare == self.whoami.bare else from_jid lib_wid = self.getLiberviaWidget(panels.ChatPanel, {'item': other}, ignoreOtherTabs=False) self.displayNotification(from_jid, msg) if msg_type == 'headline' and from_jid.full() == self._defaultDomain: try: assert extra['subject'] # subject is defined and not empty title = extra['subject'] except (KeyError, AssertionError): title = _('Announcement from %s') % from_jid.full() msg = strings.addURLToText(html_tools.XHTML2Text(msg)) dialog.InfoDialog(title, msg).show() return if lib_wid is not None: if msg_type == C.MESS_TYPE_INFO: lib_wid.printInfo(msg) else: lib_wid.printMessage(from_jid, msg, extra) if 'header_info' in extra: lib_wid.setHeaderInfo(extra['header_info']) else: # FIXME: "info" message and header info will be lost here if not self.contact_panel.isContactInRoster(other.bare): self.contact_panel.updateContact(other.bare, {}, [C.GROUP_NOT_IN_ROSTER]) # The message has not been shown, we must indicate it self.contact_panel.setContactMessageWaiting(other.bare, True) def _presenceUpdateCb(self, entity, show, priority, statuses): entity_jid = jid.JID(entity) if self.whoami and self.whoami == entity_jid: # XXX: QnD way to get our presence/status assert(isinstance(self.status_panel, panels.PresenceStatusPanel)) self.status_panel.setPresence(show) # pylint: disable=E1103 if statuses: self.status_panel.setStatus(statuses.values()[0]) # pylint: disable=E1103 else: bare_jid = entity_jid.bareJID() if bare_jid.full() in self.room_list or bare_jid in self.room_list: # as JID is a string-based class, we don't know what will please Pyjamas... wid = self.getRoomWidget(bare_jid) else: wid = self.contact_panel if show == 'unavailable': # XXX: save some resources as for now we only need 'unavailable' for plugin in self.plugins.values(): if hasattr(plugin, 'presenceReceivedTrigger'): plugin.presenceReceivedTrigger(entity_jid, show, priority, statuses) if wid: wid.setConnected(entity_jid.bare, entity_jid.resource, show, priority, statuses) def _roomJoinedCb(self, room_jid_s, room_nicks, user_nick): chat_panel = self.getOrCreateRoomWidget(jid.JID(room_jid_s)) chat_panel.setUserNick(user_nick) chat_panel.setPresents(room_nicks) chat_panel.refresh() def _roomLeftCb(self, room_jid_s, room_nicks, user_nick): try: del self.room_list[room_jid_s] except KeyError: try: # as JID is a string-based class, we don't know what will please Pyjamas... del self.room_list[jid.JID(room_jid_s)] except KeyError: pass def _roomUserJoinedCb(self, room_jid_s, user_nick, user_data): lib_wid = self.getOrCreateRoomWidget(jid.JID(room_jid_s)) if lib_wid: lib_wid.userJoined(user_nick, user_data) def _roomUserLeftCb(self, room_jid_s, user_nick, user_data): lib_wid = self.getRoomWidget(jid.JID(room_jid_s)) if lib_wid: lib_wid.userLeft(user_nick, user_data) def _roomUserChangedNickCb(self, room_jid_s, old_nick, new_nick): """Called when an user joined a MUC room""" lib_wid = self.getRoomWidget(jid.JID(room_jid_s)) if lib_wid: lib_wid.changeUserNick(old_nick, new_nick) def _tarotGameStartedCb(self, waiting, room_jid_s, referee, players): lib_wid = self.getRoomWidget(jid.JID(room_jid_s)) if lib_wid: lib_wid.startGame("Tarot", waiting, referee, players) def _tarotGameGenericCb(self, event_name, room_jid_s, args): lib_wid = self.getRoomWidget(jid.JID(room_jid_s)) if lib_wid: getattr(lib_wid.getGame("Tarot"), event_name)(*args) def _radioColStartedCb(self, waiting, room_jid_s, referee, players, queue_data): lib_wid = self.getRoomWidget(jid.JID(room_jid_s)) if lib_wid: lib_wid.startGame("RadioCol", waiting, referee, players, queue_data) def _radioColGenericCb(self, event_name, room_jid_s, args): lib_wid = self.getRoomWidget(jid.JID(room_jid_s)) if lib_wid: getattr(lib_wid.getGame("RadioCol"), event_name)(*args) def _getPresenceStatusesCb(self, presence_data): for entity in presence_data: for resource in presence_data[entity]: args = presence_data[entity][resource] full = ('%s/%s' % (jid.JID(entity).bare, resource)) if resource else entity self._presenceUpdateCb(full, *args) def _getRoomsJoinedCb(self, room_data): for room in room_data: self._roomJoinedCb(*room) def _getWaitingSubCb(self, waiting_sub): for sub in waiting_sub: self._subscribeCb(waiting_sub[sub], sub) def _subscribeCb(self, sub_type, entity): if sub_type == 'subscribed': dialog.InfoDialog('Subscription confirmation', 'The contact <b>%s</b> has added you to his/her contact list' % html_tools.html_sanitize(entity)).show() self.getEntityMBlog(entity) elif sub_type == 'unsubscribed': dialog.InfoDialog('Subscription refusal', 'The contact <b>%s</b> has refused to add you in his/her contact list' % html_tools.html_sanitize(entity)).show() #TODO: remove microblogs from panels elif sub_type == 'subscribe': #The user want to subscribe to our presence _dialog = None msg = HTML('The contact <b>%s</b> want to add you in his/her contact list, do you accept ?' % html_tools.html_sanitize(entity)) def ok_cb(ignore): self.bridge.call('subscription', None, "subscribed", entity, '', _dialog.getSelectedGroups()) def cancel_cb(ignore): self.bridge.call('subscription', None, "unsubscribed", entity, '', '') _dialog = dialog.GroupSelector([msg], self.contact_panel.getGroups(), [], "Add", ok_cb, cancel_cb) _dialog.setHTML('<b>Add contact request</b>') _dialog.show() def _contactDeletedCb(self, entity): self.contact_panel.removeContact(entity) def _newContactCb(self, contact_jid, attributes, groups): self.contact_panel.updateContact(contact_jid, attributes, groups) def _entityDataUpdatedCb(self, entity_jid_s, key, value): if key == "avatar": avatar = '/' + C.AVATARS_DIR + value self.avatars_cache[entity_jid_s] = avatar self.contact_panel.updateAvatar(entity_jid_s, avatar) for lib_wid in self.libervia_widgets: if isinstance(lib_wid, panels.MicroblogPanel): if lib_wid.isJidAccepted(entity_jid_s) or (self.whoami and entity_jid_s == self.whoami.bare): lib_wid.updateValue('avatar', entity_jid_s, avatar) def _chatStateReceivedCb(self, from_jid_s, state): """Callback when a new chat state is received. @param from_jid_s: JID of the contact who sent his state, or '@ALL@' @param state: new state (string) """ if from_jid_s == '@ALL@': for lib_wid in self.libervia_widgets: if isinstance(lib_wid, panels.ChatPanel): lib_wid.setState(state, nick=C.ALL_OCCUPANTS) return from_jid = jid.JID(from_jid_s) lib_wid = self.getLiberviaWidget(panels.ChatPanel, {'item': from_jid}, ignoreOtherTabs=False) lib_wid.setState(state, nick=from_jid.resource) def _askConfirmation(self, confirmation_id, confirmation_type, data): answer_data = {} def confirm_cb(result): self.bridge.call('confirmationAnswer', None, confirmation_id, result, answer_data) if confirmation_type == "YES/NO": dialog.ConfirmDialog(confirm_cb, text=data["message"], title=data["title"]).show() def _newAlert(self, message, title, alert_type): dialog.InfoDialog(title, message).show() def _paramUpdate(self, name, value, category, refresh=True): """This is called when the paramUpdate signal is received, but also during initialization when the UI parameters values are retrieved. @param refresh: set to True to refresh the general UI """ for param_cat, param_name in C.CACHED_PARAMS: if name == param_name and category == param_cat: self.cached_params[(category, name)] = value if refresh: self.refresh() break def getCachedParam(self, category, name): """Return a parameter cached value (e.g for refreshing the UI) @param category (str): the parameter category @pram name (str): the parameter name """ return self.cached_params[(category, name)] if (category, name) in self.cached_params else None def sendError(self, errorData): dialog.InfoDialog("Error while sending message", "Your message can't be sent", Width="400px").center() log.error("sendError: %s" % str(errorData)) def send(self, targets, text, extra={}): """Send a message to any target type. @param targets: list of tuples (type, entities, addr) with: - type in ("PUBLIC", "GROUP", "COMMENT", "STATUS" , "groupchat" , "chat") - entities could be a JID, a list groups, a node hash... depending the target - addr in ("To", "Cc", "Bcc") - ignore case @param text: the message content @param extra: options """ # FIXME: too many magic strings, we should use constants instead addresses = [] for target in targets: type_, entities, addr = target[0], target[1], 'to' if len(target) < 3 else target[2].lower() if type_ in ("PUBLIC", "GROUP"): self.bridge.call("sendMblog", None, type_, entities if type_ == "GROUP" else None, text, extra) elif type_ == "COMMENT": self.bridge.call("sendMblogComment", None, entities, text, extra) elif type_ == "STATUS": assert(isinstance(self.status_panel, panels.PresenceStatusPanel)) self.bridge.call('setStatus', None, self.status_panel.presence, text) # pylint: disable=E1103 elif type_ in ("groupchat", "chat"): addresses.append((addr, entities)) else: log.error("Unknown target type") if addresses: if len(addresses) == 1 and addresses[0][0] == 'to': to_jid_s = addresses[0][1] for plugin in self.plugins.values(): if hasattr(plugin, 'sendMessageTrigger'): if not plugin.sendMessageTrigger(jid.JID(to_jid_s), text, type_, extra): return # plugin returned False to interrupt the process self.bridge.call('sendMessage', (None, self.sendError), to_jid_s, text, '', type_, extra) else: extra.update({'address': '\n'.join([('%s:%s' % entry) for entry in addresses])}) self.bridge.call('sendMessage', (None, self.sendError), self.whoami.domain, text, '', type_, extra) def showWarning(self, type_=None, msg=None): """Display a popup information message, e.g. to notify the recipient of a message being composed. If type_ is None, a popup being currently displayed will be hidden. @type_: a type determining the CSS style to be applied (see WarningPopup.showWarning) @msg: message to be displayed """ if not hasattr(self, "warning_popup"): self.warning_popup = panels.WarningPopup() self.warning_popup.showWarning(type_, msg) if __name__ == '__main__': app = SatWebFrontend() app.onModuleLoad() pyjd.run()