Как использовать jQuery-обещание/дефис в пользовательской функции?
У меня есть функция, которая получает местоположение через navigator.geolocation
:
var getLocation = function( callback ){
navigator.geolocation.getCurrentPosition( callback || function( position ){
// Stuff with geolocation
});
};
Я хотел бы сделать так, чтобы я мог связать эту функцию, используя jQuerys Deffered, но мне все еще не удалось понять концепцию и использование Deffered.
Я ищу что-то похожее на этот Pseudo Code:
getLocation().then(function(){
drawMarkerOnMap();
});
Является ли этот синтаксис даже возможным, не перевернувшись назад и не утонув в коде?
Ответы
Ответ 1
Вам нужно создать экземпляр нового отложенного объекта и вернуть его (или его обещание) из функции. Вызовите его метод .resolve
после получения ответа:
var getLocation = function() {
var deferred = new $.Deferred();
navigator.geolocation.getCurrentPosition(function( position ){
// Stuff with geolocation
deferred.resolve(position);
});
// return promise so that outside code cannot reject/resolve the deferred
return deferred.promise();
};
Использование:
getLocation().then(drawMarkerOnMap);
Ссылка: jQuery.Deferred
Добавление
Я бы посоветовал использовать оба подхода, отложенные объекты и передавать обратные вызовы функции, чтобы упростить интерфейс. Но если вам нужно оставаться обратно совместимым, вы можете просто зарегистрировать пройденный обратный вызов на отложенном объекте:
var getLocation = function(callback) {
var deferred = new $.Deferred();
if ($.isFunction(callback)) {
deferred.then(callback);
}
navigator.geolocation.getCurrentPosition(function( position ){
// Stuff with geolocation
deferred.resolve(position);
});
// return promise so that outside code cannot reject/resolve the deferred
return deferred.promise();
};
Ответ 2
Несмотря на то, что приведенный выше пример помог мне, мне пришлось немного побольше почитать, чтобы окунуться в концепцию.
Ниже приведен пример, основанный на моем коде, который содержит комментарии, чтобы помочь мне, когда я вернусь к нему, и, надеюсь, кто-нибудь прочитает этот вопрос в Stackoverflow:
/* promise based getFilter to accommodate getting surrounding suburbs */
oSearchResult.fPromiseOfFilterSetting = function fPromiseOfFilterSetting(sId) {
var self = this;
self.oPromiseCache = self.oPromiseCache || {}; // creates a persistent cache
// across function calls
var oDeferred = $.Deferred(); // `new` keyword is optional
var oPromise = oDeferred.promise();
// leverage the cache (it ok if promise is still pending), you can key
if (self.oPromiseCache[sId] !== undefined) {
return self.oPromiseCache[sId];
}
else {
self.oPromiseCache[sId] = oPromise;
}
// do our asynchronous action below which at some point calls
// defered.resolve(...) and hence complete our promise
$.cmsRestProxy.doAjaxServiceRequest('ocms_searchProperties_Extension', {
action : 'getSurroundingSuburbs',
sSuburbIds : 'a0RO0000003BwWeMAK'
}, function(result, json) {
console.log("doAjaxServiceRequest(
'ocms_searchProperties_Extension')", json);
oDeferred.resolve(json); // `json` is our result and `.resolve(json)`
// passes the value as first argument to
// the `oPromise.done`, `oPromise.fail`
// and `oPromise.always` callback functions
})
// We can now return the promise or attach optional `oPromise.done`,
// `oPromise.fail`, and `oPromise.always` callbacks which will execute first
// in the chain.
//
// Note that `oPromise.then(doneCallback, failCallback, alwaysCallback)`
// is short form for the below
oPromise.done(function(value) { // returned by promise.resolve(...); call
console.log('will run if this Promise is resolved.', value);
})
oPromise.fail(function(value) {
console.log("will run if this Promise is rejected.", value);
});
oPromise.always(function(value) {
console.log("this will run either way.", value);
});
// return a promise instead of deferred object so that
// outside code cannot reject/resolve it
return oPromise;
}
// then to use one would do
oSearchResult.fPromiseOfFilterSetting().done(function(value) {alert(value)});
// or using $.when chaining
$.when(
oSearchResult.fPromiseOfFilterSetting()
)
.done(
function fDoneCallback(arg1, arg2, argN) {
console.debug(arguments) // `arguments` is an array of all args collected
}
);