← Back to team overview

launchpad-reviewers team mailing list archive

[Merge] lp:~wallyworld/launchpad/person-picker-extra-detail into lp:launchpad

 

Ian Booth has proposed merging lp:~wallyworld/launchpad/person-picker-extra-detail into lp:launchpad with lp:~wallyworld/launchpad/irc-nickname-formatter as a prerequisite.

Requested reviews:
  Launchpad code reviewers (launchpad-reviewers)
  Curtis Hovey (sinzui): code
Related bugs:
  Bug #669930 in Launchpad itself: "person picker isn't clear and unambiguous about the person being picked"
  https://bugs.launchpad.net/launchpad/+bug/669930

For more details, see:
https://code.launchpad.net/~wallyworld/launchpad/person-picker-extra-detail/+merge/62238

Bug 669930: person picker isn't clear and unambiguous about the person being picked

To solve the issue, add extra detail to what's displayed in the person picker:

Display Name (~lp username)
Email Address (irc nicks)

Also we need to overlay a badge/icon on the person/team sprite to indicate the affiliation the person/team has with the pillar associated with the current context. This part isn't done yet but the hooks are in place. What's required is the business logic to do the affiliation check. Once it is determined how a user is affiliated (if at all), the idea is to return the name of the badge/icon. This is used as the "image" attribute in the lazr-js picker entry object. lazr-js renders it to the left of the entry title (display name) which at the moment means it will overlay the css sprite used for team/person. It doesn't look too bad but we need to work out exactly what UI we want, which may well require a lazr-js update.

== Implementation ==

Modify the existing adapters defined in vocabulary.py to make them classes and allowing the required PickerEntry object to be constructed via an API call to getPickerEntry(associated_object). The associated_object parameter is the current context object for which the picker is being created (BugTask etc). A new IHasAffiliation interface has been created with some adapter infrastructure. The idea is that the associated_object mentioned before will be adapted to an IHasAffiliation instance and the relevant affiliation determined.

The relevant code from the PersonPickerEntryAdaptor is something like:

    def getPickerEntry(self, associated_object):
        person = self.context
        extra = super(PersonPickerEntryAdapter, self).getPickerEntry(associated_object)

        # If the person is affiliated with the associated_object then we can display a badge.
        badge_name = IHasAffiliation(associated_object).getAffiliationBadge(person)

So a simple one line call is made to figure out the affiliation and the business logic to do so can be delivered via a subsequent branch.

There's also quite a few changes due to lint cleanup.

== Demo ==

http://people.canonical.com/~ianb/person-picker-extra-detail.png

== Tests ==

test_tales.py: add new TestIRCNicknameFormatterAPI test case
vocabularies-json.txt: extend existing tests to cover the new picker display attributes

== Lint ==

Checking for conflicts and issues in changed files.

Linting changed files:
  lib/canonical/launchpad/browser/vocabulary.py
  lib/canonical/launchpad/doc/vocabulary-json.txt
  lib/canonical/launchpad/zcml/launchpad.zcml
  lib/lp/app/browser/configure.zcml
  lib/lp/app/browser/tales.py
  lib/lp/app/tests/test_tales.py
  lib/lp/registry/configure.zcml
  lib/lp/registry/vocabularies.py
  lib/lp/registry/model/pillaraffiliation.py

./lib/canonical/launchpad/doc/vocabulary-json.txt
       1: narrative uses a moin header.
      83: want exceeds 78 characters.
./lib/lp/registry/vocabularies.py
     594: E261 at least two spaces before inline comment
-- 
https://code.launchpad.net/~wallyworld/launchpad/person-picker-extra-detail/+merge/62238
Your team Launchpad code reviewers is requested to review the proposed merge of lp:~wallyworld/launchpad/person-picker-extra-detail into lp:launchpad.
=== modified file 'lib/canonical/launchpad/browser/vocabulary.py'
--- lib/canonical/launchpad/browser/vocabulary.py	2011-01-18 18:33:21 +0000
+++ lib/canonical/launchpad/browser/vocabulary.py	2011-05-25 07:18:44 +0000
@@ -6,16 +6,13 @@
 __metaclass__ = type
 
 __all__ = [
-    'branch_to_vocabularyjson',
-    'default_vocabularyjson_adapter',
     'HugeVocabularyJSONView',
     'IPickerEntry',
-    'person_to_vocabularyjson',
-    'sourcepackagename_to_vocabularyjson',
     ]
 
