Как показать загрузочный счетчик в jQuery?

412

В Prototype я могу показать изображение «loading ...» с этим кодом:

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

function showLoad () {
    ...
}

В jQuery я могу загрузить страницу сервера в элемент с помощью этого:

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

но как мне добавить к этой команде загрузочный счетчик, как я это делал в Prototype?

Эдвард Тангей
источник

Ответы:

793

Есть несколько способов. Мой предпочтительный способ - присоединить функцию к событиям ajaxStart / Stop на самом элементе.

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

Функции ajaxStart / Stop будут запускаться всякий раз, когда вы делаете какие-либо вызовы Ajax.

Обновление : Начиная с jQuery 1.8, в документации указывается, что к ней .ajaxStart/Stopследует присоединяться только document. Это преобразовало бы приведенный выше фрагмент в:

var $loading = $('#loadingDiv').hide();
$(document)
  .ajaxStart(function () {
    $loading.show();
  })
  .ajaxStop(function () {
    $loading.hide();
  });
nickf
источник
48
Это, вероятно, слишком глобально для одной простой загрузки в DIV.
Монреалист
363
слишком глобальный? сколько разных сообщений "пожалуйста, подождите" вы хотите?
nickf
23
таким образом, к сожалению, вы не можете управлять позицией загрузки div в случае, если вы не хотите просто показывать окно модальной загрузки, а показывать его рядом с элементом, ожидающим загрузки содержимого ajax в ...
glaz666
10
ajaxStart и ajaxStop являются событиями jQuery, поэтому их можно именовать в пространстве: stackoverflow.com/questions/1191485/…docs.jquery.com/Namespaced_Events
Дэвид Ся,
10
Если вы используете jQuery> = 1.9, присоедините события ajaxStart и ajaxStop к $ (document). jquery.com/upgrade-guide/1.9/…
Доменик
213

Для jQuery я использую

