jQuery에서 로딩 스피너를 표시하는 방법은 무엇입니까?


412

에서 프로토 타입 이 코드와 함께 "로드 ..."이미지를 표시 할 수 있습니다 :

var myAjax = new Ajax.Request( url, {method: 'get', parameters: pars, 
onLoading: showLoad, onComplete: showResponse} );

function showLoad () {
    ...
}

jQuery 에서는 다음 과 같이 서버 페이지를 요소에로드 할 수 있습니다.

$('#message').load('index.php?pg=ajaxFlashcard');

그러나 프로토 타입에서와 같이 로딩 스피너를이 명령에 어떻게 첨부합니까?

답변:


793

몇 가지 방법이 있습니다. 내가 선호하는 방법은 요소 자체의 ajaxStart / Stop 이벤트에 함수를 첨부하는 것입니다.

$('#loadingDiv')
    .hide()  // Hide it initially
    .ajaxStart(function() {
        $(this).show();
    })
    .ajaxStop(function() {
        $(this).hide();
    })
;

Ajax 시작 / 중지 기능은 Ajax 호출을 할 때마다 실행됩니다.

업데이트 : jQuery 1.8부터 문서에는 .ajaxStart/Stop첨부해야한다고 명시 되어 document있습니다. 위의 스 니펫을 다음과 같이 변환합니다.

var $loading = $('#loadingDiv').hide();
$(document)
  .ajaxStart(function () {
    $loading.show();
  })
  .ajaxStop(function () {
    $loading.hide();
  });

48
DIV에 한 번의 간단한로드를하기에는 아마도 너무 전역적일 것입니다.
몬트리올리스트

363
너무 글로벌? 얼마나 많은 다른 "기다려주십시오"메시지를 원하십니까?
nickf

23
이 방법은 불행히도 모달 로딩 창을 표시하지 않고 ajax 내용이로드되기를 기다리는 요소 근처에 표시하는 경우 div 위치 로딩을 제어 할 수 없습니다 ...
glaz666

10
ajaxStart 및 ajaxStop은 jQuery 이벤트이므로 네임 스페이스로 지정할 수 있습니다. stackoverflow.com/questions/1191485/… docs.jquery.com/Namespaced_Events
David Xia

10
jQuery> = 1.9를 사용하는 경우 ajaxStart 및 ajaxStop 이벤트를 $ (document)에 첨부하십시오. jquery.com/upgrade-guide/1.9/…
Domenic

213

jQuery의 경우 사용합니다

jQuery.ajaxSetup({
  beforeSend: function() {
     $('#loader').show();
  },
  complete: function(){
     $('#loader').hide();
  },
  success: function() {}
});

13
나를 위해 작동, HTML은 다음과 같이 있어야합니다 : <div id = 'loader'> <img src = "spinner.gif"/> </ div>
yigal

7
이것은 나를 위해 가장 깨끗했습니다. ajaxSetup 대신 $
Sax

4
그 가치가 무엇인지 모르지만 jQuery는 .ajaxSetup () 사용을 권장하지 않는다고 문서에 언급했습니다 . 링크
pec

여기에 주 beforeSend마다 호출하기 전에 호출됩니다 반면, ajaxStart유일한 경우 트리거되는 첫 번째 아약스 메소드가 호출된다. 마찬가지로 마지막 아약스 호출이 완료 ajaxStop되면 호출됩니다 . 동시 요청이 둘 이상인 경우이 답변의 스 니펫이 제대로 작동하지 않습니다.
nickf

Ajax 호출 시간이 초과되면 Mac 용 Firefox 28이 작동하지 않습니다.
osa

48

jQuery의 .ajax함수를 사용하고 해당 옵션을 사용하고 beforeSend로더 div와 같은 것을 보여줄 수있는 함수를 정의하고 성공 옵션에서는 해당 로더 div를 숨길 수 있습니다.

jQuery.ajax({
    type: "POST",
    url: 'YOU_URL_TO_WHICH_DATA_SEND',
    data:'YOUR_DATA_TO_SEND',
    beforeSend: function() {
        $("#loaderDiv").show();
    },
    success: function(data) {
        $("#loaderDiv").hide();
    }
});

