Monday, June 3, 2024
 Popular · Latest · Hot · Upcoming
174
rated 0 times [  175] [ 1]  / answers: 1 / hits: 58722  / 11 Years ago, sat, december 7, 2013, 12:00:00

I was looking for both single and double-click event handling with AngularJS, since AngularJS always fires only the ng-click event even if there is ng-dblclick directive set for our element.



Here is some working code for those who seek solution:



JS:



function MyController($scope) {

var waitingForSecondClick = false;
$scope.singleClickAction = function() {

executingDoubleClick = false;
if (waitingForSecondClick) {
waitingForSecondClick = false;
executingDoubleClick = true;
return $scope.doubleClickAction();
}
waitingForSecondClick = true;

setTimeout(function() {
waitingForSecondClick = false;
return singleClickOnlyAction();
}, 250); // delay

/*
* Code executed with single AND double-click goes here.
* ...
*/

var singleClickOnlyAction = function() {
if (executingDoubleClick) return;

/*
* Code executed ONLY with single-click goes here.
* ...
*/

}

}

$scope.doubleClickAction = function() {

/*
* Code executed ONLY with double-click goes here.
* ...
*/

};

}


HTML:



<div ng-controller=MyController>
<a href=# ng-click=singleClickAction()>CLICK</a>
</div>


So my question is (since I'm an AngularJS newbie): could somebody more experianced write some nice directive for handling those both events?



In my opinion the perfect way would be to change the behaviour of ng-click, ng-dblclick and add some ng-sglclick directive for handling single-click-only code.
Don't know if it is even possible, but I'd find it very useful for everyone.



Feel free to share your opinions!


More From » angularjs

 Answers
20

Greg's answer is definitely closest to the cleanest answer. I'm going to build on his answer to come up with a version where no new code needs to be written, and no new injections need to be used in your controller.



The first thing to question is why timeouts are used to hack around these kinds of problems. Essentially, they're used to make a function skip the rest of the current event loop so that the execution is clean. In angular, however, you are actually interested in how the digest loop works. It's almost the same as your classic event handler, except for some minor differences which make it great for UX. Some tools that you have on hand to mess around with the order of function execution include scope.$eval, scope.$evalAsync,scope.$apply, and scope.$applyAsync.



I believe the $applys will kick off another digest loop, which leaves the $evals. $eval will run whatever code you include immediately with the context of the current $scope and $evalAsync will queue your function to be run at the end of the digest loop. Essentially, $evalAsync is a cleaner version of $timeout with one big difference — the first has context and exists on the scope!



This means that you can, actually, handle ng-click and ng-dblclick on the same element. Note, however, that this will still trigger the single-click function before the double-click function. This should be sufficient:



<div ng-controller=MyController>
<a href=#
ng-click=$evalAsync(singleClickAction())
ng-dblclick=doubleClickAction()>
CLICK
</a>
</div>


Here's a jsfiddle with the intended functionality using Angular 1.6.4.


[#73861] Thursday, December 5, 2013, 11 Years  [reply] [flag answer]
Only authorized users can answer the question. Please sign in first, or register a free account.
tonisandyp

Total Points: 694
Total Questions: 97
Total Answers: 77

Location: British Indian Ocean Territory
Member since Tue, Feb 22, 2022
2 Years ago
;