0005088: Narrated Web Site Report sort order different Windows vs Linux. Initial commit to resolve the issues on Linux and Mac (provided in both cases PyICU is installed). Windows still needs to be tested, and there may still be some other uses of strxfrm (or strcoll) that need to be fixed.
svn: r21175
This commit is contained in:
parent
32b30ccc09
commit
99c54b5e2e
@ -31,9 +31,11 @@ Utility functions to cast types
|
|||||||
# Python modules
|
# Python modules
|
||||||
#
|
#
|
||||||
#-------------------------------------------------------------------------
|
#-------------------------------------------------------------------------
|
||||||
|
import os
|
||||||
import locale
|
import locale
|
||||||
import sys
|
import sys
|
||||||
|
import logging
|
||||||
|
LOG = logging.getLogger(".")
|
||||||
#-------------------------------------------------------------------------
|
#-------------------------------------------------------------------------
|
||||||
#
|
#
|
||||||
# Gramps modules
|
# Gramps modules
|
||||||
@ -42,19 +44,48 @@ import sys
|
|||||||
from ..datehandler import codeset
|
from ..datehandler import codeset
|
||||||
from ..constfunc import conv_to_unicode, conv_to_unicode_direct, UNITYPE, STRTYPE
|
from ..constfunc import conv_to_unicode, conv_to_unicode_direct, UNITYPE, STRTYPE
|
||||||
|
|
||||||
"""
|
try:
|
||||||
strxfrm needs it's unicode argument correctly cast before used.
|
import PyICU
|
||||||
"""
|
if os.environ.has_key("LC_COLLATE"):
|
||||||
if sys.version_info[0] < 3:
|
collation = os.environ['LC_COLLATE']
|
||||||
conv_unicode_tosrtkey = lambda x: locale.strxfrm(x.encode(codeset, 'replace'))
|
else:
|
||||||
else:
|
collation = os.environ["LANG"]
|
||||||
conv_unicode_tosrtkey = lambda x: locale.strxfrm(x)
|
language_and_country = collation.rsplit('.', 1)[0]
|
||||||
|
if language_and_country in PyICU.Collator.getAvailableLocales().keys():
|
||||||
if codeset == 'UTF-8':
|
loc = language_and_country
|
||||||
conv_str_tosrtkey = lambda x: locale.strxfrm(x)
|
else:
|
||||||
else:
|
language = collation.rsplit('_', 1)[0]
|
||||||
conv_str_tosrtkey = lambda x: locale.strxfrm(
|
if language in PyICU.Collator.getAvailableLocales().keys():
|
||||||
conv_to_unicode(x,'UTF-8').encode(codeset, 'replace'))
|
LOG.warn(_("Language and country %s not supported by ICU: "
|
||||||
|
"but language %s is supported and will be used" %
|
||||||
|
(language_and_country, language)))
|
||||||
|
loc = language
|
||||||
|
else:
|
||||||
|
LOG.warn(_("Neither Language and country %s nor language %s "
|
||||||
|
"supported by ICU: using en_GB" %
|
||||||
|
(language_and_country, language)))
|
||||||
|
loc = "en_GB"
|
||||||
|
|
||||||
|
collator = PyICU.Collator.createInstance(PyICU.Locale(loc))
|
||||||
|
# on ICU, the functions need to receive unicode
|
||||||
|
conv_unicode_tosrtkey = lambda x: collator.getCollationKey(
|
||||||
|
x).getByteArray()
|
||||||
|
conv_str_tosrtkey = lambda x: collator.getCollationKey(
|
||||||
|
x.decode("UTF-8")).getByteArray()
|
||||||
|
except:
|
||||||
|
"""
|
||||||
|
strxfrm needs it's unicode argument correctly cast before used.
|
||||||
|
"""
|
||||||
|
if sys.version_info[0] < 3:
|
||||||
|
conv_unicode_tosrtkey = lambda x: locale.strxfrm(x.encode(codeset, 'replace'))
|
||||||
|
else:
|
||||||
|
conv_unicode_tosrtkey = lambda x: locale.strxfrm(x)
|
||||||
|
|
||||||
|
if codeset == 'UTF-8':
|
||||||
|
conv_str_tosrtkey = lambda x: locale.strxfrm(x)
|
||||||
|
else:
|
||||||
|
conv_str_tosrtkey = lambda x: locale.strxfrm(
|
||||||
|
conv_to_unicode(x,'UTF-8').encode(codeset, 'replace'))
|
||||||
|
|
||||||
def conv_tosrtkey(value):
|
def conv_tosrtkey(value):
|
||||||
if isinstance(value, UNITYPE):
|
if isinstance(value, UNITYPE):
|
||||||
|
@ -112,8 +112,8 @@ class FlatNodeMap(object):
|
|||||||
the path, and a dictionary mapping hndl to index.
|
the path, and a dictionary mapping hndl to index.
|
||||||
To obtain index given a path, method real_index() is available
|
To obtain index given a path, method real_index() is available
|
||||||
|
|
||||||
..Note: If a string sortkey is used, apply conv_unicode_tosrtkey
|
..Note: conv_unicode_tosrtkey is applied to the underlying sort key,
|
||||||
on it , so as to have localized sort
|
so as to have localized sort
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
@ -456,6 +456,8 @@ class FlatBaseModel(GObject.Object, Gtk.TreeModel):
|
|||||||
"""
|
"""
|
||||||
The base class for all flat treeview models.
|
The base class for all flat treeview models.
|
||||||
It keeps a FlatNodeMap, and obtains data from database as needed
|
It keeps a FlatNodeMap, and obtains data from database as needed
|
||||||
|
..Note: conv_unicode_tosrtkey is applied to the underlying sort key,
|
||||||
|
so as to have localized sort
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, db, scol=0, order=Gtk.SortType.ASCENDING,
|
def __init__(self, db, scol=0, order=Gtk.SortType.ASCENDING,
|
||||||
@ -478,9 +480,9 @@ class FlatBaseModel(GObject.Object, Gtk.TreeModel):
|
|||||||
self.sort_map = [ f for f in sort_map if f[0]]
|
self.sort_map = [ f for f in sort_map if f[0]]
|
||||||
#we need the model col, that corresponds with scol
|
#we need the model col, that corresponds with scol
|
||||||
col = self.sort_map[scol][1]
|
col = self.sort_map[scol][1]
|
||||||
self.sort_func = self.smap[col]
|
|
||||||
else:
|
else:
|
||||||
self.sort_func = self.smap[scol]
|
col = scol
|
||||||
|
self.sort_func = lambda x: conv_unicode_tosrtkey(self.smap[col](x))
|
||||||
self.sort_col = scol
|
self.sort_col = scol
|
||||||
self.skip = skip
|
self.skip = skip
|
||||||
self._in_build = False
|
self._in_build = False
|
||||||
@ -577,15 +579,11 @@ class FlatBaseModel(GObject.Object, Gtk.TreeModel):
|
|||||||
Return the (sort_key, handle) list of all data that can maximally
|
Return the (sort_key, handle) list of all data that can maximally
|
||||||
be shown.
|
be shown.
|
||||||
This list is sorted ascending, via localized string sort.
|
This list is sorted ascending, via localized string sort.
|
||||||
conv_unicode_tosrtkey which uses strxfrm
|
|
||||||
"""
|
"""
|
||||||
# use cursor as a context manager
|
# use cursor as a context manager
|
||||||
with self.gen_cursor() as cursor:
|
with self.gen_cursor() as cursor:
|
||||||
#loop over database and store the sort field, and the handle, and
|
#loop over database and store the sort field, and the handle
|
||||||
#allow for a third iter
|
return sorted((self.sort_func(data), key) for key, data in cursor)
|
||||||
return sorted((list(map(conv_tosrtkey,
|
|
||||||
self.sort_func(data))), handle2internal(key))
|
|
||||||
for key, data in cursor)
|
|
||||||
|
|
||||||
def _rebuild_search(self, ignore=None):
|
def _rebuild_search(self, ignore=None):
|
||||||
""" function called when view must be build, given a search text
|
""" function called when view must be build, given a search text
|
||||||
@ -656,8 +654,7 @@ class FlatBaseModel(GObject.Object, Gtk.TreeModel):
|
|||||||
if self.node_map.get_path_from_handle(handle) is not None:
|
if self.node_map.get_path_from_handle(handle) is not None:
|
||||||
return # row is already displayed
|
return # row is already displayed
|
||||||
data = self.map(handle)
|
data = self.map(handle)
|
||||||
insert_val = (list(map(conv_tosrtkey, self.sort_func(data))),
|
insert_val = (self.sort_func(data), handle)
|
||||||
handle)
|
|
||||||
if not self.search or \
|
if not self.search or \
|
||||||
(self.search and self.search.match(handle, self.db)):
|
(self.search and self.search.match(handle, self.db)):
|
||||||
#row needs to be added to the model
|
#row needs to be added to the model
|
||||||
@ -692,8 +689,7 @@ class FlatBaseModel(GObject.Object, Gtk.TreeModel):
|
|||||||
return # row is not currently displayed
|
return # row is not currently displayed
|
||||||
self.clear_cache(handle)
|
self.clear_cache(handle)
|
||||||
oldsortkey = self.node_map.get_sortkey(handle)
|
oldsortkey = self.node_map.get_sortkey(handle)
|
||||||
newsortkey = list(map(conv_tosrtkey, self.sort_func(self.map(
|
newsortkey = self.sort_func(self.map(handle))
|
||||||
handle))))
|
|
||||||
if oldsortkey is None or oldsortkey != newsortkey:
|
if oldsortkey is None or oldsortkey != newsortkey:
|
||||||
#or the changed object is not present in the view due to filtering
|
#or the changed object is not present in the view due to filtering
|
||||||
#or the order of the object must change.
|
#or the order of the object must change.
|
||||||
|
Loading…
x
Reference in New Issue
Block a user