<?php namespace HashOver;

// Copyright (C) 2018 Jacob Barkdull
// This file is part of HashOver.
//
// HashOver is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as
// published by the Free Software Foundation, either version 3 of the
// License, or (at your option) any later version.
//
// HashOver 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 Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with HashOver. If not, see <http://www.gnu.org/licenses/>.


// Generates an array of various settings information
function ui_array (Setup $setup, Locale $locale)
{
// Theme names
$themes = array ();

// Themes directory
$themes_directory = $setup->getAbsolutePath ('themes');

// Get each theme directory name
foreach (glob ($themes_directory . '/*', GLOB_ONLYDIR) as $directory) {
$theme = basename ($directory);
$themes[$theme] = $theme;
}

// Return array of settings allowed to be changed
return array (
'language' => array (
'type' => 'select',
'value' => $setup->language,

'options' => array (
'en' => 'English',
'da' => 'Danish',
'el' => 'Greek',
'de' => 'German',
'es' => 'Spanish',
'fa' => 'Persian',
'fr' => 'French',
'jp' => 'Japanese',
'ko' => 'Korean',
'lt' => 'Lithuanian',
'nl' => 'Dutch',
'pl' => 'Polish',
'pt-br' => 'Brazilian Portuguese',
'ro' => 'Romanian',
'tr' => 'Turkish',
'zh-cn' => 'Simplified Chinese'
)
),
'theme' => array (
'type' => 'select',
'value' => $setup->theme,
'options' => $themes
),
'default-sorting' => array (
'type' => 'select',
'value' => $setup->defaultSorting,

'options' => array (
'ascending' => $locale->text['sort-ascending'],
'descending' => $locale->text['sort-descending'],
'by-date' => $locale->text['sort-by-date'],
'by-likes' => $locale->text['sort-by-likes'],
'by-replies' => $locale->text['sort-by-replies'],
'by-name' => $locale->text['sort-by-name'],

'sort-threads' => array (
'threaded-descending' => $locale->text['sort-descending'],
'threaded-by-date' => $locale->text['sort-by-date'],
'threaded-by-likes' => $locale->text['sort-by-likes'],
'by-popularity' => $locale->text['sort-by-popularity'],
'by-discussion' => $locale->text['sort-by-discussion'],
'threaded-by-name' => $locale->text['sort-by-name']
)
)
),
'uses-moderation' => array (
'type' => 'checkbox',
'value' => $setup->usesModeration
),
'pends-user-edits' => array (
'type' => 'checkbox',
'value' => $setup->pendsUserEdits
),
'data-format' => array (
'type' => 'select',
'value' => $setup->dataFormat,

'options' => array (
'xml' => 'XML',
'json' => 'JSON',
'sql' => 'SQL'
)
),
'mailer' => array (
'type' => 'select',
'value' => $setup->mailer,

'options' => array (
'sendmail' => 'Sendmail',
'smtp' => 'SMTP'
)
),
'mail-type' => array (
'type' => 'select',
'value' => $setup->mailType,

'options' => array (
'text' => 'Text',
'html' => 'HTML'
)
),
'noreply-email' => array (
'type' => 'text',
'value' => $setup->noreplyEmail
),
'default-name' => array (
'type' => 'text',
'value' => $setup->defaultName
),
'allows-images' => array (
'type' => 'checkbox',
'value' => $setup->allowsImages
),
'allows-login' => array (
'type' => 'checkbox',
'value' => $setup->allowsLogin
),
'allows-likes' => array (
'type' => 'checkbox',
'value' => $setup->allowsLikes
),
'allows-dislikes' => array (
'type' => 'checkbox',
'value' => $setup->allowsDislikes
),
'uses-ajax' => array (
'type' => 'checkbox',
'value' => $setup->usesAjax
),
'collapses-interface' => array (
'type' => 'checkbox',
'value' => $setup->collapsesInterface
),
'collapses-comments' => array (
'type' => 'checkbox',
'value' => $setup->collapsesComments
),
'collapse-limit' => array (
'type' => 'number',
'value' => $setup->collapseLimit
),
'reply-mode' => array (
'type' => 'select',
'value' => $setup->replyMode,

'options' => array (
'thread' => 'Threaded',
'stream' => 'Stream'
)
),
'stream-depth' => array (
'type' => 'number',
'value' => $setup->streamDepth
),
'popularity-threshold' => array (
'type' => 'number',
'value' => $setup->popularityThreshold
),
'popularity-limit' => array (
'type' => 'number',
'value' => $setup->popularityLimit
),
'uses-markdown' => array (
'type' => 'checkbox',
'value' => $setup->usesMarkdown
),
'server-timezone' => array (
'documentation' => 'https://php.net/manual/en/timezones.php',
'type' => 'text',
'value' => $setup->serverTimezone
),
'uses-user-timezone' => array (
'type' => 'checkbox',
'value' => $setup->usesUserTimezone
),
'uses-short-dates' => array (
'type' => 'checkbox',
'value' => $setup->usesShortDates
),
'time-format' => array (
'documentation' => 'https://php.net/manual/en/function.date.php',
'type' => 'text',
'value' => $setup->timeFormat
),
'date-format' => array (
'documentation' => 'https://php.net/manual/en/function.date.php',
'type' => 'text',
'value' => $setup->dateFormat
),
'displays-title' => array (
'type' => 'checkbox',
'value' => $setup->displaysTitle
),
'form-position' => array (
'type' => 'select',
'value' => $setup->formPosition,

'options' => array (
'top' => 'Top',
'bottom' => 'Bottom'
)
),
'uses-auto-login' => array (
'type' => 'checkbox',
'value' => $setup->usesAutoLogin
),
'shows-reply-count' => array (
'type' => 'checkbox',
'value' => $setup->showsReplyCount
),
'count-includes-deleted' => array (
'type' => 'checkbox',
'value' => $setup->countIncludesDeleted
),
'icon-mode' => array (
'type' => 'select',
'value' => $setup->iconMode,

'options' => array (
'image' => 'Image',
'count' => 'Count',
'none' => 'None'
)
),
'icon-size' => array (
'type' => 'number',
'value' => $setup->iconSize
),
'image-format' => array (
'type' => 'select',
'value' => $setup->imageFormat,

'options' => array (
'png' => 'PNG',
'svg' => 'SVG'
)
),
'uses-labels' => array (
'type' => 'checkbox',
'value' => $setup->usesLabels
),
'uses-cancel-buttons' => array (
'type' => 'checkbox',
'value' => $setup->usesCancelButtons
),
'appends-css' => array (
'type' => 'checkbox',
'value' => $setup->appendsCss
),
'appends-rss' => array (
'type' => 'checkbox',
'value' => $setup->appendsRss
),
'login-method' => array (
'type' => 'select',
'value' => $setup->loginMethod,

'options' => array (
'defaultLogin' => 'Default Login'
)
),
'sets-cookies' => array (
'type' => 'checkbox',
'value' => $setup->setsCookies
),
'secure-cookies' => array (
'type' => 'checkbox',
'value' => $setup->secureCookies
),
'stores-ip-address' => array (
'type' => 'checkbox',
'value' => $setup->storesIpAddress
),
'subscribes-user' => array (
'type' => 'checkbox',
'value' => $setup->subscribesUser
),
'allows-user-replies' => array (
'type' => 'checkbox',
'value' => $setup->allowsUserReplies
),
'spam-batabase' => array (
'type' => 'select',
'value' => $setup->spamDatabase,

'options' => array (
'remote' => 'StopForumSpam.com',
'local' => 'Local CSV file'
)
),
'spam-check-modes' => array (
'type' => 'select',
'value' => $setup->spamCheckModes,

'options' => array (
'json' => 'JSON',
'php' => 'PHP',
'both' => 'Both'
)
),
'gravatar-force' => array (
'type' => 'checkbox',
'value' => $setup->gravatarForce
),
'gravatar-default' => array (
'type' => 'select',
'value' => $setup->gravatarDefault,

'options' => array (
'custom' => 'Custom',
'identicon' => 'Identicon',
'monsterid' => 'Monsterid',
'wavatar' => 'Wavatar',
'retro' => 'Retro'
)
),
'minifies-javascript' => array (
'type' => 'checkbox',
'value' => $setup->minifiesJavascript
),
'minify-level' => array (
'type' => 'select',
'cast' => 'number',
'value' => $setup->minifyLevel,

'options' => array (
1 => 'Basic (removes code comments)',
2 => 'Low (removes whitespace + Basic)',
3 => 'Medium (removes newlines + Low)',
4 => 'High (removes extra bits + Medium)'
)
),
'allow-local-metadata' => array (
'type' => 'checkbox',
'value' => $setup->allowLocalMetadata
)
);
}

