launchpad-reviewers team mailing list archive
  
  - 
     launchpad-reviewers team launchpad-reviewers team
- 
    Mailing list archive
  
- 
    Message #01335
  
 [Merge]	lp:~malaria/launchpad/fixes-bug-608631 into lp:launchpad/devel
  
Nicolas Delvaux has proposed merging lp:~malaria/launchpad/fixes-bug-608631 into lp:launchpad/devel.
Requested reviews:
  Launchpad code reviewers (launchpad-reviewers)
Related bugs:
  #608631 Visual tag to represent narrow non-breaking spaces
  https://bugs.launchpad.net/bugs/608631
== Summary ==
Bug 608631 describe the fact that there is no easy way to input and to display narrow no-break spaces (U+202F) in Rosetta.
Narrow no-break spaces are mainly used in French before ";:!?»" chars and after "«", but they are also used in some other languages (e.g. for the short form of the Czech dates and others).
Input of NNBSP is a problem for example on MS Windows (users need to mess around in the registry) and on GNU/Linux if your keyboard layout does not provide a shortcut for it.
Display of NNBSP is a problem in some browsers.
All versions of MS Internet Explorer *on Windows XP* display a square instead of the regular character (it works on Vista and Seven, here the bug lays in the OS build-in text renderer). There is also a bug in Qt which means that Qt browsers (Konqueror, Rekonq...) display NNBSP as a 0 width char.
Basically, it displays well on Firefox, GTK based browsers (Epiphany, Midori...) and on IE (on Vista and Seven).
On top of that, it is sometime difficult to tell apart NNBSP from a white space or a no-break space. So, a helper to recognize NNBSP is necessary anyway.
== Proposed fix ==
As discussed with Данило Шеган, it seems that implementing a new tag (as already existing [tab] and [nbsp]) is the best answer to this.
== Pre-implementation notes ==
The obvious tag for NNBSP is [nnbsp], but it's too close to [nbsp].
Basically, NNBSP is just the no-break version of the thin space (U+2009) which HTML shortcut is  . But [nbthinsp] is too long, so I just proposed [nbthin].
So far, no one found something better.
== Implementation details ==
Implementation was quite easy with the existing code for other tags.
There is no really new code, so this should not cause anything to break (that wouldn't have broke without this patch I mean ;-)).
== Test ==
bin/test -cvv -m lp.translations -t browser-helpers.txt
== Demo and Q/A ==
To demo and Q/A this change, do the following:
- Log on as Sample Person (test@xxxxxxxxxxxxx:test)
- Visit any translation page (e.g. https://translations.launchpad.dev/evolution/trunk/+pots/evolution-2.2-test/es/+translate )
- Input something like "Test[nbthin]!" and/or it's escaped version "Test\[nbthin]!" and/or the 'literal' version "Test !"
- Save your translation(s) and see the results (better here with the above example: https://translations.launchpad.dev/evolution/trunk/+pots/evolution-2.2-test/es/+filter?person=name12 )
== Launchpad lint ==
Checking for conflicts and issues in changed files.
Linting changed files:
  lib/lp/translations/browser/browser_helpers.py
  lib/lp/translations/doc/browser-helpers.txt
  lib/lp/translations/interfaces/translations.py
I fixed all warnings in these files (most of them where not related to my code)
-- 
https://code.launchpad.net/~malaria/launchpad/fixes-bug-608631/+merge/37287
Your team Launchpad code reviewers is requested to review the proposed merge of lp:~malaria/launchpad/fixes-bug-608631 into lp:launchpad/devel.
=== modified file 'lib/lp/translations/browser/browser_helpers.py'
--- lib/lp/translations/browser/browser_helpers.py	2010-08-20 20:31:18 +0000
+++ lib/lp/translations/browser/browser_helpers.py	2010-10-01 17:01:49 +0000
@@ -31,7 +31,9 @@
     return helpers.text_replaced(text, {'[tab]': '\t',
                                         r'\[tab]': '[tab]',
                                         '[nbsp]': u'\u00a0',
-                                        r'\[nbsp]': '[nbsp]'})
+                                        r'\[nbsp]': '[nbsp]',
+                                        '[nbthin]': u'\u202f',
+                                        r'\[nbthin]': '[nbthin]'})
 
 
 def expand_rosetta_escapes(unicode_text):
@@ -39,7 +41,10 @@
     escapes = {u'\t': TranslationConstants.TAB_CHAR,
                u'[tab]': TranslationConstants.TAB_CHAR_ESCAPED,
                u'\u00a0': TranslationConstants.NO_BREAK_SPACE_CHAR,
-               u'[nbsp]': TranslationConstants.NO_BREAK_SPACE_CHAR_ESCAPED}
+               u'[nbsp]': TranslationConstants.NO_BREAK_SPACE_CHAR_ESCAPED,
+               u'\u202f': TranslationConstants.NARROW_NO_BREAK_SPACE_CHAR,
+               u'[nbthin]':
+    TranslationConstants.NARROW_NO_BREAK_SPACE_CHAR_ESCAPED}
     return helpers.text_replaced(unicode_text, escapes)
 
 
