2021-12-14 18:32:47 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
declare(strict_types=1);
|
2009-12-24 07:36:09 +00:00
|
|
|
|
2023-01-10 22:44:09 +00:00
|
|
|
namespace Shimmie2;
|
|
|
|
|
2019-06-21 20:13:47 +00:00
|
|
|
abstract class PoolsConfig
|
|
|
|
{
|
2021-12-14 18:32:47 +00:00
|
|
|
public const MAX_IMPORT_RESULTS = "poolsMaxImportResults";
|
|
|
|
public const IMAGES_PER_PAGE = "poolsImagesPerPage";
|
|
|
|
public const LISTS_PER_PAGE = "poolsListsPerPage";
|
|
|
|
public const UPDATED_PER_PAGE = "poolsUpdatedPerPage";
|
|
|
|
public const INFO_ON_VIEW_IMAGE = "poolsInfoOnViewImage";
|
|
|
|
public const ADDER_ON_VIEW_IMAGE = "poolsAdderOnViewImage";
|
|
|
|
public const SHOW_NAV_LINKS = "poolsShowNavLinks";
|
|
|
|
public const AUTO_INCREMENT_ORDER = "poolsAutoIncrementOrder";
|
2019-06-21 20:13:47 +00:00
|
|
|
}
|
|
|
|
|
2012-02-08 04:15:23 +00:00
|
|
|
/**
|
|
|
|
* This class is just a wrapper around SCoreException.
|
|
|
|
*/
|
2019-05-28 16:59:38 +00:00
|
|
|
class PoolCreationException extends SCoreException
|
|
|
|
{
|
2010-05-04 19:10:37 +00:00
|
|
|
}
|
|
|
|
|
2019-06-21 20:13:47 +00:00
|
|
|
class PoolAddPostsEvent extends Event
|
|
|
|
{
|
2021-03-14 23:43:50 +00:00
|
|
|
public int $pool_id;
|
2024-01-20 14:10:59 +00:00
|
|
|
/** @var int[] */
|
2021-03-14 23:43:50 +00:00
|
|
|
public array $posts = [];
|
2019-06-21 20:13:47 +00:00
|
|
|
|
2024-01-20 14:10:59 +00:00
|
|
|
/**
|
|
|
|
* @param int[] $posts
|
|
|
|
*/
|
2019-06-21 20:13:47 +00:00
|
|
|
public function __construct(int $pool_id, array $posts)
|
|
|
|
{
|
2020-01-26 13:19:35 +00:00
|
|
|
parent::__construct();
|
2019-06-21 20:13:47 +00:00
|
|
|
$this->pool_id = $pool_id;
|
|
|
|
$this->posts = $posts;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class PoolCreationEvent extends Event
|
|
|
|
{
|
2021-03-14 23:43:50 +00:00
|
|
|
public string $title;
|
|
|
|
public User $user;
|
|
|
|
public bool $public;
|
|
|
|
public string $description;
|
|
|
|
public int $new_id = -1;
|
2019-06-21 20:13:47 +00:00
|
|
|
|
2020-03-27 00:15:15 +00:00
|
|
|
public function __construct(
|
|
|
|
string $title,
|
|
|
|
User $pool_user = null,
|
|
|
|
bool $public = false,
|
|
|
|
string $description = ""
|
2020-04-25 20:36:28 +00:00
|
|
|
) {
|
2020-01-26 13:19:35 +00:00
|
|
|
parent::__construct();
|
2019-06-21 20:13:47 +00:00
|
|
|
global $user;
|
|
|
|
|
|
|
|
$this->title = $title;
|
|
|
|
$this->user = $pool_user ?? $user;
|
|
|
|
$this->public = $public;
|
|
|
|
$this->description = $description;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-24 21:16:18 +00:00
|
|
|
class PoolDeletionEvent extends Event
|
|
|
|
{
|
2021-03-14 23:43:50 +00:00
|
|
|
public int $pool_id;
|
2020-10-24 17:55:07 +00:00
|
|
|
|
|
|
|
public function __construct(int $pool_id)
|
|
|
|
{
|
|
|
|
parent::__construct();
|
|
|
|
$this->pool_id = $pool_id;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-25 20:36:28 +00:00
|
|
|
class Pool
|
|
|
|
{
|
2021-03-14 23:43:50 +00:00
|
|
|
public int $id;
|
|
|
|
public int $user_id;
|
|
|
|
public ?string $user_name;
|
|
|
|
public bool $public;
|
|
|
|
public string $title;
|
|
|
|
public string $description;
|
|
|
|
public string $date;
|
|
|
|
public int $posts;
|
2020-03-27 00:15:15 +00:00
|
|
|
|
2024-01-20 14:10:59 +00:00
|
|
|
/**
|
|
|
|
* @param array<string,mixed> $row
|
|
|
|
*/
|
2020-03-27 00:15:15 +00:00
|
|
|
public function __construct(array $row)
|
|
|
|
{
|
|
|
|
$this->id = (int)$row['id'];
|
|
|
|
$this->user_id = (int)$row['user_id'];
|
|
|
|
$this->user_name = $row['user_name'] ?? null;
|
|
|
|
$this->public = bool_escape($row['public']);
|
|
|
|
$this->title = $row['title'];
|
|
|
|
$this->description = $row['description'];
|
|
|
|
$this->date = $row['date'];
|
|
|
|
$this->posts = (int)$row['posts'];
|
|
|
|
}
|
|
|
|
|
2024-01-20 14:10:59 +00:00
|
|
|
/**
|
|
|
|
* @param array<string,mixed> $row
|
|
|
|
*/
|
2020-04-25 20:36:28 +00:00
|
|
|
public static function makePool(array $row): Pool
|
|
|
|
{
|
2020-03-27 00:15:15 +00:00
|
|
|
return new Pool($row);
|
|
|
|
}
|
2023-11-07 21:01:03 +00:00
|
|
|
|
2024-01-20 14:10:59 +00:00
|
|
|
public static function get_pool_id_by_title(string $poolTitle): ?int
|
2023-11-07 21:01:03 +00:00
|
|
|
{
|
|
|
|
global $database;
|
2023-11-06 05:50:33 +00:00
|
|
|
$row = $database->get_row("SELECT * FROM pools WHERE title=:title", ["title" => $poolTitle]);
|
|
|
|
if ($row != null) {
|
2023-11-07 21:01:03 +00:00
|
|
|
return $row['id'];
|
|
|
|
} else {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
2020-03-27 00:15:15 +00:00
|
|
|
}
|
|
|
|
|
2020-03-27 14:41:24 +00:00
|
|
|
function _image_to_id(Image $image): int
|
|
|
|
{
|
|
|
|
return $image->id;
|
|
|
|
}
|
|
|
|
|
2019-05-28 16:59:38 +00:00
|
|
|
class Pools extends Extension
|
|
|
|
{
|
2020-01-26 13:19:35 +00:00
|
|
|
/** @var PoolsTheme */
|
2023-06-27 14:56:49 +00:00
|
|
|
protected Themelet $theme;
|
2020-01-26 13:19:35 +00:00
|
|
|
|
2024-01-15 11:52:35 +00:00
|
|
|
public function onInitExt(InitExtEvent $event): void
|
2019-11-03 18:28:38 +00:00
|
|
|
{
|
2019-11-03 17:19:37 +00:00
|
|
|
global $config;
|
2019-05-28 16:59:38 +00:00
|
|
|
|
|
|
|
// Set the defaults for the pools extension
|
2019-06-21 20:13:47 +00:00
|
|
|
$config->set_default_int(PoolsConfig::MAX_IMPORT_RESULTS, 1000);
|
|
|
|
$config->set_default_int(PoolsConfig::IMAGES_PER_PAGE, 20);
|
|
|
|
$config->set_default_int(PoolsConfig::LISTS_PER_PAGE, 20);
|
|
|
|
$config->set_default_int(PoolsConfig::UPDATED_PER_PAGE, 20);
|
|
|
|
$config->set_default_bool(PoolsConfig::INFO_ON_VIEW_IMAGE, false);
|
|
|
|
$config->set_default_bool(PoolsConfig::ADDER_ON_VIEW_IMAGE, false);
|
|
|
|
$config->set_default_bool(PoolsConfig::SHOW_NAV_LINKS, false);
|
|
|
|
$config->set_default_bool(PoolsConfig::AUTO_INCREMENT_ORDER, false);
|
2019-11-03 17:19:37 +00:00
|
|
|
}
|
|
|
|
|
2024-01-15 11:52:35 +00:00
|
|
|
public function onDatabaseUpgrade(DatabaseUpgradeEvent $event): void
|
2019-11-03 18:28:38 +00:00
|
|
|
{
|
2020-01-26 13:19:35 +00:00
|
|
|
global $database;
|
2019-05-28 16:59:38 +00:00
|
|
|
|
|
|
|
// Create the database tables
|
2019-11-03 19:49:52 +00:00
|
|
|
if ($this->get_version("ext_pools_version") < 1) {
|
2019-05-28 16:59:38 +00:00
|
|
|
$database->create_table("pools", "
|
2009-12-25 23:05:57 +00:00
|
|
|
id SCORE_AIPK,
|
|
|
|
user_id INTEGER NOT NULL,
|
2020-10-26 22:37:25 +00:00
|
|
|
public BOOLEAN NOT NULL DEFAULT FALSE,
|
2023-08-18 21:23:40 +00:00
|
|
|
title VARCHAR(255) NOT NULL UNIQUE,
|
2009-12-25 23:05:57 +00:00
|
|
|
description TEXT,
|
2019-11-03 19:25:51 +00:00
|
|
|
date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
2023-08-18 21:23:40 +00:00
|
|
|
lastupdated TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
2009-12-25 23:05:57 +00:00
|
|
|
posts INTEGER NOT NULL DEFAULT 0,
|
2012-03-11 02:13:25 +00:00
|
|
|
FOREIGN KEY (user_id) REFERENCES users(id) ON UPDATE CASCADE ON DELETE CASCADE
|
2009-12-25 23:05:57 +00:00
|
|
|
");
|
2019-05-28 16:59:38 +00:00
|
|
|
$database->create_table("pool_images", "
|
2009-12-25 23:05:57 +00:00
|
|
|
pool_id INTEGER NOT NULL,
|
|
|
|
image_id INTEGER NOT NULL,
|
2012-03-11 02:13:25 +00:00
|
|
|
image_order INTEGER NOT NULL DEFAULT 0,
|
|
|
|
FOREIGN KEY (pool_id) REFERENCES pools(id) ON UPDATE CASCADE ON DELETE CASCADE,
|
2012-03-11 02:17:16 +00:00
|
|
|
FOREIGN KEY (image_id) REFERENCES images(id) ON UPDATE CASCADE ON DELETE CASCADE
|
2009-12-25 23:05:57 +00:00
|
|
|
");
|
2019-05-28 16:59:38 +00:00
|
|
|
$database->create_table("pool_history", "
|
2009-12-25 23:05:57 +00:00
|
|
|
id SCORE_AIPK,
|
|
|
|
pool_id INTEGER NOT NULL,
|
|
|
|
user_id INTEGER NOT NULL,
|
|
|
|
action INTEGER NOT NULL,
|
|
|
|
images TEXT,
|
|
|
|
count INTEGER NOT NULL DEFAULT 0,
|
2019-11-03 19:25:51 +00:00
|
|
|
date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
2012-03-11 02:13:25 +00:00
|
|
|
FOREIGN KEY (pool_id) REFERENCES pools(id) ON UPDATE CASCADE ON DELETE CASCADE,
|
|
|
|
FOREIGN KEY (user_id) REFERENCES users(id) ON UPDATE CASCADE ON DELETE CASCADE
|
2009-12-25 23:05:57 +00:00
|
|
|
");
|
2020-10-26 19:50:28 +00:00
|
|
|
$this->set_version("ext_pools_version", 4);
|
2019-05-28 16:59:38 +00:00
|
|
|
|
|
|
|
log_info("pools", "extension installed");
|
|
|
|
}
|
|
|
|
|
2020-10-26 19:12:20 +00:00
|
|
|
if ($this->get_version("ext_pools_version") < 4) {
|
|
|
|
$database->standardise_boolean("pools", "public");
|
|
|
|
$this->set_version("ext_pools_version", 4);
|
|
|
|
}
|
2024-01-15 23:23:33 +00:00
|
|
|
|
|
|
|
if ($this->get_version("ext_pools_version") < 5) {
|
|
|
|
// earlier versions of the table-creation code added the lastupdated
|
|
|
|
// column non-deterministically, so let's check if it is there and
|
|
|
|
// add it if needed.
|
|
|
|
$cols = $database->raw_db()->describe("pools");
|
|
|
|
if(!array_key_exists("lastupdated", $cols)) {
|
|
|
|
$database->execute("ALTER TABLE pools ADD COLUMN lastupdated TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP");
|
|
|
|
}
|
|
|
|
$this->set_version("ext_pools_version", 5);
|
|
|
|
}
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Add a block to the Board Config / Setup
|
2024-01-15 11:52:35 +00:00
|
|
|
public function onSetupBuilding(SetupBuildingEvent $event): void
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
2020-10-26 15:13:28 +00:00
|
|
|
$sb = $event->panel->create_new_block("Pools");
|
2019-06-21 20:13:47 +00:00
|
|
|
$sb->add_int_option(PoolsConfig::MAX_IMPORT_RESULTS, "Max results on import: ");
|
2020-10-26 15:22:13 +00:00
|
|
|
$sb->add_int_option(PoolsConfig::IMAGES_PER_PAGE, "<br>Posts per page: ");
|
2019-06-21 20:13:47 +00:00
|
|
|
$sb->add_int_option(PoolsConfig::LISTS_PER_PAGE, "<br>Index list items per page: ");
|
|
|
|
$sb->add_int_option(PoolsConfig::UPDATED_PER_PAGE, "<br>Updated list items per page: ");
|
|
|
|
$sb->add_bool_option(PoolsConfig::INFO_ON_VIEW_IMAGE, "<br>Show pool info on image: ");
|
|
|
|
$sb->add_bool_option(PoolsConfig::SHOW_NAV_LINKS, "<br>Show 'Prev' & 'Next' links when viewing pool images: ");
|
|
|
|
$sb->add_bool_option(PoolsConfig::AUTO_INCREMENT_ORDER, "<br>Autoincrement order when post is added to pool:");
|
|
|
|
//$sb->add_bool_option(PoolsConfig::ADDER_ON_VIEW_IMAGE, "<br>Show pool adder on image: ");
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
|
2024-01-15 11:52:35 +00:00
|
|
|
public function onPageNavBuilding(PageNavBuildingEvent $event): void
|
2019-08-02 19:54:48 +00:00
|
|
|
{
|
|
|
|
$event->add_nav_link("pool", new Link('pool/list'), "Pools");
|
|
|
|
}
|
|
|
|
|
2024-01-15 11:52:35 +00:00
|
|
|
public function onPageSubNavBuilding(PageSubNavBuildingEvent $event): void
|
2019-08-02 19:54:48 +00:00
|
|
|
{
|
2023-11-11 21:49:12 +00:00
|
|
|
if ($event->parent == "pool") {
|
2019-08-02 19:54:48 +00:00
|
|
|
$event->add_nav_link("pool_list", new Link('pool/list'), "List");
|
|
|
|
$event->add_nav_link("pool_new", new Link('pool/new'), "Create");
|
|
|
|
$event->add_nav_link("pool_updated", new Link('pool/updated'), "Changes");
|
|
|
|
$event->add_nav_link("pool_help", new Link('ext_doc/pools'), "Help");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-15 11:52:35 +00:00
|
|
|
public function onPageRequest(PageRequestEvent $event): void
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
2020-03-27 00:15:15 +00:00
|
|
|
global $config, $database, $page, $user;
|
2023-11-07 21:01:03 +00:00
|
|
|
if ($event->page_matches("pool/list")) { //index
|
2024-02-09 16:36:57 +00:00
|
|
|
if ($event->get_GET('search')) {
|
2023-11-07 21:01:03 +00:00
|
|
|
$page->set_mode(PageMode::REDIRECT);
|
2024-02-09 16:36:57 +00:00
|
|
|
$page->set_redirect(make_link('pool/list').'/'.url_escape($event->get_GET('search')).'/'.strval($event->try_page_num(1)));
|
2023-11-07 21:01:03 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (count($event->args) >= 4) { // Assume first 2 args are search and page num
|
|
|
|
$search = $event->get_arg(0); // Search is based on name comparison instead of tag search
|
|
|
|
$page_num = $event->try_page_num(1);
|
|
|
|
} else {
|
|
|
|
$search = "";
|
|
|
|
$page_num = $event->try_page_num(0);
|
|
|
|
}
|
|
|
|
$this->list_pools($page, $page_num, $search);
|
|
|
|
} elseif ($event->page_matches("pool")) {
|
2019-05-28 16:59:38 +00:00
|
|
|
// What action are we trying to perform?
|
|
|
|
switch ($event->get_arg(0)) {
|
|
|
|
case "new": // Show form for new pools
|
|
|
|
if (!$user->is_anonymous()) {
|
|
|
|
$this->theme->new_pool_composer($page);
|
|
|
|
} else {
|
|
|
|
$errMessage = "You must be registered and logged in to create a new pool.";
|
|
|
|
$this->theme->display_error(401, "Error", $errMessage);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case "create": // ADD _POST
|
|
|
|
try {
|
2019-06-21 20:13:47 +00:00
|
|
|
$title = $_POST["title"];
|
2023-02-04 20:50:26 +00:00
|
|
|
$event = send_event(new PoolCreationEvent(
|
2019-06-21 20:13:47 +00:00
|
|
|
$title,
|
|
|
|
$user,
|
2020-10-26 19:12:20 +00:00
|
|
|
bool_escape($_POST["public"]),
|
2019-09-29 13:30:55 +00:00
|
|
|
$_POST["description"]
|
2023-02-04 20:50:26 +00:00
|
|
|
));
|
2019-06-19 01:58:28 +00:00
|
|
|
$page->set_mode(PageMode::REDIRECT);
|
2019-06-21 20:13:47 +00:00
|
|
|
$page->set_redirect(make_link("pool/view/" . $event->new_id));
|
2019-05-28 16:59:38 +00:00
|
|
|
} catch (PoolCreationException $e) {
|
|
|
|
$this->theme->display_error(400, "Error", $e->error);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case "view":
|
|
|
|
$poolID = int_escape($event->get_arg(1));
|
|
|
|
$this->get_posts($event, $poolID);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case "updated":
|
2020-03-27 14:41:24 +00:00
|
|
|
$this->get_history($event->try_page_num(1));
|
2019-05-28 16:59:38 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case "revert":
|
|
|
|
if (!$user->is_anonymous()) {
|
|
|
|
$historyID = int_escape($event->get_arg(1));
|
|
|
|
$this->revert_history($historyID);
|
2019-06-19 01:58:28 +00:00
|
|
|
$page->set_mode(PageMode::REDIRECT);
|
2019-05-28 16:59:38 +00:00
|
|
|
$page->set_redirect(make_link("pool/updated"));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case "edit": // Edit the pool (remove images)
|
2024-01-20 20:48:47 +00:00
|
|
|
$pool_id = int_escape($_POST["pool_id"]);
|
|
|
|
$pool = $this->get_single_pool($pool_id);
|
|
|
|
|
2019-05-28 16:59:38 +00:00
|
|
|
if ($this->have_permission($user, $pool)) {
|
2020-03-27 00:15:15 +00:00
|
|
|
$result = $database->execute("SELECT image_id FROM pool_images WHERE pool_id=:pid ORDER BY image_order ASC", ["pid" => $pool_id]);
|
|
|
|
$images = [];
|
|
|
|
while ($row = $result->fetch()) {
|
|
|
|
$images[] = Image::by_id((int)$row["image_id"]);
|
|
|
|
}
|
|
|
|
$this->theme->edit_pool($page, $pool, $images);
|
2019-05-28 16:59:38 +00:00
|
|
|
} else {
|
2019-06-19 01:58:28 +00:00
|
|
|
$page->set_mode(PageMode::REDIRECT);
|
2019-06-21 20:13:47 +00:00
|
|
|
$page->set_redirect(make_link("pool/view/" . $pool_id));
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case "order": // Order the pool (view and change the order of images within the pool)
|
2024-01-20 20:48:47 +00:00
|
|
|
$pool_id = int_escape($_POST["pool_id"]);
|
|
|
|
$pool = $this->get_single_pool($pool_id);
|
|
|
|
|
2019-05-28 16:59:38 +00:00
|
|
|
if (isset($_POST["order_view"])) {
|
|
|
|
if ($this->have_permission($user, $pool)) {
|
2020-03-27 00:15:15 +00:00
|
|
|
$result = $database->execute(
|
|
|
|
"SELECT image_id FROM pool_images WHERE pool_id=:pid ORDER BY image_order ASC",
|
|
|
|
["pid" => $pool_id]
|
|
|
|
);
|
|
|
|
$images = [];
|
|
|
|
|
|
|
|
while ($row = $result->fetch()) {
|
|
|
|
$image = $database->get_row(
|
|
|
|
"
|
|
|
|
SELECT * FROM images AS i
|
|
|
|
INNER JOIN pool_images AS p ON i.id = p.image_id
|
|
|
|
WHERE pool_id=:pid AND i.id=:iid",
|
|
|
|
["pid" => $pool_id, "iid" => (int)$row['image_id']]
|
|
|
|
);
|
|
|
|
$images[] = ($image ? new Image($image) : null);
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->theme->edit_order($page, $pool, $images);
|
2019-05-28 16:59:38 +00:00
|
|
|
} else {
|
2019-06-19 01:58:28 +00:00
|
|
|
$page->set_mode(PageMode::REDIRECT);
|
2019-06-21 20:13:47 +00:00
|
|
|
$page->set_redirect(make_link("pool/view/" . $pool_id));
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if ($this->have_permission($user, $pool)) {
|
2020-03-27 00:15:15 +00:00
|
|
|
foreach ($_POST['imgs'] as $data) {
|
|
|
|
list($imageORDER, $imageID) = $data;
|
|
|
|
$database->execute(
|
|
|
|
"
|
|
|
|
UPDATE pool_images
|
|
|
|
SET image_order = :ord
|
|
|
|
WHERE pool_id = :pid AND image_id = :iid",
|
|
|
|
["ord" => $imageORDER, "pid" => int_escape($_POST['pool_id']), "iid" => $imageID]
|
|
|
|
);
|
|
|
|
}
|
2019-06-19 01:58:28 +00:00
|
|
|
$page->set_mode(PageMode::REDIRECT);
|
2019-06-21 20:13:47 +00:00
|
|
|
$page->set_redirect(make_link("pool/view/" . $pool_id));
|
2019-05-28 16:59:38 +00:00
|
|
|
} else {
|
|
|
|
$this->theme->display_error(403, "Permission Denied", "You do not have permission to access this page");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2023-11-07 21:01:03 +00:00
|
|
|
case "reverse":
|
2024-01-20 20:48:47 +00:00
|
|
|
$pool_id = int_escape($_POST["pool_id"]);
|
|
|
|
$pool = $this->get_single_pool($pool_id);
|
|
|
|
|
2023-11-07 21:01:03 +00:00
|
|
|
if ($this->have_permission($user, $pool)) {
|
2024-01-09 21:59:24 +00:00
|
|
|
$database->with_savepoint(function () use ($pool_id) {
|
|
|
|
global $database;
|
|
|
|
$result = $database->execute(
|
|
|
|
"SELECT image_id FROM pool_images WHERE pool_id=:pid ORDER BY image_order DESC",
|
|
|
|
["pid" => $pool_id]
|
|
|
|
);
|
|
|
|
$image_order = 1;
|
2023-11-07 21:01:03 +00:00
|
|
|
while ($row = $result->fetch()) {
|
|
|
|
$database->execute(
|
|
|
|
"
|
2024-01-09 21:59:24 +00:00
|
|
|
UPDATE pool_images
|
|
|
|
SET image_order=:ord
|
|
|
|
WHERE pool_id = :pid AND image_id = :iid",
|
2023-11-07 21:01:03 +00:00
|
|
|
["ord" => $image_order, "pid" => $pool_id, "iid" => (int)$row['image_id']]
|
|
|
|
);
|
|
|
|
$image_order = $image_order + 1;
|
|
|
|
}
|
2024-01-09 21:59:24 +00:00
|
|
|
});
|
2023-11-07 21:01:03 +00:00
|
|
|
$page->set_mode(PageMode::REDIRECT);
|
|
|
|
$page->set_redirect(make_link("pool/view/" . $pool_id));
|
|
|
|
} else {
|
|
|
|
$this->theme->display_error(403, "Permission Denied", "You do not have permission to access this page");
|
|
|
|
}
|
|
|
|
break;
|
2019-05-28 16:59:38 +00:00
|
|
|
case "import":
|
2024-01-20 20:48:47 +00:00
|
|
|
$pool_id = int_escape($_POST["pool_id"]);
|
|
|
|
$pool = $this->get_single_pool($pool_id);
|
|
|
|
|
2019-05-28 16:59:38 +00:00
|
|
|
if ($this->have_permission($user, $pool)) {
|
2023-12-14 16:33:21 +00:00
|
|
|
$images = Search::find_images(
|
2023-06-27 14:32:39 +00:00
|
|
|
limit: $config->get_int(PoolsConfig::MAX_IMPORT_RESULTS, 1000),
|
|
|
|
tags: Tag::explode($_POST["pool_tag"])
|
2020-03-27 00:15:15 +00:00
|
|
|
);
|
|
|
|
$this->theme->pool_result($page, $images, $pool);
|
2019-05-28 16:59:38 +00:00
|
|
|
} else {
|
|
|
|
$this->theme->display_error(403, "Permission Denied", "You do not have permission to access this page");
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case "add_posts":
|
2024-01-20 20:48:47 +00:00
|
|
|
$pool_id = int_escape($_POST["pool_id"]);
|
|
|
|
$pool = $this->get_single_pool($pool_id);
|
|
|
|
|
2019-05-28 16:59:38 +00:00
|
|
|
if ($this->have_permission($user, $pool)) {
|
2020-03-27 14:41:24 +00:00
|
|
|
$image_ids = array_map('intval', $_POST['check']);
|
|
|
|
send_event(new PoolAddPostsEvent($pool_id, $image_ids));
|
2019-06-19 01:58:28 +00:00
|
|
|
$page->set_mode(PageMode::REDIRECT);
|
2019-06-21 20:13:47 +00:00
|
|
|
$page->set_redirect(make_link("pool/view/" . $pool_id));
|
2019-05-28 16:59:38 +00:00
|
|
|
} else {
|
|
|
|
$this->theme->display_error(403, "Permission Denied", "You do not have permission to access this page");
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case "remove_posts":
|
2024-01-20 20:48:47 +00:00
|
|
|
$pool_id = int_escape($_POST["pool_id"]);
|
|
|
|
$pool = $this->get_single_pool($pool_id);
|
|
|
|
|
2019-05-28 16:59:38 +00:00
|
|
|
if ($this->have_permission($user, $pool)) {
|
2020-03-27 00:15:15 +00:00
|
|
|
$images = "";
|
|
|
|
foreach ($_POST['check'] as $imageID) {
|
|
|
|
$database->execute(
|
|
|
|
"DELETE FROM pool_images WHERE pool_id = :pid AND image_id = :iid",
|
|
|
|
["pid" => $pool_id, "iid" => $imageID]
|
|
|
|
);
|
|
|
|
$images .= " " . $imageID;
|
|
|
|
}
|
|
|
|
$count = (int)$database->get_one(
|
|
|
|
"SELECT COUNT(*) FROM pool_images WHERE pool_id=:pid",
|
|
|
|
["pid" => $pool_id]
|
|
|
|
);
|
|
|
|
$this->add_history($pool_id, 0, $images, $count);
|
2019-06-19 01:58:28 +00:00
|
|
|
$page->set_mode(PageMode::REDIRECT);
|
2019-06-21 20:13:47 +00:00
|
|
|
$page->set_redirect(make_link("pool/view/" . $pool_id));
|
2019-05-28 16:59:38 +00:00
|
|
|
} else {
|
|
|
|
$this->theme->display_error(403, "Permission Denied", "You do not have permission to access this page");
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case "edit_description":
|
2024-01-20 20:48:47 +00:00
|
|
|
$pool_id = int_escape($_POST["pool_id"]);
|
|
|
|
$pool = $this->get_single_pool($pool_id);
|
|
|
|
|
2019-05-28 16:59:38 +00:00
|
|
|
if ($this->have_permission($user, $pool)) {
|
2020-03-27 00:15:15 +00:00
|
|
|
$database->execute(
|
2023-08-18 21:23:40 +00:00
|
|
|
"UPDATE pools SET description=:dsc,lastupdated=CURRENT_TIMESTAMP WHERE id=:pid",
|
2020-03-27 14:41:24 +00:00
|
|
|
["dsc" => $_POST['description'], "pid" => $pool_id]
|
2020-03-27 00:15:15 +00:00
|
|
|
);
|
2019-06-19 01:58:28 +00:00
|
|
|
$page->set_mode(PageMode::REDIRECT);
|
2019-06-21 20:13:47 +00:00
|
|
|
$page->set_redirect(make_link("pool/view/" . $pool_id));
|
2019-05-28 16:59:38 +00:00
|
|
|
} else {
|
|
|
|
$this->theme->display_error(403, "Permission Denied", "You do not have permission to access this page");
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case "nuke":
|
|
|
|
// Completely remove the given pool.
|
|
|
|
// -> Only admins and owners may do this
|
2024-01-20 20:48:47 +00:00
|
|
|
$pool_id = int_escape($_POST["pool_id"]);
|
|
|
|
$pool = $this->get_single_pool($pool_id);
|
|
|
|
|
2020-03-27 14:41:24 +00:00
|
|
|
if ($user->can(Permissions::POOLS_ADMIN) || $user->id == $pool->user_id) {
|
2020-10-24 17:55:07 +00:00
|
|
|
send_event(new PoolDeletionEvent($pool_id));
|
2019-06-19 01:58:28 +00:00
|
|
|
$page->set_mode(PageMode::REDIRECT);
|
2019-05-28 16:59:38 +00:00
|
|
|
$page->set_redirect(make_link("pool/list"));
|
|
|
|
} else {
|
|
|
|
$this->theme->display_error(403, "Permission Denied", "You do not have permission to access this page");
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-15 11:52:35 +00:00
|
|
|
public function onUserBlockBuilding(UserBlockBuildingEvent $event): void
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
|
|
|
$event->add_link("Pools", make_link("pool/list"));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* When displaying an image, optionally list all the pools that the
|
|
|
|
* image is currently a member of on a side panel, as well as a link
|
|
|
|
* to the Next image in the pool.
|
|
|
|
*/
|
2024-01-15 11:52:35 +00:00
|
|
|
public function onDisplayingImage(DisplayingImageEvent $event): void
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
|
|
|
global $config;
|
|
|
|
|
2019-06-21 20:13:47 +00:00
|
|
|
if ($config->get_bool(PoolsConfig::INFO_ON_VIEW_IMAGE)) {
|
2019-05-28 16:59:38 +00:00
|
|
|
$imageID = $event->image->id;
|
|
|
|
$poolsIDs = $this->get_pool_ids($imageID);
|
|
|
|
|
2019-06-21 20:13:47 +00:00
|
|
|
$show_nav = $config->get_bool(PoolsConfig::SHOW_NAV_LINKS, false);
|
2019-05-28 16:59:38 +00:00
|
|
|
|
|
|
|
$navInfo = [];
|
|
|
|
foreach ($poolsIDs as $poolID) {
|
|
|
|
$pool = $this->get_single_pool($poolID);
|
|
|
|
|
2020-03-27 00:15:15 +00:00
|
|
|
$navInfo[$pool->id] = [
|
|
|
|
"info" => $pool,
|
|
|
|
"nav" => $show_nav ? $this->get_nav_posts($pool, $imageID) : null,
|
|
|
|
];
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
$this->theme->pool_info($navInfo);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-15 11:52:35 +00:00
|
|
|
public function onImageAdminBlockBuilding(ImageAdminBlockBuildingEvent $event): void
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
|
|
|
global $config, $database, $user;
|
2019-06-21 20:13:47 +00:00
|
|
|
if ($config->get_bool(PoolsConfig::ADDER_ON_VIEW_IMAGE) && !$user->is_anonymous()) {
|
2023-06-25 23:49:19 +00:00
|
|
|
$pools = [];
|
2019-09-29 18:00:51 +00:00
|
|
|
if ($user->can(Permissions::POOLS_ADMIN)) {
|
2023-06-25 23:49:19 +00:00
|
|
|
$pools = $database->get_pairs("SELECT id,title FROM pools ORDER BY title");
|
2019-05-28 16:59:38 +00:00
|
|
|
} else {
|
2023-06-25 23:53:44 +00:00
|
|
|
$pools = $database->get_pairs("SELECT id,title FROM pools WHERE user_id=:id ORDER BY title", ["id" => $user->id]);
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
2023-06-25 23:49:19 +00:00
|
|
|
if (count($pools) > 0) {
|
2024-01-19 18:57:02 +00:00
|
|
|
$event->add_part($this->theme->get_adder_html($event->image, $pools));
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-15 11:52:35 +00:00
|
|
|
public function onHelpPageBuilding(HelpPageBuildingEvent $event): void
|
2019-08-02 20:05:49 +00:00
|
|
|
{
|
2023-11-11 21:49:12 +00:00
|
|
|
if ($event->key === HelpPages::SEARCH) {
|
2023-07-05 00:23:11 +00:00
|
|
|
$event->add_block(new Block("Pools", $this->theme->get_help_html()));
|
2019-08-02 20:05:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-15 11:52:35 +00:00
|
|
|
public function onSearchTermParse(SearchTermParseEvent $event): void
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
2020-01-26 16:38:26 +00:00
|
|
|
if (is_null($event->term)) {
|
|
|
|
return;
|
|
|
|
}
|
2020-01-26 13:19:35 +00:00
|
|
|
|
2019-05-28 16:59:38 +00:00
|
|
|
$matches = [];
|
|
|
|
if (preg_match("/^pool[=|:]([0-9]+|any|none)$/i", $event->term, $matches)) {
|
|
|
|
$poolID = $matches[1];
|
|
|
|
|
|
|
|
if (preg_match("/^(any|none)$/", $poolID)) {
|
|
|
|
$not = ($poolID == "none" ? "NOT" : "");
|
|
|
|
$event->add_querylet(new Querylet("images.id $not IN (SELECT DISTINCT image_id FROM pool_images)"));
|
|
|
|
} else {
|
|
|
|
$event->add_querylet(new Querylet("images.id IN (SELECT DISTINCT image_id FROM pool_images WHERE pool_id = $poolID)"));
|
|
|
|
}
|
|
|
|
} elseif (preg_match("/^pool_by_name[=|:](.*)$/i", $event->term, $matches)) {
|
|
|
|
$poolTitle = str_replace("_", " ", $matches[1]);
|
|
|
|
|
|
|
|
$pool = $this->get_single_pool_from_title($poolTitle);
|
|
|
|
$poolID = 0;
|
|
|
|
if ($pool) {
|
2020-03-27 14:41:24 +00:00
|
|
|
$poolID = $pool->id;
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
$event->add_querylet(new Querylet("images.id IN (SELECT DISTINCT image_id FROM pool_images WHERE pool_id = $poolID)"));
|
2022-10-07 20:19:50 +00:00
|
|
|
} elseif (preg_match("/^pool_id[=|:](.*)$/i", $event->term, $matches)) {
|
|
|
|
$poolID = str_replace("_", " ", $matches[1]);
|
|
|
|
$event->add_querylet(new Querylet("images.id IN (SELECT DISTINCT image_id FROM pool_images WHERE pool_id = $poolID)"));
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
2023-11-07 21:01:03 +00:00
|
|
|
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
|
2024-01-15 11:52:35 +00:00
|
|
|
public function onTagTermCheck(TagTermCheckEvent $event): void
|
2020-01-29 20:22:50 +00:00
|
|
|
{
|
|
|
|
if (preg_match("/^pool[=|:]([^:]*|lastcreated):?([0-9]*)$/i", $event->term)) {
|
|
|
|
$event->metatag = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-15 11:52:35 +00:00
|
|
|
public function onTagTermParse(TagTermParseEvent $event): void
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
|
|
|
$matches = [];
|
|
|
|
if (preg_match("/^pool[=|:]([^:]*|lastcreated):?([0-9]*)$/i", $event->term, $matches)) {
|
|
|
|
global $user;
|
2019-06-21 20:13:47 +00:00
|
|
|
$poolTag = (string)str_replace("_", " ", $matches[1]);
|
2019-05-28 16:59:38 +00:00
|
|
|
|
|
|
|
$pool = null;
|
|
|
|
if ($poolTag == 'lastcreated') {
|
|
|
|
$pool = $this->get_last_userpool($user->id);
|
|
|
|
} elseif (ctype_digit($poolTag)) { //If only digits, assume PoolID
|
2020-01-26 13:19:35 +00:00
|
|
|
$pool = $this->get_single_pool((int)$poolTag);
|
2019-05-28 16:59:38 +00:00
|
|
|
} else { //assume PoolTitle
|
|
|
|
$pool = $this->get_single_pool_from_title($poolTag);
|
|
|
|
}
|
|
|
|
|
2020-03-27 00:15:15 +00:00
|
|
|
if ($pool && $this->have_permission($user, $pool)) {
|
2024-01-20 20:48:47 +00:00
|
|
|
$image_order = (int)($matches[2] ?: 0);
|
2020-03-27 14:41:24 +00:00
|
|
|
$this->add_post($pool->id, $event->image_id, true, $image_order);
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-15 11:52:35 +00:00
|
|
|
public function onBulkActionBlockBuilding(BulkActionBlockBuildingEvent $event): void
|
2019-06-21 20:13:47 +00:00
|
|
|
{
|
2019-10-02 10:23:57 +00:00
|
|
|
global $database;
|
2019-06-21 20:13:47 +00:00
|
|
|
|
2023-06-25 23:05:38 +00:00
|
|
|
$options = $database->get_pairs("SELECT id,title FROM pools ORDER BY title");
|
2019-06-21 20:13:47 +00:00
|
|
|
|
2023-06-29 05:50:32 +00:00
|
|
|
// TODO: Don't cast into strings, make BABBE accept HTMLElement instead.
|
|
|
|
$event->add_action("bulk_pool_add_existing", "Add To (P)ool", "p", "", (string)$this->theme->get_bulk_pool_selector($options));
|
|
|
|
$event->add_action("bulk_pool_add_new", "Create Pool", "", "", (string)$this->theme->get_bulk_pool_input($event->search_terms));
|
2019-06-21 20:13:47 +00:00
|
|
|
}
|
|
|
|
|
2024-01-15 11:52:35 +00:00
|
|
|
public function onBulkAction(BulkActionEvent $event): void
|
2019-06-21 20:13:47 +00:00
|
|
|
{
|
|
|
|
global $user;
|
|
|
|
|
|
|
|
switch ($event->action) {
|
|
|
|
case "bulk_pool_add_existing":
|
|
|
|
if (!isset($_POST['bulk_pool_select'])) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
$pool_id = intval($_POST['bulk_pool_select']);
|
2020-03-27 00:15:15 +00:00
|
|
|
$pool = $this->get_single_pool($pool_id);
|
2019-06-21 20:13:47 +00:00
|
|
|
|
|
|
|
if ($this->have_permission($user, $pool)) {
|
2019-07-05 16:07:27 +00:00
|
|
|
send_event(
|
2023-01-11 11:15:26 +00:00
|
|
|
new PoolAddPostsEvent($pool_id, iterator_map_to_array("Shimmie2\_image_to_id", $event->items))
|
2019-09-29 13:30:55 +00:00
|
|
|
);
|
2019-06-21 20:13:47 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case "bulk_pool_add_new":
|
|
|
|
if (!isset($_POST['bulk_pool_new'])) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
$new_pool_title = $_POST['bulk_pool_new'];
|
2023-02-04 20:50:26 +00:00
|
|
|
$pce = send_event(new PoolCreationEvent($new_pool_title));
|
2023-01-11 11:15:26 +00:00
|
|
|
send_event(new PoolAddPostsEvent($pce->new_id, iterator_map_to_array("Shimmie2\_image_to_id", $event->items)));
|
2019-06-21 20:13:47 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-28 16:59:38 +00:00
|
|
|
/* ------------------------------------------------- */
|
|
|
|
/* -------------- Private Functions -------------- */
|
|
|
|
/* ------------------------------------------------- */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check if the given user has permission to edit/change the pool.
|
|
|
|
*
|
|
|
|
* TODO: Should the user variable be global?
|
|
|
|
*/
|
2020-03-27 00:15:15 +00:00
|
|
|
private function have_permission(User $user, Pool $pool): bool
|
|
|
|
{
|
|
|
|
// If the pool is public and user is logged
|
|
|
|
// OR if the user is admin
|
|
|
|
// OR if the pool is owned by the user.
|
|
|
|
return (
|
|
|
|
($pool->public && !$user->is_anonymous()) ||
|
|
|
|
$user->can(Permissions::POOLS_ADMIN) ||
|
|
|
|
$user->id == $pool->user_id
|
|
|
|
);
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
|
2024-01-20 14:10:59 +00:00
|
|
|
private function list_pools(Page $page, int $pageNumber, ?string $search): void
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
|
|
|
global $config, $database;
|
|
|
|
|
2019-06-21 20:13:47 +00:00
|
|
|
$poolsPerPage = $config->get_int(PoolsConfig::LISTS_PER_PAGE);
|
2019-05-28 16:59:38 +00:00
|
|
|
|
|
|
|
$order_by = "";
|
|
|
|
$order = $page->get_cookie("ui-order-pool");
|
|
|
|
if ($order == "created" || is_null($order)) {
|
|
|
|
$order_by = "ORDER BY p.date DESC";
|
|
|
|
} elseif ($order == "updated") {
|
|
|
|
$order_by = "ORDER BY p.lastupdated DESC";
|
|
|
|
} elseif ($order == "name") {
|
|
|
|
$order_by = "ORDER BY p.title ASC";
|
|
|
|
} elseif ($order == "count") {
|
|
|
|
$order_by = "ORDER BY p.posts DESC";
|
|
|
|
}
|
|
|
|
|
2023-11-07 21:01:03 +00:00
|
|
|
$where_clause = "WHERE LOWER(title) like '%%'";
|
|
|
|
if ($search != null) {
|
|
|
|
$where_clause = "WHERE LOWER(title) like '%".strtolower($search)."%'";
|
|
|
|
}
|
|
|
|
|
2020-03-27 00:15:15 +00:00
|
|
|
$pools = array_map([Pool::class, "makePool"], $database->get_all("
|
|
|
|
SELECT p.*, u.name as user_name
|
2015-09-26 18:53:15 +00:00
|
|
|
FROM pools AS p
|
|
|
|
INNER JOIN users AS u
|
|
|
|
ON p.user_id = u.id
|
2023-11-06 05:50:33 +00:00
|
|
|
$where_clause
|
2015-09-26 18:53:15 +00:00
|
|
|
$order_by
|
|
|
|
LIMIT :l OFFSET :o
|
2020-03-27 00:15:15 +00:00
|
|
|
", ["l" => $poolsPerPage, "o" => $pageNumber * $poolsPerPage]));
|
2023-11-06 05:50:33 +00:00
|
|
|
$totalPages = (int)ceil((int)$database->get_one("SELECT COUNT(*) FROM pools ".$where_clause) / $poolsPerPage);
|
2019-05-28 16:59:38 +00:00
|
|
|
|
2023-11-06 05:50:33 +00:00
|
|
|
$this->theme->list_pools($page, $pools, $search, $pageNumber + 1, $totalPages);
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
|
2024-01-15 11:52:35 +00:00
|
|
|
public function onPoolCreation(PoolCreationEvent $event): void
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
|
|
|
global $user, $database;
|
|
|
|
|
|
|
|
if ($user->is_anonymous()) {
|
|
|
|
throw new PoolCreationException("You must be registered and logged in to add a image.");
|
|
|
|
}
|
2019-06-21 20:13:47 +00:00
|
|
|
if (empty($event->title)) {
|
2019-05-28 16:59:38 +00:00
|
|
|
throw new PoolCreationException("Pool title is empty.");
|
|
|
|
}
|
2019-06-21 20:13:47 +00:00
|
|
|
if ($this->get_single_pool_from_title($event->title)) {
|
2019-05-28 16:59:38 +00:00
|
|
|
throw new PoolCreationException("A pool using this title already exists.");
|
|
|
|
}
|
|
|
|
|
2023-06-25 20:31:11 +00:00
|
|
|
$database->execute(
|
2019-05-28 16:59:38 +00:00
|
|
|
"
|
2009-12-26 00:31:02 +00:00
|
|
|
INSERT INTO pools (user_id, public, title, description, date)
|
2023-06-25 20:31:11 +00:00
|
|
|
VALUES (:uid, :public, :title, :desc, now())",
|
2020-03-27 00:15:15 +00:00
|
|
|
["uid" => $event->user->id, "public" => $event->public, "title" => $event->title, "desc" => $event->description]
|
2019-05-28 16:59:38 +00:00
|
|
|
);
|
2023-06-25 20:31:11 +00:00
|
|
|
|
|
|
|
$poolID = $database->get_last_insert_id('pools_id_seq');
|
2019-05-28 16:59:38 +00:00
|
|
|
log_info("pools", "Pool {$poolID} created by {$user->name}");
|
2019-06-21 20:13:47 +00:00
|
|
|
|
|
|
|
$event->new_id = $poolID;
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve information about a pool given a pool ID.
|
|
|
|
*/
|
2020-03-27 00:15:15 +00:00
|
|
|
private function get_single_pool(int $poolID): Pool
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
|
|
|
global $database;
|
2020-03-27 00:15:15 +00:00
|
|
|
return new Pool($database->get_row("SELECT * FROM pools WHERE id=:id", ["id" => $poolID]));
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve information about a pool given a pool title.
|
|
|
|
*/
|
2020-03-27 00:15:15 +00:00
|
|
|
private function get_single_pool_from_title(string $poolTitle): ?Pool
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
|
|
|
global $database;
|
2020-03-27 00:15:15 +00:00
|
|
|
$row = $database->get_row("SELECT * FROM pools WHERE title=:title", ["title" => $poolTitle]);
|
|
|
|
return $row ? new Pool($row) : null;
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get all of the pool IDs that an image is in, given an image ID.
|
2024-01-06 21:28:06 +00:00
|
|
|
* @return int[]
|
2019-05-28 16:59:38 +00:00
|
|
|
*/
|
|
|
|
private function get_pool_ids(int $imageID): array
|
|
|
|
{
|
|
|
|
global $database;
|
2020-03-26 16:50:16 +00:00
|
|
|
$col = $database->get_col("SELECT pool_id FROM pool_images WHERE image_id=:iid", ["iid" => $imageID]);
|
|
|
|
$col = array_map('intval', $col);
|
|
|
|
return $col;
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve information about the last pool the given userID created
|
|
|
|
*/
|
2020-03-27 00:15:15 +00:00
|
|
|
private function get_last_userpool(int $userID): Pool
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
|
|
|
global $database;
|
2020-03-27 00:15:15 +00:00
|
|
|
return new Pool($database->get_row("SELECT * FROM pools WHERE user_id=:uid ORDER BY id DESC", ["uid" => $userID]));
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* HERE WE ADD CHECKED IMAGES FROM POOL AND UPDATE THE HISTORY
|
|
|
|
*/
|
2024-01-15 11:52:35 +00:00
|
|
|
public function onPoolAddPosts(PoolAddPostsEvent $event): void
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
2019-06-21 20:13:47 +00:00
|
|
|
global $database, $user;
|
2019-05-28 16:59:38 +00:00
|
|
|
|
2019-06-21 20:13:47 +00:00
|
|
|
$pool = $this->get_single_pool($event->pool_id);
|
|
|
|
if (!$this->have_permission($user, $pool)) {
|
|
|
|
return;
|
|
|
|
}
|
2009-12-25 23:05:57 +00:00
|
|
|
|
2024-01-15 13:40:18 +00:00
|
|
|
$images = [];
|
2019-06-21 20:13:47 +00:00
|
|
|
foreach ($event->posts as $post_id) {
|
|
|
|
if ($this->add_post($event->pool_id, $post_id, false)) {
|
2024-01-15 13:40:18 +00:00
|
|
|
$images[] = $post_id;
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
}
|
2009-12-25 23:05:57 +00:00
|
|
|
|
2024-01-15 13:40:18 +00:00
|
|
|
if (count($images) > 0) {
|
2020-03-27 00:15:15 +00:00
|
|
|
$count = (int)$database->get_one(
|
|
|
|
"SELECT COUNT(*) FROM pool_images WHERE pool_id=:pid",
|
|
|
|
["pid" => $event->pool_id]
|
2019-05-28 16:59:38 +00:00
|
|
|
);
|
2024-01-15 13:40:18 +00:00
|
|
|
$this->add_history($event->pool_id, 1, implode(" ", $images), $count);
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the previous and next successive images from a pool, given a pool ID and an image ID.
|
|
|
|
*
|
2024-01-20 14:10:59 +00:00
|
|
|
* @return array{prev:?int,next:?int} Array returning two elements (prev, next) in 1 dimension. Each returns ImageID or NULL if none.
|
2019-05-28 16:59:38 +00:00
|
|
|
*/
|
2024-01-20 14:10:59 +00:00
|
|
|
private function get_nav_posts(Pool $pool, int $imageID): array
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
|
|
|
global $database;
|
|
|
|
|
2020-03-27 00:15:15 +00:00
|
|
|
return $database->get_row(
|
2019-05-28 16:59:38 +00:00
|
|
|
"
|
2024-01-20 14:10:59 +00:00
|
|
|
SELECT (
|
|
|
|
SELECT image_id
|
|
|
|
FROM pool_images
|
|
|
|
WHERE pool_id = :pid
|
|
|
|
AND image_order < (
|
|
|
|
SELECT image_order
|
|
|
|
FROM pool_images
|
|
|
|
WHERE pool_id = :pid
|
|
|
|
AND image_id = :iid
|
|
|
|
LIMIT 1
|
|
|
|
)
|
|
|
|
ORDER BY image_order DESC LIMIT 1
|
|
|
|
) AS prev,
|
|
|
|
(
|
|
|
|
SELECT image_id
|
|
|
|
FROM pool_images
|
|
|
|
WHERE pool_id = :pid
|
|
|
|
AND image_order > (
|
|
|
|
SELECT image_order
|
|
|
|
FROM pool_images
|
|
|
|
WHERE pool_id = :pid
|
|
|
|
AND image_id = :iid
|
|
|
|
LIMIT 1
|
|
|
|
)
|
|
|
|
ORDER BY image_order ASC LIMIT 1
|
|
|
|
) AS next
|
|
|
|
|
|
|
|
LIMIT 1
|
|
|
|
",
|
2020-03-27 00:15:15 +00:00
|
|
|
["pid" => $pool->id, "iid" => $imageID]
|
2019-05-28 16:59:38 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve all the images in a pool, given a pool ID.
|
|
|
|
*/
|
2024-01-20 14:10:59 +00:00
|
|
|
private function get_posts(PageRequestEvent $event, int $poolID): void
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
|
|
|
global $config, $user, $database;
|
|
|
|
|
2020-03-27 14:41:24 +00:00
|
|
|
$pageNumber = $event->try_page_num(2);
|
2020-03-27 00:15:15 +00:00
|
|
|
$pool = $this->get_single_pool($poolID);
|
2019-06-21 20:13:47 +00:00
|
|
|
$imagesPerPage = $config->get_int(PoolsConfig::IMAGES_PER_PAGE);
|
2019-05-28 16:59:38 +00:00
|
|
|
|
2019-06-25 23:47:06 +00:00
|
|
|
$query = "
|
2020-03-27 00:15:15 +00:00
|
|
|
INNER JOIN images AS i ON i.id = p.image_id
|
|
|
|
WHERE p.pool_id = :pid
|
2019-06-25 23:47:06 +00:00
|
|
|
";
|
2020-10-26 23:32:56 +00:00
|
|
|
$params = [];
|
2019-06-25 23:47:06 +00:00
|
|
|
|
2019-05-28 16:59:38 +00:00
|
|
|
// WE CHECK IF THE EXTENSION RATING IS INSTALLED, WHICH VERSION AND IF IT
|
|
|
|
// WORKS TO SHOW/HIDE SAFE, QUESTIONABLE, EXPLICIT AND UNRATED IMAGES FROM USER
|
2019-08-07 19:53:59 +00:00
|
|
|
if (Extension::is_enabled(RatingsInfo::KEY)) {
|
2019-06-27 04:11:59 +00:00
|
|
|
$query .= "AND i.rating IN (".Ratings::privs_to_sql(Ratings::get_user_class_privs($user)).")";
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
2019-09-29 13:30:55 +00:00
|
|
|
if (Extension::is_enabled(TrashInfo::KEY)) {
|
2020-10-26 23:26:49 +00:00
|
|
|
$query .= " AND trash != :true";
|
2020-10-26 23:32:56 +00:00
|
|
|
$params["true"] = true;
|
2019-06-25 23:47:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
$result = $database->get_all(
|
|
|
|
"
|
|
|
|
SELECT p.image_id FROM pool_images p
|
|
|
|
$query
|
2009-12-26 00:31:02 +00:00
|
|
|
ORDER BY p.image_order ASC
|
2011-08-24 08:01:43 +00:00
|
|
|
LIMIT :l OFFSET :o",
|
2020-10-26 23:26:49 +00:00
|
|
|
[
|
|
|
|
"pid" => $poolID,
|
|
|
|
"l" => $imagesPerPage,
|
|
|
|
"o" => $pageNumber * $imagesPerPage,
|
2020-10-26 23:32:56 +00:00
|
|
|
] + $params
|
2019-06-25 23:47:06 +00:00
|
|
|
);
|
2009-12-26 00:31:02 +00:00
|
|
|
|
2020-03-26 18:52:31 +00:00
|
|
|
$totalPages = (int)ceil((int)$database->get_one(
|
|
|
|
"SELECT COUNT(*) FROM pool_images p $query",
|
2020-10-26 23:32:56 +00:00
|
|
|
["pid" => $poolID] + $params
|
2019-11-02 19:57:34 +00:00
|
|
|
) / $imagesPerPage);
|
2019-06-21 20:13:47 +00:00
|
|
|
|
2019-05-28 16:59:38 +00:00
|
|
|
$images = [];
|
|
|
|
foreach ($result as $singleResult) {
|
2020-03-26 18:52:31 +00:00
|
|
|
$images[] = Image::by_id((int)$singleResult["image_id"]);
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
$this->theme->view_pool($pool, $images, $pageNumber + 1, $totalPages);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* HERE WE NUKE ENTIRE POOL. WE REMOVE POOLS AND POSTS FROM REMOVED POOL AND HISTORIES ENTRIES FROM REMOVED POOL.
|
|
|
|
*/
|
2024-01-15 11:52:35 +00:00
|
|
|
public function onPoolDeletion(PoolDeletionEvent $event): void
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
|
|
|
global $user, $database;
|
2020-10-24 17:55:07 +00:00
|
|
|
$poolID = $event->pool_id;
|
2019-05-28 16:59:38 +00:00
|
|
|
|
2020-10-24 17:55:07 +00:00
|
|
|
$owner_id = (int)$database->get_one("SELECT user_id FROM pools WHERE id = :pid", ["pid" => $poolID]);
|
|
|
|
if ($owner_id == $user->id || $user->can(Permissions::POOLS_ADMIN)) {
|
2019-06-21 20:13:47 +00:00
|
|
|
$database->execute("DELETE FROM pool_history WHERE pool_id = :pid", ["pid" => $poolID]);
|
|
|
|
$database->execute("DELETE FROM pool_images WHERE pool_id = :pid", ["pid" => $poolID]);
|
|
|
|
$database->execute("DELETE FROM pools WHERE id = :pid", ["pid" => $poolID]);
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* HERE WE ADD A HISTORY ENTRY.
|
|
|
|
*
|
|
|
|
* $action Action=1 (one) MEANS ADDED, Action=0 (zero) MEANS REMOVED
|
|
|
|
*/
|
2024-01-20 14:10:59 +00:00
|
|
|
private function add_history(int $poolID, int $action, string $images, int $count): void
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
|
|
|
global $user, $database;
|
|
|
|
|
|
|
|
$database->execute(
|
|
|
|
"
|
2009-12-26 00:31:02 +00:00
|
|
|
INSERT INTO pool_history (pool_id, user_id, action, images, count, date)
|
2011-08-24 08:01:43 +00:00
|
|
|
VALUES (:pid, :uid, :act, :img, :count, now())",
|
2019-06-21 20:13:47 +00:00
|
|
|
["pid" => $poolID, "uid" => $user->id, "act" => $action, "img" => $images, "count" => $count]
|
2019-05-28 16:59:38 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2024-01-20 14:10:59 +00:00
|
|
|
private function get_history(int $pageNumber): void
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
|
|
|
global $config, $database;
|
|
|
|
|
2019-06-21 20:13:47 +00:00
|
|
|
$historiesPerPage = $config->get_int(PoolsConfig::UPDATED_PER_PAGE);
|
2019-05-28 16:59:38 +00:00
|
|
|
|
|
|
|
$history = $database->get_all("
|
2010-05-04 19:10:37 +00:00
|
|
|
SELECT h.id, h.pool_id, h.user_id, h.action, h.images,
|
|
|
|
h.count, h.date, u.name as user_name, p.title as title
|
|
|
|
FROM pool_history AS h
|
|
|
|
INNER JOIN pools AS p
|
|
|
|
ON p.id = h.pool_id
|
|
|
|
INNER JOIN users AS u
|
|
|
|
ON h.user_id = u.id
|
|
|
|
ORDER BY h.date DESC
|
2011-08-24 08:01:43 +00:00
|
|
|
LIMIT :l OFFSET :o
|
2019-06-21 20:13:47 +00:00
|
|
|
", ["l" => $historiesPerPage, "o" => $pageNumber * $historiesPerPage]);
|
2019-05-28 16:59:38 +00:00
|
|
|
|
2020-03-26 18:52:31 +00:00
|
|
|
$totalPages = (int)ceil((int)$database->get_one("SELECT COUNT(*) FROM pool_history") / $historiesPerPage);
|
2019-05-28 16:59:38 +00:00
|
|
|
|
|
|
|
$this->theme->show_history($history, $pageNumber + 1, $totalPages);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* HERE GO BACK IN HISTORY AND ADD OR REMOVE POSTS TO POOL.
|
|
|
|
*/
|
2024-01-20 14:10:59 +00:00
|
|
|
private function revert_history(int $historyID): void
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
|
|
|
global $database;
|
2019-06-21 20:13:47 +00:00
|
|
|
$status = $database->get_all("SELECT * FROM pool_history WHERE id=:hid", ["hid" => $historyID]);
|
2019-05-28 16:59:38 +00:00
|
|
|
|
|
|
|
foreach ($status as $entry) {
|
|
|
|
$images = trim($entry['images']);
|
|
|
|
$images = explode(" ", $images);
|
2020-03-27 14:41:24 +00:00
|
|
|
$poolID = (int)$entry['pool_id'];
|
2019-05-28 16:59:38 +00:00
|
|
|
$imageArray = "";
|
|
|
|
$newAction = -1;
|
|
|
|
|
|
|
|
if ($entry['action'] == 0) {
|
|
|
|
// READ ENTRIES
|
2022-10-28 00:45:35 +00:00
|
|
|
foreach ($images as $imageID) {
|
|
|
|
$this->add_post($poolID, int_escape($imageID));
|
2019-05-28 16:59:38 +00:00
|
|
|
|
2019-06-21 20:13:47 +00:00
|
|
|
$imageArray .= " " . $imageID;
|
2019-05-28 16:59:38 +00:00
|
|
|
$newAction = 1;
|
|
|
|
}
|
|
|
|
} elseif ($entry['action'] == 1) {
|
|
|
|
// DELETE ENTRIES
|
2020-03-27 00:15:15 +00:00
|
|
|
foreach ($images as $imageID) {
|
|
|
|
$database->execute(
|
|
|
|
"DELETE FROM pool_images WHERE pool_id = :pid AND image_id = :iid",
|
|
|
|
["pid" => $poolID, "iid" => $imageID]
|
|
|
|
);
|
2019-06-21 20:13:47 +00:00
|
|
|
$imageArray .= " " . $imageID;
|
2019-05-28 16:59:38 +00:00
|
|
|
$newAction = 0;
|
|
|
|
}
|
2020-03-27 00:15:15 +00:00
|
|
|
$this->update_count($poolID);
|
2019-05-28 16:59:38 +00:00
|
|
|
} else {
|
|
|
|
// FIXME: should this throw an exception instead?
|
|
|
|
log_error("pools", "Invalid history action.");
|
|
|
|
continue; // go on to the next one.
|
|
|
|
}
|
|
|
|
|
2020-02-01 18:11:00 +00:00
|
|
|
$count = (int)$database->get_one("SELECT COUNT(*) FROM pool_images WHERE pool_id=:pid", ["pid" => $poolID]);
|
2019-05-28 16:59:38 +00:00
|
|
|
$this->add_history($poolID, $newAction, $imageArray, $count);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* HERE WE ADD A SIMPLE POST FROM POOL.
|
|
|
|
* USED WITH FOREACH IN revert_history() & onTagTermParse().
|
|
|
|
*/
|
2019-06-21 20:13:47 +00:00
|
|
|
private function add_post(int $poolID, int $imageID, bool $history = false, int $imageOrder = 0): bool
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
|
|
|
global $database, $config;
|
|
|
|
|
2020-03-27 00:15:15 +00:00
|
|
|
$result = (int)$database->get_one(
|
|
|
|
"SELECT COUNT(*) FROM pool_images WHERE pool_id=:pid AND image_id=:iid",
|
|
|
|
["pid" => $poolID, "iid" => $imageID]
|
|
|
|
);
|
|
|
|
|
|
|
|
if ($result == 0) {
|
2019-06-21 20:13:47 +00:00
|
|
|
if ($config->get_bool(PoolsConfig::AUTO_INCREMENT_ORDER) && $imageOrder === 0) {
|
2020-03-26 18:52:31 +00:00
|
|
|
$imageOrder = (int)$database->get_one(
|
2019-05-28 16:59:38 +00:00
|
|
|
"
|
2019-06-21 20:13:47 +00:00
|
|
|
SELECT COALESCE(MAX(image_order),0) + 1
|
2014-05-30 12:12:34 +00:00
|
|
|
FROM pool_images
|
2019-06-21 20:13:47 +00:00
|
|
|
WHERE pool_id = :pid AND image_order IS NOT NULL",
|
|
|
|
["pid" => $poolID]
|
2019-05-28 16:59:38 +00:00
|
|
|
);
|
|
|
|
}
|
2014-05-30 12:12:34 +00:00
|
|
|
|
2019-05-28 16:59:38 +00:00
|
|
|
$database->execute(
|
|
|
|
"
|
2014-05-30 12:12:34 +00:00
|
|
|
INSERT INTO pool_images (pool_id, image_id, image_order)
|
|
|
|
VALUES (:pid, :iid, :ord)",
|
2019-06-21 20:13:47 +00:00
|
|
|
["pid" => $poolID, "iid" => $imageID, "ord" => $imageOrder]
|
2019-05-28 16:59:38 +00:00
|
|
|
);
|
2019-06-21 20:13:47 +00:00
|
|
|
} else {
|
|
|
|
// If the post is already added, there is nothing else to do
|
|
|
|
return false;
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
|
2019-06-21 20:13:47 +00:00
|
|
|
$this->update_count($poolID);
|
2019-05-28 16:59:38 +00:00
|
|
|
|
|
|
|
if ($history) {
|
2020-02-01 18:11:00 +00:00
|
|
|
$count = (int)$database->get_one("SELECT COUNT(*) FROM pool_images WHERE pool_id=:pid", ["pid" => $poolID]);
|
2020-01-26 13:19:35 +00:00
|
|
|
$this->add_history($poolID, 1, (string)$imageID, $count);
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
2019-06-21 20:13:47 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-01-20 14:10:59 +00:00
|
|
|
private function update_count(int $pool_id): void
|
2019-06-21 20:13:47 +00:00
|
|
|
{
|
|
|
|
global $database;
|
2020-03-27 00:15:15 +00:00
|
|
|
$database->execute(
|
2023-08-18 21:23:40 +00:00
|
|
|
"UPDATE pools SET posts=(SELECT COUNT(*) FROM pool_images WHERE pool_id=:pid),lastupdated=CURRENT_TIMESTAMP WHERE id=:pid",
|
2020-03-27 00:15:15 +00:00
|
|
|
["pid" => $pool_id]
|
|
|
|
);
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
2009-12-24 07:36:09 +00:00
|
|
|
}
|