+import simplejson
+
 from lazr.restful.interfaces import IWebServiceClientRequest
-import simplejson
 from zope.app.form.interfaces import MissingInputError
 from zope.app.schema.vocabulary import IVocabularyFactory
 from zope.component import (
@@ -25,7 +22,6 @@
 from zope.component.interfaces import ComponentLookupError
 from zope.interface import (
     Attribute,
-    implementer,
     implements,
     Interface,
     )
@@ -34,13 +30,18 @@
 from canonical.launchpad.webapp.batching import BatchNavigator
 from canonical.launchpad.webapp.interfaces import NoCanonicalUrl
 from canonical.launchpad.webapp.publisher import canonical_url
-from lp.app.browser.tales import ObjectImageDisplayAPI
+from lp.app.browser.tales import (
+    IRCNicknameFormatterAPI,
+    ObjectImageDisplayAPI,
+    )
 from canonical.launchpad.webapp.vocabulary import IHugeVocabulary
 from lp.app.errors import UnexpectedFormData
 from lp.code.interfaces.branch import IBranch
 from lp.registry.interfaces.person import IPerson
 from lp.registry.interfaces.sourcepackagename import ISourcePackageName
+from lp.registry.model.pillaraffiliation import IHasAffiliation
 from lp.registry.model.sourcepackagename import getSourcePackageDescriptions
+from lp.services.features import getFeatureFlag
 
 # XXX: EdwinGrubbs 2009-07-27 bug=405476
 # This limits the output to one line of text, since the sprite class
@@ -68,51 +69,102 @@
         self.css = css
 
 
-@implementer(IPickerEntry)
 @adapter(Interface)
-def default_pickerentry_adapter(obj):
+class DefaultPickerEntryAdapter(object):
     """Adapts Interface to IPickerEntry."""
-    extra = PickerEntry()
-    if hasattr(obj, 'summary'):
-        extra.description = obj.summary
-    display_api = ObjectImageDisplayAPI(obj)
-    extra.css = display_api.sprite_css()
-    if extra.css is None:
-        extra.css = 'sprite bullet'
-    return extra
-
-
-@implementer(IPickerEntry)
+
+    implements(IPickerEntry)
+
+    def __init__(self, context):
+        self.context = context
+
+    def getPickerEntry(self, associated_object, **kwarg):
+        """ Construct a PickerEntry for the context of this adapter.
+
+        The associated_object represents the context for which the picker is
+        being rendered. eg a picker used to select a bug task assignee will
+        have associated_object set to the bug task.
+        """
+        extra = PickerEntry()
+        if hasattr(self.context, 'summary'):
+            extra.description = self.context.summary
+        display_api = ObjectImageDisplayAPI(self.context)
+        extra.css = display_api.sprite_css()
+        if extra.css is None:
+            extra.css = 'sprite bullet'
+        return extra
+
+
 @adapter(IPerson)
-def person_to_pickerentry(person):
+class PersonPickerEntryAdapter(DefaultPickerEntryAdapter):
     """Adapts IPerson to IPickerEntry."""
-    extra = default_pickerentry_adapter(person)
-    if person.preferredemail is not None:
-        try:
-            extra.description = person.preferredemail.email
-        except Unauthorized:
-            extra.description = '<email address hidden>'
-    return extra
-
-
-@implementer(IPickerEntry)
+
+    def getPickerEntry(self, associated_object, **kwarg):
+        person = self.context
+        extra = super(PersonPickerEntryAdapter, self).getPickerEntry(
+            associated_object)
+
+        enhanced_picker_enabled = kwarg.get('enhanced_picker_enabled', False)
+        if enhanced_picker_enabled:
+            # If the person is affiliated with the associated_object then we
+            # can display a badge.
+            badge_name = IHasAffiliation(
+                associated_object).getAffiliationBadge(person)
+            if badge_name is not None:
+                extra.image = "/@@/%s" % badge_name
+
+        if person.preferredemail is not None:
+            if person.hide_email_addresses:
+                extra.description = '<email address hidden>'
+            else:
+                try:
+                    extra.description = person.preferredemail.email
+                except Unauthorized:
+                    extra.description = '<email address hidden>'
+
+        if enhanced_picker_enabled:
+            # We will display the person's irc nick(s) after their email
+            # address in the description text.
+            irc_nicks = None
+            if person.ircnicknames:
+                irc_nicks = ", ".join(
+                    [IRCNicknameFormatterAPI(ircid).displayname()
+                    for ircid in person.ircnicknames])
+            if irc_nicks:
+                if extra.description:
+                    extra.description = ("%s (%s)" %
+                        (extra.description, irc_nicks))
+                else:
+                    extra.description = "%s" % irc_nicks
+
+        return extra
+
+
 @adapter(IBranch)
-def branch_to_pickerentry(branch):
+class BranchPickerEntryAdapter(DefaultPickerEntryAdapter):
     """Adapts IBranch to IPickerEntry."""
-    extra = default_pickerentry_adapter(branch)
-    extra.description = branch.bzr_identity
-    return extra
-
-
-@implementer(IPickerEntry)
+
+    def getPickerEntry(self, associated_object, **kwarg):
+        branch = self.context
+        extra = super(BranchPickerEntryAdapter, self).getPickerEntry(
+            associated_object)
+        extra.description = branch.bzr_identity
+        return extra
+
+
 @adapter(ISourcePackageName)
-def sourcepackagename_to_pickerentry(sourcepackagename):
+class SourcePackageNamePickerEntryAdapter(DefaultPickerEntryAdapter):
     """Adapts ISourcePackageName to IPickerEntry."""
-    extra = default_pickerentry_adapter(sourcepackagename)
-    descriptions = getSourcePackageDescriptions([sourcepackagename])
-    extra.description = descriptions.get(
-        sourcepackagename.name, "Not yet built")
-    return extra
+
+    def getPickerEntry(self, associated_object, **kwarg):
+        sourcepackagename = self.context
+        extra = super(
+            SourcePackageNamePickerEntryAdapter, self).getPickerEntry(
+                associated_object)
+        descriptions = getSourcePackageDescriptions([sourcepackagename])
+        extra.description = descriptions.get(
+            sourcepackagename.name, "Not yet built")
+        return extra
 
 
 class HugeVocabularyJSONView:
@@ -126,6 +178,8 @@
     def __init__(self, context, request):
         self.context = context
         self.request = request
+        self.enhanced_picker_enabled = bool(
+            getFeatureFlag('disclosure.picker_enhancements.enabled'))
 
     def __call__(self):
         name = self.request.form.get('name')
@@ -168,11 +222,13 @@
                 # needed for inplace editing via a REST call. The
                 # form picker doesn't need the api_url.
                 entry['api_uri'] = 'Could not find canonical url.'
-            picker_entry = IPickerEntry(term.value)
+            picker_entry = IPickerEntry(term.value).getPickerEntry(
+                self.context,
+                enhanced_picker_enabled=self.enhanced_picker_enabled)
             if picker_entry.description is not None:
                 if len(picker_entry.description) > MAX_DESCRIPTION_LENGTH:
                     entry['description'] = (
-                        picker_entry.description[:MAX_DESCRIPTION_LENGTH-3]
+                        picker_entry.description[:MAX_DESCRIPTION_LENGTH - 3]
                         + '...')
                 else:
                     entry['description'] = picker_entry.description

=== modified file 'lib/canonical/launchpad/doc/vocabulary-json.txt'
--- lib/canonical/launchpad/doc/vocabulary-json.txt	2011-04-21 10:24:47 +0000
+++ lib/canonical/launchpad/doc/vocabulary-json.txt	2011-05-25 07:18:44 +0000
@@ -42,6 +42,12 @@
     ...
     UnexpectedFormData: Unknown vocabulary 'invalid-vocabulary'
 
+To use the new disclosure functionality, we need to turn on the feature flag.
+    >>> from lp.services.features.testing import FeatureFixture
+    >>> feature_flag = {'disclosure.picker_enhancements.enabled': 'on'}
+    >>> flags = FeatureFixture(feature_flag)
+    >>> flags.setUp()
+
 A successful search.
 
     >>> form = dict(name='ValidPersonOrTeam', search_text='guadamen')
@@ -52,7 +58,7 @@
             {
                 "api_uri": "/~guadamen",
                 "css": "sprite team",
-                "title": "GuadaMen",
+                "title": "GuadaMen (~guadamen)",
                 "value": "guadamen"
             }
         ],
