<?php /** * This file contains liking posts and displaying the list of who liked a post. * * Simple Machines Forum (SMF) * * @package SMF * @author Simple Machines http://www.simplemachines.org * @copyright 2013 Simple Machines and individual contributors * @license http://www.simplemachines.org/about/smf/license.php BSD * * @version 2.1 Alpha 1 */ if (!defined('SMF')) die('No direct access...'); /** * The main handler. Verifies permissions (whether the user can see the content in question) * before either liking/unliking or spitting out the list of likers. * Accessed from index.php?action=likes */ function Likes() { global $context, $smcFunc; // Zerothly, they did indicate some kind of content to like, right? $like_type = isset($_GET['ltype']) ? $_GET['ltype'] : ''; preg_match('~^([a-z0-9\-\_]{1,6})~i', $like_type, $matches); $like_type = isset($matches[1]) ? $matches[1] : ''; $like_content = isset($_GET['like']) ? (int) $_GET['like'] : 0; if ($like_type == '' || $like_content <= 0) fatal_lang_error(isset($_GET['view']) ? 'cannot_view_likes' : 'cannot_like_content', false); // First we need to verify if the user can see the type of content or not. This is set up to be extensible, // so we'll check for the one type we do know about, and if it's not that, we'll defer to any hooks. if ($like_type == 'msg') { // So we're doing something off a like. We need to verify that it exists, and that the current user can see it. // Fortunately for messages, this is quite easy to do - and we'll get the topic id while we're at it, because // we need this later for other things. $request = $smcFunc['db_query']('', ' SELECT m.id_topic FROM {db_prefix}messages AS m INNER JOIN {db_prefix}boards AS b ON (m.id_board = b.id_board) WHERE {query_see_board} AND m.id_msg = {int:msg}', array( 'msg' => $like_content, ) ); if ($smcFunc['db_num_rows']($request) == 1) list ($id_topic) = $smcFunc['db_fetch_row']($request); $smcFunc['db_free_result']($request); if (empty($id_topic)) fatal_lang_error(isset($_GET['view']) ? 'cannot_view_likes' : 'cannot_like_content', false); // So we know what topic it's in and more importantly we know the user can see it. // If we're not viewing, we need some info set up. if (!isset($_GET['view'])) { $context['flush_cache'] = 'likes_topic_' . $id_topic . '_' . $context['user']['id']; $context['redirect_from_like'] = 'topic=' . $id_topic . '.msg' . $like_content . '#msg' . $like_content; add_integration_function('integrate_issue_like', 'msg_issue_like', '', false); } } else { // Modders: This will give you whatever the user offers up in terms of liking, e.g. $like_type=msg, $like_content=1 // When you hook this, check $like_type first. If it is not something your mod worries about, return false. // Otherwise, determine (however you need to) that the user can see the relevant liked content (and it exists). // If the user cannot see it, return false. If the user can see it and can like it, you MUST return your $like_type back. // See also issueLike() for further notes. $can_like = call_integration_hook('integrate_valid_likes', array($like_type, $like_content)); $found = false; if (!empty($can_like)) { $can_like = (array) $can_like; foreach ($can_like as $result) { if ($result !== false) { $like_type = $result; $found = true; break; } } } if (!$found) fatal_lang_error(isset($_GET['view']) ? 'cannot_view_likes' : 'cannot_like_content', false); } // So at this point, whatever type of like the user supplied and the item of content in question, // we know it exists, now we need to figure out what we're doing with that. if (isset($_GET['view'])) viewLikes($like_type, $like_content); else { // Only registered users may actually like content. is_not_guest(); checkSession('get'); issueLike($like_type, $like_content); } } /** * @param string $like_type The type of content being liked * @param integer $like_content The ID of the content being liked */ function issueLike($like_type, $like_content) { global $context, $smcFunc; // Do we already like this? $request = $smcFunc['db_query']('', ' SELECT content_id, content_type, id_member FROM {db_prefix}user_likes WHERE content_id = {int:like_content} AND content_type = {string:like_type} AND id_member = {int:id_member}', array( 'like_content' => $like_content, 'like_type' => $like_type, 'id_member' => $context['user']['id'], ) ); $already_liked = $smcFunc['db_num_rows']($request) != 0; $smcFunc['db_free_result']($request); if ($already_liked) { $smcFunc['db_query']('', ' DELETE FROM {db_prefix}user_likes WHERE content_id = {int:like_content} AND content_type = {string:like_type} AND id_member = {int:id_member}', array( 'like_content' => $like_content, 'like_type' => $like_type, 'id_member' => $context['user']['id'], ) ); } else { // Insert the like. $smcFunc['db_insert']('insert', '{db_prefix}user_likes', array('content_id' => 'int', 'content_type' => 'string-6', 'id_member' => 'int', 'like_time' => 'int'), array($like_content, $like_type, $context['user']['id'], time()), array('content_id', 'content_type', 'id_member') ); // Add a background task to process sending alerts. $smcFunc['db_insert']('insert', '{db_prefix}background_tasks', array('task_file' => 'string', 'task_class' => 'string', 'task_data' => 'string', 'claimed_time' => 'int'), array('$sourcedir/tasks/Likes-Notify.php', 'Likes_Notify_Background', serialize(array( 'content_id' => $like_content, 'content_type' => $like_type, 'sender_id' => $context['user']['id'], 'sender_name' => $context['user']['name'], 'time' => time(), )), 0), array('id_task') ); } // Now, how many people like this content now? We *could* just +1 / -1 the relevant container but that has proven to become unstable. $request = $smcFunc['db_query']('', ' SELECT COUNT(id_member) FROM {db_prefix}user_likes WHERE content_id = {int:like_content} AND content_type = {string:like_type}', array( 'like_content' => $like_content, 'like_type' => $like_type, ) ); list ($num_likes) = $smcFunc['db_fetch_row']($request); $smcFunc['db_free_result']($request); // Sometimes there might be other things that need updating after we do this like. call_integration_hook('integrate_issue_like', array($like_type, $like_content, $num_likes)); // Now some clean up. This is provided here for any like handlers that want to do any cache flushing. // This way a like handler doesn't need to explicitly declare anything in integrate_issue_like, but do so // in integrate_valid_likes where it absolutely has to exist. if (!empty($context['flush_cache'])) cache_put_data($context['flush_cache'], null); if (!empty($context['redirect_from_like'])) redirectexit($context['redirect_from_like']); else redirectexit(); // Because we have to go *somewhere*. } /** * Callback attached to integrate_issue_like. * Partly it indicates how it's supposed to work and partly it deals with updating the count of likes * attached to this message now. * @param string $like_type The type of content being liked - should always be 'msg' * @param int $like_content The ID of the post being liked * @param int $num_likes The number of likes this message has received */ function msg_issue_like($like_type, $like_content, $num_likes) { global $smcFunc; if ($like_type !== 'msg') return; $smcFunc['db_query']('', ' UPDATE {db_prefix}messages SET likes = {int:num_likes} WHERE id_msg = {int:id_msg}', array( 'id_msg' => $like_content, 'num_likes' => $num_likes, ) ); // Note that we could just as easily have cleared the cache here, or set up the redirection address // but if your liked content doesn't need to do anything other than have the record in smf_user_likes, // there's no point in creating another function unnecessarily. } /** * This is for viewing the people who liked a thing. * Accessed from index.php?action=likes;view and should generally load in a popup. * We use a template for this in case themers want to style it. * @param string $like_type The type of content being liked * @param integer $like_content The ID of the content being liked */ function viewLikes($like_type, $like_content) { global $smcFunc, $txt, $context, $memberContext; // Firstly, load what we need. We already know we can see this, so that's something. $context['likers'] = array(); $request = $smcFunc['db_query']('', ' SELECT id_member, like_time FROM {db_prefix}user_likes WHERE content_id = {int:like_content} AND content_type = {string:like_type} ORDER BY like_time DESC', array( 'like_content' => $like_content, 'like_type' => $like_type, ) ); while ($row = $smcFunc['db_fetch_assoc']($request)) $context['likers'][$row['id_member']] = array('timestamp' => $row['like_time']); // Now to get member data, including avatars and so on. $members = array_keys($context['likers']); $loaded = loadMemberData($members); if (count($loaded) != count($members)) { $members = array_diff($members, $loaded); foreach ($members as $not_loaded) unset ($context['likers'][$not_loaded]); } foreach ($context['likers'] as $liker => $dummy) { $loaded = loadMemberContext($liker); if (!$loaded) { unset ($context['likers'][$liker]); continue; } $context['likers'][$liker]['profile'] = &$memberContext[$liker]; $context['likers'][$liker]['time'] = timeformat($dummy['timestamp']); } $count = count($context['likers']); $title_base = isset($txt['likes_' . $count]) ? 'likes_' . $count : 'likes_n'; $context['page_title'] = strip_tags(sprintf($txt[$title_base], '', comma_format($count))); // Lastly, setting up for display loadTemplate('Likes'); loadLanguage('Help'); // for the close window button $context['template_layers'] = array(); $context['sub_template'] = 'popup'; } /** * What's this? I dunno, what are you talking about? Never seen this before, nope. No sir. */ function BookOfUnknown() { global $context, $scripturl; echo '<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> <html xmlns="http://www.w3.org/1999/xhtml"', $context['right_to_left'] ? ' dir="rtl"' : '', '> <head> <title>The Book of Unknown, ', @$_GET['verse'] == '2:18' ? '2:18' : '4:16', '</title> <style type="text/css"> em { font-size: 1.3em; line-height: 0; } </style> </head> <body style="background-color: #444455; color: white; font-style: italic; font-family: serif;"> <div style="margin-top: 12%; font-size: 1.1em; line-height: 1.4; text-align: center;">'; if (!isset($_GET['verse']) || ($_GET['verse'] != '2:18' && $_GET['verse'] != '22:1-2')) $_GET['verse'] = '4:16'; if ($_GET['verse'] == '2:18') echo ' Woe, it was that his name wasn\'t <em>known</em>, that he came in mystery, and was recognized by none. And it became to be in those days <em>something</em>. Something not yet <em id="unknown" name="[Unknown]">unknown</em> to mankind. And thus what was to be known the <em>secret project</em> began into its existence. Henceforth the opposition was only <em>weary</em> and <em>fearful</em>, for now their match was at arms against them.'; elseif ($_GET['verse'] == '4:16') echo ' And it came to pass that the <em>unbelievers</em> dwindled in number and saw rise of many <em>proselytizers</em>, and the opposition found fear in the face of the <em>x</em> and the <em>j</em> while those who stood with the <em>something</em> grew stronger and came together. Still, this was only the <em>beginning</em>, and what lay in the future was <em id="unknown" name="[Unknown]">unknown</em> to all, even those on the right side.'; elseif ($_GET['verse'] == '22:1-2') echo ' <p>Now <em>behold</em>, that which was once the secret project was <em id="unknown" name="[Unknown]">unknown</em> no longer. Alas, it needed more than <em>only one</em>, but yet even thought otherwise. It became that the opposition <em>rumored</em> and lied, but still to no avail. Their match, though not <em>perfect</em>, had them outdone.</p> <p style="margin: 2ex 1ex 0 1ex; font-size: 1.05em; line-height: 1.5; text-align: center;">Let it continue. <em>The end</em>.</p>'; echo ' </div> <div style="margin-top: 2ex; font-size: 2em; text-align: right;">'; if ($_GET['verse'] == '2:18') echo ' from <span style="font-family: Georgia, serif;"><strong><a href="', $scripturl, '?action=about:unknown;verse=4:16" style="color: white; text-decoration: none; cursor: text;">The Book of Unknown</a></strong>, 2:18</span>'; elseif ($_GET['verse'] == '4:16') echo ' from <span style="font-family: Georgia, serif;"><strong><a href="', $scripturl, '?action=about:unknown;verse=22:1-2" style="color: white; text-decoration: none; cursor: text;">The Book of Unknown</a></strong>, 4:16</span>'; elseif ($_GET['verse'] == '22:1-2') echo ' from <span style="font-family: Georgia, serif;"><strong>The Book of Unknown</strong>, 22:1-2</span>'; echo ' </div> </body> </html>'; obExit(false); } ?>