Remove broken `debouncePromise` and clean up some deprecations
This commit is contained in:
parent
6fafebc4c8
commit
d1d3e5dd8c
|
@ -1,10 +1,14 @@
|
||||||
|
import searchForTerm from 'discourse/lib/search-for-term';
|
||||||
|
|
||||||
|
var _dontSearch = false;
|
||||||
|
|
||||||
export default Em.Controller.extend(Discourse.Presence, {
|
export default Em.Controller.extend(Discourse.Presence, {
|
||||||
|
|
||||||
contextChanged: function(){
|
contextChanged: function(){
|
||||||
if(this.get('searchContextEnabled')){
|
if (this.get('searchContextEnabled')) {
|
||||||
this._dontSearch = true;
|
_dontSearch = true;
|
||||||
this.set('searchContextEnabled', false);
|
this.set('searchContextEnabled', false);
|
||||||
this._dontSearch = false;
|
_dontSearch = false;
|
||||||
}
|
}
|
||||||
}.observes("searchContext"),
|
}.observes("searchContext"),
|
||||||
|
|
||||||
|
@ -23,7 +27,7 @@ export default Em.Controller.extend(Discourse.Presence, {
|
||||||
}.property('searchContext'),
|
}.property('searchContext'),
|
||||||
|
|
||||||
searchContextEnabledChanged: function(){
|
searchContextEnabledChanged: function(){
|
||||||
if(this._dontSearch){ return; }
|
if (_dontSearch) { return; }
|
||||||
this.newSearchNeeded();
|
this.newSearchNeeded();
|
||||||
}.observes('searchContextEnabled'),
|
}.observes('searchContextEnabled'),
|
||||||
|
|
||||||
|
@ -33,14 +37,15 @@ export default Em.Controller.extend(Discourse.Presence, {
|
||||||
var term = (this.get('term') || '').trim();
|
var term = (this.get('term') || '').trim();
|
||||||
if (term.length >= Discourse.SiteSettings.min_search_term_length) {
|
if (term.length >= Discourse.SiteSettings.min_search_term_length) {
|
||||||
this.set('loading', true);
|
this.set('loading', true);
|
||||||
this.searchTerm(term, this.get('typeFilter'));
|
|
||||||
|
Ember.run.debounce(this, 'searchTerm', term, this.get('typeFilter'), 400);
|
||||||
} else {
|
} else {
|
||||||
this.setProperties({ content: null });
|
this.setProperties({ content: null });
|
||||||
}
|
}
|
||||||
this.set('selectedIndex', 0);
|
this.set('selectedIndex', 0);
|
||||||
}.observes('term', 'typeFilter'),
|
}.observes('term', 'typeFilter'),
|
||||||
|
|
||||||
searchTerm: Discourse.debouncePromise(function(term, typeFilter) {
|
searchTerm: function(term, typeFilter) {
|
||||||
var self = this;
|
var self = this;
|
||||||
|
|
||||||
var context;
|
var context;
|
||||||
|
@ -48,7 +53,7 @@ export default Em.Controller.extend(Discourse.Presence, {
|
||||||
context = this.get('searchContext');
|
context = this.get('searchContext');
|
||||||
}
|
}
|
||||||
|
|
||||||
return Discourse.Search.forTerm(term, {
|
searchForTerm(term, {
|
||||||
typeFilter: typeFilter,
|
typeFilter: typeFilter,
|
||||||
searchContext: context
|
searchContext: context
|
||||||
}).then(function(results) {
|
}).then(function(results) {
|
||||||
|
@ -57,7 +62,7 @@ export default Em.Controller.extend(Discourse.Presence, {
|
||||||
}).catch(function() {
|
}).catch(function() {
|
||||||
self.set('loading', false);
|
self.set('loading', false);
|
||||||
});
|
});
|
||||||
}, 400),
|
},
|
||||||
|
|
||||||
showCancelFilter: function() {
|
showCancelFilter: function() {
|
||||||
if (this.get('loading')) return false;
|
if (this.get('loading')) return false;
|
||||||
|
|
|
@ -21,31 +21,3 @@ Discourse.debounce = function(func, wait) {
|
||||||
Ember.run.debounce(null, later, wait);
|
Ember.run.debounce(null, later, wait);
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
|
||||||
Debounce a javascript function that returns a promise. If it's called too soon it
|
|
||||||
will return a promise that is never resolved.
|
|
||||||
|
|
||||||
@method debouncePromise
|
|
||||||
@module Discourse
|
|
||||||
@param {function} func The function to debounce
|
|
||||||
@param {Number} wait how long to wait
|
|
||||||
**/
|
|
||||||
Discourse.debouncePromise = function(func, wait) {
|
|
||||||
var self, args, promise;
|
|
||||||
var later = function() {
|
|
||||||
func.apply(self, args).then(function (funcResult) {
|
|
||||||
promise.resolve(funcResult);
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
return function() {
|
|
||||||
self = this;
|
|
||||||
args = arguments;
|
|
||||||
promise = Ember.Deferred.create();
|
|
||||||
|
|
||||||
Ember.run.debounce(null, later, wait);
|
|
||||||
|
|
||||||
return promise;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
|
@ -0,0 +1,66 @@
|
||||||
|
export default function searchForTerm(term, opts) {
|
||||||
|
if (!opts) opts = {};
|
||||||
|
|
||||||
|
// Only include the data we have
|
||||||
|
var data = { term: term, include_blurbs: 'true' };
|
||||||
|
if (opts.typeFilter) data.type_filter = opts.typeFilter;
|
||||||
|
if (opts.searchForId) data.search_for_id = true;
|
||||||
|
|
||||||
|
if (opts.searchContext) {
|
||||||
|
data.search_context = {
|
||||||
|
type: opts.searchContext.type,
|
||||||
|
id: opts.searchContext.id
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return Discourse.ajax('/search/', { data: data }).then(function(results){
|
||||||
|
// Topics might not be included
|
||||||
|
if (!results.topics) { results.topics = []; }
|
||||||
|
if (!results.users) { results.users = []; }
|
||||||
|
if (!results.posts) { results.posts = []; }
|
||||||
|
if (!results.categories) { results.categories = []; }
|
||||||
|
|
||||||
|
var topicMap = {};
|
||||||
|
results.topics = results.topics.map(function(topic){
|
||||||
|
topic = Discourse.Topic.create(topic);
|
||||||
|
topicMap[topic.id] = topic;
|
||||||
|
return topic;
|
||||||
|
});
|
||||||
|
|
||||||
|
results.posts = results.posts.map(function(post){
|
||||||
|
post = Discourse.Post.create(post);
|
||||||
|
post.set('topic', topicMap[post.topic_id]);
|
||||||
|
return post;
|
||||||
|
});
|
||||||
|
|
||||||
|
results.users = results.users.map(function(user){
|
||||||
|
user = Discourse.User.create(user);
|
||||||
|
return user;
|
||||||
|
});
|
||||||
|
|
||||||
|
results.categories = results.categories.map(function(category){
|
||||||
|
category = Discourse.Category.list().findProperty('id', category.id);
|
||||||
|
return category;
|
||||||
|
});
|
||||||
|
|
||||||
|
var r = results.grouped_search_result;
|
||||||
|
results.resultTypes = [];
|
||||||
|
|
||||||
|
// TODO: consider refactoring front end to take a better structure
|
||||||
|
[['topic','posts'],['user','users'],['category','categories']].forEach(function(pair){
|
||||||
|
var type = pair[0], name = pair[1];
|
||||||
|
if(results[name].length > 0) {
|
||||||
|
results.resultTypes.push({
|
||||||
|
results: results[name],
|
||||||
|
displayType: (opts.searchContext && opts.searchContext.type === 'topic' && type === 'topic') ? 'post' : type,
|
||||||
|
type: type,
|
||||||
|
more: r['more_' + name]
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
var noResults = !!((results.topics.length === 0) && (results.posts.length === 0) && (results.categories.length === 0));
|
||||||
|
|
||||||
|
return noResults ? null : Em.Object.create(results);
|
||||||
|
});
|
||||||
|
}
|
|
@ -1,88 +0,0 @@
|
||||||
/**
|
|
||||||
This component helps with Searching
|
|
||||||
|
|
||||||
@class Search
|
|
||||||
@namespace Discourse
|
|
||||||
@module Discourse
|
|
||||||
**/
|
|
||||||
Discourse.Search = {
|
|
||||||
|
|
||||||
/**
|
|
||||||
Search for a term, with an optional filter.
|
|
||||||
|
|
||||||
@method forTerm
|
|
||||||
@param {String} term The term to search for
|
|
||||||
@param {Object} opts Options for searching
|
|
||||||
@param {String} opts.typeFilter Filter our results to one type only
|
|
||||||
@param {Ember.Object} opts.searchContext data to help searching within a context (say, a category or user)
|
|
||||||
@return {Promise} a promise that resolves the search results
|
|
||||||
**/
|
|
||||||
forTerm: function(term, opts) {
|
|
||||||
if (!opts) opts = {};
|
|
||||||
|
|
||||||
// Only include the data we have
|
|
||||||
var data = { term: term, include_blurbs: 'true' };
|
|
||||||
if (opts.typeFilter) data.type_filter = opts.typeFilter;
|
|
||||||
if (opts.searchForId) data.search_for_id = true;
|
|
||||||
|
|
||||||
if (opts.searchContext) {
|
|
||||||
data.search_context = {
|
|
||||||
type: opts.searchContext.type,
|
|
||||||
id: opts.searchContext.id
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
var promise = Discourse.ajax('/search', { data: data });
|
|
||||||
|
|
||||||
promise.then(function(results){
|
|
||||||
// Topics might not be included
|
|
||||||
if (!results.topics) { results.topics = []; }
|
|
||||||
|
|
||||||
var topicMap = {};
|
|
||||||
results.topics = results.topics.map(function(topic){
|
|
||||||
topic = Discourse.Topic.create(topic);
|
|
||||||
topicMap[topic.id] = topic;
|
|
||||||
return topic;
|
|
||||||
});
|
|
||||||
|
|
||||||
results.posts = results.posts.map(function(post){
|
|
||||||
post = Discourse.Post.create(post);
|
|
||||||
post.set('topic', topicMap[post.topic_id]);
|
|
||||||
return post;
|
|
||||||
});
|
|
||||||
|
|
||||||
results.users = results.users.map(function(user){
|
|
||||||
user = Discourse.User.create(user);
|
|
||||||
return user;
|
|
||||||
});
|
|
||||||
|
|
||||||
results.categories = results.categories.map(function(category){
|
|
||||||
category = Discourse.Category.list().findProperty('id', category.id);
|
|
||||||
return category;
|
|
||||||
});
|
|
||||||
|
|
||||||
var r = results.grouped_search_result;
|
|
||||||
results.resultTypes = [];
|
|
||||||
|
|
||||||
// TODO: consider refactoring front end to take a better structure
|
|
||||||
[['topic','posts'],['user','users'],['category','categories']].forEach(function(pair){
|
|
||||||
var type = pair[0], name = pair[1];
|
|
||||||
if(results[name].length > 0) {
|
|
||||||
results.resultTypes.push({
|
|
||||||
results: results[name],
|
|
||||||
displayType: (opts.searchContext && opts.searchContext.type === 'topic' && type === 'topic') ? 'post' : type,
|
|
||||||
type: type,
|
|
||||||
more: r['more_' + name]
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
var noResults = !!((results.topics.length === 0) && (results.posts.length === 0) && (results.categories.length === 0));
|
|
||||||
|
|
||||||
return noResults ? null : Em.Object.create(results);
|
|
||||||
});
|
|
||||||
|
|
||||||
return promise;
|
|
||||||
}
|
|
||||||
|
|
||||||
};
|
|
|
@ -1,11 +1,5 @@
|
||||||
/**
|
import searchForTerm from 'discourse/lib/search-for-term';
|
||||||
This view presents the user with a widget to choose a topic.
|
|
||||||
|
|
||||||
@class ChooseTopicView
|
|
||||||
@extends Discourse.View
|
|
||||||
@namespace Discourse
|
|
||||||
@module Discourse
|
|
||||||
**/
|
|
||||||
export default Discourse.View.extend({
|
export default Discourse.View.extend({
|
||||||
templateName: 'choose_topic',
|
templateName: 'choose_topic',
|
||||||
|
|
||||||
|
@ -30,7 +24,7 @@ export default Discourse.View.extend({
|
||||||
self.setProperties({ topics: null, loading: false });
|
self.setProperties({ topics: null, loading: false });
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
Discourse.Search.forTerm(title, {typeFilter: 'topic', searchForId: true}).then(function (results) {
|
searchForTerm(title, {typeFilter: 'topic', searchForId: true}).then(function (results) {
|
||||||
if (results && results.posts && results.posts.length > 0) {
|
if (results && results.posts && results.posts.length > 0) {
|
||||||
self.set('topics', results.posts.mapBy('topic'));
|
self.set('topics', results.posts.mapBy('topic'));
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -1,94 +0,0 @@
|
||||||
var clock, original, debounced, originalPromiseResolvesWith, callback;
|
|
||||||
|
|
||||||
var nothingFired = function(additionalMessage) {
|
|
||||||
ok(!original.called, "original function is not called " + additionalMessage);
|
|
||||||
ok(!callback.called, "debounced promise is not resolved " + additionalMessage);
|
|
||||||
};
|
|
||||||
|
|
||||||
var originalAndCallbackFiredOnce = function(additionalMessage) {
|
|
||||||
ok(original.calledOnce, "original function is called once " + additionalMessage);
|
|
||||||
ok(callback.calledOnce, "debounced promise is resolved once " + additionalMessage);
|
|
||||||
};
|
|
||||||
|
|
||||||
module("Discourse.debouncePromise", {
|
|
||||||
setup: function() {
|
|
||||||
clock = sinon.useFakeTimers();
|
|
||||||
|
|
||||||
originalPromiseResolvesWith = null;
|
|
||||||
original = sinon.spy(function() {
|
|
||||||
var promise = Ember.Deferred.create();
|
|
||||||
promise.resolve(originalPromiseResolvesWith);
|
|
||||||
return promise;
|
|
||||||
});
|
|
||||||
|
|
||||||
debounced = Discourse.debouncePromise(original, 100);
|
|
||||||
callback = sinon.spy();
|
|
||||||
},
|
|
||||||
|
|
||||||
teardown: function() {
|
|
||||||
clock.restore();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
test("delays execution till the end of the timeout", function() {
|
|
||||||
debounced().then(callback);
|
|
||||||
nothingFired("immediately after calling debounced function");
|
|
||||||
|
|
||||||
clock.tick(99);
|
|
||||||
nothingFired("just before the end of the timeout");
|
|
||||||
|
|
||||||
clock.tick(1);
|
|
||||||
originalAndCallbackFiredOnce("exactly at the end of the timeout");
|
|
||||||
});
|
|
||||||
|
|
||||||
test("executes only once, no matter how many times debounced function is called during the timeout", function() {
|
|
||||||
debounced().then(callback);
|
|
||||||
debounced().then(callback);
|
|
||||||
|
|
||||||
clock.tick(100);
|
|
||||||
originalAndCallbackFiredOnce("(second call was supressed)");
|
|
||||||
});
|
|
||||||
|
|
||||||
test("prolongs the timeout when the debounced function is called for the second time during the timeout", function() {
|
|
||||||
debounced().then(callback);
|
|
||||||
|
|
||||||
clock.tick(50);
|
|
||||||
debounced().then(callback);
|
|
||||||
|
|
||||||
clock.tick(50);
|
|
||||||
nothingFired("at the end of the original timeout");
|
|
||||||
|
|
||||||
clock.tick(50);
|
|
||||||
originalAndCallbackFiredOnce("exactly at the end of the prolonged timeout");
|
|
||||||
});
|
|
||||||
|
|
||||||
test("preserves last call's context and params when executing delayed function", function() {
|
|
||||||
var firstObj = {};
|
|
||||||
var secondObj = {};
|
|
||||||
|
|
||||||
debounced.call(firstObj, "first");
|
|
||||||
debounced.call(secondObj, "second");
|
|
||||||
|
|
||||||
clock.tick(100);
|
|
||||||
ok(original.calledOn(secondObj), "the context of the second of two subsequent calls is preserved");
|
|
||||||
ok(original.calledWithExactly("second"), "param passed during the second of two subsequent calls is preserved");
|
|
||||||
});
|
|
||||||
|
|
||||||
test("can be called again after timeout passes", function() {
|
|
||||||
debounced().then(callback);
|
|
||||||
|
|
||||||
clock.tick(100);
|
|
||||||
debounced().then(callback);
|
|
||||||
|
|
||||||
clock.tick(100);
|
|
||||||
ok(original.calledTwice, "original function is called for the second time");
|
|
||||||
ok(callback.calledTwice, "debounced promise is resolved for the second time");
|
|
||||||
});
|
|
||||||
|
|
||||||
test("passes resolved value from the original promise as a param to the debounced promise's callback", function() {
|
|
||||||
originalPromiseResolvesWith = "original promise return value";
|
|
||||||
debounced().then(callback);
|
|
||||||
|
|
||||||
clock.tick(100);
|
|
||||||
ok(callback.calledWith("original promise return value"));
|
|
||||||
});
|
|
Loading…
Reference in New Issue