Skip to content
This repository was archived by the owner on Oct 2, 2019. It is now read-only.

fix(uiSelectCtrl): properly calculate container width #2045

Open
wants to merge 1 commit into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
20 changes: 20 additions & 0 deletions src/common.js
Original file line number Diff line number Diff line change
Expand Up @@ -172,4 +172,24 @@ var uis = angular.module('ui.select', [])
left: boundingClientRect.left + ($window.pageXOffset || $document[0].documentElement.scrollLeft)
};
};
}])

/**
* Gets an elements inner width (width minus padding)
*/
.factory('uisElementInnerWidth',
['$window',
function ($window) {
return $window.jQuery ? getInnerWidthJQuery : getInnerWidth;

function getInnerWidthJQuery(element) {
return element.width();
}

function getInnerWidth(element) {
var style = $window.getComputedStyle(element[0]);
var paddingLeft = parseFloat(style.getPropertyValue('padding-left'));
var paddingRight = parseFloat(style.getPropertyValue('padding-right'));
return element[0].clientWidth - paddingLeft - paddingRight;
}
}]);
9 changes: 4 additions & 5 deletions src/uiSelectController.js
Original file line number Diff line number Diff line change
Expand Up @@ -5,8 +5,8 @@
* put as much logic in the controller (instead of the link functions) as possible so it can be easily tested.
*/
uis.controller('uiSelectCtrl',
['$scope', '$element', '$timeout', '$filter', '$$uisDebounce', 'uisRepeatParser', 'uiSelectMinErr', 'uiSelectConfig', '$parse', '$injector', '$window',
function($scope, $element, $timeout, $filter, $$uisDebounce, RepeatParser, uiSelectMinErr, uiSelectConfig, $parse, $injector, $window) {
['$scope', '$element', '$timeout', '$filter', '$$uisDebounce', 'uisRepeatParser', 'uiSelectMinErr', 'uiSelectConfig', '$parse', '$injector', '$window', 'uisElementInnerWidth',
function($scope, $element, $timeout, $filter, $$uisDebounce, RepeatParser, uiSelectMinErr, uiSelectConfig, $parse, $injector, $window, uisElementInnerWidth) {

var ctrl = this;

Expand Down Expand Up @@ -534,10 +534,9 @@ uis.controller('uiSelectCtrl',
ctrl.sizeSearchInput = function() {

var input = ctrl.searchInput[0],
container = ctrl.$element[0],
calculateContainerWidth = function() {
// Return the container width only if the search input is visible
return container.clientWidth * !!input.offsetParent;
return uisElementInnerWidth(ctrl.$element) * !!input.offsetParent;
},
updateIfVisible = function(containerWidth) {
if (containerWidth === 0) {
Expand All @@ -549,7 +548,7 @@ uis.controller('uiSelectCtrl',
return true;
};

ctrl.searchInput.css('width', '10px');
ctrl.searchInput.css('width', '50px');
$timeout(function() { //Give tags time to render correctly
if (sizeWatch === null && !updateIfVisible(calculateContainerWidth())) {
sizeWatch = $scope.$watch(function() {
Expand Down
56 changes: 56 additions & 0 deletions test/select.spec.js
Original file line number Diff line number Diff line change
Expand Up @@ -2060,6 +2060,62 @@ describe('ui-select tests', function () {

});

it('input size should properly account for container paddings if box-sizing is set to border-box', function () {
var el = createUiSelectMultiple({
tagging: '',
taggingLabel: 'false'
});

angular.element(document.body).append(el);
// Set fixed match item width for easier testing
var style = $(
'<style> \
* { box-sizing: border-box; } \
.ui-select-container { position: relative; } \
.ui-select-container > div { font-size: 0; } \
.ui-select-container > div > span, \
.ui-select-container > div > input { font-size: 14px; } \
.ui-select-search { border: 0; } \
.ui-select-match-item { display: inline-block; overflow: hidden; width: 260px; white-space: nowrap; } \
.ui-select-search { outline: 0; border: 0; margin: 0; padding: 0; } \
</style>'
);

$('head').append(style);

var searchInput = el.find('.ui-select-search');

el.css({
paddingLeft: '6px',
paddingRight: '6px'
});

$timeout.flush();

var fullWidth = searchInput.outerWidth();
var matchItemWidth = 260;

expect(searchInput.outerWidth()).toBe(el.width() - 6); // Full width minus padding due to nested div

clickItem(el, 'Wladimir');
$timeout.flush();
// 1 items selected, input should be less than full width minus the invisible text node and one item with
expect(searchInput.outerWidth()).toBe(fullWidth - matchItemWidth ); // remaining width of the row

clickItem(el, 'Samantha');
$timeout.flush();
// Input should be smaller than before
expect(searchInput.outerWidth()).toBe(fullWidth - (2 * matchItemWidth));

clickItem(el, 'Adrian');
$timeout.flush();
// Minimum input width is 50px, we should be on a new line now
expect(searchInput.outerWidth()).toBe(fullWidth);

el.remove();
style.remove();
});

it('should update size of search input use container width', function () {
scope.selection.selectedMultiple = [scope.people[4], scope.people[5]]; //Wladimir & Samantha
var el = createUiSelectMultiple({
Expand Down