Remove broken `debouncePromise` and clean up some deprecations

This commit is contained in:
Robin Ward 2014-09-23 16:16:19 -04:00
parent 6fafebc4c8
commit d1d3e5dd8c
6 changed files with 81 additions and 226 deletions

View File

@ -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;

View File

@ -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;
};
};

View File

@ -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);
});
}

View File

@ -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;
}
};

View File

@ -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 {

View File

@ -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"));
});