From da6754bf8aef7ae70c95b4f7fa77788f46f1b636 Mon Sep 17 00:00:00 2001
From: Denis Fomin <fominde@gmail.com>
Date: Fri, 26 Nov 2010 17:50:20 +0300
Subject: [PATCH] Correct display hyperlinks in XHTML-IM content. Fixed the
 ability to run htmltextview.py as separate application. Cut long lines. Fixed
 # 6045

---
 src/conversation_textview.py |  65 ++++++++++-------
 src/htmltextview.py          | 138 ++++++++++++++++++++++-------------
 2 files changed, 126 insertions(+), 77 deletions(-)

diff --git a/src/conversation_textview.py b/src/conversation_textview.py
index 34ebc0840c..5a0ea06fb4 100644
--- a/src/conversation_textview.py
+++ b/src/conversation_textview.py
@@ -191,7 +191,7 @@ class ConversationTextview(gobject.GObject):
 
         # no need to inherit TextView, use it as atrribute is safer
         self.tv = HtmlTextView()
-        self.tv.html_hyperlink_handler = self.html_hyperlink_handler
+        self.tv.hyperlink_handler = self.hyperlink_handler
 
         # set properties
         self.tv.set_border_width(1)
@@ -946,10 +946,24 @@ class ConversationTextview(gobject.GObject):
             # we get the end of the tag
             while not end_iter.ends_tag(texttag):
                 end_iter.forward_char()
