mirror of
https://github.com/godotengine/godot-question2answer.git
synced 2026-01-01 01:48:37 +03:00
This project lived only on the server without version control. This is now the starting point for the repository.
476 lines
17 KiB
PHP
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;
|