@@ -80,7 +86,7 @@
             {
                 "api_uri": "/~commercial-admins",
                 "css": "sprite team",
-                "title": "Commercial Subscription Admins",
+                "title": "Commercial Subscription Admins (~commercial-admins)",
                 "value": "commercial-admins"
             }
         ],
@@ -97,7 +103,7 @@
                 "api_uri": "/~name16",
                 "css": "sprite person",
                 "description": "<email address hidden>",
-                "title": "Foo Bar",
+                "title": "Foo Bar (~name16)",
                 "value": "name16"
             }
         ],
@@ -135,9 +141,31 @@
                 "api_uri": "/~name12",
                 "css": "sprite person",
                 "description": "<email address hidden>",
-                "title": "Sample Person",
+                "title": "Sample Person (~name12)",
                 "value": "name12"
             }
         ],
         "total_size": 1
     }
+
+IRC nicknames should be displayed after any email address.
+
+    >>> form = dict(name='ValidPersonOrTeam', search_text='mark',
+    ...             start='0', batch='1')
+    >>> view = create_vocabulary_view(form)
+    >>> print_json(view())
+    {
+        "entries": [
+            {
+                "api_uri": "/~mark",
+                "css": "sprite person",
+                "description": "<email address hidden> (mark on irc.freenode.net)",
+                "title": "Mark Shuttleworth (~mark)",
+                "value": "mark"
+            }
+        ],
+        "total_size": 1
+    }
+
+Clean up the feature flag.
+    >>> flags.cleanUp()