try {
// View setup
require (realpath ('../view-setup.php'));

// Get array of UI elements to create
$ui = ui_array ($hashover->setup, $hashover->locale);

// Check if the form has been submitted
if (isset ($_POST['save'])) {
// Settings JSON file path
$settings_file = $hashover->setup->getAbsolutePath ('config/settings.json');

// Read JSON settings file
$json = $data_files->readJSON ($settings_file);

// Existing JSON settings or an empty array
$settings = ($json !== false) ? $json : array ();

// Run through configurable settings
foreach ($ui as $name => $setting) {
// Use specified type or optional cast
$type = !empty ($setting['cast']) ? 'cast' : 'type';

switch ($setting[$type]) {
// Set value to boolean based on POST data
case 'checkbox': {
$settings[$name] = isset ($_POST[$name]);
break;
}

// Cast number values to integers
case 'number': {
$settings[$name] = (int)($_POST[$name]);
break;
}

// All other values are strings
default: {
// Check if setting has a value
if (!empty ($_POST[$name])) {
// If so, cast it to string before setting it
$settings[$name] = (string)($_POST[$name]);
} else {
// If not, remove the setting entirely
unset ($settings[$name]);
}

break;
}
}
}

// Check if the user login is admin
if ($hashover->login->verifyAdmin () === true) {
// If so, attempt to save the settings
$saved = $data_files->saveJSON ($settings_file, $settings);

// If saved successfully, redirect with success indicator
if ($saved === true) {
redirect ('index.php?status=success');
}
}

// Otherwise, redirect with failure indicator
redirect ('index.php?status=failure');
}

// Otherwise, create settings table
$table = new HTMLTag ('table', array (
'id' => 'settings',
'class' => 'p-spaced',
'cellspacing' => '0',
'cellpadding' => '4'
));

// Create settings table
foreach ($ui as $name => $setting) {
// Create table row
$tr = new HTMLTag ('tr');

// Create setting description cell
$description = new HTMLTag ('td');

// Create description label
$label = new HTMLTag ('label', array (
'for' => $name,
'innerHTML' => $hashover->locale->text['setting-' . $name]
), false);

// Check for documentation URL
if (!empty ($setting['documentation'])) {
// Create documentation link
$docs = new HTMLTag ('a', array (
'href' => $setting['documentation'],
'target' => '_blank',
'innerHTML' => mb_strtolower ($hashover->locale->text['documentation'])
), false);

// Append documentation in parentheses
$label->appendInnerHTML ('(' . $docs->asHTML () . ')');
}

// Append label to description cell
$description->appendChild ($label);

// Append description cell to settings table row
$tr->appendChild ($description);

// Create setting value cell
$field = new HTMLTag ('td');

switch ($setting['type']) {
case 'checkbox': {
// Create checkbox for enabling/disabling the setting
$element = new HTMLTag ('input', array (
'id' => $name,
'type' => 'checkbox',
'name' => $name
), false, true);

// Set check based on current setting
if ($setting['value'] !== false) {
$element->createAttribute ('checked', 'true');
}

break;
}

// Create text/number box for entering the setting value
case 'number' : case 'text': {
$element = new HTMLTag ('input', array (
'id' => $name,
'type' => $setting['type'],
'name' => $name,
'value' => $setting['value'],
'size' => ($setting['type'] === 'text') ? '25' : '10'
), false, true);

break;
}

// Create dropdown menu for selecting the setting value
case 'select': {
// Create wrapper element for dropdown menu
$element = new HTMLTag ('span', array (
'class' => 'select-wrapper'
));

// Create dropdown menu
$select = new HTMLTag ('select', array (
'id' => $name,
'name' => $name,
'size' => 1
));

foreach ($setting['options'] as $value => $data) {
// Check if the current option is an array
if (is_array ($data)) {
// If so, add an option group spacer to menu
$select->appendChild (new HTMLTag ('optgroup', array (
'label' => '&nbsp;'
)));

// Create an option group with localized label
$optgroup = new HTMLTag ('optgroup', array (
'label' => $hashover->locale->text[$value]
));

// Run through each optgroup option
foreach ($data as $opt_value => $opt_text) {
// Create setting option
$option = new HTMLTag ('option', array (
'value' => $opt_value,
'innerHTML' => $opt_text
), false);

// Select proper option
if ($opt_value === $setting['value']) {
$option->createAttribute ('selected', 'true');
}

// Append option to optgroup
$optgroup->appendChild ($option);
}

// And append optgroup to menu
$select->appendChild ($optgroup);
} else {
// If not, create setting option
$option = new HTMLTag ('option', array (
'value' => $value,
'innerHTML' => $data
), false);

// Select proper option
if ($value === $setting['value']) {
$option->createAttribute ('selected', 'true');
}

// Append option to menu
$select->appendChild ($option);
}
}

// Append dropdown menu to wrapper element
$element->appendChild ($select);

break;
}
}

// Append the setting value element to the setting value cell
$field->appendChild ($element);

// Append the setting value cell to the settings table row
$tr->appendChild ($field);

// Add row to settings table
$table->appendChild ($tr);
}

// Template data
$template = array (
'title' => $hashover->locale->text['settings'],
'logout' => $logout->asHTML ("\t\t\t"),
'sub-title' => $hashover->locale->text['settings-sub'],
'message' => $form_message,
'settings' => $table->asHTML ("\t\t\t"),
'save-button' => $hashover->locale->text['save']
);

// Load and parse HTML template
echo $hashover->templater->parseTemplate ('settings.html', $template);

} catch (\
Exception $error) {
echo Misc::displayError ($error->getMessage ());
}