Files
Emi 07ec659385 Importing project into Git
This project lived only on the server without version control. This is now the starting point for the repository.
2023-05-23 20:03:24 +02:00

476 lines
17 KiB
PHP

<?php
/*
Question2Answer by Gideon Greenspan and contributors
http://www.question2answer.org/
Description: Controller for question page (only viewing functionality here)
This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License
as published by the Free Software Foundation; either version 2
of the License, or (at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
More about this license: http://www.question2answer.org/license.php
*/
if (!defined('QA_VERSION')) { // don't allow this page to be requested directly from browser
header('Location: ../../');
exit;
}
require_once QA_INCLUDE_DIR . 'app/cookies.php';
require_once QA_INCLUDE_DIR . 'app/format.php';
require_once QA_INCLUDE_DIR . 'db/selects.php';
require_once QA_INCLUDE_DIR . 'util/sort.php';
require_once QA_INCLUDE_DIR . 'util/string.php';
require_once QA_INCLUDE_DIR . 'app/captcha.php';
require_once QA_INCLUDE_DIR . 'pages/question-view.php';
require_once QA_INCLUDE_DIR . 'app/updates.php';
$questionid = qa_request_part(0);
$userid = qa_get_logged_in_userid();
$cookieid = qa_cookie_get();
$pagestate = qa_get_state();
// Get information about this question
$cacheDriver = Q2A_Storage_CacheFactory::getCacheDriver();
$cacheKey = "question:$questionid";
$useCache = $userid === null && $cacheDriver->isEnabled() && !qa_is_http_post() && empty($pagestate);
$saveCache = false;
if ($useCache) {
$questionData = $cacheDriver->get($cacheKey);
}
if (!isset($questionData)) {
$questionData = qa_db_select_with_pending(
qa_db_full_post_selectspec($userid, $questionid),
qa_db_full_child_posts_selectspec($userid, $questionid),
qa_db_full_a_child_posts_selectspec($userid, $questionid),
qa_db_post_parent_q_selectspec($questionid),
qa_db_post_close_post_selectspec($questionid),
qa_db_post_duplicates_selectspec($questionid),
qa_db_post_meta_selectspec($questionid, 'qa_q_extra'),
qa_db_category_nav_selectspec($questionid, true, true, true),
isset($userid) ? qa_db_is_favorite_selectspec($userid, QA_ENTITY_QUESTION, $questionid) : null
);
// whether to save the cache (actioned below, after basic checks)
$saveCache = $useCache;
}
list($question, $childposts, $achildposts, $parentquestion, $closepost, $duplicateposts, $extravalue, $categories, $favorite) = $questionData;
if (isset($question['basetype']) && $question['basetype'] != 'Q') // don't allow direct viewing of other types of post
$question = null;
if (isset($question)) {
$q_request = qa_q_request($questionid, $question['title']);
if (trim($q_request, '/') !== trim(qa_request(), '/')) {
// redirect if the current URL is incorrect
qa_redirect($q_request);
}
$question['extra'] = $extravalue;
$answers = qa_page_q_load_as($question, $childposts);
$commentsfollows = qa_page_q_load_c_follows($question, $childposts, $achildposts, $duplicateposts);
$question = $question + qa_page_q_post_rules($question, null, null, $childposts + $duplicateposts); // array union
if ($question['selchildid'] && (@$answers[$question['selchildid']]['type'] != 'A'))
$question['selchildid'] = null; // if selected answer is hidden or somehow not there, consider it not selected
foreach ($answers as $key => $answer) {
$answers[$key] = $answer + qa_page_q_post_rules($answer, $question, $answers, $achildposts);
$answers[$key]['isselected'] = ($answer['postid'] == $question['selchildid']);
}
foreach ($commentsfollows as $key => $commentfollow) {
$parent = ($commentfollow['parentid'] == $questionid) ? $question : @$answers[$commentfollow['parentid']];
$commentsfollows[$key] = $commentfollow + qa_page_q_post_rules($commentfollow, $parent, $commentsfollows, null);
}
}
// Deal with question not found or not viewable, otherwise report the view event
if (!isset($question))
return include QA_INCLUDE_DIR . 'qa-page-not-found.php';
if (!$question['viewable']) {
$qa_content = qa_content_prepare();
if ($question['queued'])
$qa_content['error'] = qa_lang_html('question/q_waiting_approval');
elseif ($question['flagcount'] && !isset($question['lastuserid']))
$qa_content['error'] = qa_lang_html('question/q_hidden_flagged');
elseif ($question['authorlast'])
$qa_content['error'] = qa_lang_html('question/q_hidden_author');
else
$qa_content['error'] = qa_lang_html('question/q_hidden_other');
$qa_content['suggest_next'] = qa_html_suggest_qs_tags(qa_using_tags());
return $qa_content;
}
$permiterror = qa_user_post_permit_error('permit_view_q_page', $question, null, false);
if ($permiterror && (qa_is_human_probably() || !qa_opt('allow_view_q_bots'))) {
$qa_content = qa_content_prepare();
$topage = qa_q_request($questionid, $question['title']);
switch ($permiterror) {
case 'login':
$qa_content['error'] = qa_insert_login_links(qa_lang_html('main/view_q_must_login'), $topage);
break;
case 'confirm':
$qa_content['error'] = qa_insert_login_links(qa_lang_html('main/view_q_must_confirm'), $topage);
break;
case 'approve':
$qa_content['error'] = strtr(qa_lang_html('main/view_q_must_be_approved'), array(
'^1' => '<a href="' . qa_path_html('account') . '">',
'^2' => '</a>',
));
break;
default:
$qa_content['error'] = qa_lang_html('users/no_permission');
break;
}
return $qa_content;
}
// Save question data to cache (if older than configured limit)
if ($saveCache) {
$questionAge = qa_opt('db_time') - $question['created'];
if ($questionAge > 86400 * qa_opt('caching_q_start')) {
$cacheDriver->set($cacheKey, $questionData, qa_opt('caching_q_time'));
}
}
// Determine if captchas will be required
$captchareason = qa_user_captcha_reason(qa_user_level_for_post($question));
$usecaptcha = ($captchareason != false);
// If we're responding to an HTTP POST, include file that handles all posting/editing/etc... logic
// This is in a separate file because it's a *lot* of logic, and will slow down ordinary page views
$pagestart = qa_get_start();
$showid = qa_get('show');
$pageerror = null;
$formtype = null;
$formpostid = null;
$jumptoanchor = null;
$commentsall = null;
if (substr($pagestate, 0, 13) == 'showcomments-') {
$commentsall = substr($pagestate, 13);
$pagestate = null;
} elseif (isset($showid)) {
foreach ($commentsfollows as $comment) {
if ($comment['postid'] == $showid) {
$commentsall = $comment['parentid'];
break;
}
}
}
if (qa_is_http_post() || strlen($pagestate))
require QA_INCLUDE_DIR . 'pages/question-post.php';
$formrequested = isset($formtype);
if (!$formrequested && $question['answerbutton']) {
$immedoption = qa_opt('show_a_form_immediate');
if ($immedoption == 'always' || ($immedoption == 'if_no_as' && !$question['isbyuser'] && !$question['acount']))
$formtype = 'a_add'; // show answer form by default
}
// Get information on the users referenced
$usershtml = qa_userids_handles_html(array_merge(array($question), $answers, $commentsfollows), true);
// Prepare content for theme
$qa_content = qa_content_prepare(true, array_keys(qa_category_path($categories, $question['categoryid'])));
if (isset($userid) && !$formrequested)
$qa_content['favorite'] = qa_favorite_form(QA_ENTITY_QUESTION, $questionid, $favorite,
qa_lang($favorite ? 'question/remove_q_favorites' : 'question/add_q_favorites'));
if (isset($pageerror))
$qa_content['error'] = $pageerror; // might also show voting error set in qa-index.php
elseif ($question['queued'])
$qa_content['error'] = $question['isbyuser'] ? qa_lang_html('question/q_your_waiting_approval') : qa_lang_html('question/q_waiting_your_approval');
if ($question['hidden'])
$qa_content['hidden'] = true;
qa_sort_by($commentsfollows, 'created');
// Prepare content for the question...
if ($formtype == 'q_edit') { // ...in edit mode
$qa_content['title'] = qa_lang_html($question['editable'] ? 'question/edit_q_title' :
(qa_using_categories() ? 'question/recat_q_title' : 'question/retag_q_title'));
$qa_content['form_q_edit'] = qa_page_q_edit_q_form($qa_content, $question, @$qin, @$qerrors, $completetags, $categories);
$qa_content['q_view']['raw'] = $question;
} else { // ...in view mode
$qa_content['q_view'] = qa_page_q_question_view($question, $parentquestion, $closepost, $usershtml, $formrequested);
$qa_content['title'] = $qa_content['q_view']['title'];
$qa_content['description'] = qa_html(qa_shorten_string_line(qa_viewer_text($question['content'], $question['format']), 150));
$categorykeyword = @$categories[$question['categoryid']]['title'];
$qa_content['keywords'] = qa_html(implode(',', array_merge(
(qa_using_categories() && strlen($categorykeyword)) ? array($categorykeyword) : array(),
qa_tagstring_to_tags($question['tags'])
))); // as far as I know, META keywords have zero effect on search rankings or listings, but many people have asked for this
}
$microdata = qa_opt('use_microdata');
if ($microdata) {
$qa_content['head_lines'][] = '<meta itemprop="name" content="' . qa_html($qa_content['q_view']['raw']['title']) . '">';
$qa_content['html_tags'] .= ' itemscope itemtype="https://schema.org/QAPage"';
$qa_content['wrapper_tags'] = ' itemprop="mainEntity" itemscope itemtype="https://schema.org/Question"';
}
// Prepare content for an answer being edited (if any) or to be added
if ($formtype == 'a_edit') {
$qa_content['a_form'] = qa_page_q_edit_a_form($qa_content, 'a' . $formpostid, $answers[$formpostid],
$question, $answers, $commentsfollows, @$aeditin[$formpostid], @$aediterrors[$formpostid]);
$qa_content['a_form']['c_list'] = qa_page_q_comment_follow_list($question, $answers[$formpostid],
$commentsfollows, true, $usershtml, $formrequested, $formpostid);
$jumptoanchor = 'a' . $formpostid;
} elseif ($formtype == 'a_add' || ($question['answerbutton'] && !$formrequested)) {
$qa_content['a_form'] = qa_page_q_add_a_form($qa_content, 'anew', $captchareason, $question, @$anewin, @$anewerrors, $formtype == 'a_add', $formrequested);
if ($formrequested) {
$jumptoanchor = 'anew';
} elseif ($formtype == 'a_add') {
$qa_content['script_onloads'][] = array(
"qa_element_revealed=document.getElementById('anew');"
);
}
}
// Prepare content for comments on the question, plus add or edit comment forms
if ($formtype == 'q_close') {
$qa_content['q_view']['c_form'] = qa_page_q_close_q_form($qa_content, $question, 'close', @$closein, @$closeerrors);
$jumptoanchor = 'close';
} elseif (($formtype == 'c_add' && $formpostid == $questionid) || ($question['commentbutton'] && !$formrequested)) { // ...to be added
$qa_content['q_view']['c_form'] = qa_page_q_add_c_form($qa_content, $question, $question, 'c' . $questionid,
$captchareason, @$cnewin[$questionid], @$cnewerrors[$questionid], $formtype == 'c_add');
if ($formtype == 'c_add' && $formpostid == $questionid) {
$jumptoanchor = 'c' . $questionid;
$commentsall = $questionid;
}
} elseif ($formtype == 'c_edit' && @$commentsfollows[$formpostid]['parentid'] == $questionid) { // ...being edited
$qa_content['q_view']['c_form'] = qa_page_q_edit_c_form($qa_content, 'c' . $formpostid, $commentsfollows[$formpostid],
@$ceditin[$formpostid], @$cediterrors[$formpostid]);
$jumptoanchor = 'c' . $formpostid;
$commentsall = $questionid;
}
$qa_content['q_view']['c_list'] = qa_page_q_comment_follow_list($question, $question, $commentsfollows,
$commentsall == $questionid, $usershtml, $formrequested, $formpostid); // ...for viewing
// Prepare content for existing answers (could be added to by Ajax)
$qa_content['a_list'] = array(
'tags' => 'id="a_list"',
'as' => array(),
);
// sort according to the site preferences
if (qa_opt('sort_answers_by') == 'votes') {
foreach ($answers as $answerid => $answer)
$answers[$answerid]['sortvotes'] = $answer['downvotes'] - $answer['upvotes'];
qa_sort_by($answers, 'sortvotes', 'created');
} else {
qa_sort_by($answers, 'created');
}
// further changes to ordering to deal with queued, hidden and selected answers
$countfortitle = (int) $question['acount'];
$nextposition = 10000;
$answerposition = array();
foreach ($answers as $answerid => $answer) {
if ($answer['viewable']) {
$position = $nextposition++;
if ($answer['hidden'])
$position += 10000;
elseif ($answer['queued']) {
$position -= 10000;
$countfortitle++; // include these in displayed count
} elseif ($answer['isselected'] && qa_opt('show_selected_first'))
$position -= 5000;
$answerposition[$answerid] = $position;
}
}
asort($answerposition, SORT_NUMERIC);
// extract IDs and prepare for pagination
$answerids = array_keys($answerposition);
$countforpages = count($answerids);
$pagesize = qa_opt('page_size_q_as');
// see if we need to display a particular answer
if (isset($showid)) {
if (isset($commentsfollows[$showid]))
$showid = $commentsfollows[$showid]['parentid'];
$position = array_search($showid, $answerids);
if (is_numeric($position))
$pagestart = floor($position / $pagesize) * $pagesize;
}
// set the canonical url based on possible pagination
$qa_content['canonical'] = qa_path_html(qa_q_request($question['postid'], $question['title']),
($pagestart > 0) ? array('start' => $pagestart) : null, qa_opt('site_url'));
// build the actual answer list
$answerids = array_slice($answerids, $pagestart, $pagesize);
foreach ($answerids as $answerid) {
$answer = $answers[$answerid];
if (!($formtype == 'a_edit' && $formpostid == $answerid)) {
$a_view = qa_page_q_answer_view($question, $answer, $answer['isselected'], $usershtml, $formrequested);
// Prepare content for comments on this answer, plus add or edit comment forms
if (($formtype == 'c_add' && $formpostid == $answerid) || ($answer['commentbutton'] && !$formrequested)) { // ...to be added
$a_view['c_form'] = qa_page_q_add_c_form($qa_content, $question, $answer, 'c' . $answerid,
$captchareason, @$cnewin[$answerid], @$cnewerrors[$answerid], $formtype == 'c_add');
if ($formtype == 'c_add' && $formpostid == $answerid) {
$jumptoanchor = 'c' . $answerid;
$commentsall = $answerid;
}
} elseif ($formtype == 'c_edit' && @$commentsfollows[$formpostid]['parentid'] == $answerid) { // ...being edited
$a_view['c_form'] = qa_page_q_edit_c_form($qa_content, 'c' . $formpostid, $commentsfollows[$formpostid],
@$ceditin[$formpostid], @$cediterrors[$formpostid]);
$jumptoanchor = 'c' . $formpostid;
$commentsall = $answerid;
}
$a_view['c_list'] = qa_page_q_comment_follow_list($question, $answer, $commentsfollows,
$commentsall == $answerid, $usershtml, $formrequested, $formpostid); // ...for viewing
// Add the answer to the list
$qa_content['a_list']['as'][] = $a_view;
}
}
if ($question['basetype'] == 'Q') {
$qa_content['a_list']['title_tags'] = 'id="a_list_title"';
$split = $countfortitle == 1
? qa_lang_html_sub_split('question/1_answer_title', '1', '1')
: qa_lang_html_sub_split('question/x_answers_title', $countfortitle);
if ($microdata) {
$split['data'] = '<span itemprop="answerCount">' . $split['data'] . '</span>';
}
$qa_content['a_list']['title'] = $split['prefix'] . $split['data'] . $split['suffix'];
if ($countfortitle == 0) {
$qa_content['a_list']['title_tags'] .= ' style="display:none;" ';
}
}
if (!$formrequested) {
$qa_content['page_links'] = qa_html_page_links(qa_request(), $pagestart, $pagesize, $countforpages, qa_opt('pages_prev_next'), array(), false, 'a_list_title');
}
// Some generally useful stuff
if (qa_using_categories() && count($categories)) {
$qa_content['navigation']['cat'] = qa_category_navigation($categories, $question['categoryid']);
}
if (isset($jumptoanchor)) {
$qa_content['script_onloads'][] = array(
'qa_scroll_page_to($("#"+' . qa_js($jumptoanchor) . ').offset().top);'
);
}
// Determine whether this request should be counted for page view statistics.
// The lastviewip check is now part of the hotness query in order to bypass caching.
if (qa_opt('do_count_q_views') && !$formrequested && !qa_is_http_post() && qa_is_human_probably() &&
(!$question['views'] || (
// if it has more than zero views, then it must be different IP & user & cookieid from the creator
(@inet_ntop($question['createip']) != qa_remote_ip_address() || !isset($question['createip'])) &&
($question['userid'] != $userid || !isset($question['userid'])) &&
($question['cookieid'] != $cookieid || !isset($question['cookieid']))
))
) {
$qa_content['inc_views_postid'] = $questionid;
}
return $qa_content;