=== modified file 'lib/canonical/launchpad/zcml/launchpad.zcml'
--- lib/canonical/launchpad/zcml/launchpad.zcml	2011-04-21 02:10:29 +0000
+++ lib/canonical/launchpad/zcml/launchpad.zcml	2011-05-25 07:18:44 +0000
@@ -136,19 +136,19 @@
     />
 
   <adapter
-    factory="canonical.launchpad.browser.vocabulary.default_pickerentry_adapter"
-    />
-
-  <adapter
-    factory="canonical.launchpad.browser.vocabulary.person_to_pickerentry"
-    />
-
-  <adapter
-    factory="canonical.launchpad.browser.vocabulary.branch_to_pickerentry"
-    />
-
-  <adapter
-    factory="canonical.launchpad.browser.vocabulary.sourcepackagename_to_pickerentry"
+    factory="canonical.launchpad.browser.vocabulary.DefaultPickerEntryAdapter"
+    />
+
+  <adapter
+    factory="canonical.launchpad.browser.vocabulary.PersonPickerEntryAdapter"
+    />
+
+  <adapter
+    factory="canonical.launchpad.browser.vocabulary.BranchPickerEntryAdapter"
+    />
+
+  <adapter
+    factory="canonical.launchpad.browser.vocabulary.SourcePackageNamePickerEntryAdapter"
     />
 
 <facet facet="overview">

=== modified file 'lib/lp/registry/configure.zcml'
--- lib/lp/registry/configure.zcml	2011-05-14 15:02:13 +0000
+++ lib/lp/registry/configure.zcml	2011-05-25 07:18:44 +0000
@@ -862,6 +862,14 @@
             factory="lp.registry.browser.person.TeamBreadcrumb"
             permission="zope.Public"/>
 
+        <adapter
+            factory="lp.registry.model.pillaraffiliation.BugTaskPillarAffiliation"
+            />
+
+        <adapter
+            factory="lp.registry.model.pillaraffiliation.PillarAffiliation"
+            />
+
         <!-- Using
 
     permission="Public" here causes some tests to fail

=== added file 'lib/lp/registry/model/pillaraffiliation.py'
--- lib/lp/registry/model/pillaraffiliation.py	1970-01-01 00:00:00 +0000
+++ lib/lp/registry/model/pillaraffiliation.py	2011-05-25 07:18:44 +0000
@@ -0,0 +1,66 @@
+# Copyright 2011 Canonical Ltd.  This software is licensed under the
+# GNU Affero General Public License version 3 (see the file LICENSE).
+
+"""Adapters to figure out affiliations between people and pillars/bugs etc.
+
+When using a person in a given context, for example as a selection item in a
+picker used to choose a bug task assignee, it is important to provide an
+indication as to how that person may be affiliated with the context. Amongst
+other reasons, this provides a visual cue that the correct person is being
+selected for example.
+
+The adapters herein are provided for various contexts so that for a given
+person, the relevant affiliation details may be determined.
+
+"""
+
+__metaclass__ = type
+
+__all__ = [
+    'IHasAffiliation',
+    ]
+
+from zope.component import adapter
+from zope.interface import (
+    implements,
+    Interface,
+    )
+
+from lp.bugs.interfaces.bugtask import IBugTask
+
+
+class IHasAffiliation(Interface):
+    """The affiliation status of a person with a context."""
+
+    def getAffiliationBadge(person):
+        """Return the badge name for the type of affiliation the person has.
+
+        If the person has no affiliation with this object, return None.
+        """
+
+
+@adapter(Interface)
+class PillarAffiliation(object):
+    """Default affiliation adapter.
+
+    No affiliation is returned.
+    """
+
+    implements(IHasAffiliation)
+
+    def __init__(self, context):
+        self.context = context
+
+    def getAffiliationBadge(self, person):
+        return None
+
+
+# XXX: wallyworld 2011-05-24 bug=81692: TODO Work is required to determine
+# exactly what is required in terms of figuring out affiliation..
+
+@adapter(IBugTask)
+class BugTaskPillarAffiliation(PillarAffiliation):
+    """An affiliation adapter for bug tasks."""
+
+    def getAffiliationBadge(self, person):
+        return None