Spinning Gif 이미지를 가질 수 있습니다. 다음은 귀하의 색 구성표에 따라 훌륭한 AJAX 로더 생성기 인 웹 사이트입니다. http://ajaxload.info/


16
success"당신은 항상 받게됩니다 오류가있는 경우 호출되지 않습니다,하지만 complete에 따라, 심지어 동기 요청, 콜백" jQuery를 아약스 이벤트 .
LeeGee

23

AJAX 호출 직전에 애니메이션 이미지를 DOM에 삽입하고 인라인 함수를 사용하여 제거 할 수 있습니다.

$("#myDiv").html('<img src="images/spinner.gif" alt="Wait" />');
$('#message').load('index.php?pg=ajaxFlashcard', null, function() {
  $("#myDiv").html('');
});

그러면 후속 요청에서 애니메이션이 동일한 프레임에서 시작됩니다 (필요한 경우). 이전 버전의 IE 는 애니메이션에 어려움이 있을 수 있습니다.

행운을 빕니다!


13
팁 : 디스플레이가 none으로 설정된 div에 spinner.gif를 미리로드하십시오. 그렇지 않으면 스피너가 여전히 다운로드되므로 약간의 스피너 효과가 지연 될 수 있습니다.
uriDium

좋은 팁, 다른 사람의 플러그인을 사용하는 것보다 훨씬 간단
Gordon Thompson

좋은! 하지만 친애하는 한 가지 더, 페이지가 이미로드 된 경우에도 2 초 동안 스피너를 표시하고 싶습니다.
양쪽 FM

5
@BothFM 사람들은 물건을 기다리는 것을 좋아하기 때문에 ...?
Josh Stodola

21
$('#message').load('index.php?pg=ajaxFlashcard', null, showResponse);
showLoad();

function showResponse() {
    hideLoad();
    ...
}

http://docs.jquery.com/Ajax/load#urldatacallback


거기에 타이밍 문제가있을 수 있습니까?
Tim Büthe

2
@ UltimateBrent 나는 당신이 잘못 생각합니다. showLoad ()가 콜백이 아닌 이유는 무엇입니까? JavaScript는 내용을 비동기 적으로로드합니다. 내가 올바른 경우 내용을로드하기 전에 showLoad ()가 작동합니다.
Jaseem

2
@ Jaseem 나는 비슷한 방법을 사용하며 실제로 비동기 호출에 의존합니다. 나는 아약스 호출 앞에 showLoad를 배치 할 것이지만 요점은 스피너를 보여주고 JS가 호출을 시작하게하고 AJAX가 완료 된 후 콜백에서 스피너를 죽이는 것입니다.
Nenotlep

17

사용하는 경우 다음 $.ajax()과 같은 것을 사용할 수 있습니다.

$.ajax({
        url: "destination url",
        success: sdialog,
        error: edialog,
        // shows the loader element before sending.
        beforeSend: function () { $("#imgSpinner1").show(); },
        // hides the loader after completion of request, whether successfull or failor.             
        complete: function () { $("#imgSpinner1").hide(); },             
        type: 'POST', dataType: 'json'
    });  

11

로딩 플러그인을 사용하십시오 : http://plugins.jquery.com/project/loading

$.loading.onAjax({img:'loading.gif'});

10
이 간단한 작업을위한 하나의 플러그인? 정말 좋은 해결책은 아닙니다? 누가 페이지에 100 개의 스크립트를로드하고 싶습니까?
Jaseem

2
동의했다. 더 나은 성능을 원하면 압축하고 연결하십시오.
Nathan Bubna

9
동의하지 않음 : 작업을 유지하고 다른 사람들도 그렇게하려면, 분리 및 모듈화, 재사용 코드. 사용자가 연결 상태가 좋지 않아 클라이언트가 플러그인뿐만 아니라 코드를로드 할 수없는 것은 무엇입니까?
LeeGee

@NathanBubna 링크가 죽었습니다
SpringLearner

"사용자가 연결 상태가 좋지 않아 클라이언트가 플러그인뿐만 아니라 코드를로드 할 수없는 가능성은 무엇입니까?" 내가 말하고 싶은 게 좋겠다
Andrew oxenburgh