jQuery.ajaxSetup({
  beforeSend: function() {
     $('#loader').show();
  },
  complete: function(){
     $('#loader').hide();
  },
  success: function() {}
});
kr00lix
источник
13
работает для меня, HTML должен иметь что-то вроде: <div id = 'loader'> <img src = "spinner.gif" /> </ div>
Игл
7
Этот был самым чистым для меня. Вместо ajaxSetup я поместил beforeSend: и complete: внутри оператора $ ajax ({....
kenswdev
4
Не знаю, чего это стоит, но в документации jQuery упоминает, что использование .ajaxSetup () не рекомендуется. ссылка
Печ
Обратите внимание, что он beforeSendвызывается перед каждым вызовом, а ajaxStartсрабатывает только при вызове первого метода ajax. Аналогично ajaxStop, он вызывается, когда завершается последний вызов ajax. Если у вас более одного одновременного запроса, фрагмент этого ответа не будет работать должным образом.
Nickf
У меня не работает, если время ожидания ajax истекло (Firefox 28 для Mac).
osa
48

Вы можете просто использовать .ajaxфункцию jQuery и использовать ее опцию 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 Loader Generator в соответствии с вашей цветовой схемой: http://ajaxload.info/

Искатель
источник
16
successНе будет вызываться , если есть ошибка, но «вы всегда получите completeобратный вызов, даже для синхронных запросов,» в соответствии с JQuery Ajax событий .
LeeGee
23

Вы можете вставить анимированное изображение в DOM прямо перед вызовом AJAX и выполнить встроенную функцию для его удаления ...

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

Это обеспечит, чтобы ваша анимация начиналась с того же кадра при последующих запросах (если это имеет значение). Обратите внимание, что старые версии IE могут иметь проблемы с анимацией.

Удачи!

Джош Стодола
источник
13
Подсказка: предварительно загрузите spinner.gif в div с показом none. В противном случае вы можете получить эффект замедленного вращения, так как загрузка продолжается.
UriDium
хороший совет, гораздо проще, чем использовать чей-то плагин
Гордон Томпсон
Ницца! но дорогой, еще одна вещь, которую я хочу, чтобы отобразить счетчик в течение 2 секунд, даже если страница уже загружена.
Оба FM
5
@BothFM Потому что людям нравится ждать чего-то ...?
Джош Стодола
21
$('#message').load('index.php?pg=ajaxFlashcard', null, showResponse);
showLoad();

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

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

казарка
источник
У вас могут быть проблемы со временем, не так ли?
Тим Бюте
2
@ UltimateBrent Я думаю, вы ошиблись. Почему showLoad () не является обратным вызовом? JavaScript будет загружать контент асинхронно. showLoad () будет работать даже до загрузки контента, если я прав.
Джасим
2
@Jaseem Я тоже использую подобный метод, и на самом деле он использует асинхронные вызовы. Я бы разместил showLoad где-нибудь перед вызовом ajax, но суть в том, чтобы показать счетчик, позволить JS начать вызов и убить счетчик в обратном вызове после завершения AJAX.
Ненотлеп 12.12.12
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 скриптов на страницу?
Джасим
2
Согласовано. Сожмите и объедините, если вы хотите лучшую производительность.
Натан Бубна
9
Не согласны: разделите и модульно, повторно используйте код, если вы хотите иметь возможность поддерживать свою работу, а другие тоже. Какова вероятность того, что у вашего пользователя будет такое плохое соединение, его клиент не сможет загрузить плагин так же, как и ваш код?
LeeGee
@NathanBubna Ссылка не работает
SpringLearner
«Какова вероятность того, что у вашего пользователя такое плохое соединение, его клиент не может загрузить плагин так же, как и ваш код?» Довольно хорошо, я бы сказал.
Эндрю Оксенбург
9

Вариант: у меня есть значок с id = "логотип" в левом верхнем углу главной страницы; GIF вращающегося затем накладывается сверху (с прозрачностью), когда работает AJAX.

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. Это затрудняет фильтрацию только некоторых запросов AJAX. Soo ...
  2. Переключение на ajaxSend и ajaxComplete позволяет проверить текущий запрос 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кажется, срабатывает преждевременно , когда запросы повторить, поэтому я использовал комбинацию ajaxSendи ajaxStopи он прекрасно работает.
Mahn
@SchweizerSchoggi: есть другие темы, которые отвечают на этот вопрос, используйте поиск!
Эмиль Стенстрём,
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');

Вы также можете заменить тег span на тег image.

Умеш Кумар
источник
1
Так будет сложно управлять пользовательским интерфейсом. Возможно, нам понадобятся совершенно разные стили для «загрузки» текста и окончательного сообщения. Мы можем использовать метод обратного вызова, упомянутый выше для решения этой проблемы
Jaseem
6

Помимо установки глобальных значений по умолчанию для событий AJAX, вы можете установить поведение для определенных элементов. Возможно, достаточно просто поменять свой класс?

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

Пример CSS, чтобы скрыть #myForm с помощью спиннера:

.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;  
}
LeeGee
источник
1
Это должен быть выбранный ответ на мой взгляд. Спасибо @LeeGee.
Nam G VU
Действительно отличный ответ
Раскольников
4

Обратите внимание, что вы должны использовать асинхронные вызовы для работы счетчиков (по крайней мере, это заставило мой не показывать до окончания вызова ajax, а затем быстро ушел, когда вызов завершился и удалил счетчик).

$.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 с прозрачным фоном
Изабела Скибинска
2

Я использовал следующее с JQuery UI Dialog. (Может быть, это работает с другими обратными вызовами Ajax?)

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

Содержит анимированный GIF-файл загрузки, пока его содержимое не будет заменено после завершения вызова ajax.

Куинн Комендант
источник
2

Это лучший способ для меня:

JQuery :

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

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

Кофе :

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

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

Документы: ajaxStart , ajaxStop

Фред К
источник
Точно, и вы можете использовать его вместе с плагином, упомянутым в этом ответе, если хотите ...
Мэтт
2

JavaScript

$.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

andcl
источник
Действительно мило! Особенно, если вы посмотрите на демоверсии ... Отключите элементы DOM, In Tag, Full Overlay, Element Overlay ... он предоставляет все, что вам нужно. И вы можете легко объединить Full Overlay с этим ответом, чтобы он заработал мгновенно.
Мэтт
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 не влияет на саму страницу. (фон сохранить, например). (вы всегда можете отключить его для определенного вызова ajax, передав «global»: false - см. документацию в 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-код выглядит следующим образом, по сути, я только что закомментировал строку async: false, и показывается счетчик.

$.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();
        }
      });

Я показываю счетчик внутри функции перед кодом ajax:

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

Для HTML я использовал класс font классный:

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

Надеюсь, это кому-нибудь поможет.

Jaggan_j
источник
0

Вы всегда можете использовать плагин Block UI jQuery, который сделает все за вас, и даже блокирует страницу любого ввода во время загрузки ajax. Если плагин, кажется, не работает, вы можете прочитать о том, как правильно использовать его в этом ответе. Проверьте это.

SamyCode
источник