=== modified file 'lib/lp/registry/vocabularies.py'
--- lib/lp/registry/vocabularies.py	2011-05-14 15:02:13 +0000
+++ lib/lp/registry/vocabularies.py	2011-05-25 07:18:44 +0000
@@ -50,7 +50,6 @@
     'SourcePackageNameVocabulary',
     'UserTeamsParticipationPlusSelfVocabulary',
     'UserTeamsParticipationVocabulary',
-    'ValidPersonOrClosedTeamVocabulary',
     'ValidPersonOrTeamVocabulary',
     'ValidPersonVocabulary',
     'ValidTeamMemberVocabulary',
@@ -100,6 +99,9 @@
     SQLBase,
     sqlvalues,
     )
+from canonical.launchpad.components.decoratedresultset import (
+    DecoratedResultSet,
+    )
 from canonical.launchpad.database.emailaddress import EmailAddress
 from canonical.launchpad.helpers import (
     ensure_unicode,
@@ -167,7 +169,7 @@
 from lp.registry.model.karma import KarmaCategory
 from lp.registry.model.mailinglist import MailingList
 from lp.registry.model.milestone import Milestone
-from lp.registry.model.person import Person
+from lp.registry.model.person import Person, IrcID
 from lp.registry.model.pillar import PillarName
 from lp.registry.model.product import Product
 from lp.registry.model.productrelease import ProductRelease
@@ -175,6 +177,8 @@
 from lp.registry.model.projectgroup import ProjectGroup
 from lp.registry.model.sourcepackagename import SourcePackageName
 from lp.registry.model.teammembership import TeamParticipation
+from lp.services.database import bulk
+from lp.services.features import getFeatureFlag
 from lp.services.propertycache import cachedproperty
 
 
@@ -183,10 +187,19 @@
 
     _table = Person
 
+    def __init__(self, context):
+        self.enhanced_picker_enabled = bool(
+            getFeatureFlag('disclosure.picker_enhancements.enabled'))
+
     def toTerm(self, obj):
         """Return the term for this object."""
         try:
-            return SimpleTerm(obj, obj.name, obj.displayname)
+            if self.enhanced_picker_enabled:
+                # Display the person's Launchpad id next to their name.
+                title = "%s (~%s)" % (obj.displayname, obj.name)
+            else:
+                title = obj.displayname
+            return SimpleTerm(obj, obj.name, title)
         except Unauthorized:
             return None
 
@@ -580,7 +593,6 @@
                       EmailAddressStatus.PREFERRED.value,
                       self.LIMIT))
 
-
             public_result = self.store.using(*public_tables).find(
                 Person,
                 And(
@@ -644,7 +656,14 @@
         else:
             result.order_by(Person.displayname, Person.name)
         result.config(limit=self.LIMIT)
-        return result
+
+        # We will be displaying the person's irc nic(s) in the description
+        # so we need to bulk load them in one query for performance.
+        def pre_iter_hook(rows):
+            persons = set(obj for obj in rows)
+            bulk.load_referencing(IrcID, persons, ['personID'])
+
+        return DecoratedResultSet(result, pre_iter_hook=pre_iter_hook)
 
     def search(self, text):
         """Return people/teams whose fti or email address match :text:."""
@@ -1433,7 +1452,7 @@
 
     def __iter__(self):
         series = self._table.select(
-            DistroSeries.q.distributionID==Distribution.q.id,
+            DistroSeries.q.distributionID == Distribution.q.id,
             orderBy=self._orderBy, clauseTables=self._clauseTables)
         for series in sorted(series, key=attrgetter('sortkey')):
             yield self.toTerm(series)


Follow ups