9

변형 : 기본 페이지 왼쪽 상단에 id = "logo"아이콘이 있습니다. 그런 다음 Ajax가 작동 할 때 스피너 gif가 투명하게 겹쳐집니다.

jQuery.ajaxSetup({
  beforeSend: function() {
     $('#logo').css('background', 'url(images/ajax-loader.gif) no-repeat')
  },
  complete: function(){
     $('#logo').css('background', 'none')
  },
  success: function() {}
});

9

나는 원래 회신에 두 가지 변경으로 끝났다 .

  1. jQuery 1.8부터 ajaxStart 및 ajaxStop은에만 첨부해야 document합니다. 이로 인해 일부 아약스 요청 만 필터링하기가 더 어려워집니다. 수 ...
  2. ajaxSendajaxComplete로 전환 하면 스피너를 표시하기 전에 현재 ajax 요청을 조사 할 수 있습니다.

다음은 이러한 변경 후의 코드입니다.

$(document)
    .hide()  // hide it initially
    .ajaxSend(function(event, jqxhr, settings) {
        if (settings.url !== "ajax/request.php") return;
        $(".spinner").show();
    })
    .ajaxComplete(function(event, jqxhr, settings) {
        if (settings.url !== "ajax/request.php") return;
        $(".spinner").hide();
    })

좋은 해결 방법. ajaxComplete요청을 다시 시도 할 때 나는의 조합을 사용했습니다, 그래서 조기에 화재 것으로 보인다 ajaxSendajaxStop그것은 잘 작동합니다.
Mahn

@ SchweizerSchoggi : 그 질문에 대답하는 다른 주제가 있습니다. 검색을 사용하십시오!
Emil Stenström

8

또한이 답변에 기여하고 싶습니다. jQuery에서 비슷한 것을 찾고 있었으며 결국에는 결국 사용했습니다.

http://ajaxload.info/ 에서 로딩 스피너를 얻었습니다 . 내 솔루션은 http://christierney.com/2011/03/23/global-ajax-loading-spinners/의 간단한 답변을 기반으로합니다 . .

기본적으로 HTML 마크 업과 CSS는 다음과 같습니다.

<style>
     #ajaxSpinnerImage {
          display: none;
     }
</style>

<div id="ajaxSpinnerContainer">
     <img src="~/Content/ajax-loader.gif" id="ajaxSpinnerImage" title="working..." />
</div>

그런 다음 jQuery 코드는 다음과 같습니다.

<script>
     $(document).ready(function () {
          $(document)
          .ajaxStart(function () {
               $("#ajaxSpinnerImage").show();
          })
          .ajaxStop(function () {
               $("#ajaxSpinnerImage").hide();
          });

          var owmAPI = "http://api.openweathermap.org/data/2.5/weather?q=London,uk&APPID=YourAppID";
          $.getJSON(owmAPI)
          .done(function (data) {
               alert(data.coord.lon);
          })
          .fail(function () {
               alert('error');
          });
     });
</script>

그것은 간단합니다 :)


8

로더 이미지를 나중에 Ajax 호출을 사용하여 컨텐츠를로드 할 동일한 태그에 지정하면됩니다.

$("#message").html('<span>Loading...</span>');

$('#message').load('index.php?pg=ajaxFlashcard');

스팬 태그를 이미지 태그로 교체 할 수도 있습니다.


1
이 방법으로 UI를 관리하기는 어렵습니다. "로드"텍스트와 최종 메시지에 대해 완전히 다른 스타일이 필요할 수 있습니다. 우리는이 문제를 처리하기 위해 위에서 언급 한 콜백 메소드를 사용할 수 있습니다
Jaseem

6

아약스 이벤트의 전역 기본값을 설정하는 것 외에도 특정 요소에 대한 동작을 설정할 수 있습니다. 아마도 수업을 바꾸는 것만으로 충분할까요?

$('#myForm').ajaxSend( function() {
    $(this).addClass('loading');
});
$('#myForm').ajaxComplete( function(){
    $(this).removeClass('loading');
});

스피너로 #myForm을 숨기는 CSS 예제 :

