Skip to content
This repository has been archived by the owner on Sep 5, 2024. It is now read-only.

Commit

Permalink
fix(autocomplete): tap outside options panel on iOS does not close panel
Browse files Browse the repository at this point in the history
Fixes #9581.
  • Loading branch information
Splaktar committed Jan 31, 2019
1 parent 3317180 commit 9500539
Show file tree
Hide file tree
Showing 3 changed files with 50 additions and 22 deletions.
28 changes: 26 additions & 2 deletions src/components/autocomplete/js/autocompleteController.js
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,15 @@ function MdAutocompleteCtrl ($scope, $element, $mdUtil, $mdConstant, $mdTheming,
debouncedOnResize = $mdUtil.debounce(onWindowResize),
mode = MODE_VIRTUAL; // default

/**
* The root document element. This is used for attaching a top-level click handler to
* close the options panel when a click outside said panel occurs. We use `documentElement`
* instead of body because, when scrolling is disabled, some browsers consider the body element
* to be completely off the screen and propagate events directly to the html element.
* @type {!angular.JQLite}
*/
ctrl.documentElement = angular.element(document.documentElement);

// Public Exported Variables with handlers
defineProperty('hidden', handleHiddenChange, true);

Expand Down Expand Up @@ -346,8 +355,10 @@ function MdAutocompleteCtrl ($scope, $element, $mdUtil, $mdConstant, $mdTheming,
if (elements) {
$mdUtil.disableScrollAround(elements.ul);
enableWrapScroll = disableElementScrollEvents(angular.element(elements.wrap));
ctrl.documentElement.on('click', handleClickOutside);
}
} else if (hidden && !oldHidden) {
ctrl.documentElement.off('click', handleClickOutside);
$mdUtil.enableScrolling();

if (enableWrapScroll) {
Expand All @@ -357,6 +368,15 @@ function MdAutocompleteCtrl ($scope, $element, $mdUtil, $mdConstant, $mdTheming,
}
}

/**
* Handling click events that bubble up to the document is required for closing the dropdown
* panel on click outside of the panel on iOS.
* @param {Event} $event
*/
function handleClickOutside($event) {
ctrl.hidden = true;
}

/**
* Disables scrolling for a specific element
*/
Expand Down Expand Up @@ -529,7 +549,7 @@ function MdAutocompleteCtrl ($scope, $element, $mdUtil, $mdConstant, $mdTheming,

/**
* Force blur on input element
* @param forceBlur
* @param {boolean} forceBlur
*/
function doBlur(forceBlur) {
if (forceBlur) {
Expand Down Expand Up @@ -823,8 +843,12 @@ function MdAutocompleteCtrl ($scope, $element, $mdUtil, $mdConstant, $mdTheming,

/**
* Clears the searchText value and selected item.
* @param {Event} $event
*/
function clearValue () {
function clearValue ($event) {
if ($event) {
$event.stopPropagation();
}
clearSelectedItem();
clearSearchText();
}
Expand Down
10 changes: 5 additions & 5 deletions src/components/bottomSheet/bottom-sheet.js
Original file line number Diff line number Diff line change
Expand Up @@ -244,14 +244,14 @@ function MdBottomSheetProvider($$interimElementProvider) {
// Add a backdrop that will close on click
backdrop = $mdUtil.createBackdrop(scope, "md-bottom-sheet-backdrop md-opaque");

// Prevent mouse focus on backdrop; ONLY programatic focus allowed.
// This allows clicks on backdrop to propogate to the $rootElement and
// Prevent mouse focus on backdrop; ONLY programmatic focus allowed.
// This allows clicks on backdrop to propagate to the $rootElement and
// ESC key events to be detected properly.
backdrop[0].tabIndex = -1;

if (options.clickOutsideToClose) {
backdrop.on('click', function() {
$mdUtil.nextTick($mdBottomSheet.cancel,true);
$mdUtil.nextTick($mdBottomSheet.cancel, true);
});
}

Expand All @@ -277,7 +277,7 @@ function MdBottomSheetProvider($$interimElementProvider) {
if (options.escapeToClose) {
options.rootElementKeyupCallback = function(e) {
if (e.keyCode === $mdConstant.KEY_CODE.ESCAPE) {
$mdUtil.nextTick($mdBottomSheet.cancel,true);
$mdUtil.nextTick($mdBottomSheet.cancel, true);
}
};

Expand Down Expand Up @@ -337,7 +337,7 @@ function MdBottomSheetProvider($$interimElementProvider) {
var distanceRemaining = element.prop('offsetHeight') - ev.pointer.distanceY;
var transitionDuration = Math.min(distanceRemaining / ev.pointer.velocityY * 0.75, 500);
element.css($mdConstant.CSS.TRANSITION_DURATION, transitionDuration + 'ms');
$mdUtil.nextTick($mdBottomSheet.cancel,true);
$mdUtil.nextTick($mdBottomSheet.cancel, true);
} else {
element.css($mdConstant.CSS.TRANSITION_DURATION, '');
element.css($mdConstant.CSS.TRANSFORM, '');
Expand Down
34 changes: 19 additions & 15 deletions src/core/util/util.js
Original file line number Diff line number Diff line change
Expand Up @@ -209,10 +209,10 @@ function UtilFactory($document, $timeout, $compile, $rootScope, $$mdAnimate, $in

/**
* Disables scroll around the passed parent element.
* @param element Unused
* @param {!Element|!angular.JQLite} element Origin Element (not used)
* @param {!Element|!angular.JQLite} parent Element to disable scrolling within.
* Defaults to body if none supplied.
* @param options Object of options to modify functionality
* @param {Object=} options Object of options to modify functionality
* - disableScrollMask Boolean of whether or not to create a scroll mask element or
* use the passed parent element.
*/
Expand All @@ -228,7 +228,7 @@ function UtilFactory($document, $timeout, $compile, $rootScope, $$mdAnimate, $in

var body = $document[0].body;
var restoreBody = disableBodyScroll();
var restoreElement = disableElementScroll(parent);
var restoreElement = disableElementScroll(parent, options);

return $mdUtil.disableScrollAround._restoreScroll = function() {
if (--$mdUtil.disableScrollAround._count <= 0) {
Expand All @@ -240,21 +240,29 @@ function UtilFactory($document, $timeout, $compile, $rootScope, $$mdAnimate, $in

/**
* Creates a virtual scrolling mask to prevent touchmove, keyboard, scrollbar clicking,
* and wheel events
* and wheel events.
* @param {!Element|!angular.JQLite} elementToDisable
* @param {Object=} scrollMaskOptions Object of options to modify functionality
* - disableScrollMask Boolean of whether or not to create a scroll mask element or
* use the passed parent element.
* @returns {Function}
*/
function disableElementScroll(element) {
element = angular.element(element || body);

function disableElementScroll(elementToDisable, scrollMaskOptions) {
var scrollMask;
var wrappedElementToDisable = angular.element(elementToDisable || body);

if (options.disableScrollMask) {
scrollMask = element;
if (scrollMaskOptions.disableScrollMask) {
scrollMask = wrappedElementToDisable;
} else {
scrollMask = angular.element(
'<div class="md-scroll-mask">' +
' <div class="md-scroll-mask-bar"></div>' +
'</div>');
element.append(scrollMask);
wrappedElementToDisable.append(scrollMask);
}

function preventDefault(e) {
e.preventDefault();
}

scrollMask.on('wheel', preventDefault);
Expand All @@ -264,14 +272,10 @@ function UtilFactory($document, $timeout, $compile, $rootScope, $$mdAnimate, $in
scrollMask.off('wheel');
scrollMask.off('touchmove');

if (!options.disableScrollMask && scrollMask[0].parentNode) {
if (!scrollMaskOptions.disableScrollMask && scrollMask[0].parentNode) {
scrollMask[0].parentNode.removeChild(scrollMask[0]);
}
};

function preventDefault(e) {
e.preventDefault();
}
}

// Converts the body to a position fixed block and translate it to the proper scroll position
Expand Down

0 comments on commit 9500539

Please sign in to comment.