-            word = self.tv.get_buffer().get_text(begin_iter, end_iter).decode(
+
+            # Detect XHTML-IM link
+            word = getattr(texttag, 'href', None)
+            if word:
+                if word.startswith('xmpp'):
+                    kind = 'xmpp'
+                elif word.startswith('mailto:'):
+                    kind = 'mail'
+                elif gajim.interface.sth_at_sth_dot_sth_re.match(word):
+                    # it's a JID or mail
+                    kind = 'sth_at_sth'
+            else:
+                word = self.tv.get_buffer().get_text(begin_iter, end_iter).decode(
                     'utf-8')
+
             if event.button == 3: # right click
                 self.make_link_menu(event, kind, word)
+                return True
             else:
                 # we launch the correct application
                 if kind == 'xmpp':
@@ -965,16 +979,6 @@ class ConversationTextview(gobject.GObject):
                 else:
                     helpers.launch_browser_mailer(kind, word)
 
-    def html_hyperlink_handler(self, texttag, widget, event, iter_, kind, href):
-        if event.type == gtk.gdk.BUTTON_PRESS:
-            if event.button == 3: # right click
-                self.make_link_menu(event, kind, href)
-                return True
-            else:
-                # we launch the correct application
-                helpers.launch_browser_mailer(kind, href)
-
-
     def detect_and_print_special_text(self, otext, other_tags, graphics=True):
         """
         Detect special text (emots & links & formatting), print normal text
@@ -1039,6 +1043,14 @@ class ConversationTextview(gobject.GObject):
                 gajim.config.get('show_ascii_formatting_chars')
         buffer_ = self.tv.get_buffer()
 
+        # Detect XHTML-IM link
+        ttt = buffer_.get_tag_table()
+        tags_ = [(ttt.lookup(t) if isinstance(t, str) else t) for t in other_tags]
+        for t in tags_:
+            is_xhtml_link = getattr(t, 'href', None)
+            if is_xhtml_link:
+                break
+
         # Check if we accept this as an uri
         schemes = gajim.config.get('uri_schemes').split()
         for scheme in schemes:
@@ -1062,21 +1074,18 @@ class ConversationTextview(gobject.GObject):
             self.images.append(img)
             # add with possible animation
             self.tv.add_child_at_anchor(img, anchor)
-        elif special_text.startswith('www.') or \
-        special_text.startswith('ftp.') or \
-        text_is_valid_uri:
-            tags.append('url')
-            use_other_tags = False
-        elif special_text.startswith('mailto:'):
-            tags.append('mail')
-            use_other_tags = False
-        elif special_text.startswith('xmpp:'):
-            tags.append('xmpp')
-            use_other_tags = False
-        elif gajim.interface.sth_at_sth_dot_sth_re.match(special_text):
-            # it's a JID or mail
-            tags.append('sth_at_sth')
-            use_other_tags = False
+        elif not is_xhtml_link:
+            if special_text.startswith('www.') or \
+            special_text.startswith('ftp.') or \
+            text_is_valid_uri:
+                tags.append('url')
+            elif special_text.startswith('mailto:'):
+                tags.append('mail')
+            elif special_text.startswith('xmpp:'):
+                tags.append('xmpp')
+            elif gajim.interface.sth_at_sth_dot_sth_re.match(special_text):
+                # it's a JID or mail
+                tags.append('sth_at_sth')
         elif special_text.startswith('*'): # it's a bold text
             tags.append('bold')
             if special_text[1] == '/' and special_text[-2] == '/' and\
@@ -1163,7 +1172,6 @@ class ConversationTextview(gobject.GObject):
             if use_other_tags:
                 all_tags += other_tags
             # convert all names to TextTag
-            ttt = buffer_.get_tag_table()
             all_tags = [(ttt.lookup(t) if isinstance(t, str) else t) for t in all_tags]
             buffer_.insert_with_tags(end_iter, special_text, *all_tags)
 
@@ -1361,6 +1369,7 @@ class ConversationTextview(gobject.GObject):
                 self.tv.display_html(xhtml.encode('utf-8'), self)
                 return
             except Exception, e:
+                print e
                 gajim.log.debug('Error processing xhtml' + str(e))
                 gajim.log.debug('with |' + xhtml + '|')
 
diff --git a/src/htmltextview.py b/src/htmltextview.py
index 4883058433..52e77e2bf8 100644
--- a/src/htmltextview.py
+++ b/src/htmltextview.py
@@ -50,7 +50,9 @@ import operator
 if __name__ == '__main__':
     from common import i18n
     import common.configpaths
+    common.configpaths.gajimpaths.init_profile()
     common.configpaths.gajimpaths.init(None)
+    import gtkgui_helpers
 from common import gajim
 
 import tooltips
@@ -182,8 +184,7 @@ for name in BLOCK_HEAD:
     size = (num-1) // 2
     weigth = (num - 1) % 2
     element_styles[name] = '; font-size: %s; %s' % ( ('large', 'medium', 'small')[size],
-                                                                                                    ('font-weight: bold', 'font-style: oblique')[weigth],
-                                                                                      )
+        ('font-weight: bold', 'font-style: oblique')[weigth],)
 
 def _parse_css_color(color):
     if color.startswith('rgb(') and color.endswith(')'):
@@ -215,7 +216,7 @@ class HtmlHandler(xml.sax.handler.ContentHandler):
         self.preserve = False
         self.styles = [] # a gtk.TextTag or None, for each span level
         self.list_counters = [] # stack (top at head) of list
-                                                        # counters, or None for unordered list
+                                # counters, or None for unordered list
 
     def _parse_style_color(self, tag, value):
         color = _parse_css_color(value)
@@ -234,11 +235,12 @@ class HtmlHandler(xml.sax.handler.ContentHandler):
         self.iter.forward_char()
         return attrs
 
-    def __parse_length_frac_size_allocate(self, textview, allocation,
-                                                                              frac, callback, args):
+    def __parse_length_frac_size_allocate(self, textview, allocation, frac,
+        callback, args):
         callback(allocation.width*frac, *args)
 
-    def _parse_length(self, value, font_relative, block_relative, minl, maxl, callback, *args):
+    def _parse_length(self, value, font_relative, block_relative, minl, maxl,
+        callback, *args):
         """
         Parse/calc length, converting to pixels, calls callback(length, *args)
         when the length is first computed or changes
@@ -260,10 +262,10 @@ class HtmlHandler(xml.sax.handler.ContentHandler):
                 # textview width instead; a reasonable approximation..
                 alloc = self.textview.get_allocation()
                 self.__parse_length_frac_size_allocate(self.textview, alloc,
-                                                                                           frac, callback, args)
+                    frac, callback, args)
                 self.textview.connect('size-allocate',
-                                                          self.__parse_length_frac_size_allocate,
-                                                          frac, callback, args)
+                    self.__parse_length_frac_size_allocate,
+                    frac, callback, args)
             else:
                 callback(frac, *args)
             return
@@ -335,7 +337,8 @@ class HtmlHandler(xml.sax.handler.ContentHandler):
             tag.set_property('scale', pango.SCALE_LARGE)
             return
         # font relative (5 ~ 4pt, 110 ~ 72pt)
-        self._parse_length(value, True, False, 5, 110, self.__parse_font_size_cb, tag)
+        self._parse_length(value, True, False, 5, 110,self.__parse_font_size_cb,
+            tag)
 
     def _parse_style_font_style(self, tag, value):
         try:
@@ -358,13 +361,13 @@ class HtmlHandler(xml.sax.handler.ContentHandler):
 
     def _parse_style_margin_left(self, tag, value):
         # block relative
-        self._parse_length(value, False, True, 1, 1000, self.__frac_length_tag_cb,
-                                           tag, 'left-margin')
+        self._parse_length(value, False, True, 1, 1000,
+            self.__frac_length_tag_cb, tag, 'left-margin')
 
     def _parse_style_margin_right(self, tag, value):
         # block relative
-        self._parse_length(value, False, True, 1, 1000, self.__frac_length_tag_cb,
-                                           tag, 'right-margin')
+        self._parse_length(value, False, True, 1, 1000,
+            self.__frac_length_tag_cb, tag, 'right-margin')
 
     def _parse_style_font_weight(self, tag, value):
         # TODO: missing 'bolder' and 'lighter'
@@ -440,20 +443,20 @@ class HtmlHandler(xml.sax.handler.ContentHandler):
         if value == 'auto':
             return
         self._parse_length(value, False, False, 1, 1000, self.__length_tag_cb,
-                                           tag, "width")
+            tag, "width")
     def _parse_style_height(self, tag, value):
         if value == 'auto':
             return
         self._parse_length(value, False, False, 1, 1000, self.__length_tag_cb,
-                                           tag, "height")
+            tag, "height")
 
 
     # build a dictionary mapping styles to methods, for greater speed
     __style_methods = dict()
     for style in ('background-color', 'color', 'font-family', 'font-size',
-                              'font-style', 'font-weight', 'margin-left', 'margin-right',
-                              'text-align', 'text-decoration', 'white-space', 'display',
-                              'width', 'height' ):
+                  'font-style', 'font-weight', 'margin-left', 'margin-right',
+                  'text-align', 'text-decoration', 'white-space', 'display',
+                  'width', 'height' ):
         try:
             method = locals()['_parse_style_%s' % style.replace('-', '_')]
         except KeyError:
@@ -473,7 +476,7 @@ class HtmlHandler(xml.sax.handler.ContentHandler):
         if href and href[0] != '#':
             tag.href = href
             tag.type_ = type_ # to be used by the URL handler
-            tag.connect('event', self.textview.html_hyperlink_handler, 'url', href)
+            tag.connect('event', self.textview.hyperlink_handler, 'url')
             tag.set_property('foreground', gajim.config.get('urlmsgcolor'))
             tag.set_property('underline', pango.UNDERLINE_SINGLE)
             tag.is_anchor = True
@@ -507,7 +510,7 @@ class HtmlHandler(xml.sax.handler.ContentHandler):
                 while True:
                     if time.time() > deadline:
                         gajim.log.debug(str('Timeout loading image %s ' % \
-                                attrs['src'] + ex))
+                            attrs['src'] + ex))
                         mem = ''
                         alt = attrs.get('alt', '')
                         if alt:
@@ -517,8 +520,8 @@ class HtmlHandler(xml.sax.handler.ContentHandler):
                     try:
                         temp = f.read(100)
                     except socket.timeout, ex:
-                        gajim.log.debug('Timeout loading image %s ' % attrs['src'] + \
-                                str(ex))
+                        gajim.log.debug('Timeout loading image %s ' % \
+                            attrs['src'] + str(ex))
                         alt = attrs.get('alt', '')
                         if alt:
                             alt += '\n'
@@ -618,7 +621,7 @@ class HtmlHandler(xml.sax.handler.ContentHandler):
                 method = self.__style_methods[attr]
             except KeyError:
                 warnings.warn('Style attribute "%s" requested '
-                                          'but not yet implemented' % attr)
+                    'but not yet implemented' % attr)
             else:
                 method(self, tag, val)
         self.styles.append(tag)
@@ -658,14 +661,14 @@ class HtmlHandler(xml.sax.handler.ContentHandler):
         return False
 
     def handle_specials(self, text):
-        self.iter = self.conv_textview.detect_and_print_special_text(text,                                                      self._get_style_tags())
+        self.iter = self.conv_textview.detect_and_print_special_text(text,
+            self._get_style_tags())
 
     def characters(self, content):
         if self.preserve:
             self.text += content
             return
         if allwhitespace_rx.match(content) is not None and self._starts_line():