.loading {
    display: block;
    background: url(spinner.gif) no-repeat center middle;
    width: 124px;
    height: 124px;
    margin: 0 auto;
}
/* Hide all the children of the 'loading' element */
.loading * {
    display: none;  
}

1
이것은 내 관점에서 선택된 대답이어야합니다. @LeeGee에게 감사합니다.
Nam G VU

정말 좋은 답변
Raskolnikov

4

스피너가 작동하려면 비동기 호출을 사용해야합니다 (적어도 그것은 아약스 호출이 끝날 때까지 표시되지 않은 다음 호출이 완료되고 스피너가 제거됨에 따라 신속하게 사라졌습니다).

$.ajax({
        url: requestUrl,
        data: data,
        dataType: 'JSON',
        processData: false,
        type: requestMethod,
        async: true,                         <<<<<<------ set async to true
        accepts: 'application/json',
        contentType: 'application/json',
        success: function (restResponse) {
            // something here
        },
        error: function (restResponse) {
            // something here                
        }
    });

4
$('#loading-image').html('<img src="/images/ajax-loader.gif"> Sending...');

        $.ajax({
            url:  uri,
            cache: false,
            success: function(){
                $('#loading-image').html('');           
            },

           error:   function(jqXHR, textStatus, errorThrown) {
            var text =  "Error has occured when submitting the job: "+jqXHR.status+ " Contact IT dept";
           $('#loading-image').html('<span style="color:red">'+text +'  </span>');

            }
        });

가장 짧은 방법은 투명한 배경으로 ajaxload.info 에서 다운로드 한 이미지로 이미지 태그를 삽입하는 것입니다.
Izabela Skibinska

2

jQuery UI Dialog와 함께 다음을 사용했습니다. (아마 다른 아약스 콜백과 작동합니까?)

$('<div><img src="/i/loading.gif" id="loading" /></div>').load('/ajax.html').dialog({
    height: 300,
    width: 600,
    title: 'Wait for it...'
});

는 ajax 호출이 완료 될 때 내용이 교체 될 때까지 애니메이션 로딩 gif를 포함합니다.


2

이것은 나에게 가장 좋은 방법입니다.

jQuery :

$(document).ajaxStart(function() {
  $(".loading").show();
});

$(document).ajaxStop(function() {
  $(".loading").hide();
});

커피 :

  $(document).ajaxStart ->
    $(".loading").show()

  $(document).ajaxStop ->
    $(".loading").hide()

문서 : ajaxStart , ajaxStop


원하는 경우이 답변에 언급 된 플러그인과 함께 사용할 수 있습니다 .
Matt

2

자바 스크립트

$.listen('click', '#captcha', function() {
    $('#captcha-block').html('<div id="loading" style="width: 70px; height: 40px; display: inline-block;" />');
    $.get("/captcha/new", null, function(data) {
        $('#captcha-block').html(data);
    }); 
    return false;
});

CSS

#loading { background: url(/image/loading.gif) no-repeat center; }

2

이것은 특정 목적을위한 매우 간단하고 똑똑한 플러그인입니다 : https://github.com/hekigan/is-loading


정말 좋습니다! 특히 데모를 보면 ... DOM 요소 비활성화, 태그, 전체 오버레이, 요소 오버레이 ... 필요한 모든 것을 제공합니다. 또한 풀 오버레이를 이 답변 과 쉽게 결합하여 즉시 작동하도록 할 수 있습니다 .
Matt

1

나는 이것을한다:

var preloaderdiv = '<div class="thumbs_preloader">Loading...</div>';
           $('#detail_thumbnails').html(preloaderdiv);
             $.ajax({
                        async:true,
                        url:'./Ajaxification/getRandomUser?top='+ $(sender).css('top') +'&lef='+ $(sender).css('left'),
                        success:function(data){
                            $('#detail_thumbnails').html(data);
                        }
             });

1

그 쪽이 맞는 거 같아요. 이 방법은 너무 전역 적입니다 ...