=== modified file 'lib/lp/translations/doc/browser-helpers.txt'
--- lib/lp/translations/doc/browser-helpers.txt	2009-09-07 07:25:18 +0000
+++ lib/lp/translations/doc/browser-helpers.txt	2010-10-01 17:01:49 +0000
@@ -1,9 +1,11 @@
-= Translation message helper functions =
+Translation message helper functions
+====================================
 
 For rendering translations in the TranslationMessageView a number of
 helper functions exist. The following sections cover them in detail.
 
-== contract_rosetta_escapes ==
+contract_rosetta_escapes
+------------------------
 
     >>> from lp.translations.browser.browser_helpers import (
     ...     contract_rosetta_escapes)
@@ -45,8 +47,20 @@
     >>> contract_rosetta_escapes('foo\\[nbsp]bar')
     'foo[nbsp]bar'
 
-
-== expand_rosetta_escapes ==
+Similarly, string '[nbthin]' gets converted to narrow no-break space
+character.
+
+    >>> contract_rosetta_escapes('foo[nbthin]bar')
+    u'foo\u202fbar'
+
+The string '\[nbthin]' gets converted to a literal '[nbthin]'.
+
+    >>> contract_rosetta_escapes('foo\\[nbthin]bar')
+    'foo[nbthin]bar'
+
+
+expand_rosetta_escapes
+----------------------
 
     >>> from lp.translations.browser.browser_helpers import (
     ...     expand_rosetta_escapes)
@@ -93,8 +107,22 @@
     >>> expand_rosetta_escapes(u'foo[nbsp]bar')
     u'foo<code>\\[nbsp]</code>bar'
 
-
-== parse_cformat_string ==
+Similarly, narrow no-break spaces get converted to a special constant
+TranslationConstants.NARROW_NO_BREAK_SPACE_CHAR which renders as below:
+
+    >>> expand_rosetta_escapes(u'foo\u202fbar')
+    u'foo<code>[nbthin]</code>bar'
+
+Literal occurrences of u'[nbthin]' get escaped to a special constant
+TranslationConstants.NARROW_NO_BREAK_SPACE_CHAR_ESCAPED which renders them
+as below:
+
+    >>> expand_rosetta_escapes(u'foo[nbthin]bar')
+    u'foo<code>\\[nbthin]</code>bar'
+
+
+parse_cformat_string
+--------------------
 
     >>> from lp.translations.browser.browser_helpers import (
     ...     parse_cformat_string)
@@ -112,7 +140,8 @@
     UnrecognisedCFormatString: %
 
 
-== text_to_html ==
+text_to_html
+------------
 
     >>> from lp.translations.browser.browser_helpers import (
     ...     text_to_html)
@@ -178,7 +207,8 @@
     u'foo<img alt="" src="/@@/translation-newline" /><br/>\nbar'
 
 
-== convert_newlines_to_web_form ==
+convert_newlines_to_web_form
+----------------------------
 
     >>> from lp.translations.browser.browser_helpers import (
     ...     convert_newlines_to_web_form)
@@ -194,7 +224,8 @@
     u'foo\r\nbar'
 
 
-== count_lines ==
+count_lines
+-----------
 
     >>> from lp.translations.browser.browser_helpers import count_lines
     >>> count_lines("foo")
@@ -216,8 +247,8 @@
     ...     "1234566789a123456789a")
     2
     >>> count_lines(
-    ...     "123456789abc123456789abc123456789abc123456789abc123456789abc123456\n"
-    ...     "789a123456789a123456789a")
+    ...     "123456789abc123456789abc123456789abc123456789abc123456789abc"
+    ...     "123456\n789a123456789a123456789a")
     3
     >>> count_lines(
     ...     "123456789abc123456789abc123456789abc123456789abc123456789abc"
=== modified file 'lib/lp/translations/interfaces/translations.py'
--- lib/lp/translations/interfaces/translations.py	2010-08-20 20:31:18 +0000
+++ lib/lp/translations/interfaces/translations.py	2010-10-01 17:01:49 +0000
@@ -16,6 +16,7 @@
     'TranslationsBranchImportMode',
     )
 
+
 class TranslationConstants:
     """Set of constants used inside the context of translations."""
 
@@ -31,6 +32,8 @@
     TAB_CHAR_ESCAPED = '<code>' + r'\[tab]' + '</code>'
     NO_BREAK_SPACE_CHAR = '<code>[nbsp]</code>'
     NO_BREAK_SPACE_CHAR_ESCAPED = '<code>' + r'\[nbsp]' + '</code>'
+    NARROW_NO_BREAK_SPACE_CHAR = '<code>[nbthin]</code>'
+    NARROW_NO_BREAK_SPACE_CHAR_ESCAPED = '<code>' + r'\[nbthin]' + '</code>'
 
 
 class TranslationsBranchImportMode(DBEnumeratedType):
@@ -54,5 +57,3 @@
         Import all translation files (templates and translations)
         found in the branch.
         """)
-
-