diff --git a/public/language/en_GB/topic.json b/public/language/en_GB/topic.json
index ab2031be4d..c26d0c05f8 100644
--- a/public/language/en_GB/topic.json
+++ b/public/language/en_GB/topic.json
@@ -32,7 +32,6 @@
"bookmark_instructions" : "Click here to return to the last unread post in this thread.",
"flag_title": "Flag this post for moderation",
- "flag_confirm": "Are you sure you want to flag this post?",
"flag_success": "This post has been flagged for moderation.",
"deleted_message": "This topic has been deleted. Only users with topic management privileges can see it.",
@@ -117,5 +116,10 @@
"most_votes": "Most votes",
"most_posts": "Most posts",
- "stale_topic_warning": "The topic you are replying to is quite old. Would you like to create a new topic instead, and reference this one in your reply?"
+ "stale_topic_warning": "The topic you are replying to is quite old. Would you like to create a new topic instead, and reference this one in your reply?",
+
+ "spam": "Spam",
+ "offensive": "Offensive",
+ "custom-flag-reason": "Enter a flagging reason"
+
}
diff --git a/public/src/client/topic/flag.js b/public/src/client/topic/flag.js
new file mode 100644
index 0000000000..3101c0cd1c
--- /dev/null
+++ b/public/src/client/topic/flag.js
@@ -0,0 +1,64 @@
+'use strict';
+
+/* globals define, app, socket, templates, translator */
+
+define('forum/topic/flag', [], function() {
+
+ var Flag = {},
+ flagModal,
+ flagCommit;
+
+ Flag.showFlagModal = function(pid) {
+ parseModal(function(html) {
+ flagModal = $(html);
+
+ flagModal.on('hidden.bs.modal', function() {
+ flagModal.remove();
+ });
+
+ flagCommit = flagModal.find('#flag-post-commit');
+
+ flagModal.on('click', '.flag-reason', function() {
+ flagPost(pid, $(this).text());
+ });
+
+ flagCommit.on('click', function() {
+ flagPost(pid, flagModal.find('#flag-reason-custom').val());
+ });
+
+ flagModal.modal('show');
+
+ flagModal.find('#flag-reason-custom').on('keyup blur change', checkFlagButtonEnable);
+ });
+ };
+
+ function parseModal(callback) {
+ templates.parse('partials/modals/flag_post_modal', {}, function(html) {
+ translator.translate(html, callback);
+ });
+ }
+
+ function flagPost(pid, reason) {
+ if (!pid || !reason) {
+ return;
+ }
+ socket.emit('posts.flag', {pid: pid, reason: reason}, function(err) {
+ if (err) {
+ return app.alertError(err.message);
+ }
+
+ flagModal.modal('hide');
+ app.alertSuccess('[[topic:flag_success]]');
+ });
+ }
+
+ function checkFlagButtonEnable() {
+ if (flagModal.find('#flag-reason-custom').val()) {
+ flagCommit.removeAttr('disabled');
+ } else {
+ flagCommit.attr('disabled', true);
+ }
+ }
+
+ return Flag;
+});
diff --git a/public/src/client/topic/postTools.js b/public/src/client/topic/postTools.js
index d6052798d7..4140d5e557 100644
--- a/public/src/client/topic/postTools.js
+++ b/public/src/client/topic/postTools.js
@@ -1,6 +1,6 @@
'use strict';
-/* globals define, app, ajaxify, bootbox, socket, templates, utils */
+/* globals define, app, ajaxify, bootbox, socket, templates, utils, config */
define('forum/topic/postTools', ['share', 'navigator', 'components', 'translator'], function(share, navigator, components, translator) {
@@ -110,33 +110,36 @@ define('forum/topic/postTools', ['share', 'navigator', 'components', 'translator
});
postContainer.on('click', '[component="post/flag"]', function() {
- flagPost(getData($(this), 'data-pid'));
+ var pid = getData($(this), 'data-pid');
+ require(['forum/topic/flag'], function(flag) {
+ flag.showFlagModal(pid);
+ });
});
- postContainer.on('click', '[component="post/edit"]', function(e) {
+ postContainer.on('click', '[component="post/edit"]', function() {
var btn = $(this);
$(window).trigger('action:composer.post.edit', {
pid: getData(btn, 'data-pid')
});
});
- postContainer.on('click', '[component="post/delete"]', function(e) {
+ postContainer.on('click', '[component="post/delete"]', function() {
togglePostDelete($(this), tid);
});
- postContainer.on('click', '[component="post/restore"]', function(e) {
+ postContainer.on('click', '[component="post/restore"]', function() {
togglePostDelete($(this), tid);
});
- postContainer.on('click', '[component="post/purge"]', function(e) {
+ postContainer.on('click', '[component="post/purge"]', function() {
purgePost($(this), tid);
});
- postContainer.on('click', '[component="post/move"]', function(e) {
+ postContainer.on('click', '[component="post/move"]', function() {
openMovePostModal($(this));
});
- postContainer.on('click', '[component="post/chat"]', function(e) {
+ postContainer.on('click', '[component="post/chat"]', function() {
openChat($(this));
});
}
@@ -369,22 +372,7 @@ define('forum/topic/postTools', ['share', 'navigator', 'components', 'translator
});
}
- function flagPost(pid) {
- translator.translate('[[topic:flag_confirm]]', function(message) {
- bootbox.confirm(message, function(confirm) {
- if (!confirm) {
- return;
- }
- socket.emit('posts.flag', pid, function(err) {
- if (err) {
- return app.alertError(err.message);
- }
- app.alertSuccess('[[topic:flag_success]]');
- });
- });
- });
- }
function openChat(button) {
var post = button.parents('[data-pid]');
diff --git a/src/posts/flags.js b/src/posts/flags.js
index 47959a8d88..3adb6541f2 100644
--- a/src/posts/flags.js
+++ b/src/posts/flags.js
@@ -9,7 +9,10 @@ var async = require('async'),
module.exports = function(Posts) {
- Posts.flag = function(post, uid, callback) {
+ Posts.flag = function(post, uid, reason, callback) {
+ if (!parseInt(uid, 10) || !reason) {
+ return callback();
+ }
async.parallel({
hasFlagged: async.apply(hasFlagged, post.pid, uid),
exists: async.apply(Posts.exists, post.pid)
@@ -36,6 +39,9 @@ module.exports = function(Posts) {
function(next) {
db.sortedSetAdd('pid:' + post.pid + ':flag:uids', now, uid, next);
},
+ function(next) {
+ db.sortedSetAdd('pid:' + post.pid + ':flag:uid:reason', 0, uid + ':' + reason, next);
+ },
function(next) {
if (parseInt(post.uid, 10)) {
db.sortedSetAdd('uid:' + post.uid + ':flag:pids', now, post.pid, next);
@@ -50,7 +56,7 @@ module.exports = function(Posts) {
next();
}
}
- ], function(err, results) {
+ ], function(err) {
callback(err);
});
});
@@ -80,8 +86,11 @@ module.exports = function(Posts) {
},
function(next) {
db.delete('pid:' + pid + ':flag:uids', next);
+ },
+ function(next) {
+ db.delete('pid:' + pid + ':flag:uid:reason', next);
}
- ], function(err, results) {
+ ], function(err) {
callback(err);
});
};
@@ -96,15 +105,56 @@ module.exports = function(Posts) {
};
Posts.getFlags = function(set, uid, start, stop, callback) {
- db.getSortedSetRevRange(set, start, stop, function(err, pids) {
- if (err) {
- return callback(err);
+ async.waterfall([
+ function (next) {
+ db.getSortedSetRevRange(set, start, stop, next);
+ },
+ function (pids, next) {
+ getFlaggedPostsWithReasons(pids, uid, next);
}
-
- Posts.getPostSummaryByPids(pids, uid, {stripTags: false, extraFields: ['flags']}, callback);
- });
+ ], callback);
};
+ function getFlaggedPostsWithReasons(pids, uid, callback) {
+ async.waterfall([
+ function (next) {
+ async.parallel({
+ uidsReasons: function(next) {
+ async.map(pids, function(pid, next) {
+ db.getSortedSetRange('pid:' + pid + ':flag:uid:reason', 0, -1, next);
+ }, next);
+ },
+ posts: function(next) {
+ Posts.getPostSummaryByPids(pids, uid, {stripTags: false, extraFields: ['flags']}, next);
+ }
+ }, next);
+ },
+ function (results, next) {
+ async.map(results.uidsReasons, function(uidReasons, next) {
+ async.map(uidReasons, function(uidReason, next) {
+ var uid = uidReason.split(':')[0];
+ var reason = uidReason.substr(uidReason.indexOf(':') + 1);
+ user.getUserFields(uid, ['username', 'userslug', 'picture'], function(err, userData) {
+ next(err, {user: userData, reason: reason});
+ });
+ }, next);
+ }, function(err, reasons) {
+ if (err) {
+ return callback(err);
+ }
+
+ results.posts.forEach(function(post, index) {
+ if (post) {
+ post.flagReasons = reasons[index];
+ }
+ });
+
+ next(null, results.posts);
+ });
+ }
+ ], callback);
+ }
+
Posts.getUserFlags = function(byUsername, sortBy, callerUID, start, stop, callback) {
async.waterfall([
function(next) {
@@ -117,7 +167,7 @@ module.exports = function(Posts) {
db.getSortedSetRevRange('uid:' + uid + ':flag:pids', 0, -1, next);
},
function(pids, next) {
- Posts.getPostSummaryByPids(pids, callerUID, {stripTags: false, extraFields: ['flags']}, next);
+ getFlaggedPostsWithReasons(pids, callerUID, next);
},
function(posts, next) {
if (sortBy === 'count') {
@@ -125,6 +175,7 @@ module.exports = function(Posts) {
return b.flags - a.flags;
});
}
+
next(null, posts.slice(start, stop));
}
], callback);
diff --git a/src/socket.io/posts/flag.js b/src/socket.io/posts/flag.js
index 201ec350a5..3e5230db6b 100644
--- a/src/socket.io/posts/flag.js
+++ b/src/socket.io/posts/flag.js
@@ -13,17 +13,21 @@ var meta = require('../../meta');
module.exports = function(SocketPosts) {
- SocketPosts.flag = function(socket, pid, callback) {
+ SocketPosts.flag = function(socket, data, callback) {
if (!socket.uid) {
return callback(new Error('[[error:not-logged-in]]'));
}
+ if (!data || !data.pid || !data.reason) {
+ return callback(new Error('[[error:invalid-data]]'));
+ }
+
var flaggingUser = {},
post;
async.waterfall([
function (next) {
- posts.getPostFields(pid, ['pid', 'tid', 'uid', 'content', 'deleted'], next);
+ posts.getPostFields(data.pid, ['pid', 'tid', 'uid', 'content', 'deleted'], next);
},
function (postData, next) {
if (parseInt(postData.deleted, 10) === 1) {
@@ -55,7 +59,7 @@ module.exports = function(SocketPosts) {
flaggingUser = user.userData;
flaggingUser.uid = socket.uid;
- posts.flag(post, socket.uid, next);
+ posts.flag(post, socket.uid, data.reason, next);
},
function (next) {
async.parallel({
@@ -74,8 +78,8 @@ module.exports = function(SocketPosts) {
notifications.create({
bodyShort: '[[notifications:user_flagged_post_in, ' + flaggingUser.username + ', ' + post.topic.title + ']]',
bodyLong: post.content,
- pid: pid,
- nid: 'post_flag:' + pid + ':uid:' + socket.uid,
+ pid: data.pid,
+ nid: 'post_flag:' + data.pid + ':uid:' + socket.uid,
from: socket.uid
}, function(err, notification) {
if (err || !notification) {
diff --git a/src/views/admin/manage/flags.tpl b/src/views/admin/manage/flags.tpl
index d3ea08f121..2b5ac35f2b 100644
--- a/src/views/admin/manage/flags.tpl
+++ b/src/views/admin/manage/flags.tpl
@@ -58,6 +58,11 @@
{posts.flags}
+
+
+
{../user.username}: "{../reason}"
+
+