/**
* @file manage policy for common thread management actions
* @author Andrew Sayers
* This file defines thread management policy in the abstract,
* but sadly needs to be tightly coupled to the interface.
*/
/**
* @summary manage policy for common thread management actions
* @constructor
* @example
* var policy = new ThreadManagementPolicy({
*
* // objects:
* v : v, // Variables object
* bb: bb, // BulletinBoard object
* mod_team_bb: bb, // BulletinBoard object for the moderation team
* mc: mc, // MiscellaneousCache object
* vi: vi, // Violations object
*
* // configuration:
* thread_id : 1234,
* thread_desc : 'thread description',
* loading_html : 'loading, please wait...',
* user : { username: 'thread creator username', user_id: 12345 }
* callback: function( action, summary, template, has_post, has_pm ) { ... };
*
* // widget placement:
* post_selector_args: { container: $( '.post_selector_container') },
* pm_selector_args: { container: $( '.pm_selector_container') },
* template_selector_args: { container: $('.template_selector_container') }
* message_selector_args: { container: $( '.message_selector_container') }
* });
*/
function ThreadManagementPolicy(args) {
Policy.call( this, args );
var policy = this;
var action_data = {
thread: { thread_id: args.thread_id, thread_desc: args.thread_desc, forum_id: args.forum_id },
template: 'no template'
};
var variable_suffix = [action_data.template];
this.variable_suffix = variable_suffix;
var forum_has_icons;
var unmerge_data;
var macro_defaults = { template: 'no template' };
var recent_macros = new RecentList({ ss: args.ss, name: 'recent thread management macros', });
var thread_creator = this.user = {
is_target: true,
username: args.first_post.username,
user_id : args.first_post.user_id
};
this.default_keys = [
{ type: 'thread' , name: 'old thread' , value: { thread_id: args.thread_id, thread_desc: args.thread_desc } },
{ type: 'thread' , name: 'new thread' , value: { thread_id: args.thread_id, thread_desc: args.thread_desc } },
{ type: 'forum' , name: 'old forum' , value: { forum_id: args. forum_id, forum_desc: args. forum_desc } },
{ type: 'forum' , name: 'new forum' , value: { forum_id: args. forum_id, forum_desc: args. forum_desc } },
{ type: 'literal' , name: 'new prefix' , value: '' },
{ type: 'image' , name: 'new icon' , value: '' },
{ type: 'literal' , name: 'user-friendly description of actions', value: '' },
{ type: 'literal' , name: 'mod-friendly description of actions', value: '' },
{ type: 'username' , name: 'username' , value: thread_creator },
{ type: 'username' , name: 'thread creator' , value: thread_creator },
{ type: 'literal' , name: 'mod team user id', value: args.mod_team_user.user_id },
{ type: 'literal' , name: 'first post id' , value: args.first_post.post_id },
{ type: 'action data', name: 'action data' , value: action_data },
{ type: 'literal' , name: 'template' , value: 'no template' }
];
/*
* BUILD CALLBACK
*/
var original, metadata;
function callback_timeout() {
cb_timeout = null;
var actions = [];
var has_post = message_selector.find('[name="post"]').prop('checked');
var has_pm = message_selector.find('[name="pm"]' ).prop('checked');
variable_suffix.splice(1);
if ( status_widget.val() == 'merged' ) { // Merging is a completely different process to everything else
variable_suffix.push('merge');
var merge_post_id;
var edit_action = new Action( 'close thread', {
fire: function(keys) {
var destination_thread = title_widget.val();
post.thread_id = destination_thread.target_thread_id;
action_data.destination_thread = destination_thread = {
thread_id: destination_thread.target_thread_id,
thread_desc: destination_thread.target_thread_desc,
forum_id: destination_thread.target_forum_id
};
post.bb = args.mod_team_bb; // merge posts are always sent from the mod account, so they can be searched for
return args.bb.thread_edit({
thread_id : args.thread_id,
title : args.thread_desc,
notes : policy.resolve('edit notes', keys ),
close_thread: true
}).then(function(html) { // Get information from the closed thread
return args.bb.thread_posts( args.thread_id, html ).then(function (posts) {
var return_keys = {
'list of posts in merged thread':
'[TABLE="class: outer_border"]\n' +
'[TR][TH]Post #[/TH][TH]author[/TH][TH]summary[/TH][/TR]\n' +
posts.map(function(post, index) {
return (
'[TR][TD][center][post=' + post.post_id + ']#' + (index+1) + '[/post][/center][/TD][TD]' +
'[URL="' + location.origin + args.bb.url_for.user_show({ user_id: post.user_id }) + '"]' + post.username + '[/URL][/TD][TD]' +
args.bb.post_summary(post) + '[/TD][/TR]\n'
);
}).join('') +
'[/TABLE]',
'merge data': args.bb.stringify( 'merge data', {
date: new Date().getTime(),
source_thread: unmerge_data,
destination_thread: destination_thread,
posts: posts.map(function(post) { return post.post_id })
})
};
post.add_known_keys( return_keys );
pm .add_known_keys( return_keys );
return { keys: return_keys };
});
});
},
description: function() {
return [
{
type: 'change thread status',
target: { thread_id: args.thread_id, thread_desc: args.thread_desc, forum_id: args.forum_id }
}
];
},
blockers: function() {
if ( !title_widget.val().target_thread_id ) return [ 'Please select a valid merge target' ];
}
});
var log_action = new Action( 'post in merge log', {
fire: function(keys) {
return args.bb.thread_reply({ // Notify/save all posts in thread
thread_id: args.v.resolve('frequently used posts/threads', 'merge log'),
title : policy.resolve('merge title' , keys),
bbcode : policy.resolve('merge bbcode', keys)
}).then(function(post_id) {
merge_post_id = action_data.merge_log_post_id = post_id;
return { keys: { 'merge log post id': merge_post_id } };
});
},
description: function() {
var destination_thread = title_widget.val();
destination_thread = {
thread_id: destination_thread.target_thread_id,
thread_desc: destination_thread.target_thread_desc,
forum_id: destination_thread.target_forum_id
};
return [
{
type: 'post',
target: {
thread_desc: 'the merge log',
thread_id : args.v.resolve('frequently used posts/threads', 'merge log'),
post_id : merge_post_id
}
},
];
}
});
var merge_action = new Action( 'merge threads', {
fire: function(keys) {
var destination_thread = title_widget.val();
return args.bb.thread_merge({
forum_id : destination_thread.target_forum_id,
thread_ids: [ destination_thread.target_thread_id, args.thread_id ],
});
},
description: function() {
var destination_thread = title_widget.val();
destination_thread = {
thread_id: destination_thread.target_thread_id,
thread_desc: destination_thread.target_thread_desc,
forum_id: destination_thread.target_forum_id
};
return [
{
type: 'merge threads',
source_thread: { thread_id: args.thread_id, thread_desc: args.thread_desc, forum_id: args.forum_id },
thread_creator: { username: args.first_post.username, user_id : args.first_post.user_id },
destination_thread: destination_thread
}
];
}
});
actions[0] = new Action( 'merge wrapper', edit_action.then(log_action.then(merge_action)) )
policy.default_keys[7].value = ['merge'];
if ( has_pm ) {
pm .val( policy.notification_selector_args({ html: 'PM', type: 'PM' }) );
actions[0].then( policy.notification_selector_action(pm ) );
}
if ( has_post ) {
post.val( policy.notification_selector_args({ html: 'post', type: 'post' }) );
actions[0].then( policy.notification_selector_action(post) );
}
} else {
var edit_actions = [],
userfriendly_description = [],
edit_target = { thread_id: args.thread_id, thread_desc: args.thread_desc, forum_id: args.forum_id }
;
if ( status_widget.val() != original.status ) {
var status = status_widget.val();
if ( status == 'closed temporarily' ) {
action_data.deadline = status_widget.find('option:selected').data('deadline');
var thread_id = args.v.resolve('frequently used posts/threads', 'Moderation Chase-Up Thread' );
var post_id;
actions.push( new Action( 'chase-up post', {
fire: function(keys) {
return args.bb.thread_reply({
thread_id: thread_id,
title : policy.resolve( 'deadline post title' , keys ),
bbcode : policy.resolve( 'deadline post bbcode', keys ),
}).then(function(_post_id) {
action_data.chaseup_post_id = post_id = _post_id;
return { keys: {
'chase-up post id': post_id,
'extra actions' : '[post=' + post_id + ']replied to the chase-up thread[/post]'
}};
});
},
description: function() { // description
return [{
type: 'post',
target: {
thread_desc: 'the chase-up thread',
thread_id : thread_id,
post_id : post_id
}
}];
}
}));
}
edit_actions.push( { type: 'change thread status', target: edit_target } );
variable_suffix.push('change status', status);
userfriendly_description.push('{{'+policy._namespace+': change status: ' + status + '}}');
}
if ( (prefix_widget.val()||'') != original.prefix ) {
edit_actions.push( { type: 'change thread prefix', target: edit_target } );
variable_suffix.push('change prefix');
userfriendly_description.push('{{'+policy._namespace+': change prefix}}');
}
if ( forum_has_icons && icon_widget.getSelectedIndex() != original.icon ) {
edit_actions.push( { type: 'change thread icon', target: edit_target } );
variable_suffix.push('change icon');
userfriendly_description.push('{{'+policy._namespace+': change icon}}');
}
if ( forum_widget.val() != original.forum_id ) {
variable_suffix.push('change forum');
userfriendly_description.push('{{'+policy._namespace+': change forum}}');
if ( title_widget.val().thread_desc != original.title.thread_desc ) {
variable_suffix.push('change title');
userfriendly_description.push('{{'+policy._namespace+': change title}}');
}
actions.push( new Action(
'move thread',
{
fire: function(keys) {
var option = forum_widget.find(':selected');
action_data.destination_forum_id = option.val();
return args.bb.thread_move({
thread_id : args.thread_id,
title: title_widget.val().thread_desc,
redirect_title: original.title.thread_desc,
forum_id : option.val(),
redirect : parse_duration( policy.resolve( 'redirect deadline', keys ) )
});
},
description: function() {
var option = forum_widget.find(':selected');
var change_forum = {
type: 'change thread forum',
target: $.extend({
forum_id: option.val(),
forum_desc: option.text()
}, edit_target )
};
if ( title_widget.val().thread_desc == original.title.thread_desc )
return [change_forum];
else
return [
change_forum,
{ type: 'change thread title', target: edit_target }
];
},
blockers: function() {
if ( !forum_widget.val() ) return [ 'Please select a target forum' ];
}
}
));
} else if ( title_widget.val().thread_desc != original.title.thread_desc ) {
// we can retitle a thread at the same time as moving it
edit_actions.push( { type: 'change thread title', target: edit_target } );
variable_suffix.push('change title');
userfriendly_description.push('{{'+policy._namespace+': change title}}');
}
policy.default_keys[6].value = userfriendly_description;
policy.default_keys[7].value = userfriendly_description
.map(function(d) { return d.replace(/{{'+policy._namespace+': (.*)}}/, '$1') });
if ( edit_actions.length )
actions.push( new Action(
'edit thread',
{
fire: function(keys) {
var edit_info = action_data.edit_info = {
thread_id : args.thread_id,
title : title_widget.val().thread_desc,
notes : policy.resolve('edit notes', keys ),
icon_id : forum_has_icons ? icon_widget.getSelectedValue() : undefined,
prefix_id : prefix_widget.val(),
close_thread : status_widget.val() == 'closed' || status_widget.val() == 'closed temporarily',
unapprove_thread: status_widget.val() == 'moderated',
delete_thread : status_widget.val() == 'deleted',
delete_reason : policy.resolve('delete reason', keys ),
};
return args.bb.thread_edit(edit_info);
},
description: function() {
return edit_actions;
},
}
));
// need to do this before building the messages...
var bump_value = bump_selector.find(':checked').val();
//if ( bump_value ) variable_suffix.push(bump_value); // informing users about this would only cause problems
if ( has_pm ) {
pm .val( policy.notification_selector_args({ html: 'PM', type: 'PM' }) );
actions.push( policy.notification_selector_action(pm ) );
}
if ( has_post ) {
post.val( policy.notification_selector_args({ html: 'post', type: 'post' }) );
actions.push( policy.notification_selector_action(post) );
}
// ... but de-bumping must happen *after* any post action bumps the thread
switch ( bump_value ) {
case 'bump':
if ( !has_post ) {
// bumping is redundant if you reply to a thread
actions.push( new Action(
'bump thread',
{
fire: function(keys) { return args.bb.thread_bump(args.thread_id) },
description: function() {
return [{ type: 'bump thread', target: edit_target }];
}
}
));
}
break;
case 'debump':
var debump_action = new Action(
'debump thread',
{
fire: function(keys) { return args.bb.thread_debump(args.thread_id) },
description: function() {
return [{ type: 'debump thread', target: edit_target }];
}
}
);
if ( has_post )
actions[actions.length-1].then( debump_action );
else
actions.push(debump_action);
break;
case '': // nothing to do
}
}
if ( has_pm ) policy.default_keys[7].value.push('PM');
if ( has_post ) policy.default_keys[7].value.push('reply');
policy.default_keys[7].value = policy.default_keys[7].value.join('; ');
if ( macro_defaults.template && policy.check( [ 'macro' ].concat(variable_suffix) ) ) {
var macro_values = { template: macro_defaults.template };
switch ( status_widget.val() ) {
case 'closed temporarily':
var deadline = status_widget.find('option:selected').data('deadline');
if ( (macro_defaults.deadline||'') != deadline ) {
macro_values.status = 'closed temporarily';
macro_values.deadline = deadline;
}
break;
case 'merged':
var target_thread_id = title_widget.val().target_thread_id;
if ( (macro_defaults.target_thread_id||NaN) != target_thread_id ) {
macro_values.status = 'merged';
macro_values.target_thread_id = target_thread_id;
}
case macro_defaults.status:
break;
default:
macro_values.status = status_widget.val();
}
if ( macro_defaults. forum_id != forum_widget.val() ) macro_values.forum_id = parseInt( forum_widget.val(), 10 );
if ( macro_defaults.prefix_id != (prefix_widget.val()||'') ) macro_values.prefix_id = prefix_widget.val()||'';
if ( macro_defaults. icon != icon_widget.getSelectedIndex() && forum_has_icons )
macro_values.icon = icon_widget.getSelectedIndex();
if ( macro_defaults.has_pm != has_pm ) macro_values.has_pm = has_pm ;
if ( macro_defaults.has_post != has_post ) macro_values.has_post = has_post;
if ( Object.keys(macro_values).length > 1 ) {
actions.push(new Action( 'register macro', {
fire: function(keys) {
macro_values.name = policy.resolve( [ 'macro' ].concat(variable_suffix), keys );
return recent_macros.push(macro_values);
}
}));
}
}
if ( actions.length ) {
args.callback(
new Action(
'root action',
// this has to be split out from the notification actions because they're performed with the mod team account:
new Action( policy._namespace + ' wrapper', actions ).then(new Action( 'add usernote', {
fire: function(keys) {
var return_keys = {};
if ( !keys.hasOwnProperty( 'pm result') ) return_keys[ 'pm result'] = 'no action';
if ( !keys.hasOwnProperty('post result') ) return_keys['post result'] = 'no action';
$.extend( keys, return_keys );
return args.bb.usernote_add(
thread_creator.user_id,
policy.resolve( 'combined note title' , keys ),
policy.resolve( 'combined note bbcode', keys )
).then(function() {
if ( keys['notification error' ] == 'fail' ) {
return $.Deferred()
.reject('Notification failed, but the error was postponed so we could add a usernote:\n' + error)
.promise();
} else {
return { keys: return_keys };
}
});
},
description: function() {
return [{ type: 'usernote', target: thread_creator }];
}
}))
),
'manage [thread=' + args.thread_id + ']' + args.thread_desc + '[/thread]',
has_post, has_pm,
status_widget.val() == 'merged' && sender_selector.find('input:checked').val() == 'personal'
);
} else {
args.callback( null, '', has_post, has_pm );
}
}
var cb_timeout, callback = (
args.callback
? function() {
if ( cb_timeout ) clearTimeout(cb_timeout);
cb_timeout = setTimeout(callback_timeout, 0 );
}
: function() {}
);
/*
* INITIALISE WIDGETS
*/
var was_merged = false;
var status_widget = $(
'<select title="set the thread status">' +
'<option title="open the thread to replies from ordinary users" value="open">Open: </option>' +
this.resolve( 'close periods', {}, 'array of items' ).map(function(period) {
period = period.value.split(/\s*:\s*/, 2 );
return '<option title="close the thread and make a note to reopen it" value="closed temporarily" data-deadline="' + period[0] + '">' + period[1] + ': </options>';
}).join('') +
'<option title="permanently close the thread" value="closed" data-deadline="permanent">Closed: </option>' +
'<option title="merge this with the specified thread" value="merged">Merged with: </option>' +
'<option title="unapprove this thread" value="moderated">Moderated: </option>' +
'<option title="(soft-)delete this thread" value="deleted">Deleted: </option>' +
'</select>'
)
.appendTo(args.status_args.container)
.change(function() {
if ( this.value == 'merged' ) {
forum_widget .prop( 'disabled', true );
prefix_widget .prop( 'disabled', true );
bump_selector.find('input').prop( 'disabled', true );
title_widget.val({ mode: 'merge', target_thread_id: null, target_thread_desc: '' });
title_widget.focus();
$('#mod-friend-thread-metadata-icons').addClass('disabled');
was_merged = true;
} else {
if ( was_merged ) {
forum_widget .prop( 'disabled', false );
status_widget .prop( 'disabled', false );
bump_selector.find('input').prop( 'disabled', false );
$('#mod-friend-thread-metadata-icons').removeClass('disabled');
title_widget.val({ mode: 'edit', target_thread_id: args.thread_id });
forum_widget.val(metadata.forum_id);
if ( metadata.prefixes.length ) {
prefix_widget.empty().show().append(metadata.prefixes.map(function(option) {
return $('<option>').val(option.prefix_id).text(option.text).prop( 'checked', option.checked );
}));
} else {
prefix_widget.empty().hide();
}
}
}
callback();
});
args.icon_args.container.append('<span title="specify the thread icon" id="mod-friend-thread-metadata-icons"></span>' );
var icon_widget = new IconSelect('mod-friend-thread-metadata-icons', {
selectedIconWidth: 16,
selectedIconHeight: 16,
iconsWidth: 16,
iconsHeight: 16,
boxIconSpace: 0,
vectoralIconNumber: 3,
horizontalIconNumber: 5
});
var prefix_widget = $('<select title="specify the thread prefix"></select>').appendTo(args.prefix_args.container).hide();
prefix_widget.change(function() {
policy.default_keys[4].value = $( ':selected', this ).text();
callback();
});
var prev_title_thread_id = args.thread_id;
var title_widget = new ThreadTitleSelector($.extend({}, args, args.title_args, {
mode: 'edit',
callback: function(thread) {
if ( !forum_widget ) return; // skip during creation
if ( thread ) {
policy.default_keys[1].value = thread;
if ( prev_title_thread_id != thread.thread_id ) {
prev_title_thread_id = thread.thread_id;
forum_widget.val( thread.forum_id ).change();
}
} else {
policy.default_keys[1].value = { thread_id: args.thread_id, thread_desc: args.thread_desc, forum_id: args.forum_id };
forum_widget.val( args.forum_id ).change();
}
callback();
}
}));
var forum_widget = $('<select title="move this thread to another forum" required><option value="">Please select a forum...</option></select>').appendTo(args.forum_args.container);
var forum_metadata = {}, stored_prefixes = {}, old_forum_id;
forum_widget.change(function() {
if ( !this.value ) return;
policy.default_keys[3].value.forum_id = this.value;
policy.default_keys[3].value.forum_desc = $( ':selected', this ).text().replace( /^\s+/, '' );
var old_prefix = stored_prefixes[old_forum_id] = prefix_widget.val();
var stored_prefix = stored_prefixes[this.value];
old_forum_id = this.value;
if ( !forum_metadata[this.value] ) forum_metadata[this.value] = args.bb.forum_metadata(this.value);
forum_metadata[this.value].then(function(forum_metadata) {
metadata = forum_metadata;
if ( forum_has_icons ) {
forum_has_icons = forum_metadata.icons.length;
if ( !forum_has_icons )
$('#mod-friend-thread-metadata-icons').hide();
} else {
forum_has_icons = forum_metadata.icons.length;
if ( forum_has_icons ) {
var default_icon;
icon_widget.refresh(
forum_metadata.icons.map(function(icon, index) {
if ( !icon.file ) default_icon = index;
return {
'iconFilePath': icon.file || '',
'iconValue': icon.icon_id
};
})
);
icon_widget.setSelectedIndex( ( original.icon == -1 ) ? default_icon : original.icon );
$('#mod-friend-thread-metadata-icons').show();
}
}
if ( forum_metadata.prefixes.length ) {
var has_old_prefix, has_stored_prefix;
prefix_widget.show().empty().append(forum_metadata.prefixes.map(function(option) {
if ( old_prefix && option.prefix_id == old_prefix ) has_old_prefix = true;
if ( stored_prefix && option.prefix_id == stored_prefix ) has_stored_prefix = true;
return $('<option>').val(option.prefix_id).text(option.text);
})).val(
has_old_prefix ? old_prefix :
has_stored_prefix ? stored_prefix :
''
);
macro_defaults.prefix_id = prefix_widget.val()||'';
} else {
prefix_widget.empty().hide();
}
callback();
});
});
var post = new NotificationSelector(this.notification_selector_args(
{ html: 'post', type: 'post' },
args.post_selector_args,
{ known_keys: ['list of posts in merged thread'], key_prefix: 'post ' }
));
var pm = new NotificationSelector(this.notification_selector_args(
{ html: 'PM', type: 'PM' },
args.pm_selector_args,
{ known_keys: ['list of posts in merged thread'], key_prefix: 'pm ' }
));
var message_selector = $(
'<div>' +
'<label title="post a new reply in this thread"><input type="checkbox" name="post">post to thread</label>' +
'<label><input type="checkbox" name="pm">PM the thread creator</label>' +
'</div>'
).appendTo(args.message_selector_args.container);
message_selector.find('input').change(callback)
.last().parent().attr( 'title', 'send a private message to ' + thread_creator.username );
var sender_selector = $(
'<div>' +
'<label title="more official, protects your privacy"><input type="radio" value="team" name="sender" checked>send from mod team account</label>' +
'<label title="more individual, can ease tensions"><input type="radio" value="personal" name="sender">send from personal account</label>' +
'</div>'
).appendTo(args.sender_selector_args.container);
sender_selector.find('input').change(function() {
if ( this.checked == (this.value == 'team' ) )
post.bb = pm.bb = args.mod_team_bb;
else
post.bb = pm.bb = args.bb;
}).filter(':checked').change();
var bump_selector = $(
'<div>' +
'<label title="push this thread up to the top of the listings"><input type="radio" name="bump" value="bump">bump thread</label>' +
'<label><input type="radio" name="bump" checked>no bump</label>' +
'<label title="hide this thread far down in the listings"><input type="radio" name="bump" value="debump">de-bump thread</label>' +
'</div>'
).appendTo(args.bump_selector_args.container);
bump_selector.find('input').change(callback);
var template_selector = $(
'<select title="select a template reply or specify actions by hand" name="response_template"><option value="No template">Please choose a template...</option></select>'
).appendTo(args.template_selector_args.container);
var seen_macros = {};
var macros_to_insert = recent_macros.get().reverse().filter(function(macro) {
if ( seen_macros.hasOwnProperty(macro.name) ) return false;
seen_macros[macro.name] = true;
return true;
});
if ( macros_to_insert.length ) {
$('<optgroup>').attr( 'label', 'Recent' )
.appendTo(template_selector)
.append(
macros_to_insert.map(function(macro) {
return $('<option>')
.text (macro.name)
.val (macro.template)
.attr( 'title', policy.resolve(['hint',macro.template]) )
.data( 'macro', macro )
})
);
}
template_selector.append(
this.resolve( 'visible templates', {}, 'array of items' ).map(function(templates) {
templates = templates.value.split( /\s*[:,]\s*/g );
var name = templates.shift();
var optgroup = $('<optgroup>').attr( 'label', name );
optgroup.append( templates.map(function(t) {
return $('<option>')
.text(t)
.val (t)
.attr( 'title', policy.resolve(['hint',t]) )
}) );
return optgroup;
})
).change(function() {
/*
* Update everything based on the new template
*/
policy.default_keys[policy.default_keys.length-1].value
= variable_suffix[0]
= action_data.template
= $(this).val();
title_widget.val(original.title);
if ( policy.check( [ 'new thread id' ].concat(variable_suffix) ) ) {
status_widget.val( 'merged' ).change();
title_widget.val({
target_thread_id: parseInt( policy.resolve([ 'new thread id' ].concat(variable_suffix)).toLowerCase(), 10 )
});
} else if ( policy.check( [ 'deadline' ].concat(variable_suffix) ) ) {
var deadline = policy.resolve([ 'deadline' ].concat(variable_suffix)).toLowerCase();
status_widget.find('option')
.prop( 'selected', false )
.filter(function() { return $(this).data('deadline') == deadline })
.prop( 'selected', true )
.change()
;
} else if ( policy.check( [ 'new status' ].concat(variable_suffix) ) ) {
status_widget.val( policy.resolve([ 'new status' ].concat(variable_suffix)).toLowerCase() ).change();
} else {
status_widget.val( original.status ).change();
}
if ( policy.check( [ 'new thread title' ].concat(variable_suffix) ) ) {
title_widget.val({
target_thread_id : args.thread_id,
target_thread_desc: policy.resolve( [ 'new thread title' ].concat(variable_suffix), { 'old thread title': args.thread_desc } ),
target_forum_id : args.forum_id,
});
}
if ( policy.check( [ 'new icon' ].concat(variable_suffix) ) ) {
var icon_name = policy.resolve( [ 'new icon' ].concat(variable_suffix) );
metadata.icons.map(function(icon, index) {
if ( icon.name == icon_name ) icon_widget.setSelectedIndex(index);
});
} else if ( forum_has_icons ) {
if ( original.icon == -1 ) {
metadata.icons.map(function(icon, index) {
if ( !icon.file ) icon_widget.setSelectedIndex(index);
});
} else {
icon_widget.setSelectedIndex(original.icon);
}
}
if ( policy.check( [ 'new forum' ].concat(variable_suffix) ) ) {
var new_forum = policy.resolve( [ 'new forum' ].concat(variable_suffix) ).toLowerCase();
forum_widget.find('option')
.prop( 'selected', false )
.filter(function() { return this.textContent.toLowerCase() == new_forum })
.prop( 'selected', true )
;
} else {
forum_widget.val( original.forum_id );
}
forum_widget.change();
if ( policy.check( [ 'new prefix' ].concat(variable_suffix) ) ) {
var new_prefix = policy.resolve( [ 'new prefix' ].concat(variable_suffix) );
prefix_widget.find('option')
.prop( 'selected', false )
.filter(function() { return this.textContent.toLowerCase() == new_prefix })
.prop( 'selected', true )
;
} else {
prefix_widget.val( original.prefix );
}
prefix_widget.change();
message_selector.find('[name="post"]').prop( 'checked', policy.check( [ 'post bbcode' ].concat(variable_suffix) ) );
post.val( policy.notification_selector_args({ html: 'post', type: 'post' }) );
message_selector.find('[name="pm"]' ).prop( 'checked', policy.check( [ 'pm bbcode' ].concat(variable_suffix) ) );
pm .val( policy.notification_selector_args({ html: 'PM', type: 'PM' }) );
// record macro defaults even if this action came from a macro, so frequent actions are frequently updated:
macro_defaults = {
template: variable_suffix[0],
status: status_widget.val(),
deadline: status_widget.find('option:selected').data('deadline') || '',
has_post: message_selector.find('[name="post"]').prop( 'checked' ),
has_pm : message_selector.find('[name="pm"]' ).prop( 'checked' ),
icon : forum_has_icons ? icon_widget.getSelectedIndex() : -1,
forum_id: forum_widget.find('option:selected').val() || '',
prefix_id: prefix_widget.val()||'',
target_thread_id: title_widget.val().target_thread_id || NaN
};
var macro = $( 'option:selected', this ).data( 'macro' );
if ( macro ) {
switch ( macro.status ) {
case undefined: break;
case 'closed temporarily':
status_widget.find('option')
.prop( 'selected', false )
.filter(function() { return $(this).data('deadline') == macro.deadline })
.prop( 'selected', true )
;
status_widget.change();
title_widget.val({ mode: 'edit' });
break;
case 'merged':
status_widget.val('merged').change();
title_widget.val({ mode: 'merge', target_thread_id: macro.target_thread_id });
break;
default:
title_widget.val({ mode: 'edit' });
status_widget.val( macro[status] ).change();
}
if ( macro.hasOwnProperty('icon' ) ) icon_widget.setSelectedIndex( macro.icon );
if ( macro.hasOwnProperty('has_pm' ) ) message_selector.find('[name="pm"]' ).prop('checked', macro.has_pm ).change();
if ( macro.hasOwnProperty('has_post') ) message_selector.find('[name="post"]').prop('checked', macro.has_post).change();
if ( macro.hasOwnProperty('prefix_id') ) {
if ( macro.hasOwnProperty('forum_id') ) {
forum_widget.val(macro.forum_id);
forum_metadata[macro.forum_id].then(function() {
prefix_widget.val(macro.prefix_id).change();
});
} else {
prefix_widget.val(macro.prefix_id).change();
}
} else if ( macro.hasOwnProperty('forum_id') ) {
forum_widget.val(macro.forum_id).change();
}
}
callback();
});
/*
* Download information and initialise a few last things
*/
this.promise = $.when(
args.bb.thread_metadata(args.thread_id),
args.bb.forums()
).then(function(_metadata, forums) {
metadata = _metadata;
status_widget.val(
metadata.deleted ? 'deleted' :
metadata.moderated ? 'moderated' :
metadata.open ? 'open' :
'closed'
);
var selected_index;
forum_has_icons = metadata.icons.length;
if ( forum_has_icons ) {
icon_widget.refresh(
metadata.icons.map(function(icon, index) {
if ( icon.icon_id == metadata.icon_id ) selected_index = index;
return {
'iconFilePath': icon.file || '',
'iconValue': icon.icon_id
};
})
);
icon_widget.setSelectedIndex(selected_index);
$('#mod-friend-thread-metadata-icons').show();
} else {
$('#mod-friend-thread-metadata-icons').hide();
}
$('#mod-friend-thread-metadata-icons').on( 'changed', function() {
policy.default_keys[5].value = (
metadata.icons[icon_widget.getSelectedIndex()].file
? location.origin + '/' + metadata.icons[icon_widget.getSelectedIndex()].file
: ''
);
callback();
});
if ( metadata.prefixes.length ) {
prefix_widget
.show()
.empty()
.append(metadata.prefixes.map(function(option) { return $('<option>').val(option.prefix_id).text(option.text) }))
.val( metadata.prefix_id );
}
(function build_forums(options, prefix, optgroup) {
options.forEach(function(option) {
if ( option.forum_id ) {
var node = $('<option>').val(option.forum_id);
if ( option.forum_id == args.forum_id )
node.text(prefix + option.name + ' (current forum)').addClass('current-forum');
else
node.text(prefix + option.name);
optgroup.append( node );
if ( option.children )
build_forums( option.children, prefix + '\xA0\xA0\xA0\xA0', optgroup );
} else if ( option.children ) {
build_forums( option.children, prefix, $('<optgroup>').attr( 'label', prefix + option.name ).appendTo(forum_widget) );
}
});
})( forums, '', forum_widget );
old_forum_id = metadata.forum[metadata.forum.length-1].forum_id;
forum_widget.val( metadata.forum[metadata.forum.length-1].forum_id );
original = action_data.original = {
thread_id: args.thread_id,
status : status_widget.val(),
title : $.extend( {}, title_widget.val() ),
forum_id : parseInt( forum_widget.val(), 10 ),
prefix : prefix_widget.val() || '',
icon : icon_widget.getSelectedIndex(),
icon_id : forum_has_icons ? icon_widget.getSelectedValue() : undefined,
};
unmerge_data = {
forum_id: original.forum_id,
thread_id: args.thread_id,
title : original.title.thread_desc,
icon_id: original.icon_id,
prefix : original.prefix,
status : original.status,
close_thread: metadata.open ? undefined : true,
delete_thread: metadata.deleted ? true : undefined,
unapprove_thread: metadata.moderated ? true : undefined,
delete_reason: metadata.delete_reason,
notes : metadata.notes,
};
template_selector.change();
});
}
ThreadManagementPolicy.prototype = Object.create(Policy.prototype, {
_namespace: { writable: false, configurable: false, value: 'thread management' },
});
ThreadManagementPolicy.prototype.constructor = ThreadManagementPolicy;
/**
* @summary build an action to unmerge a thread
* @param {BulletinBoard} bb Bulletin Board to manipulate
* @param {BulletinBoard} mod_team_bb Bulletin Board to manipulate
* @param {Variables} v Variables to use
* @param {Object} unmerge_data 'merge data' created during merging
* @return {Action}
*/
ThreadManagementPolicy.prototype.unmerge_action = function(bb, mod_team_bb, v, unmerge_data) {
var merge_log = v.resolve('frequently used posts/threads', 'merge log'), merge_post_id;
return new Action(
'root action',
new Action(
'unmerge wrapper',
new Action( 'create thread', {
fire: function(keys) {
return mod_team_bb.thread_create($.extend(
{
bbcode: v.resolve('thread management', 'unmerge notification bbcode', $.extend( keys, unmerge_data.source_thread ) ),
},
unmerge_data.source_thread
)).then(function(new_thread_id) {
unmerge_data.source_thread.thread_id = new_thread_id;
return { keys: {
'new thread id': new_thread_id,
'old thread title with link': '[thread=' + new_thread_id + ']' + keys['old thread title'] + '[/thread]'
}};
});
},
description: function() {
return [{ type: 'create thread', target: unmerge_data.source_thread }];
}
}).then(
new Action( 'move posts', {
fire: function(keys) {
return bb.posts_move( keys['new thread id'], unmerge_data.posts );
},
description: function() {
return [{ type: 'move posts', target: { thread: unmerge_data.source_thread, posts: unmerge_data.posts } }];
}
}),
new Action( 'post to merge log', {
fire: function(keys) {
return bb.thread_reply({
thread_id: merge_log,
title : v.resolve('thread management', 'unmerge title' , keys),
bbcode : v.resolve('thread management', 'unmerge bbcode', keys),
}).then(function(post_id) {
merge_post_id = post_id;
});
},
description: function() {
return [
{
type: 'post',
target: {
thread_desc: 'the merge log',
thread_id : merge_log,
post_id : merge_post_id
}
},
];
}
}),
( unmerge_data.source_thread.delete_thread || unmerge_data.source_thread.unapprove_thread ) ? new Action( 'change thread status', {
fire: function(keys) {
return bb.thread_edit(unmerge_data.source_thread)
},
description: function() {
return [{ type: 'change thread status', target: unmerge_data.source_thread }];
}
}) : undefined
)
).then(
unmerge_data.thread_creator
? new Action( 'add usernote', {
fire: function(keys) {
var return_keys = {};
if ( !keys.hasOwnProperty( 'pm result') ) return_keys[ 'pm result'] = 'no action';
if ( !keys.hasOwnProperty('post result') ) return_keys['post result'] = 'no action';
$.extend( keys, return_keys );
return args.bb.usernote_add(
thread_creator.user_id,
policy.resolve( 'combined note title' , keys ),
policy.resolve( 'combined note bbcode', keys )
).then(function() {
if ( keys['notification error' ] == 'fail' ) {
return $.Deferred()
.reject('Notification failed, but the error was postponed so we could add a usernote:\n' + error)
.promise();
} else {
return { keys: return_keys };
}
});
},
description: function() {
return [{ type: 'usernote', target: thread_creator }];
}
})
: new Action( 'fix results', {
fire: function(keys) {
var return_keys = {};
if ( !keys.hasOwnProperty( 'pm result') ) return_keys[ 'pm result'] = 'no action';
if ( !keys.hasOwnProperty('post result') ) return_keys['post result'] = 'no action';
return { keys: return_keys };
}
})
)
);
}