mirror of
https://github.com/codeninjasllc/discourse.git
synced 2024-11-29 02:26:18 -05:00
555 lines
17 KiB
JavaScript
555 lines
17 KiB
JavaScript
/**
|
|
A data model for representing the composer's current state
|
|
|
|
@class Composer
|
|
@extends Discourse.Model
|
|
@namespace Discourse
|
|
@module Discourse
|
|
**/
|
|
|
|
var CLOSED, CREATE_TOPIC, DRAFT, EDIT, OPEN, PRIVATE_MESSAGE, REPLY, REPLY_AS_NEW_TOPIC_KEY, SAVING;
|
|
|
|
CLOSED = 'closed';
|
|
SAVING = 'saving';
|
|
OPEN = 'open';
|
|
DRAFT = 'draft';
|
|
|
|
// The actions the composer can take
|
|
CREATE_TOPIC = 'createTopic';
|
|
PRIVATE_MESSAGE = 'privateMessage';
|
|
REPLY = 'reply';
|
|
EDIT = 'edit';
|
|
REPLY_AS_NEW_TOPIC_KEY = "reply_as_new_topic";
|
|
|
|
Discourse.Composer = Discourse.Model.extend({
|
|
archetypesBinding: 'Discourse.site.archetypes',
|
|
|
|
init: function() {
|
|
this._super();
|
|
var val = Discourse.KeyValueStore.get('composer.showPreview') || 'true';
|
|
this.set('showPreview', val === 'true');
|
|
this.set('archetypeId', Discourse.get('site.default_archetype'));
|
|
},
|
|
|
|
creatingTopic: (function() {
|
|
return this.get('action') === CREATE_TOPIC;
|
|
}).property('action'),
|
|
|
|
creatingPrivateMessage: (function() {
|
|
return this.get('action') === PRIVATE_MESSAGE;
|
|
}).property('action'),
|
|
|
|
editingPost: (function() {
|
|
return this.get('action') === EDIT;
|
|
}).property('action'),
|
|
|
|
replyingToTopic: (function() {
|
|
return this.get('action') === REPLY;
|
|
}).property('action'),
|
|
|
|
viewOpen: (function() {
|
|
return this.get('composeState') === OPEN;
|
|
}).property('composeState'),
|
|
|
|
archetype: (function() {
|
|
return this.get('archetypes').findProperty('id', this.get('archetypeId'));
|
|
}).property('archetypeId'),
|
|
|
|
archetypeChanged: (function() {
|
|
return this.set('metaData', Em.Object.create());
|
|
}).observes('archetype'),
|
|
|
|
editTitle: (function() {
|
|
if (this.get('creatingTopic') || this.get('creatingPrivateMessage')) return true;
|
|
if (this.get('editingPost') && this.get('post.post_number') === 1) return true;
|
|
return false;
|
|
}).property('editingPost', 'creatingTopic', 'post.post_number'),
|
|
|
|
togglePreview: function() {
|
|
this.toggleProperty('showPreview');
|
|
return Discourse.KeyValueStore.set({ key: 'showPreview', value: this.get('showPreview') });
|
|
},
|
|
|
|
// Import a quote from the post
|
|
importQuote: function() {
|
|
var post = this.get('post');
|
|
|
|
// If we don't have a post, check the topic for the first one
|
|
if (!post) {
|
|
var posts = this.get('topic.posts');
|
|
if (posts && posts.length > 0) {
|
|
post = posts[0];
|
|
}
|
|
}
|
|
|
|
if (post) {
|
|
this.set('loading', true);
|
|
var composer = this;
|
|
Discourse.Post.load(post.get('id')).then(function(result) {
|
|
composer.appendText(Discourse.BBCode.buildQuoteBBCode(post, result.get('raw')));
|
|
composer.set('loading', false);
|
|
});
|
|
}
|
|
},
|
|
|
|
appendText: function(text) {
|
|
this.set('reply', (this.get('reply') || '') + text);
|
|
},
|
|
|
|
// Determine the appropriate title for this action
|
|
actionTitle: (function() {
|
|
var topic = this.get('topic');
|
|
|
|
var postLink, topicLink;
|
|
if (topic) {
|
|
var postNumber = this.get('post.post_number');
|
|
postLink = "<a href='" + (topic.get('url')) + "/" + postNumber + "'>" +
|
|
Em.String.i18n("post.post_number", { number: postNumber }) + "</a>";
|
|
topicLink = "<a href='" + (topic.get('url')) + "'> " + (Handlebars.Utils.escapeExpression(topic.get('title'))) + "</a>";
|
|
}
|
|
|
|
var postDescription,
|
|
post = this.get('post');
|
|
|
|
if (post) {
|
|
|
|
postDescription = Em.String.i18n('post.' + this.get('action'), {
|
|
link: postLink,
|
|
replyAvatar: Discourse.Utilities.tinyAvatar(post.get('username')),
|
|
username: this.get('post.username')
|
|
});
|
|
|
|
var replyUsername = post.get('reply_to_user.username');
|
|
if (replyUsername && this.get('action') === EDIT) {
|
|
postDescription += " " + Em.String.i18n("post.in_reply_to") + " " +
|
|
Discourse.Utilities.tinyAvatar(replyUsername) + " " + replyUsername;
|
|
}
|
|
|
|
}
|
|
|
|
switch (this.get('action')) {
|
|
case PRIVATE_MESSAGE:
|
|
return Em.String.i18n('topic.private_message');
|
|
case CREATE_TOPIC:
|
|
return Em.String.i18n('topic.create_long');
|
|
case REPLY:
|
|
case EDIT:
|
|
if (postDescription) return postDescription;
|
|
if (topic) return Em.String.i18n('post.reply_topic', { link: topicLink });
|
|
}
|
|
|
|
}).property('action', 'post', 'topic', 'topic.title'),
|
|
|
|
toggleText: (function() {
|
|
return this.get('showPreview') ? Em.String.i18n('composer.hide_preview') : Em.String.i18n('composer.show_preview');
|
|
}).property('showPreview'),
|
|
|
|
hidePreview: (function() {
|
|
return !this.get('showPreview');
|
|
}).property('showPreview'),
|
|
|
|
// Whether to disable the post button
|
|
cantSubmitPost: (function() {
|
|
|
|
// Can't submit while loading
|
|
if (this.get('loading')) return true;
|
|
|
|
// Title is required when:
|
|
// - creating a new topic
|
|
// - editing the 1st post
|
|
// - creating a private message
|
|
if (this.get('editTitle') && this.get('titleLength') < Discourse.SiteSettings.min_topic_title_length) return true;
|
|
|
|
// Need at least one user when sending a private message
|
|
if (this.get('creatingPrivateMessage') && (this.get('targetUsernames').trim() + ',').indexOf(',') === 0) return true;
|
|
|
|
// reply is always required
|
|
if (this.get('replyLength') < Discourse.SiteSettings.min_post_length) return true;
|
|
|
|
return false;
|
|
}).property('loading', 'editTitle', 'titleLength', 'targetUsernames', 'replyLength'),
|
|
|
|
// The text for the save button
|
|
saveText: (function() {
|
|
switch (this.get('action')) {
|
|
case EDIT:
|
|
return Em.String.i18n('composer.save_edit');
|
|
case REPLY:
|
|
return Em.String.i18n('composer.reply');
|
|
case CREATE_TOPIC:
|
|
return Em.String.i18n('composer.create_topic');
|
|
case PRIVATE_MESSAGE:
|
|
return Em.String.i18n('composer.create_pm');
|
|
}
|
|
}).property('action'),
|
|
|
|
hasMetaData: (function() {
|
|
var metaData = this.get('metaData');
|
|
if (!this.get('metaData')) {
|
|
return false;
|
|
}
|
|
return Em.empty(Em.keys(this.get('metaData')));
|
|
}).property('metaData'),
|
|
|
|
wouldLoseChanges: function() {
|
|
return this.get('reply') !== this.get('originalText');
|
|
},
|
|
|
|
/*
|
|
Open a composer
|
|
|
|
opts:
|
|
action - The action we're performing: edit, reply or createTopic
|
|
post - The post we're replying to, if present
|
|
topic - The topic we're replying to, if present
|
|
quote - If we're opening a reply from a quote, the quote we're making
|
|
*/
|
|
open: function(opts) {
|
|
var topicId;
|
|
if (!opts) opts = {};
|
|
|
|
this.set('loading', false);
|
|
if (opts.topic) {
|
|
topicId = opts.topic.get('id');
|
|
}
|
|
|
|
var replyBlank = (this.get("reply") || "") === "";
|
|
var composer = this;
|
|
if (!replyBlank &&
|
|
(opts.action !== this.get('action') || ((opts.reply || opts.action === this.EDIT) && this.get('reply') !== this.get('originalText'))) &&
|
|
!opts.tested) {
|
|
opts.tested = true;
|
|
this.cancel(function() {
|
|
return composer.open(opts);
|
|
});
|
|
return;
|
|
}
|
|
|
|
this.set('draftKey', opts.draftKey);
|
|
this.set('draftSequence', opts.draftSequence);
|
|
if (!opts.draftKey) throw 'draft key is required';
|
|
if (opts.draftSequence === null) throw 'draft sequence is required';
|
|
|
|
this.set('composeState', opts.composerState || OPEN);
|
|
this.set('action', opts.action);
|
|
this.set('topic', opts.topic);
|
|
this.set('targetUsernames', opts.usernames);
|
|
if (opts.post) {
|
|
this.set('post', opts.post);
|
|
if (!this.get('topic')) {
|
|
this.set('topic', opts.post.get('topic'));
|
|
}
|
|
}
|
|
|
|
this.set('categoryName', opts.categoryName || this.get('topic.category.name'));
|
|
this.set('archetypeId', opts.archetypeId || Discourse.get('site.default_archetype'));
|
|
this.set('metaData', opts.metaData ? Em.Object.create(opts.metaData) : null);
|
|
this.set('reply', opts.reply || this.get("reply") || "");
|
|
if (opts.postId) {
|
|
this.set('loading', true);
|
|
Discourse.Post.load(opts.postId).then(function(result) {
|
|
console.log(result);
|
|
composer.set('post', result);
|
|
composer.set('loading', false);
|
|
});
|
|
}
|
|
|
|
// If we are editing a post, load it.
|
|
if (opts.action === EDIT && opts.post) {
|
|
this.set('title', this.get('topic.title'));
|
|
this.set('loading', true);
|
|
Discourse.Post.load(opts.post.get('id')).then(function(result) {
|
|
composer.set('reply', result.get('raw'));
|
|
composer.set('originalText', composer.get('reply'));
|
|
composer.set('loading', false);
|
|
});
|
|
}
|
|
|
|
if (opts.title) {
|
|
this.set('title', opts.title);
|
|
}
|
|
|
|
if (opts.draft) {
|
|
this.set('originalText', '');
|
|
} else if (opts.reply) {
|
|
this.set('originalText', this.get('reply'));
|
|
}
|
|
return false;
|
|
},
|
|
|
|
save: function(opts) {
|
|
if (this.get('editingPost')) return this.editPost(opts);
|
|
return this.createPost(opts);
|
|
},
|
|
|
|
// When you edit a post
|
|
editPost: function(opts) {
|
|
var post = this.get('post');
|
|
var oldCooked = post.get('cooked');
|
|
var composer = this;
|
|
|
|
// Update the title if we've changed it
|
|
if (this.get('title') && post.get('post_number') === 1) {
|
|
var topic = this.get('topic');
|
|
topic.set('title', this.get('title'));
|
|
topic.set('fancy_title', this.get('title'));
|
|
topic.set('categoryName', this.get('categoryName'));
|
|
topic.save();
|
|
}
|
|
|
|
post.set('raw', this.get('reply'));
|
|
post.set('imageSizes', opts.imageSizes);
|
|
post.set('cooked', $('#wmd-preview').html());
|
|
this.set('composeState', CLOSED);
|
|
|
|
return Ember.Deferred.promise(function(promise) {
|
|
post.save(function(savedPost) {
|
|
var posts = composer.get('topic.posts');
|
|
|
|
// perhaps our post came from elsewhere eg. draft
|
|
var idx = -1;
|
|
var postNumber = post.get('post_number');
|
|
posts.each(function(p, i) {
|
|
if (p.get('post_number') === postNumber) {
|
|
idx = i;
|
|
}
|
|
});
|
|
if (idx > -1) {
|
|
savedPost.set('topic', composer.get('topic'));
|
|
posts.replace(idx, 1, [savedPost]);
|
|
promise.resolve({ post: post });
|
|
composer.set('topic.draft_sequence', savedPost.draft_sequence);
|
|
}
|
|
}, function(error) {
|
|
var errors;
|
|
errors = $.parseJSON(error.responseText).errors;
|
|
promise.reject(errors[0]);
|
|
post.set('cooked', oldCooked);
|
|
return composer.set('composeState', OPEN);
|
|
});
|
|
});
|
|
},
|
|
|
|
// Create a new Post
|
|
createPost: function(opts) {
|
|
var post = this.get('post'),
|
|
topic = this.get('topic'),
|
|
currentUser = Discourse.get('currentUser'),
|
|
addedToStream = false;
|
|
|
|
// Build the post object
|
|
var createdPost = Discourse.Post.create({
|
|
raw: this.get('reply'),
|
|
title: this.get('title'),
|
|
category: this.get('categoryName'),
|
|
topic_id: this.get('topic.id'),
|
|
reply_to_post_number: post ? post.get('post_number') : null,
|
|
imageSizes: opts.imageSizes,
|
|
post_number: this.get('topic.highest_post_number') + 1,
|
|
cooked: $('#wmd-preview').html(),
|
|
reply_count: 0,
|
|
display_username: currentUser.get('name'),
|
|
username: currentUser.get('username'),
|
|
metaData: this.get('metaData'),
|
|
archetype: this.get('archetypeId'),
|
|
post_type: Discourse.get('site.post_types.regular'),
|
|
target_usernames: this.get('targetUsernames'),
|
|
actions_summary: Em.A(),
|
|
moderator: currentUser.get('moderator'),
|
|
yours: true,
|
|
newPost: true
|
|
});
|
|
|
|
// If we're in a topic, we can append the post instantly.
|
|
if (topic) {
|
|
|
|
// Increase the reply count
|
|
if (post) {
|
|
post.set('reply_count', (post.get('reply_count') || 0) + 1);
|
|
}
|
|
topic.set('posts_count', topic.get('posts_count') + 1);
|
|
|
|
// Update last post
|
|
topic.set('last_posted_at', new Date());
|
|
topic.set('highest_post_number', createdPost.get('post_number'));
|
|
topic.set('last_poster', Discourse.get('currentUser'));
|
|
|
|
// Set the topic view for the new post
|
|
createdPost.set('topic', topic);
|
|
createdPost.set('created_at', new Date());
|
|
|
|
// If we're near the end of the topic, load new posts
|
|
var lastPost = topic.posts.last();
|
|
if (lastPost) {
|
|
var diff = topic.get('highest_post_number') - lastPost.get('post_number');
|
|
|
|
// If the new post is within a threshold of the end of the topic,
|
|
// add it and scroll there instead of adding the link.
|
|
|
|
if (diff < 5) {
|
|
createdPost.set('scrollToAfterInsert', createdPost.get('post_number'));
|
|
topic.pushPosts([createdPost]);
|
|
addedToStream = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Save callback
|
|
var composer = this;
|
|
return Ember.Deferred.promise(function(promise) {
|
|
createdPost.save(function(result) {
|
|
var addedPost = false,
|
|
saving = true;
|
|
createdPost.updateFromSave(result);
|
|
if (topic) {
|
|
// It's no longer a new post
|
|
createdPost.set('newPost', false);
|
|
topic.set('draft_sequence', result.draft_sequence);
|
|
} else {
|
|
// We created a new topic, let's show it.
|
|
composer.set('composeState', CLOSED);
|
|
saving = false;
|
|
}
|
|
composer.set('reply', '');
|
|
composer.set('createdPost', createdPost);
|
|
if (addedToStream) {
|
|
composer.set('composeState', CLOSED);
|
|
} else if (saving) {
|
|
composer.set('composeState', SAVING);
|
|
}
|
|
return promise.resolve({ post: result });
|
|
}, function(error) {
|
|
// If an error occurs
|
|
if (topic) {
|
|
topic.posts.removeObject(createdPost);
|
|
}
|
|
promise.reject($.parseJSON(error.responseText).errors[0]);
|
|
composer.set('composeState', OPEN);
|
|
});
|
|
});
|
|
},
|
|
|
|
saveDraft: function() {
|
|
// Do not save when drafts are disabled
|
|
if (this.get('disableDrafts')) return;
|
|
// Do not save when there is no reply
|
|
if (!this.get('reply')) return;
|
|
// Do not save when the reply's length is too small
|
|
if (this.get('replyLength') < Discourse.SiteSettings.min_post_length) return;
|
|
|
|
var data = {
|
|
reply: this.get('reply'),
|
|
action: this.get('action'),
|
|
title: this.get('title'),
|
|
categoryName: this.get('categoryName'),
|
|
postId: this.get('post.id'),
|
|
archetypeId: this.get('archetypeId'),
|
|
metaData: this.get('metaData'),
|
|
usernames: this.get('targetUsernames')
|
|
};
|
|
|
|
this.set('draftStatus', Em.String.i18n('composer.saving_draft_tip'));
|
|
|
|
var composer = this;
|
|
return Discourse.Draft.save(this.get('draftKey'), this.get('draftSequence'), data).then((function() {
|
|
composer.set('draftStatus', Em.String.i18n('composer.saved_draft_tip'));
|
|
}), (function() {
|
|
composer.set('draftStatus', Em.String.i18n('composer.drafts_offline'));
|
|
}));
|
|
},
|
|
|
|
resetDraftStatus: (function() {
|
|
// 'title' is focused
|
|
if ($('#reply-title').is(':focus')) {
|
|
var titleDiff = Discourse.SiteSettings.min_topic_title_length - this.get('titleLength');
|
|
if (titleDiff > 0) {
|
|
return this.set('draftStatus', Em.String.i18n('composer.min_length.need_more_for_title', { n: titleDiff }));
|
|
}
|
|
// 'reply' is focused
|
|
} else if ($('#wmd-input').is(':focus')) {
|
|
var replyDiff = Discourse.SiteSettings.min_post_length - this.get('replyLength');
|
|
if (replyDiff > 0) {
|
|
return this.set('draftStatus', Em.String.i18n('composer.min_length.need_more_for_reply', { n: replyDiff }));
|
|
}
|
|
}
|
|
// hide the counters if the currently focused text field is OK
|
|
this.set('draftStatus', null);
|
|
|
|
}).observes('replyLength', 'titleLength'),
|
|
|
|
/**
|
|
Computes the length of the title minus non-significant whitespaces
|
|
|
|
@property titleLength
|
|
**/
|
|
titleLength: function() {
|
|
var title = this.get('title') || "";
|
|
return title.replace(/\s+/img, " ").trim().length;
|
|
}.property('title'),
|
|
|
|
/**
|
|
Computes the length of the reply minus the quote(s) and non-significant whitespaces
|
|
|
|
@property replyLength
|
|
**/
|
|
replyLength: function() {
|
|
var reply = this.get('reply') || "";
|
|
while (Discourse.BBCode.QUOTE_REGEXP.test(reply)) { reply = reply.replace(Discourse.BBCode.QUOTE_REGEXP, ""); }
|
|
return reply.replace(/\s+/img, " ").trim().length;
|
|
}.property('reply')
|
|
|
|
});
|
|
|
|
Discourse.Composer.reopenClass({
|
|
|
|
open: function(opts) {
|
|
var composer = Discourse.Composer.create();
|
|
composer.open(opts);
|
|
return composer;
|
|
},
|
|
|
|
loadDraft: function(draftKey, draftSequence, draft, topic) {
|
|
var composer;
|
|
try {
|
|
if (draft && typeof draft === 'string') {
|
|
draft = JSON.parse(draft);
|
|
}
|
|
} catch (error) {
|
|
draft = null;
|
|
Discourse.Draft.clear(draftKey, draftSequence);
|
|
}
|
|
if (draft && ((draft.title && draft.title !== '') || (draft.reply && draft.reply !== ''))) {
|
|
composer = this.open({
|
|
draftKey: draftKey,
|
|
draftSequence: draftSequence,
|
|
topic: topic,
|
|
action: draft.action,
|
|
title: draft.title,
|
|
categoryName: draft.categoryName,
|
|
postId: draft.postId,
|
|
archetypeId: draft.archetypeId,
|
|
reply: draft.reply,
|
|
metaData: draft.metaData,
|
|
usernames: draft.usernames,
|
|
draft: true,
|
|
composerState: DRAFT
|
|
});
|
|
}
|
|
return composer;
|
|
},
|
|
|
|
// The status the compose view can have
|
|
CLOSED: CLOSED,
|
|
SAVING: SAVING,
|
|
OPEN: OPEN,
|
|
DRAFT: DRAFT,
|
|
|
|
// The actions the composer can take
|
|
CREATE_TOPIC: CREATE_TOPIC,
|
|
PRIVATE_MESSAGE: PRIVATE_MESSAGE,
|
|
REPLY: REPLY,
|
|
EDIT: EDIT,
|
|
|
|
// Draft key
|
|
REPLY_AS_NEW_TOPIC_KEY: REPLY_AS_NEW_TOPIC_KEY
|
|
});
|