launchpad-reviewers team mailing list archive
-
launchpad-reviewers team
-
Mailing list archive
-
Message #03994
[Merge] lp:~jcsackett/launchpad/package-pickers-navigation-jumps into lp:launchpad
j.c.sackett has proposed merging lp:~jcsackett/launchpad/package-pickers-navigation-jumps into lp:launchpad.
Requested reviews:
Launchpad code reviewers (launchpad-reviewers)
Related bugs:
Bug #698020 in Launchpad itself: ""Choose..." source package search results ◀ navigation ▶ jumps around"
https://bugs.launchpad.net/launchpad/+bug/698020
Bug #698024 in Launchpad itself: ""Choose..." source package search results have strange blue bullet icon"
https://bugs.launchpad.net/launchpad/+bug/698024
For more details, see:
https://code.launchpad.net/~jcsackett/launchpad/package-pickers-navigation-jumps/+merge/65048
Summary
=======
Package pickers have a variety of visual issues.
* Navigation links change position as you navigate through batches because the picker can resize for longer package names
* BinaryAndSourcePackageNames aren't recognized as a type of Package name, so they get the bullet point sprite css, instead of package sprite css.
This fixes those two issues.
Proposed Fix
============
* Set the picker widget to use a fixed width, so that it doesn't resize as the batch loads.
* Update the code selecting the sprite css to provide the package sprite for IBinaryAndSourcePackageName
Preimplementation
=================
Spoke with Curtis Hovey about the CSS fixes for the picker.
Implementation
==============
lib/lp/app/browser/tales.py
---------------------------
* Updated the code that provides sprite_css to provide the appropriate sprite for IBInaryAndSourcePackageNames
lib/canonical/launchpad/icing/style-3-0.css
-------------------------------------------
* Changed yui3-picker's width to 40%, to closely mimic its previous 'auto' behavior but have a fixed width at a given window size.
Tests
=====
No new tests, as these were trivial cosmetic changes.
QA
==
Follow through the instructions on the linked bugs, which provide "What should happen" examples for QA.
Lint
====
make lint was not generating output for me. I will get it running and fix any lint before landing.
--
https://code.launchpad.net/~jcsackett/launchpad/package-pickers-navigation-jumps/+merge/65048
Your team Launchpad code reviewers is requested to review the proposed merge of lp:~jcsackett/launchpad/package-pickers-navigation-jumps into lp:launchpad.
=== modified file 'lib/canonical/launchpad/icing/style-3-0.css'
--- lib/canonical/launchpad/icing/style-3-0.css 2011-06-09 13:49:54 +0000
+++ lib/canonical/launchpad/icing/style-3-0.css 2011-06-17 19:12:35 +0000
@@ -1939,6 +1939,10 @@
}
/* Picker styles */
+.yui3-picker {
+ width: 40%;
+ }
+
.yui3-picker-results div.badge {
position: absolute;
top: 3px;
=== modified file 'lib/lp/answers/browser/tests/test_questiontarget.py'
--- lib/lp/answers/browser/tests/test_questiontarget.py 2011-06-15 17:07:22 +0000
+++ lib/lp/answers/browser/tests/test_questiontarget.py 2011-06-17 19:12:35 +0000
@@ -219,7 +219,7 @@
None, content.find(True, id=target_widget.show_widget_id))
text = str(content)
picker_script = (
- "vocabulary_name: 'DistributionOrProductOrProjectGroup'")
+ "Y.lp.app.picker.create('DistributionOrProductOrProjectGroup'")
self.assertIn(picker_script, text)
focus_script = "setFocusByName('field.search_text')"
self.assertIn(focus_script, text)
=== modified file 'lib/lp/app/browser/tales.py'
--- lib/lp/app/browser/tales.py 2011-06-01 13:25:21 +0000
+++ lib/lp/app/browser/tales.py 2011-06-17 19:12:35 +0000
@@ -64,6 +64,9 @@
from lp.soyuz.enums import ArchivePurpose
from lp.soyuz.interfaces.archive import IPPA
from lp.soyuz.interfaces.archivesubscriber import IArchiveSubscriberSet
+from lp.soyuz.interfaces.binarypackagename import (
+ IBinaryAndSourcePackageName,
+ )
from lp.registry.interfaces.distribution import IDistribution
from lp.registry.interfaces.distributionsourcepackage import (
IDistributionSourcePackage,
@@ -741,6 +744,8 @@
return 'sprite branch'
elif ISpecification.providedBy(context):
return 'sprite blueprint'
+ elif IBinaryAndSourcePackageName.providedBy(context):
+ return 'sprite package-source'
return None
def default_logo_resource(self, context):
=== modified file 'lib/lp/app/widgets/popup.py'
--- lib/lp/app/widgets/popup.py 2011-06-14 20:00:19 +0000
+++ lib/lp/app/widgets/popup.py 2011-06-17 19:12:35 +0000
@@ -43,28 +43,6 @@
# Defaults to self.vocabulary.displayname.
header = None
- @property
- def widget_rendered(self):
- # We manually create the widget name since we don't want different
- # popups colliding, and we have to put the attr on the request so all
- # instances of the popup on this request can check the attr.
- attr = self.__class__.__name__ + '_rendered'
- if hasattr(self.request, attr):
- return getattr(self.request, attr)
- else:
- # The widget has not rendered before this. We return False, but
- # set the attribute to true, so future checks will see that it
- # has rendered. It would be better to set this in __init__ or
- # similar, but then this first check also sees True, and we never
- # render the part of the template this is used to guard.
- setattr(self.request, attr, True)
- return False
-
- @widget_rendered.setter
- def widget_rendered(self, val):
- attr = self.__class__.__name__ + '_rendered'
- setattr(self.request, attr, val)
-
@cachedproperty
def matches(self):
"""Return a list of matches (as ITokenizedTerm) to whatever the
=== modified file 'lib/lp/app/widgets/templates/form-picker-macros.pt'
--- lib/lp/app/widgets/templates/form-picker-macros.pt 2011-06-14 19:30:29 +0000
+++ lib/lp/app/widgets/templates/form-picker-macros.pt 2011-06-17 19:12:35 +0000
@@ -29,19 +29,21 @@
</tal:search_results>
<tal:chooseLink replace="structure view/chooseLink" />
-
- <script
- tal:condition="not: view/widget_rendered"
- tal:content="structure string:
+ <script tal:content="structure string:
LPS.use('node', 'lp.app.picker', 'plugin', function(Y) {
if (Y.UA.ie) {
return;
}
- namespace = Y.namespace('lp.macros');
+
// The vocabulary picker, created when used for the first time.
- function make_picker(config) {
- var vocab_name = config.vocabulary_name;
- var picker = Y.lp.app.picker.create(vocab_name, config);
+ function make_picker() {
+ var config = {
+ header: ${view/header_text},
+ step_title: ${view/step_title_text},
+ extra_no_results_message: ${view/extra_no_results_message},
+ picker_type: '${view/picker_type}'
+ };
+ var picker = Y.lp.app.picker.create('${view/vocabulary_name}', config);
if (config.extra_no_results_message !== null) {
picker.before('resultsChange', function (e) {
var new_results = e.details[0].newVal;
@@ -55,27 +57,10 @@
});
}
picker.plug(Y.lazr.TextFieldPickerPlugin,
- {input_element: '[id=\x22' + config.input_id + '\x22]'});
+ {input_element: '[id=\x22${view/input_id}\x22]'});
return picker;
}
- namespace.make_picker = make_picker;
- });
- "/>
-
- <script tal:content="structure string:
- LPS.use('node', 'lp.app.picker', 'plugin', function(Y) {
- if (Y.UA.ie) {
- return;
- }
var picker = null;
- var config = {
- vocabulary_name: '${view/vocabulary_name}',
- header: ${view/header_text},
- step_title: ${view/step_title_text},
- extra_no_results_message: ${view/extra_no_results_message},
- picker_type: '${view/picker_type}',
- input_id: '${view/input_id}'
- };
Y.on('domready', function(e) {
// Sort out the Choose... link.
var show_widget_node = Y.one('#${view/show_widget_id}');
@@ -85,7 +70,7 @@
show_widget_node.get('parentNode').removeClass('unseen');
show_widget_node.on('click', function (e) {
if (picker === null) {
- picker = Y.lp.macros.make_picker(config);
+ picker = make_picker();
}
picker.show();
e.preventDefault();
=== modified file 'lib/lp/app/widgets/tests/test_popup.py'
--- lib/lp/app/widgets/tests/test_popup.py 2011-06-15 17:16:40 +0000
+++ lib/lp/app/widgets/tests/test_popup.py 2011-06-17 19:12:35 +0000
@@ -24,8 +24,6 @@
"test_invalid_chars+":
Choice(vocabulary='ValidTeamOwner'),
"test_valid.item":
- Choice(vocabulary='ValidTeamOwner'),
- "test_valid.second_item":
Choice(vocabulary='ValidTeamOwner')}
super(TestMetaClass, self).__init__(
name, bases=bases, attrs=attrs, __doc__=__doc__,
@@ -49,24 +47,6 @@
self.context, 'ValidTeamOwner')
self.request = LaunchpadTestRequest()
- def test_widget_rendered_one(self):
- field = ITest['test_valid.item']
- bound_field = field.bind(self.context)
- picker_widget = VocabularyPickerWidget(
- bound_field, self.vocabulary, self.request)
- self.assertFalse(picker_widget.widget_rendered)
- self.assertTrue(self.request.VocabularyPickerWidget_rendered)
-
- def test_widget_rendered_multiple(self):
- fields = (ITest['test_valid.item'], ITest['test_valid.second_item'])
- bound_fields = [field.bind(self.context) for field in fields]
- picker_widgets = [VocabularyPickerWidget(
- bound_field, self.vocabulary, self.request)
- for bound_field in bound_fields]
- self.assertFalse(picker_widgets[0].widget_rendered)
- self.assertTrue(self.request.VocabularyPickerWidget_rendered)
- self.assertTrue(picker_widgets[1].widget_rendered)
-
def test_widget_template_properties(self):
# Check the picker widget is correctly set up for a field which has a
# name containing only valid HTML ID characters.
@@ -91,8 +71,8 @@
self.assertEqual(
simplejson.dumps(None), picker_widget.extra_no_results_message)
markup = picker_widget()
- self.assertIn("vocabulary_name: 'ValidTeamOwner'", markup)
- self.assertIn("Y.lp.app.picker.create(vocab_name, config);", markup)
+ self.assertIn(
+ "Y.lp.app.picker.create('ValidTeamOwner', config);", markup)
def test_widget_fieldname_with_invalid_html_chars(self):
# Check the picker widget is correctly set up for a field which has a
=== modified file 'lib/lp/blueprints/browser/tests/test_specificationtarget.py'
--- lib/lp/blueprints/browser/tests/test_specificationtarget.py 2011-06-15 17:16:40 +0000
+++ lib/lp/blueprints/browser/tests/test_specificationtarget.py 2011-06-17 19:12:35 +0000
@@ -300,7 +300,7 @@
None, content.find(True, id=target_widget.show_widget_id))
text = str(content)
picker_script = (
- "vocabulary_name: 'DistributionOrProductOrProjectGroup'")
+ "Y.lp.app.picker.create('DistributionOrProductOrProjectGroup'")
self.assertIn(picker_script, text)
focus_script = "setFocusByName('field.search_text')"
self.assertIn(focus_script, text)
=== modified file 'lib/lp/bugs/browser/tests/test_bugs.py'
--- lib/lp/bugs/browser/tests/test_bugs.py 2011-06-15 17:16:40 +0000
+++ lib/lp/bugs/browser/tests/test_bugs.py 2011-06-17 19:12:35 +0000
@@ -87,7 +87,7 @@
None, content.find(True, id=target_widget.show_widget_id))
text = str(content)
picker_script = (
- "vocabulary_name: 'DistributionOrProductOrProjectGroup'")
+ "Y.lp.app.picker.create('DistributionOrProductOrProjectGroup'")
self.assertIn(picker_script, text)
focus_script = "setFocusByName('field.searchtext')"
self.assertIn(focus_script, text)