-            self.text += ' '
             return
         self.text += content
         self.starting = False
@@ -763,7 +766,8 @@ class HtmlHandler(xml.sax.handler.ContentHandler):
             #FIXME: plenty of unused attributes (width, height,...) :)
             self._jump_line()
             try:
-                self.textbuf.insert_pixbuf(self.iter, self.textview.focus_out_line_pixbuf)
+                self.textbuf.insert_pixbuf(self.iter,
+                    self.textview.focus_out_line_pixbuf)
                 #self._insert_text(u'\u2550'*40)
                 self._jump_line()
             except Exception, e:
@@ -859,7 +863,8 @@ class HtmlTextView(gtk.TextView):
             window = widget.get_window(gtk.TEXT_WINDOW_TEXT)
             window.set_cursor(gtk.gdk.Cursor(gtk.gdk.HAND2))
             self._changed_cursor = True
-            self.tooltip.timeout = gobject.timeout_add(500, self.show_tooltip, anchor_tags[0])
+            self.tooltip.timeout = gobject.timeout_add(500, self.show_tooltip,
+                anchor_tags[0])
         elif self._changed_cursor and not anchor_tags:
             window = widget.get_window(gtk.TEXT_WINDOW_TEXT)
             window.set_cursor(gtk.gdk.Cursor(gtk.gdk.XTERM))
@@ -886,10 +891,12 @@ class HtmlTextView(gtk.TextView):
         self.emit_stop_by_name('copy-clipboard')
 
     def on_html_text_view_realized(self, unused_data):
-        self.get_buffer().remove_selection_clipboard(self.get_clipboard(gtk.gdk.SELECTION_PRIMARY))
+        self.get_buffer().remove_selection_clipboard(self.get_clipboard(
+            gtk.gdk.SELECTION_PRIMARY))
 
     def on_html_text_view_unrealized(self, unused_data):
-        self.get_buffer().add_selection_clipboard(self.get_clipboard(gtk.gdk.SELECTION_PRIMARY))
+        self.get_buffer().add_selection_clipboard(self.get_clipboard(
+            gtk.gdk.SELECTION_PRIMARY))
 
     def on_text_buffer_mark_set(self, location, mark, unused_data):
         bounds = self.get_buffer().get_selection_bounds()
