Chris@76: 'SplitSelectTopics', Chris@76: 'execute' => 'SplitExecute', Chris@76: 'index' => 'SplitIndex', Chris@76: 'splitSelection' => 'SplitSelectionExecute', Chris@76: ); Chris@76: Chris@76: // ?action=splittopics;sa=LETSBREAKIT won't work, sorry. Chris@76: if (empty($_REQUEST['sa']) || !isset($subActions[$_REQUEST['sa']])) Chris@76: SplitIndex(); Chris@76: else Chris@76: $subActions[$_REQUEST['sa']](); Chris@76: } Chris@76: Chris@76: // Part 1: General stuff. Chris@76: function SplitIndex() Chris@76: { Chris@76: global $txt, $topic, $context, $smcFunc, $modSettings; Chris@76: Chris@76: // Validate "at". Chris@76: if (empty($_GET['at'])) Chris@76: fatal_lang_error('numbers_one_to_nine', false); Chris@76: $_GET['at'] = (int) $_GET['at']; Chris@76: Chris@76: // Retrieve the subject and stuff of the specific topic/message. Chris@76: $request = $smcFunc['db_query']('', ' Chris@76: SELECT m.subject, t.num_replies, t.unapproved_posts, t.id_first_msg, t.approved Chris@76: FROM {db_prefix}messages AS m Chris@76: INNER JOIN {db_prefix}topics AS t ON (t.id_topic = {int:current_topic}) Chris@76: WHERE m.id_msg = {int:split_at}' . (!$modSettings['postmod_active'] || allowedTo('approve_posts') ? '' : ' Chris@76: AND m.approved = 1') . ' Chris@76: AND m.id_topic = {int:current_topic} Chris@76: LIMIT 1', Chris@76: array( Chris@76: 'current_topic' => $topic, Chris@76: 'split_at' => $_GET['at'], Chris@76: ) Chris@76: ); Chris@76: if ($smcFunc['db_num_rows']($request) == 0) Chris@76: fatal_lang_error('cant_find_messages'); Chris@76: list ($_REQUEST['subname'], $num_replies, $unapproved_posts, $id_first_msg, $approved) = $smcFunc['db_fetch_row']($request); Chris@76: $smcFunc['db_free_result']($request); Chris@76: Chris@76: // If not approved validate they can see it. Chris@76: if ($modSettings['postmod_active'] && !$approved) Chris@76: isAllowedTo('approve_posts'); Chris@76: Chris@76: // If this topic has unapproved posts, we need to count them too... Chris@76: if ($modSettings['postmod_active'] && allowedTo('approve_posts')) Chris@76: $num_replies += $unapproved_posts - ($approved ? 0 : 1); Chris@76: Chris@76: // Check if there is more than one message in the topic. (there should be.) Chris@76: if ($num_replies < 1) Chris@76: fatal_lang_error('topic_one_post', false); Chris@76: Chris@76: // Check if this is the first message in the topic (if so, the first and second option won't be available) Chris@76: if ($id_first_msg == $_GET['at']) Chris@76: return SplitSelectTopics(); Chris@76: Chris@76: // Basic template information.... Chris@76: $context['message'] = array( Chris@76: 'id' => $_GET['at'], Chris@76: 'subject' => $_REQUEST['subname'] Chris@76: ); Chris@76: $context['sub_template'] = 'ask'; Chris@76: $context['page_title'] = $txt['split']; Chris@76: } Chris@76: Chris@76: // Alright, you've decided what you want to do with it.... now to do it. Chris@76: function SplitExecute() Chris@76: { Chris@76: global $txt, $board, $topic, $context, $user_info, $smcFunc, $modSettings; Chris@76: Chris@76: // Check the session to make sure they meant to do this. Chris@76: checkSession(); Chris@76: Chris@76: // Clean up the subject. Chris@76: if (!isset($_POST['subname']) || $_POST['subname'] == '') Chris@76: $_POST['subname'] = $txt['new_topic']; Chris@76: Chris@76: // Redirect to the selector if they chose selective. Chris@76: if ($_POST['step2'] == 'selective') Chris@76: { Chris@76: $_REQUEST['subname'] = $_POST['subname']; Chris@76: return SplitSelectTopics(); Chris@76: } Chris@76: Chris@76: $_POST['at'] = (int) $_POST['at']; Chris@76: $messagesToBeSplit = array(); Chris@76: Chris@76: if ($_POST['step2'] == 'afterthis') Chris@76: { Chris@76: // Fetch the message IDs of the topic that are at or after the message. Chris@76: $request = $smcFunc['db_query']('', ' Chris@76: SELECT id_msg Chris@76: FROM {db_prefix}messages Chris@76: WHERE id_topic = {int:current_topic} Chris@76: AND id_msg >= {int:split_at}', Chris@76: array( Chris@76: 'current_topic' => $topic, Chris@76: 'split_at' => $_POST['at'], Chris@76: ) Chris@76: ); Chris@76: while ($row = $smcFunc['db_fetch_assoc']($request)) Chris@76: $messagesToBeSplit[] = $row['id_msg']; Chris@76: $smcFunc['db_free_result']($request); Chris@76: } Chris@76: // Only the selected message has to be split. That should be easy. Chris@76: elseif ($_POST['step2'] == 'onlythis') Chris@76: $messagesToBeSplit[] = $_POST['at']; Chris@76: // There's another action?! Chris@76: else Chris@76: fatal_lang_error('no_access', false); Chris@76: Chris@76: $context['old_topic'] = $topic; Chris@76: $context['new_topic'] = splitTopic($topic, $messagesToBeSplit, $_POST['subname']); Chris@76: $context['page_title'] = $txt['split']; Chris@76: } Chris@76: Chris@76: // Get a selective list of topics... Chris@76: function SplitSelectTopics() Chris@76: { Chris@76: global $txt, $scripturl, $topic, $context, $modSettings, $original_msgs, $smcFunc, $options; Chris@76: Chris@76: $context['page_title'] = $txt['split'] . ' - ' . $txt['select_split_posts']; Chris@76: Chris@76: // Haven't selected anything have we? Chris@76: $_SESSION['split_selection'][$topic] = empty($_SESSION['split_selection'][$topic]) ? array() : $_SESSION['split_selection'][$topic]; Chris@76: Chris@76: $context['not_selected'] = array( Chris@76: 'num_messages' => 0, Chris@76: 'start' => empty($_REQUEST['start']) ? 0 : (int) $_REQUEST['start'], Chris@76: 'messages' => array(), Chris@76: ); Chris@76: Chris@76: $context['selected'] = array( Chris@76: 'num_messages' => 0, Chris@76: 'start' => empty($_REQUEST['start2']) ? 0 : (int) $_REQUEST['start2'], Chris@76: 'messages' => array(), Chris@76: ); Chris@76: Chris@76: $context['topic'] = array( Chris@76: 'id' => $topic, Chris@76: 'subject' => urlencode($_REQUEST['subname']), Chris@76: ); Chris@76: Chris@76: // Some stuff for our favorite template. Chris@76: $context['new_subject'] = $_REQUEST['subname']; Chris@76: Chris@76: // Using the "select" sub template. Chris@76: $context['sub_template'] = isset($_REQUEST['xml']) ? 'split' : 'select'; Chris@76: Chris@76: // Are we using a custom messages per page? Chris@76: $context['messages_per_page'] = empty($modSettings['disableCustomPerPage']) && !empty($options['messages_per_page']) ? $options['messages_per_page'] : $modSettings['defaultMaxMessages']; Chris@76: Chris@76: // Get the message ID's from before the move. Chris@76: if (isset($_REQUEST['xml'])) Chris@76: { Chris@76: $original_msgs = array( Chris@76: 'not_selected' => array(), Chris@76: 'selected' => array(), Chris@76: ); Chris@76: $request = $smcFunc['db_query']('', ' Chris@76: SELECT id_msg Chris@76: FROM {db_prefix}messages Chris@76: WHERE id_topic = {int:current_topic}' . (empty($_SESSION['split_selection'][$topic]) ? '' : ' Chris@76: AND id_msg NOT IN ({array_int:no_split_msgs})') . (!$modSettings['postmod_active'] || allowedTo('approve_posts') ? '' : ' Chris@76: AND approved = {int:is_approved}') . ' Chris@76: ORDER BY id_msg DESC Chris@76: LIMIT {int:start}, {int:messages_per_page}', Chris@76: array( Chris@76: 'current_topic' => $topic, Chris@76: 'no_split_msgs' => empty($_SESSION['split_selection'][$topic]) ? array() : $_SESSION['split_selection'][$topic], Chris@76: 'is_approved' => 1, Chris@76: 'start' => $context['not_selected']['start'], Chris@76: 'messages_per_page' => $context['messages_per_page'], Chris@76: ) Chris@76: ); Chris@76: // You can't split the last message off. Chris@76: if (empty($context['not_selected']['start']) && $smcFunc['db_num_rows']($request) <= 1 && $_REQUEST['move'] == 'down') Chris@76: $_REQUEST['move'] = ''; Chris@76: while ($row = $smcFunc['db_fetch_assoc']($request)) Chris@76: $original_msgs['not_selected'][] = $row['id_msg']; Chris@76: $smcFunc['db_free_result']($request); Chris@76: if (!empty($_SESSION['split_selection'][$topic])) Chris@76: { Chris@76: $request = $smcFunc['db_query']('', ' Chris@76: SELECT id_msg Chris@76: FROM {db_prefix}messages Chris@76: WHERE id_topic = {int:current_topic} Chris@76: AND id_msg IN ({array_int:split_msgs})' . (!$modSettings['postmod_active'] || allowedTo('approve_posts') ? '' : ' Chris@76: AND approved = {int:is_approved}') . ' Chris@76: ORDER BY id_msg DESC Chris@76: LIMIT {int:start}, {int:messages_per_page}', Chris@76: array( Chris@76: 'current_topic' => $topic, Chris@76: 'split_msgs' => $_SESSION['split_selection'][$topic], Chris@76: 'is_approved' => 1, Chris@76: 'start' => $context['selected']['start'], Chris@76: 'messages_per_page' => $context['messages_per_page'], Chris@76: ) Chris@76: ); Chris@76: while ($row = $smcFunc['db_fetch_assoc']($request)) Chris@76: $original_msgs['selected'][] = $row['id_msg']; Chris@76: $smcFunc['db_free_result']($request); Chris@76: } Chris@76: } Chris@76: Chris@76: // (De)select a message.. Chris@76: if (!empty($_REQUEST['move'])) Chris@76: { Chris@76: $_REQUEST['msg'] = (int) $_REQUEST['msg']; Chris@76: Chris@76: if ($_REQUEST['move'] == 'reset') Chris@76: $_SESSION['split_selection'][$topic] = array(); Chris@76: elseif ($_REQUEST['move'] == 'up') Chris@76: $_SESSION['split_selection'][$topic] = array_diff($_SESSION['split_selection'][$topic], array($_REQUEST['msg'])); Chris@76: else Chris@76: $_SESSION['split_selection'][$topic][] = $_REQUEST['msg']; Chris@76: } Chris@76: Chris@76: // Make sure the selection is still accurate. Chris@76: if (!empty($_SESSION['split_selection'][$topic])) Chris@76: { Chris@76: $request = $smcFunc['db_query']('', ' Chris@76: SELECT id_msg Chris@76: FROM {db_prefix}messages Chris@76: WHERE id_topic = {int:current_topic} Chris@76: AND id_msg IN ({array_int:split_msgs})' . (!$modSettings['postmod_active'] || allowedTo('approve_posts') ? '' : ' Chris@76: AND approved = {int:is_approved}'), Chris@76: array( Chris@76: 'current_topic' => $topic, Chris@76: 'split_msgs' => $_SESSION['split_selection'][$topic], Chris@76: 'is_approved' => 1, Chris@76: ) Chris@76: ); Chris@76: $_SESSION['split_selection'][$topic] = array(); Chris@76: while ($row = $smcFunc['db_fetch_assoc']($request)) Chris@76: $_SESSION['split_selection'][$topic][] = $row['id_msg']; Chris@76: $smcFunc['db_free_result']($request); Chris@76: } Chris@76: Chris@76: // Get the number of messages (not) selected to be split. Chris@76: $request = $smcFunc['db_query']('', ' Chris@76: SELECT ' . (empty($_SESSION['split_selection'][$topic]) ? '0' : 'm.id_msg IN ({array_int:split_msgs})') . ' AS is_selected, COUNT(*) AS num_messages Chris@76: FROM {db_prefix}messages AS m Chris@76: WHERE m.id_topic = {int:current_topic}' . (!$modSettings['postmod_active'] || allowedTo('approve_posts') ? '' : ' Chris@76: AND approved = {int:is_approved}') . (empty($_SESSION['split_selection'][$topic]) ? '' : ' Chris@76: GROUP BY is_selected'), Chris@76: array( Chris@76: 'current_topic' => $topic, Chris@76: 'split_msgs' => !empty($_SESSION['split_selection'][$topic]) ? $_SESSION['split_selection'][$topic] : array(), Chris@76: 'is_approved' => 1, Chris@76: ) Chris@76: ); Chris@76: while ($row = $smcFunc['db_fetch_assoc']($request)) Chris@76: $context[empty($row['is_selected']) ? 'not_selected' : 'selected']['num_messages'] = $row['num_messages']; Chris@76: $smcFunc['db_free_result']($request); Chris@76: Chris@76: // Fix an oversized starting page (to make sure both pageindexes are properly set). Chris@76: if ($context['selected']['start'] >= $context['selected']['num_messages']) Chris@76: $context['selected']['start'] = $context['selected']['num_messages'] <= $context['messages_per_page'] ? 0 : ($context['selected']['num_messages'] - (($context['selected']['num_messages'] % $context['messages_per_page']) == 0 ? $context['messages_per_page'] : ($context['selected']['num_messages'] % $context['messages_per_page']))); Chris@76: Chris@76: // Build a page list of the not-selected topics... Chris@76: $context['not_selected']['page_index'] = constructPageIndex($scripturl . '?action=splittopics;sa=selectTopics;subname=' . strtr(urlencode($_REQUEST['subname']), array('%' => '%%')) . ';topic=' . $topic . '.%1$d;start2=' . $context['selected']['start'], $context['not_selected']['start'], $context['not_selected']['num_messages'], $context['messages_per_page'], true); Chris@76: // ...and one of the selected topics. Chris@76: $context['selected']['page_index'] = constructPageIndex($scripturl . '?action=splittopics;sa=selectTopics;subname=' . strtr(urlencode($_REQUEST['subname']), array('%' => '%%')) . ';topic=' . $topic . '.' . $context['not_selected']['start'] . ';start2=%1$d', $context['selected']['start'], $context['selected']['num_messages'], $context['messages_per_page'], true); Chris@76: Chris@76: // Get the messages and stick them into an array. Chris@76: $request = $smcFunc['db_query']('', ' Chris@76: SELECT m.subject, IFNULL(mem.real_name, m.poster_name) AS real_name, m.poster_time, m.body, m.id_msg, m.smileys_enabled Chris@76: FROM {db_prefix}messages AS m Chris@76: LEFT JOIN {db_prefix}members AS mem ON (mem.id_member = m.id_member) Chris@76: WHERE m.id_topic = {int:current_topic}' . (empty($_SESSION['split_selection'][$topic]) ? '' : ' Chris@76: AND id_msg NOT IN ({array_int:no_split_msgs})') . (!$modSettings['postmod_active'] || allowedTo('approve_posts') ? '' : ' Chris@76: AND approved = {int:is_approved}') . ' Chris@76: ORDER BY m.id_msg DESC Chris@76: LIMIT {int:start}, {int:messages_per_page}', Chris@76: array( Chris@76: 'current_topic' => $topic, Chris@76: 'no_split_msgs' => !empty($_SESSION['split_selection'][$topic]) ? $_SESSION['split_selection'][$topic] : array(), Chris@76: 'is_approved' => 1, Chris@76: 'start' => $context['not_selected']['start'], Chris@76: 'messages_per_page' => $context['messages_per_page'], Chris@76: ) Chris@76: ); Chris@76: $context['messages'] = array(); Chris@76: for ($counter = 0; $row = $smcFunc['db_fetch_assoc']($request); $counter ++) Chris@76: { Chris@76: censorText($row['subject']); Chris@76: censorText($row['body']); Chris@76: Chris@76: $row['body'] = parse_bbc($row['body'], $row['smileys_enabled'], $row['id_msg']); Chris@76: Chris@76: $context['not_selected']['messages'][$row['id_msg']] = array( Chris@76: 'id' => $row['id_msg'], Chris@76: 'alternate' => $counter % 2, Chris@76: 'subject' => $row['subject'], Chris@76: 'time' => timeformat($row['poster_time']), Chris@76: 'timestamp' => forum_time(true, $row['poster_time']), Chris@76: 'body' => $row['body'], Chris@76: 'poster' => $row['real_name'], Chris@76: ); Chris@76: } Chris@76: $smcFunc['db_free_result']($request); Chris@76: Chris@76: // Now get the selected messages. Chris@76: if (!empty($_SESSION['split_selection'][$topic])) Chris@76: { Chris@76: // Get the messages and stick them into an array. Chris@76: $request = $smcFunc['db_query']('', ' Chris@76: SELECT m.subject, IFNULL(mem.real_name, m.poster_name) AS real_name, m.poster_time, m.body, m.id_msg, m.smileys_enabled Chris@76: FROM {db_prefix}messages AS m Chris@76: LEFT JOIN {db_prefix}members AS mem ON (mem.id_member = m.id_member) Chris@76: WHERE m.id_topic = {int:current_topic} Chris@76: AND m.id_msg IN ({array_int:split_msgs})' . (!$modSettings['postmod_active'] || allowedTo('approve_posts') ? '' : ' Chris@76: AND approved = {int:is_approved}') . ' Chris@76: ORDER BY m.id_msg DESC Chris@76: LIMIT {int:start}, {int:messages_per_page}', Chris@76: array( Chris@76: 'current_topic' => $topic, Chris@76: 'split_msgs' => $_SESSION['split_selection'][$topic], Chris@76: 'is_approved' => 1, Chris@76: 'start' => $context['selected']['start'], Chris@76: 'messages_per_page' => $context['messages_per_page'], Chris@76: ) Chris@76: ); Chris@76: $context['messages'] = array(); Chris@76: for ($counter = 0; $row = $smcFunc['db_fetch_assoc']($request); $counter ++) Chris@76: { Chris@76: censorText($row['subject']); Chris@76: censorText($row['body']); Chris@76: Chris@76: $row['body'] = parse_bbc($row['body'], $row['smileys_enabled'], $row['id_msg']); Chris@76: Chris@76: $context['selected']['messages'][$row['id_msg']] = array( Chris@76: 'id' => $row['id_msg'], Chris@76: 'alternate' => $counter % 2, Chris@76: 'subject' => $row['subject'], Chris@76: 'time' => timeformat($row['poster_time']), Chris@76: 'timestamp' => forum_time(true, $row['poster_time']), Chris@76: 'body' => $row['body'], Chris@76: 'poster' => $row['real_name'] Chris@76: ); Chris@76: } Chris@76: $smcFunc['db_free_result']($request); Chris@76: } Chris@76: Chris@76: // The XMLhttp method only needs the stuff that changed, so let's compare. Chris@76: if (isset($_REQUEST['xml'])) Chris@76: { Chris@76: $changes = array( Chris@76: 'remove' => array( Chris@76: 'not_selected' => array_diff($original_msgs['not_selected'], array_keys($context['not_selected']['messages'])), Chris@76: 'selected' => array_diff($original_msgs['selected'], array_keys($context['selected']['messages'])), Chris@76: ), Chris@76: 'insert' => array( Chris@76: 'not_selected' => array_diff(array_keys($context['not_selected']['messages']), $original_msgs['not_selected']), Chris@76: 'selected' => array_diff(array_keys($context['selected']['messages']), $original_msgs['selected']), Chris@76: ), Chris@76: ); Chris@76: Chris@76: $context['changes'] = array(); Chris@76: foreach ($changes as $change_type => $change_array) Chris@76: foreach ($change_array as $section => $msg_array) Chris@76: { Chris@76: if (empty($msg_array)) Chris@76: continue; Chris@76: Chris@76: foreach ($msg_array as $id_msg) Chris@76: { Chris@76: $context['changes'][$change_type . $id_msg] = array( Chris@76: 'id' => $id_msg, Chris@76: 'type' => $change_type, Chris@76: 'section' => $section, Chris@76: ); Chris@76: if ($change_type == 'insert') Chris@76: $context['changes']['insert' . $id_msg]['insert_value'] = $context[$section]['messages'][$id_msg]; Chris@76: } Chris@76: } Chris@76: } Chris@76: } Chris@76: Chris@76: // Actually and selectively split the topics out. Chris@76: function SplitSelectionExecute() Chris@76: { Chris@76: global $txt, $board, $topic, $context, $user_info; Chris@76: Chris@76: // Make sure the session id was passed with post. Chris@76: checkSession(); Chris@76: Chris@76: // Default the subject in case it's blank. Chris@76: if (!isset($_POST['subname']) || $_POST['subname'] == '') Chris@76: $_POST['subname'] = $txt['new_topic']; Chris@76: Chris@76: // You must've selected some messages! Can't split out none! Chris@76: if (empty($_SESSION['split_selection'][$topic])) Chris@76: fatal_lang_error('no_posts_selected', false); Chris@76: Chris@76: $context['old_topic'] = $topic; Chris@76: $context['new_topic'] = splitTopic($topic, $_SESSION['split_selection'][$topic], $_POST['subname']); Chris@76: $context['page_title'] = $txt['split']; Chris@76: } Chris@76: Chris@76: // Split a topic in two topics. Chris@76: function splitTopic($split1_ID_TOPIC, $splitMessages, $new_subject) Chris@76: { Chris@76: global $user_info, $topic, $board, $modSettings, $smcFunc, $txt; Chris@76: Chris@76: // Nothing to split? Chris@76: if (empty($splitMessages)) Chris@76: fatal_lang_error('no_posts_selected', false); Chris@76: Chris@76: // Get some board info. Chris@76: $request = $smcFunc['db_query']('', ' Chris@76: SELECT id_board, approved Chris@76: FROM {db_prefix}topics Chris@76: WHERE id_topic = {int:id_topic} Chris@76: LIMIT 1', Chris@76: array( Chris@76: 'id_topic' => $split1_ID_TOPIC, Chris@76: ) Chris@76: ); Chris@76: list ($id_board, $split1_approved) = $smcFunc['db_fetch_row']($request); Chris@76: $smcFunc['db_free_result']($request); Chris@76: Chris@76: // Find the new first and last not in the list. (old topic) Chris@76: $request = $smcFunc['db_query']('', ' Chris@76: SELECT Chris@76: MIN(m.id_msg) AS myid_first_msg, MAX(m.id_msg) AS myid_last_msg, COUNT(*) AS message_count, m.approved Chris@76: FROM {db_prefix}messages AS m Chris@76: INNER JOIN {db_prefix}topics AS t ON (t.id_topic = {int:id_topic}) Chris@76: WHERE m.id_msg NOT IN ({array_int:no_msg_list}) Chris@76: AND m.id_topic = {int:id_topic} Chris@76: GROUP BY m.approved Chris@76: ORDER BY m.approved DESC Chris@76: LIMIT 2', Chris@76: array( Chris@76: 'id_topic' => $split1_ID_TOPIC, Chris@76: 'no_msg_list' => $splitMessages, Chris@76: ) Chris@76: ); Chris@76: // You can't select ALL the messages! Chris@76: if ($smcFunc['db_num_rows']($request) == 0) Chris@76: fatal_lang_error('selected_all_posts', false); Chris@76: while ($row = $smcFunc['db_fetch_assoc']($request)) Chris@76: { Chris@76: // Get the right first and last message dependant on approved state... Chris@76: if (empty($split1_first_msg) || $row['myid_first_msg'] < $split1_first_msg) Chris@76: $split1_first_msg = $row['myid_first_msg']; Chris@76: if (empty($split1_last_msg) || $row['approved']) Chris@76: $split1_last_msg = $row['myid_last_msg']; Chris@76: Chris@76: // Get the counts correct... Chris@76: if ($row['approved']) Chris@76: { Chris@76: $split1_replies = $row['message_count'] - 1; Chris@76: $split1_unapprovedposts = 0; Chris@76: } Chris@76: else Chris@76: { Chris@76: if (!isset($split1_replies)) Chris@76: $split1_replies = 0; Chris@76: // If the topic isn't approved then num replies must go up by one... as first post wouldn't be counted. Chris@76: elseif (!$split1_approved) Chris@76: $split1_replies++; Chris@76: Chris@76: $split1_unapprovedposts = $row['message_count']; Chris@76: } Chris@76: } Chris@76: $smcFunc['db_free_result']($request); Chris@76: $split1_firstMem = getMsgMemberID($split1_first_msg); Chris@76: $split1_lastMem = getMsgMemberID($split1_last_msg); Chris@76: Chris@76: // Find the first and last in the list. (new topic) Chris@76: $request = $smcFunc['db_query']('', ' Chris@76: SELECT MIN(id_msg) AS myid_first_msg, MAX(id_msg) AS myid_last_msg, COUNT(*) AS message_count, approved Chris@76: FROM {db_prefix}messages Chris@76: WHERE id_msg IN ({array_int:msg_list}) Chris@76: AND id_topic = {int:id_topic} Chris@76: GROUP BY id_topic, approved Chris@76: ORDER BY approved DESC Chris@76: LIMIT 2', Chris@76: array( Chris@76: 'msg_list' => $splitMessages, Chris@76: 'id_topic' => $split1_ID_TOPIC, Chris@76: ) Chris@76: ); Chris@76: while ($row = $smcFunc['db_fetch_assoc']($request)) Chris@76: { Chris@76: // As before get the right first and last message dependant on approved state... Chris@76: if (empty($split2_first_msg) || $row['myid_first_msg'] < $split2_first_msg) Chris@76: $split2_first_msg = $row['myid_first_msg']; Chris@76: if (empty($split2_last_msg) || $row['approved']) Chris@76: $split2_last_msg = $row['myid_last_msg']; Chris@76: Chris@76: // Then do the counts again... Chris@76: if ($row['approved']) Chris@76: { Chris@76: $split2_approved = true; Chris@76: $split2_replies = $row['message_count'] - 1; Chris@76: $split2_unapprovedposts = 0; Chris@76: } Chris@76: else Chris@76: { Chris@76: // Should this one be approved?? Chris@76: if ($split2_first_msg == $row['myid_first_msg']) Chris@76: $split2_approved = false; Chris@76: Chris@76: if (!isset($split2_replies)) Chris@76: $split2_replies = 0; Chris@76: // As before, fix number of replies. Chris@76: elseif (!$split2_approved) Chris@76: $split2_replies++; Chris@76: Chris@76: $split2_unapprovedposts = $row['message_count']; Chris@76: } Chris@76: } Chris@76: $smcFunc['db_free_result']($request); Chris@76: $split2_firstMem = getMsgMemberID($split2_first_msg); Chris@76: $split2_lastMem = getMsgMemberID($split2_last_msg); Chris@76: Chris@76: // No database changes yet, so let's double check to see if everything makes at least a little sense. Chris@76: if ($split1_first_msg <= 0 || $split1_last_msg <= 0 || $split2_first_msg <= 0 || $split2_last_msg <= 0 || $split1_replies < 0 || $split2_replies < 0 || $split1_unapprovedposts < 0 || $split2_unapprovedposts < 0 || !isset($split1_approved) || !isset($split2_approved)) Chris@76: fatal_lang_error('cant_find_messages'); Chris@76: Chris@76: // You cannot split off the first message of a topic. Chris@76: if ($split1_first_msg > $split2_first_msg) Chris@76: fatal_lang_error('split_first_post', false); Chris@76: Chris@76: // We're off to insert the new topic! Use 0 for now to avoid UNIQUE errors. Chris@76: $smcFunc['db_insert']('', Chris@76: '{db_prefix}topics', Chris@76: array( Chris@76: 'id_board' => 'int', Chris@76: 'id_member_started' => 'int', Chris@76: 'id_member_updated' => 'int', Chris@76: 'id_first_msg' => 'int', Chris@76: 'id_last_msg' => 'int', Chris@76: 'num_replies' => 'int', Chris@76: 'unapproved_posts' => 'int', Chris@76: 'approved' => 'int', Chris@76: 'is_sticky' => 'int', Chris@76: ), Chris@76: array( Chris@76: (int) $id_board, $split2_firstMem, $split2_lastMem, 0, Chris@76: 0, $split2_replies, $split2_unapprovedposts, (int) $split2_approved, 0, Chris@76: ), Chris@76: array('id_topic') Chris@76: ); Chris@76: $split2_ID_TOPIC = $smcFunc['db_insert_id']('{db_prefix}topics', 'id_topic'); Chris@76: if ($split2_ID_TOPIC <= 0) Chris@76: fatal_lang_error('cant_insert_topic'); Chris@76: Chris@76: // Move the messages over to the other topic. Chris@76: $new_subject = strtr($smcFunc['htmltrim']($smcFunc['htmlspecialchars']($new_subject)), array("\r" => '', "\n" => '', "\t" => '')); Chris@76: // Check the subject length. Chris@76: if ($smcFunc['strlen']($new_subject) > 100) Chris@76: $new_subject = $smcFunc['substr']($new_subject, 0, 100); Chris@76: // Valid subject? Chris@76: if ($new_subject != '') Chris@76: { Chris@76: $smcFunc['db_query']('', ' Chris@76: UPDATE {db_prefix}messages Chris@76: SET Chris@76: id_topic = {int:id_topic}, Chris@76: subject = CASE WHEN id_msg = {int:split_first_msg} THEN {string:new_subject} ELSE {string:new_subject_replies} END Chris@76: WHERE id_msg IN ({array_int:split_msgs})', Chris@76: array( Chris@76: 'split_msgs' => $splitMessages, Chris@76: 'id_topic' => $split2_ID_TOPIC, Chris@76: 'new_subject' => $new_subject, Chris@76: 'split_first_msg' => $split2_first_msg, Chris@76: 'new_subject_replies' => $txt['response_prefix'] . $new_subject, Chris@76: ) Chris@76: ); Chris@76: Chris@76: // Cache the new topics subject... we can do it now as all the subjects are the same! Chris@76: updateStats('subject', $split2_ID_TOPIC, $new_subject); Chris@76: } Chris@76: Chris@76: // Any associated reported posts better follow... Chris@76: $smcFunc['db_query']('', ' Chris@76: UPDATE {db_prefix}log_reported Chris@76: SET id_topic = {int:id_topic} Chris@76: WHERE id_msg IN ({array_int:split_msgs})', Chris@76: array( Chris@76: 'split_msgs' => $splitMessages, Chris@76: 'id_topic' => $split2_ID_TOPIC, Chris@76: ) Chris@76: ); Chris@76: Chris@76: // Mess with the old topic's first, last, and number of messages. Chris@76: $smcFunc['db_query']('', ' Chris@76: UPDATE {db_prefix}topics Chris@76: SET Chris@76: num_replies = {int:num_replies}, Chris@76: id_first_msg = {int:id_first_msg}, Chris@76: id_last_msg = {int:id_last_msg}, Chris@76: id_member_started = {int:id_member_started}, Chris@76: id_member_updated = {int:id_member_updated}, Chris@76: unapproved_posts = {int:unapproved_posts} Chris@76: WHERE id_topic = {int:id_topic}', Chris@76: array( Chris@76: 'num_replies' => $split1_replies, Chris@76: 'id_first_msg' => $split1_first_msg, Chris@76: 'id_last_msg' => $split1_last_msg, Chris@76: 'id_member_started' => $split1_firstMem, Chris@76: 'id_member_updated' => $split1_lastMem, Chris@76: 'unapproved_posts' => $split1_unapprovedposts, Chris@76: 'id_topic' => $split1_ID_TOPIC, Chris@76: ) Chris@76: ); Chris@76: Chris@76: // Now, put the first/last message back to what they should be. Chris@76: $smcFunc['db_query']('', ' Chris@76: UPDATE {db_prefix}topics Chris@76: SET Chris@76: id_first_msg = {int:id_first_msg}, Chris@76: id_last_msg = {int:id_last_msg} Chris@76: WHERE id_topic = {int:id_topic}', Chris@76: array( Chris@76: 'id_first_msg' => $split2_first_msg, Chris@76: 'id_last_msg' => $split2_last_msg, Chris@76: 'id_topic' => $split2_ID_TOPIC, Chris@76: ) Chris@76: ); Chris@76: Chris@76: // If the new topic isn't approved ensure the first message flags this just in case. Chris@76: if (!$split2_approved) Chris@76: $smcFunc['db_query']('', ' Chris@76: UPDATE {db_prefix}messages Chris@76: SET approved = {int:approved} Chris@76: WHERE id_msg = {int:id_msg} Chris@76: AND id_topic = {int:id_topic}', Chris@76: array( Chris@76: 'approved' => 0, Chris@76: 'id_msg' => $split2_first_msg, Chris@76: 'id_topic' => $split2_ID_TOPIC, Chris@76: ) Chris@76: ); Chris@76: Chris@76: // The board has more topics now (Or more unapproved ones!). Chris@76: $smcFunc['db_query']('', ' Chris@76: UPDATE {db_prefix}boards Chris@76: SET ' . ($split2_approved ? ' Chris@76: num_topics = num_topics + 1' : ' Chris@76: unapproved_topics = unapproved_topics + 1') . ' Chris@76: WHERE id_board = {int:id_board}', Chris@76: array( Chris@76: 'id_board' => $id_board, Chris@76: ) Chris@76: ); Chris@76: Chris@76: // Copy log topic entries. Chris@76: // !!! This should really be chunked. Chris@76: $request = $smcFunc['db_query']('', ' Chris@76: SELECT id_member, id_msg Chris@76: FROM {db_prefix}log_topics Chris@76: WHERE id_topic = {int:id_topic}', Chris@76: array( Chris@76: 'id_topic' => (int) $split1_ID_TOPIC, Chris@76: ) Chris@76: ); Chris@76: if ($smcFunc['db_num_rows']($request) > 0) Chris@76: { Chris@76: $replaceEntries = array(); Chris@76: while ($row = $smcFunc['db_fetch_assoc']($request)) Chris@76: $replaceEntries[] = array($row['id_member'], $split2_ID_TOPIC, $row['id_msg']); Chris@76: Chris@76: $smcFunc['db_insert']('ignore', Chris@76: '{db_prefix}log_topics', Chris@76: array('id_member' => 'int', 'id_topic' => 'int', 'id_msg' => 'int'), Chris@76: $replaceEntries, Chris@76: array('id_member', 'id_topic') Chris@76: ); Chris@76: unset($replaceEntries); Chris@76: } Chris@76: $smcFunc['db_free_result']($request); Chris@76: Chris@76: // Housekeeping. Chris@76: updateStats('topic'); Chris@76: updateLastMessages($id_board); Chris@76: Chris@76: logAction('split', array('topic' => $split1_ID_TOPIC, 'new_topic' => $split2_ID_TOPIC, 'board' => $id_board)); Chris@76: Chris@76: // Notify people that this topic has been split? Chris@76: sendNotifications($split1_ID_TOPIC, 'split'); Chris@76: Chris@76: // Return the ID of the newly created topic. Chris@76: return $split2_ID_TOPIC; Chris@76: } Chris@76: Chris@76: // Merge two topics into one topic... useful if they have the same basic subject. Chris@76: function MergeTopics() Chris@76: { Chris@76: // Load the template.... Chris@76: loadTemplate('SplitTopics'); Chris@76: Chris@76: $subActions = array( Chris@76: 'done' => 'MergeDone', Chris@76: 'execute' => 'MergeExecute', Chris@76: 'index' => 'MergeIndex', Chris@76: 'options' => 'MergeExecute', Chris@76: ); Chris@76: Chris@76: // ?action=mergetopics;sa=LETSBREAKIT won't work, sorry. Chris@76: if (empty($_REQUEST['sa']) || !isset($subActions[$_REQUEST['sa']])) Chris@76: MergeIndex(); Chris@76: else Chris@76: $subActions[$_REQUEST['sa']](); Chris@76: } Chris@76: Chris@76: // Merge two topics together. Chris@76: function MergeIndex() Chris@76: { Chris@76: global $txt, $board, $context, $smcFunc; Chris@76: global $scripturl, $topic, $user_info, $modSettings; Chris@76: Chris@76: if (!isset($_GET['from'])) Chris@76: fatal_lang_error('no_access', false); Chris@76: $_GET['from'] = (int) $_GET['from']; Chris@76: Chris@76: $_REQUEST['targetboard'] = isset($_REQUEST['targetboard']) ? (int) $_REQUEST['targetboard'] : $board; Chris@76: $context['target_board'] = $_REQUEST['targetboard']; Chris@76: Chris@76: // Prepare a handy query bit for approval... Chris@76: if ($modSettings['postmod_active']) Chris@76: { Chris@76: $can_approve_boards = boardsAllowedTo('approve_posts'); Chris@76: $onlyApproved = $can_approve_boards !== array(0) && !in_array($_REQUEST['targetboard'], $can_approve_boards); Chris@76: } Chris@76: else Chris@76: $onlyApproved = false; Chris@76: Chris@76: // How many topics are on this board? (used for paging.) Chris@76: $request = $smcFunc['db_query']('', ' Chris@76: SELECT COUNT(*) Chris@76: FROM {db_prefix}topics AS t Chris@76: WHERE t.id_board = {int:id_board}' . ($onlyApproved ? ' Chris@76: AND t.approved = {int:is_approved}' : ''), Chris@76: array( Chris@76: 'id_board' => $_REQUEST['targetboard'], Chris@76: 'is_approved' => 1, Chris@76: ) Chris@76: ); Chris@76: list ($topiccount) = $smcFunc['db_fetch_row']($request); Chris@76: $smcFunc['db_free_result']($request); Chris@76: Chris@76: // Make the page list. Chris@76: $context['page_index'] = constructPageIndex($scripturl . '?action=mergetopics;from=' . $_GET['from'] . ';targetboard=' . $_REQUEST['targetboard'] . ';board=' . $board . '.%1$d', $_REQUEST['start'], $topiccount, $modSettings['defaultMaxTopics'], true); Chris@76: Chris@76: // Get the topic's subject. Chris@76: $request = $smcFunc['db_query']('', ' Chris@76: SELECT m.subject Chris@76: FROM {db_prefix}topics AS t Chris@76: INNER JOIN {db_prefix}messages AS m ON (m.id_msg = t.id_first_msg) Chris@76: WHERE t.id_topic = {int:id_topic} Chris@76: AND t.id_board = {int:current_board}' . ($onlyApproved ? ' Chris@76: AND t.approved = {int:is_approved}' : '') . ' Chris@76: LIMIT 1', Chris@76: array( Chris@76: 'current_board' => $board, Chris@76: 'id_topic' => $_GET['from'], Chris@76: 'is_approved' => 1, Chris@76: ) Chris@76: ); Chris@76: if ($smcFunc['db_num_rows']($request) == 0) Chris@76: fatal_lang_error('no_board'); Chris@76: list ($subject) = $smcFunc['db_fetch_row']($request); Chris@76: $smcFunc['db_free_result']($request); Chris@76: Chris@76: // Tell the template a few things.. Chris@76: $context['origin_topic'] = $_GET['from']; Chris@76: $context['origin_subject'] = $subject; Chris@76: $context['origin_js_subject'] = addcslashes(addslashes($subject), '/'); Chris@76: $context['page_title'] = $txt['merge']; Chris@76: Chris@76: // Check which boards you have merge permissions on. Chris@76: $merge_boards = boardsAllowedTo('merge_any'); Chris@76: Chris@76: if (empty($merge_boards)) Chris@76: fatal_lang_error('cannot_merge_any', 'user'); Chris@76: Chris@76: // Get a list of boards they can navigate to to merge. Chris@76: $request = $smcFunc['db_query']('order_by_board_order', ' Chris@76: SELECT b.id_board, b.name AS board_name, c.name AS cat_name Chris@76: FROM {db_prefix}boards AS b Chris@76: LEFT JOIN {db_prefix}categories AS c ON (c.id_cat = b.id_cat) Chris@76: WHERE {query_see_board}' . (!in_array(0, $merge_boards) ? ' Chris@76: AND b.id_board IN ({array_int:merge_boards})' : ''), Chris@76: array( Chris@76: 'merge_boards' => $merge_boards, Chris@76: ) Chris@76: ); Chris@76: $context['boards'] = array(); Chris@76: while ($row = $smcFunc['db_fetch_assoc']($request)) Chris@76: $context['boards'][] = array( Chris@76: 'id' => $row['id_board'], Chris@76: 'name' => $row['board_name'], Chris@76: 'category' => $row['cat_name'] Chris@76: ); Chris@76: $smcFunc['db_free_result']($request); Chris@76: Chris@76: // Get some topics to merge it with. Chris@76: $request = $smcFunc['db_query']('', ' Chris@76: SELECT t.id_topic, m.subject, m.id_member, IFNULL(mem.real_name, m.poster_name) AS poster_name Chris@76: FROM {db_prefix}topics AS t Chris@76: INNER JOIN {db_prefix}messages AS m ON (m.id_msg = t.id_first_msg) Chris@76: LEFT JOIN {db_prefix}members AS mem ON (mem.id_member = m.id_member) Chris@76: WHERE t.id_board = {int:id_board} Chris@76: AND t.id_topic != {int:id_topic}' . ($onlyApproved ? ' Chris@76: AND t.approved = {int:is_approved}' : '') . ' Chris@76: ORDER BY {raw:sort} Chris@76: LIMIT {int:offset}, {int:limit}', Chris@76: array( Chris@76: 'id_board' => $_REQUEST['targetboard'], Chris@76: 'id_topic' => $_GET['from'], Chris@76: 'sort' => (!empty($modSettings['enableStickyTopics']) ? 't.is_sticky DESC, ' : '') . 't.id_last_msg DESC', Chris@76: 'offset' => $_REQUEST['start'], Chris@76: 'limit' => $modSettings['defaultMaxTopics'], Chris@76: 'is_approved' => 1, Chris@76: ) Chris@76: ); Chris@76: $context['topics'] = array(); Chris@76: while ($row = $smcFunc['db_fetch_assoc']($request)) Chris@76: { Chris@76: censorText($row['subject']); Chris@76: Chris@76: $context['topics'][] = array( Chris@76: 'id' => $row['id_topic'], Chris@76: 'poster' => array( Chris@76: 'id' => $row['id_member'], Chris@76: 'name' => $row['poster_name'], Chris@76: 'href' => empty($row['id_member']) ? '' : $scripturl . '?action=profile;u=' . $row['id_member'], Chris@76: 'link' => empty($row['id_member']) ? $row['poster_name'] : '' . $row['poster_name'] . '' Chris@76: ), Chris@76: 'subject' => $row['subject'], Chris@76: 'js_subject' => addcslashes(addslashes($row['subject']), '/') Chris@76: ); Chris@76: } Chris@76: $smcFunc['db_free_result']($request); Chris@76: Chris@76: if (empty($context['topics']) && count($context['boards']) <= 1) Chris@76: fatal_lang_error('merge_need_more_topics'); Chris@76: Chris@76: $context['sub_template'] = 'merge'; Chris@76: } Chris@76: Chris@76: // Now that the topic IDs are known, do the proper merging. Chris@76: function MergeExecute($topics = array()) Chris@76: { Chris@76: global $user_info, $txt, $context, $scripturl, $sourcedir; Chris@76: global $smcFunc, $language, $modSettings; Chris@76: Chris@76: // Check the session. Chris@76: checkSession('request'); Chris@76: Chris@76: // Handle URLs from MergeIndex. Chris@76: if (!empty($_GET['from']) && !empty($_GET['to'])) Chris@76: $topics = array((int) $_GET['from'], (int) $_GET['to']); Chris@76: Chris@76: // If we came from a form, the topic IDs came by post. Chris@76: if (!empty($_POST['topics']) && is_array($_POST['topics'])) Chris@76: $topics = $_POST['topics']; Chris@76: Chris@76: // There's nothing to merge with just one topic... Chris@76: if (empty($topics) || !is_array($topics) || count($topics) == 1) Chris@76: fatal_lang_error('merge_need_more_topics'); Chris@76: Chris@76: // Make sure every topic is numeric, or some nasty things could be done with the DB. Chris@76: foreach ($topics as $id => $topic) Chris@76: $topics[$id] = (int) $topic; Chris@76: Chris@76: // Joy of all joys, make sure they're not pi**ing about with unapproved topics they can't see :P Chris@76: if ($modSettings['postmod_active']) Chris@76: $can_approve_boards = boardsAllowedTo('approve_posts'); Chris@76: Chris@76: // Get info about the topics and polls that will be merged. Chris@76: $request = $smcFunc['db_query']('', ' Chris@76: SELECT Chris@76: t.id_topic, t.id_board, t.id_poll, t.num_views, t.is_sticky, t.approved, t.num_replies, t.unapproved_posts, Chris@76: m1.subject, m1.poster_time AS time_started, IFNULL(mem1.id_member, 0) AS id_member_started, IFNULL(mem1.real_name, m1.poster_name) AS name_started, Chris@76: m2.poster_time AS time_updated, IFNULL(mem2.id_member, 0) AS id_member_updated, IFNULL(mem2.real_name, m2.poster_name) AS name_updated Chris@76: FROM {db_prefix}topics AS t Chris@76: INNER JOIN {db_prefix}messages AS m1 ON (m1.id_msg = t.id_first_msg) Chris@76: INNER JOIN {db_prefix}messages AS m2 ON (m2.id_msg = t.id_last_msg) Chris@76: LEFT JOIN {db_prefix}members AS mem1 ON (mem1.id_member = m1.id_member) Chris@76: LEFT JOIN {db_prefix}members AS mem2 ON (mem2.id_member = m2.id_member) Chris@76: WHERE t.id_topic IN ({array_int:topic_list}) Chris@76: ORDER BY t.id_first_msg Chris@76: LIMIT ' . count($topics), Chris@76: array( Chris@76: 'topic_list' => $topics, Chris@76: ) Chris@76: ); Chris@76: if ($smcFunc['db_num_rows']($request) < 2) Chris@76: fatal_lang_error('no_topic_id'); Chris@76: $num_views = 0; Chris@76: $is_sticky = 0; Chris@76: $boardTotals = array(); Chris@76: $boards = array(); Chris@76: $polls = array(); Chris@76: while ($row = $smcFunc['db_fetch_assoc']($request)) Chris@76: { Chris@76: // Make a note for the board counts... Chris@76: if (!isset($boardTotals[$row['id_board']])) Chris@76: $boardTotals[$row['id_board']] = array( Chris@76: 'posts' => 0, Chris@76: 'topics' => 0, Chris@76: 'unapproved_posts' => 0, Chris@76: 'unapproved_topics' => 0 Chris@76: ); Chris@76: Chris@76: // We can't see unapproved topics here? Chris@76: if ($modSettings['postmod_active'] && !$row['approved'] && $can_approve_boards != array(0) && in_array($row['id_board'], $can_approve_boards)) Chris@76: continue; Chris@76: elseif (!$row['approved']) Chris@76: $boardTotals[$row['id_board']]['unapproved_topics']++; Chris@76: else Chris@76: $boardTotals[$row['id_board']]['topics']++; Chris@76: Chris@76: $boardTotals[$row['id_board']]['unapproved_posts'] += $row['unapproved_posts']; Chris@76: $boardTotals[$row['id_board']]['posts'] += $row['num_replies'] + ($row['approved'] ? 1 : 0); Chris@76: Chris@76: $topic_data[$row['id_topic']] = array( Chris@76: 'id' => $row['id_topic'], Chris@76: 'board' => $row['id_board'], Chris@76: 'poll' => $row['id_poll'], Chris@76: 'num_views' => $row['num_views'], Chris@76: 'subject' => $row['subject'], Chris@76: 'started' => array( Chris@76: 'time' => timeformat($row['time_started']), Chris@76: 'timestamp' => forum_time(true, $row['time_started']), Chris@76: 'href' => empty($row['id_member_started']) ? '' : $scripturl . '?action=profile;u=' . $row['id_member_started'], Chris@76: 'link' => empty($row['id_member_started']) ? $row['name_started'] : '' . $row['name_started'] . '' Chris@76: ), Chris@76: 'updated' => array( Chris@76: 'time' => timeformat($row['time_updated']), Chris@76: 'timestamp' => forum_time(true, $row['time_updated']), Chris@76: 'href' => empty($row['id_member_updated']) ? '' : $scripturl . '?action=profile;u=' . $row['id_member_updated'], Chris@76: 'link' => empty($row['id_member_updated']) ? $row['name_updated'] : '' . $row['name_updated'] . '' Chris@76: ) Chris@76: ); Chris@76: $num_views += $row['num_views']; Chris@76: $boards[] = $row['id_board']; Chris@76: Chris@76: // If there's no poll, id_poll == 0... Chris@76: if ($row['id_poll'] > 0) Chris@76: $polls[] = $row['id_poll']; Chris@76: // Store the id_topic with the lowest id_first_msg. Chris@76: if (empty($firstTopic)) Chris@76: $firstTopic = $row['id_topic']; Chris@76: Chris@76: $is_sticky = max($is_sticky, $row['is_sticky']); Chris@76: } Chris@76: $smcFunc['db_free_result']($request); Chris@76: Chris@76: // If we didn't get any topics then they've been messing with unapproved stuff. Chris@76: if (empty($topic_data)) Chris@76: fatal_lang_error('no_topic_id'); Chris@76: Chris@76: $boards = array_values(array_unique($boards)); Chris@76: Chris@76: // The parameters of MergeExecute were set, so this must've been an internal call. Chris@76: if (!empty($topics)) Chris@76: { Chris@76: isAllowedTo('merge_any', $boards); Chris@76: loadTemplate('SplitTopics'); Chris@76: } Chris@76: Chris@76: // Get the boards a user is allowed to merge in. Chris@76: $merge_boards = boardsAllowedTo('merge_any'); Chris@76: if (empty($merge_boards)) Chris@76: fatal_lang_error('cannot_merge_any', 'user'); Chris@76: Chris@76: // Make sure they can see all boards.... Chris@76: $request = $smcFunc['db_query']('', ' Chris@76: SELECT b.id_board Chris@76: FROM {db_prefix}boards AS b Chris@76: WHERE b.id_board IN ({array_int:boards}) Chris@76: AND {query_see_board}' . (!in_array(0, $merge_boards) ? ' Chris@76: AND b.id_board IN ({array_int:merge_boards})' : '') . ' Chris@76: LIMIT ' . count($boards), Chris@76: array( Chris@76: 'boards' => $boards, Chris@76: 'merge_boards' => $merge_boards, Chris@76: ) Chris@76: ); Chris@76: // If the number of boards that's in the output isn't exactly the same as we've put in there, you're in trouble. Chris@76: if ($smcFunc['db_num_rows']($request) != count($boards)) Chris@76: fatal_lang_error('no_board'); Chris@76: $smcFunc['db_free_result']($request); Chris@76: Chris@76: if (empty($_REQUEST['sa']) || $_REQUEST['sa'] == 'options') Chris@76: { Chris@76: if (count($polls) > 1) Chris@76: { Chris@76: $request = $smcFunc['db_query']('', ' Chris@76: SELECT t.id_topic, t.id_poll, m.subject, p.question Chris@76: FROM {db_prefix}polls AS p Chris@76: INNER JOIN {db_prefix}topics AS t ON (t.id_poll = p.id_poll) Chris@76: INNER JOIN {db_prefix}messages AS m ON (m.id_msg = t.id_first_msg) Chris@76: WHERE p.id_poll IN ({array_int:polls}) Chris@76: LIMIT ' . count($polls), Chris@76: array( Chris@76: 'polls' => $polls, Chris@76: ) Chris@76: ); Chris@76: while ($row = $smcFunc['db_fetch_assoc']($request)) Chris@76: $context['polls'][] = array( Chris@76: 'id' => $row['id_poll'], Chris@76: 'topic' => array( Chris@76: 'id' => $row['id_topic'], Chris@76: 'subject' => $row['subject'] Chris@76: ), Chris@76: 'question' => $row['question'], Chris@76: 'selected' => $row['id_topic'] == $firstTopic Chris@76: ); Chris@76: $smcFunc['db_free_result']($request); Chris@76: } Chris@76: if (count($boards) > 1) Chris@76: { Chris@76: $request = $smcFunc['db_query']('', ' Chris@76: SELECT id_board, name Chris@76: FROM {db_prefix}boards Chris@76: WHERE id_board IN ({array_int:boards}) Chris@76: ORDER BY name Chris@76: LIMIT ' . count($boards), Chris@76: array( Chris@76: 'boards' => $boards, Chris@76: ) Chris@76: ); Chris@76: while ($row = $smcFunc['db_fetch_assoc']($request)) Chris@76: $context['boards'][] = array( Chris@76: 'id' => $row['id_board'], Chris@76: 'name' => $row['name'], Chris@76: 'selected' => $row['id_board'] == $topic_data[$firstTopic]['board'] Chris@76: ); Chris@76: $smcFunc['db_free_result']($request); Chris@76: } Chris@76: Chris@76: $context['topics'] = $topic_data; Chris@76: foreach ($topic_data as $id => $topic) Chris@76: $context['topics'][$id]['selected'] = $topic['id'] == $firstTopic; Chris@76: Chris@76: $context['page_title'] = $txt['merge']; Chris@76: $context['sub_template'] = 'merge_extra_options'; Chris@76: return; Chris@76: } Chris@76: Chris@76: // Determine target board. Chris@76: $target_board = count($boards) > 1 ? (int) $_REQUEST['board'] : $boards[0]; Chris@76: if (!in_array($target_board, $boards)) Chris@76: fatal_lang_error('no_board'); Chris@76: Chris@76: // Determine which poll will survive and which polls won't. Chris@76: $target_poll = count($polls) > 1 ? (int) $_POST['poll'] : (count($polls) == 1 ? $polls[0] : 0); Chris@76: if ($target_poll > 0 && !in_array($target_poll, $polls)) Chris@76: fatal_lang_error('no_access', false); Chris@76: $deleted_polls = empty($target_poll) ? $polls : array_diff($polls, array($target_poll)); Chris@76: Chris@76: // Determine the subject of the newly merged topic - was a custom subject specified? Chris@76: if (empty($_POST['subject']) && isset($_POST['custom_subject']) && $_POST['custom_subject'] != '') Chris@76: { Chris@76: $target_subject = strtr($smcFunc['htmltrim']($smcFunc['htmlspecialchars']($_POST['custom_subject'])), array("\r" => '', "\n" => '', "\t" => '')); Chris@76: // Keep checking the length. Chris@76: if ($smcFunc['strlen']($target_subject) > 100) Chris@76: $target_subject = $smcFunc['substr']($target_subject, 0, 100); Chris@76: Chris@76: // Nothing left - odd but pick the first topics subject. Chris@76: if ($target_subject == '') Chris@76: $target_subject = $topic_data[$firstTopic]['subject']; Chris@76: } Chris@76: // A subject was selected from the list. Chris@76: elseif (!empty($topic_data[(int) $_POST['subject']]['subject'])) Chris@76: $target_subject = $topic_data[(int) $_POST['subject']]['subject']; Chris@76: // Nothing worked? Just take the subject of the first message. Chris@76: else Chris@76: $target_subject = $topic_data[$firstTopic]['subject']; Chris@76: Chris@76: // Get the first and last message and the number of messages.... Chris@76: $request = $smcFunc['db_query']('', ' Chris@76: SELECT approved, MIN(id_msg) AS first_msg, MAX(id_msg) AS last_msg, COUNT(*) AS message_count Chris@76: FROM {db_prefix}messages Chris@76: WHERE id_topic IN ({array_int:topics}) Chris@76: GROUP BY approved Chris@76: ORDER BY approved DESC', Chris@76: array( Chris@76: 'topics' => $topics, Chris@76: ) Chris@76: ); Chris@76: $topic_approved = 1; Chris@76: while ($row = $smcFunc['db_fetch_assoc']($request)) Chris@76: { Chris@76: // If this is approved, or is fully unapproved. Chris@76: if ($row['approved'] || !isset($first_msg)) Chris@76: { Chris@76: $first_msg = $row['first_msg']; Chris@76: $last_msg = $row['last_msg']; Chris@76: if ($row['approved']) Chris@76: { Chris@76: $num_replies = $row['message_count'] - 1; Chris@76: $num_unapproved = 0; Chris@76: } Chris@76: else Chris@76: { Chris@76: $topic_approved = 0; Chris@76: $num_replies = 0; Chris@76: $num_unapproved = $row['message_count']; Chris@76: } Chris@76: } Chris@76: else Chris@76: { Chris@76: // If this has a lower first_msg then the first post is not approved and hence the number of replies was wrong! Chris@76: if ($first_msg > $row['first_msg']) Chris@76: { Chris@76: $first_msg = $row['first_msg']; Chris@76: $num_replies++; Chris@76: $topic_approved = 0; Chris@76: } Chris@76: $num_unapproved = $row['message_count']; Chris@76: } Chris@76: } Chris@76: $smcFunc['db_free_result']($request); Chris@76: Chris@76: // Ensure we have a board stat for the target board. Chris@76: if (!isset($boardTotals[$target_board])) Chris@76: { Chris@76: $boardTotals[$target_board] = array( Chris@76: 'posts' => 0, Chris@76: 'topics' => 0, Chris@76: 'unapproved_posts' => 0, Chris@76: 'unapproved_topics' => 0 Chris@76: ); Chris@76: } Chris@76: Chris@76: // Fix the topic count stuff depending on what the new one counts as. Chris@76: if ($topic_approved) Chris@76: $boardTotals[$target_board]['topics']--; Chris@76: else Chris@76: $boardTotals[$target_board]['unapproved_topics']--; Chris@76: Chris@76: $boardTotals[$target_board]['unapproved_posts'] -= $num_unapproved; Chris@76: $boardTotals[$target_board]['posts'] -= $topic_approved ? $num_replies + 1 : $num_replies; Chris@76: Chris@76: // Get the member ID of the first and last message. Chris@76: $request = $smcFunc['db_query']('', ' Chris@76: SELECT id_member Chris@76: FROM {db_prefix}messages Chris@76: WHERE id_msg IN ({int:first_msg}, {int:last_msg}) Chris@76: ORDER BY id_msg Chris@76: LIMIT 2', Chris@76: array( Chris@76: 'first_msg' => $first_msg, Chris@76: 'last_msg' => $last_msg, Chris@76: ) Chris@76: ); Chris@76: list ($member_started) = $smcFunc['db_fetch_row']($request); Chris@76: list ($member_updated) = $smcFunc['db_fetch_row']($request); Chris@76: // First and last message are the same, so only row was returned. Chris@76: if ($member_updated === NULL) Chris@76: $member_updated = $member_started; Chris@76: Chris@76: $smcFunc['db_free_result']($request); Chris@76: Chris@76: // Assign the first topic ID to be the merged topic. Chris@76: $id_topic = min($topics); Chris@76: Chris@76: // Delete the remaining topics. Chris@76: $deleted_topics = array_diff($topics, array($id_topic)); Chris@76: $smcFunc['db_query']('', ' Chris@76: DELETE FROM {db_prefix}topics Chris@76: WHERE id_topic IN ({array_int:deleted_topics})', Chris@76: array( Chris@76: 'deleted_topics' => $deleted_topics, Chris@76: ) Chris@76: ); Chris@76: $smcFunc['db_query']('', ' Chris@76: DELETE FROM {db_prefix}log_search_subjects Chris@76: WHERE id_topic IN ({array_int:deleted_topics})', Chris@76: array( Chris@76: 'deleted_topics' => $deleted_topics, Chris@76: ) Chris@76: ); Chris@76: Chris@76: // Asssign the properties of the newly merged topic. Chris@76: $smcFunc['db_query']('', ' Chris@76: UPDATE {db_prefix}topics Chris@76: SET Chris@76: id_board = {int:id_board}, Chris@76: id_member_started = {int:id_member_started}, Chris@76: id_member_updated = {int:id_member_updated}, Chris@76: id_first_msg = {int:id_first_msg}, Chris@76: id_last_msg = {int:id_last_msg}, Chris@76: id_poll = {int:id_poll}, Chris@76: num_replies = {int:num_replies}, Chris@76: unapproved_posts = {int:unapproved_posts}, Chris@76: num_views = {int:num_views}, Chris@76: is_sticky = {int:is_sticky}, Chris@76: approved = {int:approved} Chris@76: WHERE id_topic = {int:id_topic}', Chris@76: array( Chris@76: 'id_board' => $target_board, Chris@76: 'is_sticky' => $is_sticky, Chris@76: 'approved' => $topic_approved, Chris@76: 'id_topic' => $id_topic, Chris@76: 'id_member_started' => $member_started, Chris@76: 'id_member_updated' => $member_updated, Chris@76: 'id_first_msg' => $first_msg, Chris@76: 'id_last_msg' => $last_msg, Chris@76: 'id_poll' => $target_poll, Chris@76: 'num_replies' => $num_replies, Chris@76: 'unapproved_posts' => $num_unapproved, Chris@76: 'num_views' => $num_views, Chris@76: ) Chris@76: ); Chris@76: Chris@76: // Grab the response prefix (like 'Re: ') in the default forum language. Chris@76: if (!isset($context['response_prefix']) && !($context['response_prefix'] = cache_get_data('response_prefix'))) Chris@76: { Chris@76: if ($language === $user_info['language']) Chris@76: $context['response_prefix'] = $txt['response_prefix']; Chris@76: else Chris@76: { Chris@76: loadLanguage('index', $language, false); Chris@76: $context['response_prefix'] = $txt['response_prefix']; Chris@76: loadLanguage('index'); Chris@76: } Chris@76: cache_put_data('response_prefix', $context['response_prefix'], 600); Chris@76: } Chris@76: Chris@76: // Change the topic IDs of all messages that will be merged. Also adjust subjects if 'enforce subject' was checked. Chris@76: $smcFunc['db_query']('', ' Chris@76: UPDATE {db_prefix}messages Chris@76: SET Chris@76: id_topic = {int:id_topic}, Chris@76: id_board = {int:target_board}' . (empty($_POST['enforce_subject']) ? '' : ', Chris@76: subject = {string:subject}') . ' Chris@76: WHERE id_topic IN ({array_int:topic_list})', Chris@76: array( Chris@76: 'topic_list' => $topics, Chris@76: 'id_topic' => $id_topic, Chris@76: 'target_board' => $target_board, Chris@76: 'subject' => $context['response_prefix'] . $target_subject, Chris@76: ) Chris@76: ); Chris@76: Chris@76: // Any reported posts should reflect the new board. Chris@76: $smcFunc['db_query']('', ' Chris@76: UPDATE {db_prefix}log_reported Chris@76: SET Chris@76: id_topic = {int:id_topic}, Chris@76: id_board = {int:target_board} Chris@76: WHERE id_topic IN ({array_int:topics_list})', Chris@76: array( Chris@76: 'topics_list' => $topics, Chris@76: 'id_topic' => $id_topic, Chris@76: 'target_board' => $target_board, Chris@76: ) Chris@76: ); Chris@76: Chris@76: // Change the subject of the first message... Chris@76: $smcFunc['db_query']('', ' Chris@76: UPDATE {db_prefix}messages Chris@76: SET subject = {string:target_subject} Chris@76: WHERE id_msg = {int:first_msg}', Chris@76: array( Chris@76: 'first_msg' => $first_msg, Chris@76: 'target_subject' => $target_subject, Chris@76: ) Chris@76: ); Chris@76: Chris@76: // Adjust all calendar events to point to the new topic. Chris@76: $smcFunc['db_query']('', ' Chris@76: UPDATE {db_prefix}calendar Chris@76: SET Chris@76: id_topic = {int:id_topic}, Chris@76: id_board = {int:target_board} Chris@76: WHERE id_topic IN ({array_int:deleted_topics})', Chris@76: array( Chris@76: 'deleted_topics' => $deleted_topics, Chris@76: 'id_topic' => $id_topic, Chris@76: 'target_board' => $target_board, Chris@76: ) Chris@76: ); Chris@76: Chris@76: // Merge log topic entries. Chris@76: $request = $smcFunc['db_query']('', ' Chris@76: SELECT id_member, MIN(id_msg) AS new_id_msg Chris@76: FROM {db_prefix}log_topics Chris@76: WHERE id_topic IN ({array_int:topics}) Chris@76: GROUP BY id_member', Chris@76: array( Chris@76: 'topics' => $topics, Chris@76: ) Chris@76: ); Chris@76: if ($smcFunc['db_num_rows']($request) > 0) Chris@76: { Chris@76: $replaceEntries = array(); Chris@76: while ($row = $smcFunc['db_fetch_assoc']($request)) Chris@76: $replaceEntries[] = array($row['id_member'], $id_topic, $row['new_id_msg']); Chris@76: Chris@76: $smcFunc['db_insert']('replace', Chris@76: '{db_prefix}log_topics', Chris@76: array('id_member' => 'int', 'id_topic' => 'int', 'id_msg' => 'int'), Chris@76: $replaceEntries, Chris@76: array('id_member', 'id_topic') Chris@76: ); Chris@76: unset($replaceEntries); Chris@76: Chris@76: // Get rid of the old log entries. Chris@76: $smcFunc['db_query']('', ' Chris@76: DELETE FROM {db_prefix}log_topics Chris@76: WHERE id_topic IN ({array_int:deleted_topics})', Chris@76: array( Chris@76: 'deleted_topics' => $deleted_topics, Chris@76: ) Chris@76: ); Chris@76: } Chris@76: $smcFunc['db_free_result']($request); Chris@76: Chris@76: // Merge topic notifications. Chris@76: $notifications = isset($_POST['notifications']) && is_array($_POST['notifications']) ? array_intersect($topics, $_POST['notifications']) : array(); Chris@76: if (!empty($notifications)) Chris@76: { Chris@76: $request = $smcFunc['db_query']('', ' Chris@76: SELECT id_member, MAX(sent) AS sent Chris@76: FROM {db_prefix}log_notify Chris@76: WHERE id_topic IN ({array_int:topics_list}) Chris@76: GROUP BY id_member', Chris@76: array( Chris@76: 'topics_list' => $notifications, Chris@76: ) Chris@76: ); Chris@76: if ($smcFunc['db_num_rows']($request) > 0) Chris@76: { Chris@76: $replaceEntries = array(); Chris@76: while ($row = $smcFunc['db_fetch_assoc']($request)) Chris@76: $replaceEntries[] = array($row['id_member'], $id_topic, 0, $row['sent']); Chris@76: Chris@76: $smcFunc['db_insert']('replace', Chris@76: '{db_prefix}log_notify', Chris@76: array('id_member' => 'int', 'id_topic' => 'int', 'id_board' => 'int', 'sent' => 'int'), Chris@76: $replaceEntries, Chris@76: array('id_member', 'id_topic', 'id_board') Chris@76: ); Chris@76: unset($replaceEntries); Chris@76: Chris@76: $smcFunc['db_query']('', ' Chris@76: DELETE FROM {db_prefix}log_topics Chris@76: WHERE id_topic IN ({array_int:deleted_topics})', Chris@76: array( Chris@76: 'deleted_topics' => $deleted_topics, Chris@76: ) Chris@76: ); Chris@76: } Chris@76: $smcFunc['db_free_result']($request); Chris@76: } Chris@76: Chris@76: // Get rid of the redundant polls. Chris@76: if (!empty($deleted_polls)) Chris@76: { Chris@76: $smcFunc['db_query']('', ' Chris@76: DELETE FROM {db_prefix}polls Chris@76: WHERE id_poll IN ({array_int:deleted_polls})', Chris@76: array( Chris@76: 'deleted_polls' => $deleted_polls, Chris@76: ) Chris@76: ); Chris@76: $smcFunc['db_query']('', ' Chris@76: DELETE FROM {db_prefix}poll_choices Chris@76: WHERE id_poll IN ({array_int:deleted_polls})', Chris@76: array( Chris@76: 'deleted_polls' => $deleted_polls, Chris@76: ) Chris@76: ); Chris@76: $smcFunc['db_query']('', ' Chris@76: DELETE FROM {db_prefix}log_polls Chris@76: WHERE id_poll IN ({array_int:deleted_polls})', Chris@76: array( Chris@76: 'deleted_polls' => $deleted_polls, Chris@76: ) Chris@76: ); Chris@76: } Chris@76: Chris@76: // Cycle through each board... Chris@76: foreach ($boardTotals as $id_board => $stats) Chris@76: { Chris@76: $smcFunc['db_query']('', ' Chris@76: UPDATE {db_prefix}boards Chris@76: SET Chris@76: num_topics = CASE WHEN {int:topics} > num_topics THEN 0 ELSE num_topics - {int:topics} END, Chris@76: unapproved_topics = CASE WHEN {int:unapproved_topics} > unapproved_topics THEN 0 ELSE unapproved_topics - {int:unapproved_topics} END, Chris@76: num_posts = CASE WHEN {int:posts} > num_posts THEN 0 ELSE num_posts - {int:posts} END, Chris@76: unapproved_posts = CASE WHEN {int:unapproved_posts} > unapproved_posts THEN 0 ELSE unapproved_posts - {int:unapproved_posts} END Chris@76: WHERE id_board = {int:id_board}', Chris@76: array( Chris@76: 'id_board' => $id_board, Chris@76: 'topics' => $stats['topics'], Chris@76: 'unapproved_topics' => $stats['unapproved_topics'], Chris@76: 'posts' => $stats['posts'], Chris@76: 'unapproved_posts' => $stats['unapproved_posts'], Chris@76: ) Chris@76: ); Chris@76: } Chris@76: Chris@76: // Determine the board the final topic resides in Chris@76: $request = $smcFunc['db_query']('', ' Chris@76: SELECT id_board Chris@76: FROM {db_prefix}topics Chris@76: WHERE id_topic = {int:id_topic} Chris@76: LIMIT 1', Chris@76: array( Chris@76: 'id_topic' => $id_topic, Chris@76: ) Chris@76: ); Chris@76: list($id_board) = $smcFunc['db_fetch_row']($request); Chris@76: $smcFunc['db_free_result']($request); Chris@76: Chris@76: require_once($sourcedir . '/Subs-Post.php'); Chris@76: Chris@76: // Update all the statistics. Chris@76: updateStats('topic'); Chris@76: updateStats('subject', $id_topic, $target_subject); Chris@76: updateLastMessages($boards); Chris@76: Chris@76: logAction('merge', array('topic' => $id_topic, 'board' => $id_board)); Chris@76: Chris@76: // Notify people that these topics have been merged? Chris@76: sendNotifications($id_topic, 'merge'); Chris@76: Chris@76: // Send them to the all done page. Chris@76: redirectexit('action=mergetopics;sa=done;to=' . $id_topic . ';targetboard=' . $target_board); Chris@76: } Chris@76: Chris@76: // Tell the user the move was done properly. Chris@76: function MergeDone() Chris@76: { Chris@76: global $txt, $context; Chris@76: Chris@76: // Make sure the template knows everything... Chris@76: $context['target_board'] = (int) $_GET['targetboard']; Chris@76: $context['target_topic'] = (int) $_GET['to']; Chris@76: Chris@76: $context['page_title'] = $txt['merge']; Chris@76: $context['sub_template'] = 'merge_done'; Chris@76: } Chris@76: Chris@76: ?>