Skip to content Skip to sidebar Skip to footer

Angularjs Bootstrap Modal Closing Call When Clicking Outside/esc

I am using the Angular-ui/bootstrap modal in my project. Here is my modal: $scope.toggleModal = function () { $scope.theModal = $modal.open({ animation: true, t

Solution 1:

Yes you can. It causes a dismiss event and the promise is rejected in that case. Also, note that the $modal.open() method returns an object that has a result property that is a promise.

With the promise you can...

//This will run when modal dismisses itself when clicked outside or//when you explicitly dismiss the modal using .dismiss function.$scope.theModal.result.catch(function(){
    //Do stuff with respect to dismissal
});

//Runs when modal is closed without being dismissed, i.e when you close it//via $scope.theModal.close(...);$scope.theModal.result.then(function(datapassedinwhileclosing){
    //Do stuff with respect to closure
});

as a shortcut you could write:

$scope.theModal.result.then(doClosureFn, doDismissFn);

See ref

The open method returns a modal instance, an object with the following properties:

  • close(result) - a method that can be used to close a modal, passing a result
  • dismiss(reason) - a method that can be used to dismiss a modal, passing a reason
  • result - a promise that is resolved when a modal is closed and rejected when a modal is dismissed
  • opened - a promise that is resolved when a modal gets opened after downloading content's template and resolving all variables 'rendered' - a promise that is resolved when a modal is rendered.

Solution 2:

Old question, but if you want to add confirmation dialogs on various close actions, add this to your modal instance controller:

$scope.$on('modal.closing', function(event, reason, closed) {
    console.log('modal.closing: ' + (closed ? 'close' : 'dismiss') + '(' + reason + ')');
    var message = "You are about to leave the edit view. Uncaught reason. Are you sure?";
    switch (reason){
        // clicked outsidecase"backdrop click":
            message = "Any changes will be lost, are you sure?";
            break;

        // cancel buttoncase"cancel":
            message = "Any changes will be lost, are you sure?";
            break;

        // escape keycase"escape key press":
            message = "Any changes will be lost, are you sure?";
            break;
    }
    if (!confirm(message)) {
        event.preventDefault();
    }
});

I have a close button on the top right of mine, which triggers the "cancel" action. Clicking on the backdrop (if enabled), triggers the cancel action. You can use that to use different messages for various close events. Thought I'd share in case it's helpful for others.

Solution 3:

You can use the "result" promise returned by $modal.open() method. As bellow:

$scope.toggleModal = function () {
      $scope.theModal = $modal.open({
          animation: true,
          templateUrl: 'pages/templates/modal.html',
          size: "sm",
          scope: $scope
      });

      $scope.theModal.result.then(function(){
          console.log("Modal Closed!!!");
      }, function(){
          console.log("Modal Dismissed!!!");
      });
 }

Also you can use "finally" callback of "result" promise as below:

$scope.theModal.result.finally(function(){
          console.log("Modal Closed!!!");
      });

Solution 4:

In my case, when clicking off the modal, we wanted to display a prompt warning the user that doing so would discard all unsaved data in the modal form. To do this, set the following options on the modal:

varmyModal=$uibModal.open({controller:'MyModalController',controllerAs:'modal',templateUrl:'views/myModal.html',backdrop:'static',keyboard:false,scope:modalScope,bindToController:true,});

This prevents the modal from closing when clicking off:

backdrop:'static'

And this prevents the modal from closing when hitting 'esc':

keyboard:false

Then in the modal controller, add a custom "cancel" function - in my case a sweet alert pops up asking if the user wishes to close the modal:

  modal.cancel = function () {
    $timeout(function () {
      swal({
        title: 'Attention',
        text: 'Do you wish to discard this data?',
        type: 'warning',
        confirmButtonText: 'Yes',
        cancelButtonText: 'No',
        showCancelButton: true,
      }).then(function (confirm) {
        if (confirm) {
          $uibModalInstance.dismiss('cancel');
        }
      });
    })
  };

And lastly, inside the modal controller, add the following event listeners:

var myModal = document.getElementsByClassName('modal');
  var myModalDialog = document.getElementsByClassName('modal-dialog');

  $timeout(function () {
    myModal[0].addEventListener("click", function () {
      console.log('clicked')
      modal.cancel();
    })

    myModalDialog[0].addEventListener("click", function (e) {
      console.log('dialog clicked')
      e.stopPropagation();
    })
  }, 100);

"myModal" is the element you want to call the modal.cancel() callback function on. "myModalDialog" is the modal content window - we stop the event propagation for this element so it won't bubble up to "myModal".

This only works for clicking off the modal (in other words clicking the backdrop). Hitting 'esc' will not trigger this callback.

Solution 5:

Instead of ng-click="closeModal()" you can try ng-click="$dismiss()"

<button ng-click="$dismiss()">Close</button>

Post a Comment for "Angularjs Bootstrap Modal Closing Call When Clicking Outside/esc"