@@ -955,19 +962,19 @@ if __name__ == '__main__':
         """
         global change_cursor
         pointer_x, pointer_y = htmlview.tv.window.get_pointer()[0:2]
-        x, y = htmlview.tv.window_to_buffer_coords(gtk.TEXT_WINDOW_TEXT, pointer_x,
-                                                           pointer_y)
+        x, y = htmlview.tv.window_to_buffer_coords(gtk.TEXT_WINDOW_TEXT,
+            pointer_x, pointer_y)
         tags = htmlview.tv.get_iter_at_location(x, y).get_tags()
         if change_cursor:
             htmlview.tv.get_window(gtk.TEXT_WINDOW_TEXT).set_cursor(
-                             gtk.gdk.Cursor(gtk.gdk.XTERM))
+                gtk.gdk.Cursor(gtk.gdk.XTERM))
             change_cursor = None
         tag_table = htmlview.tv.get_buffer().get_tag_table()
         for tag in tags:
             try:
                 if tag.is_anchor:
                     htmlview.tv.get_window(gtk.TEXT_WINDOW_TEXT).set_cursor(
-                                                            gtk.gdk.Cursor(gtk.gdk.HAND2))
+                        gtk.gdk.Cursor(gtk.gdk.HAND2))
                     change_cursor = tag
                 elif tag == tag_table.lookup('focus-out-line'):
                     over_line = True
@@ -987,22 +994,30 @@ if __name__ == '__main__':
 
     htmlview.tv.connect('motion_notify_event', on_textview_motion_notify_event)
 
-    def handler(texttag, widget, event, iter_, kind, href):
+    def handler(texttag, widget, event, iter_, kind):
         if event.type == gtk.gdk.BUTTON_PRESS:
-            print href
+            pass
 
-    htmlview.tv.html_hyperlink_handler = handler
+    htmlview.tv.hyperlink_handler = htmlview.hyperlink_handler
 
-    htmlview.print_real_text(None, xhtml='<div><span style="color: red; text-decoration:underline">Hello</span><br/>\n'
-                                              '  <img src="http://images.slashdot.org/topics/topicsoftware.gif"/><br/>\n'
-                                              '  <span style="font-size: 500%; font-family: serif">World</span>\n'
-                                              '</div>\n')
+    htmlview.print_real_text(None, xhtml='<div>'
+    '<span style="color: red; text-decoration:underline">Hello</span><br/>\n'
+      '  <img src="http://images.slashdot.org/topics/topicsoftware.gif"/><br/>\n'
+    '<span style="font-size: 500%; font-family: serif">World</span>\n'
+      '</div>\n')
     htmlview.print_real_text(None, xhtml='<hr />')
-    htmlview.print_real_text(None, xhtml='''<body xmlns='http://www.w3.org/1999/xhtml'><p xmlns='http://www.w3.org/1999/xhtml'>a:b<a href='http://google.com/' xmlns='http://www.w3.org/1999/xhtml'>Google</a></p><br/></body>''')
+    htmlview.print_real_text(None, xhtml='''
+    <body xmlns='http://www.w3.org/1999/xhtml'>
+     <p xmlns='http://www.w3.org/1999/xhtml'>a:b
+       <a href='http://google.com/' xmlns='http://www.w3.org/1999/xhtml'>Google
+       </a>
+     </p><br/>
+    </body>''')
     htmlview.print_real_text(None, xhtml='''
      <body xmlns='http://www.w3.org/1999/xhtml'>
       <p style='font-size:large'>
-            <span style='font-style: italic'>O<span style='font-size:larger'>M</span>G</span>,
+            <span style='font-style: italic'>O
+            <span style='font-size:larger'>M</span>G</span>,
             I&apos;m <span style='color:green'>green</span>
             with <span style='font-weight: bold'>envy</span>!
       </p>
@@ -1017,7 +1032,8 @@ if __name__ == '__main__':
     htmlview.print_real_text(None, xhtml='<hr />')
     htmlview.print_real_text(None, xhtml='''
     <body xmlns='http://www.w3.org/1999/xhtml'>
-      <p>As Emerson said in his essay <span style='font-style: italic; background-color:cyan'>Self-Reliance</span>:</p>
+      <p>As Emerson said in his essay <span style='
+        font-style: italic; background-color:cyan'>Self-Reliance</span>:</p>
       <p style='margin-left: 5px; margin-right: 2%'>
             &quot;A foolish consistency is the hobgoblin of little minds.&quot;
       </p>
@@ -1026,11 +1042,13 @@ if __name__ == '__main__':
     htmlview.print_real_text(None, xhtml='<hr />')
     htmlview.print_real_text(None, xhtml='''
     <body xmlns='http://www.w3.org/1999/xhtml'>
-      <p style='text-align:center'>Hey, are you licensed to <a href='http://www.jabber.org/'>Jabber</a>?</p>
-      <p style='text-align:right'><img src='http://www.jabber.org/images/psa-license.jpg'
-                      alt='A License to Jabber'
-                      width='50%' height='50%'
-                      /></p>
+      <p style='text-align:center'>
+        Hey, are you licensed to <a href='http://www.jabber.org/'>Jabber</a>?
+      </p>
+      <p style='text-align:right'>
+        <img src='http://www.xmpp.org/images/psa-license.jpg'
+        alt='A License to Jabber' width='50%' height='50%'/>
+      </p>
     </body>
             ''')
     htmlview.print_real_text(None, xhtml='<hr />')
@@ -1062,6 +1080,28 @@ return faciter(n,1)</pre>
        <li> Three </li></ol>
     </body>
             ''')
+    htmlview.print_real_text(None, xhtml='<hr />')
+    htmlview.print_real_text(None, xhtml='''
+    <body xmlns='http://www.w3.org/1999/xhtml'>
+    <p>
+      <strong>
+        <a href='xmpp:example@example.org'>xmpp link</a>
+      </strong>: </p>
+    <div xmlns='http://www.w3.org/1999/xhtml'>
+      <cite style='margin: 7px;' title='xmpp:examples@example.org'>
+        <p>
+          <strong>examples@example.org wrote:</strong>
+        </p>
+        <p>this cite - bla bla bla, smile- :-)  ...</p>
+      </cite>
+      <div>
+        <p>some text</p>
+      </div>
+    </div>
+    <p/>
+    <p>#232/1</p>
+    </body>
+    ''')
     htmlview.tv.show()
     sw = gtk.ScrolledWindow()
     sw.set_property('hscrollbar-policy', gtk.POLICY_AUTOMATIC)
-- 
GitLab