그러나 AJAX 호출이 페이지 자체에 영향을 미치지 않는 경우에 좋은 기본값입니다. (예를 들어 배경 저장). ( "global": false를 전달하여 특정 ajax 호출에 대해 항상 끌 수 있습니다.- jquery의 설명서를 참조하십시오.

AJAX 호출이 페이지의 일부를 새로 고치려는 경우 "로드 중"이미지가 새로 고친 섹션에만 적용되는 것을 좋아합니다. 어느 부분이 새로 고쳐 졌는지보고 싶습니다.

단순히 다음과 같이 쓸 수 있다면 얼마나 멋진 지 상상해보십시오.

$("#component_to_refresh").ajax( { ... } ); 

그리고이 섹션에 "로드 중"이 표시됩니다. 아래는 필자가 작성한 "로드"디스플레이를 처리하는 함수이지만 ajax에서 새로 고치는 영역에 따라 다릅니다.

먼저 사용법을 보여 드리겠습니다

<!-- assume you have this HTML and you would like to refresh 
      it / load the content with ajax -->

<span id="email" name="name" class="ajax-loading">
</span>

<!-- then you have the following javascript --> 

$(document).ready(function(){
     $("#email").ajax({'url':"/my/url", load:true, global:false});
 })

그리고 이것은 당신이 원하는대로 향상시킬 수있는 기본 시작 기능입니다. 매우 유연합니다.

jQuery.fn.ajax = function(options)
{
    var $this = $(this);
    debugger;
    function invokeFunc(func, arguments)
    {
        if ( typeof(func) == "function")
        {
            func( arguments ) ;
        }
    }

    function _think( obj, think )
    {
        if ( think )
        {
            obj.html('<div class="loading" style="background: url(/public/images/loading_1.gif) no-repeat; display:inline-block; width:70px; height:30px; padding-left:25px;"> Loading ... </div>');
        }
        else
        {
            obj.find(".loading").hide();
        }
    }

    function makeMeThink( think )
    {
        if ( $this.is(".ajax-loading") )
        {
            _think($this,think);
        }
        else
        {
            _think($this, think);
        }
    }

    options = $.extend({}, options); // make options not null - ridiculous, but still.
    // read more about ajax events
    var newoptions = $.extend({
        beforeSend: function()
        {
            invokeFunc(options.beforeSend, null);
            makeMeThink(true);
        },

        complete: function()
        {
            invokeFunc(options.complete);
            makeMeThink(false);
        },
        success:function(result)
        {
            invokeFunc(options.success);
            if ( options.load )
            {
                $this.html(result);
            }
        }

    }, options);

    $.ajax(newoptions);
};


1

서버 요청을 할 때마다 로더를 사용하려는 경우 다음 패턴을 사용할 수 있습니다.

 jTarget.ajaxloader(); // (re)start the loader
 $.post('/libs/jajaxloader/demo/service/service.php', function (content) {
     jTarget.append(content); // or do something with the content
 })
 .always(function () {
     jTarget.ajaxloader("stop");
 });

이 코드는 특히 jajaxloader 플러그인 (방금 만든)을 사용합니다.

https://github.com/lingtalfi/JAjaxLoader/


1

내 아약스 코드는 다음과 같습니다. 실제로 비동기 : 주석 라인을 주석 처리했으며 스피너가 나타납니다.

$.ajax({
        url: "@Url.Action("MyJsonAction", "Home")",
        type: "POST",
        dataType: "json",
        data: {parameter:variable},
        //async: false, 

        error: function () {
        },

        success: function (data) {
          if (Object.keys(data).length > 0) {
          //use data 
          }
          $('#ajaxspinner').hide();
        }
      });

아약스 코드 전에 함수 내에서 스피너를 표시하고 있습니다.

$("#MyDropDownID").change(function () {
        $('#ajaxspinner').show();

HTML의 경우 글꼴 멋진 클래스를 사용했습니다.

<i id="ajaxspinner" class="fas fa-spinner fa-spin fa-3x fa-fw" style="display:none"></i>

그것이 누군가를 돕기를 바랍니다.


0

항상 모든 것을 수행하는 UI jQuery 플러그인 차단을 사용할 수 있으며 ajax 가로 드되는 동안 모든 입력 페이지를 차단합니다. 플러그인이 작동하지 않는 것 같으면 이 답변에서 플러그인을 사용하는 올바른 방법에 대해 읽을 수 있습니다 . 확인 해봐.

당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.