This repository has been archived on 2024-09-05. You can view files and clone it, but cannot push or open issues or pull requests.
shimmie2/core/event.php

293 lines
6.7 KiB
PHP
Raw Normal View History

2021-12-14 18:32:47 +00:00
<?php
declare(strict_types=1);
namespace Shimmie2;
2009-07-19 07:38:13 +00:00
/**
* Generic parent class for all events.
*
* An event is anything that can be passed around via send_event($blah)
*/
abstract class Event
{
public bool $stop_processing = false;
public function __construct()
{
}
2019-12-07 22:49:02 +00:00
public function __toString(): string
2019-12-07 22:49:02 +00:00
{
return var_export($this, true);
}
}
2009-07-19 07:38:13 +00:00
/**
2009-07-21 03:18:40 +00:00
* A wake-up call for extensions. Upon recieving an InitExtEvent an extension
* should check that it's database tables are there and install them if not,
* and set any defaults with Config::set_default_int() and such.
*
* This event is sent before $user is set to anything
*/
class InitExtEvent extends Event
{
}
2009-07-19 07:38:13 +00:00
/**
* A signal that a page has been requested.
*
* User requests /view/42 -> an event is generated with $args = array("view",
* "42"); when an event handler asks $event->page_matches("view"), it returns
* true and ignores the matched part, such that $event->count_args() = 1 and
* $event->get_arg(0) = "42"
*/
class PageRequestEvent extends Event
{
public string $method;
/**
* @var string[]
*/
public $args;
public int $arg_count;
public int $part_count;
public function __construct(string $method, string $path)
{
2020-01-26 13:19:35 +00:00
parent::__construct();
global $config;
$this->method = $method;
// trim starting slashes
$path = ltrim($path, "/");
// if path is not specified, use the default front page
if (empty($path)) { /* empty is faster than strlen */
2019-08-02 19:40:03 +00:00
$path = $config->get_string(SetupConfig::FRONT_PAGE);
}
// break the path into parts
$args = explode('/', $path);
$this->args = $args;
$this->arg_count = count($args);
}
/**
* Test if the requested path matches a given pattern.
*
* If it matches, store the remaining path elements in $args
*/
public function page_matches(string $name): bool
{
$parts = explode("/", $name);
$this->part_count = count($parts);
if ($this->part_count > $this->arg_count) {
return false;
}
2023-11-11 21:49:12 +00:00
for ($i = 0; $i < $this->part_count; $i++) {
if ($parts[$i] != $this->args[$i]) {
return false;
}
}
return true;
}
/**
* Get the n th argument of the page request (if it exists.)
*/
public function get_arg(int $n): string
{
$offset = $this->part_count + $n;
if ($offset >= 0 && $offset < $this->arg_count) {
return rawurldecode($this->args[$offset]);
} else {
$nm1 = $this->arg_count - 1;
2021-11-16 14:55:37 +00:00
throw new UserErrorException("Requested an invalid page argument {$offset} / {$nm1}");
}
}
/**
* If page arg $n is set, then treat that as a 1-indexed page number
* and return a 0-indexed page number less than $max; else return 0
*/
2023-11-11 21:49:12 +00:00
public function try_page_num(int $n, ?int $max = null): int
2019-11-04 00:42:06 +00:00
{
if ($this->count_args() > $n) {
$i = $this->get_arg($n);
2024-01-15 20:34:53 +00:00
if (is_numberish($i) && int_escape($i) > 0) {
return page_number($i, $max);
2019-11-04 00:42:06 +00:00
} else {
return 0;
}
2019-11-04 00:42:06 +00:00
} else {
return 0;
}
}
/**
* Returns the number of arguments the page request has.
*/
public function count_args(): int
{
2020-01-26 13:19:35 +00:00
return $this->arg_count - $this->part_count;
}
/*
* Many things use these functions
*/
public function get_search_terms(): array
{
$search_terms = [];
if ($this->count_args() === 2) {
$str = $this->get_arg(0);
// decode legacy caret-encoding just in case
// somebody has bookmarked such a URL
$from_caret = [
"^" => "^",
"s" => "/",
"b" => "\\",
"q" => "?",
"a" => "&",
"d" => ".",
];
$out = "";
$length = strlen($str);
for ($i = 0; $i < $length; $i++) {
if ($str[$i] == "^") {
$i++;
$out .= $from_caret[$str[$i]] ?? '';
} else {
$out .= $str[$i];
}
}
$str = $out;
// end legacy
$search_terms = Tag::explode($str);
}
return $search_terms;
}
public function get_page_number(): int
{
$page_number = 1;
if ($this->count_args() === 1) {
$page_number = int_escape($this->get_arg(0));
} elseif ($this->count_args() === 2) {
$page_number = int_escape($this->get_arg(1));
}
if ($page_number === 0) {
$page_number = 1;
} // invalid -> 0
return $page_number;
}
public function get_page_size(): int
{
global $config;
2019-08-16 14:18:14 +00:00
return $config->get_int(IndexConfig::IMAGES);
}
}
class CliGenEvent extends Event
{
public function __construct(
public \Symfony\Component\Console\Application $app
) {
2020-01-26 13:19:35 +00:00
parent::__construct();
}
}
2009-07-19 07:38:13 +00:00
/**
* A signal that some text needs formatting, the event carries
* both the text and the result
*/
class TextFormattingEvent extends Event
{
/**
* For reference
*/
public string $original;
/**
* with formatting applied
*/
public string $formatted;
/**
* with formatting removed
*/
public string $stripped;
public function __construct(string $text)
{
2020-01-26 13:19:35 +00:00
parent::__construct();
2020-02-01 20:01:25 +00:00
// We need to escape before formatting, instead of at display time,
// because formatters will add their own HTML tags into the mix and
// we don't want to escape those.
$h_text = html_escape(trim($text));
$this->original = $h_text;
$this->formatted = $h_text;
$this->stripped = $h_text;
}
}
2009-05-08 10:52:29 +00:00
2009-07-19 07:38:13 +00:00
/**
* A signal that something needs logging
2009-05-08 10:52:29 +00:00
*/
class LogEvent extends Event
{
/**
* a category, normally the extension name
*/
public string $section;
/**
* See python...
*/
public int $priority = 0;
/**
* Free text to be logged
*/
public string $message;
/**
* The time that the event was created
*/
public int $time;
/**
* Extra data to be held separate
*
* @var string[]
*/
public array $args;
public function __construct(string $section, int $priority, string $message)
{
2020-01-26 13:19:35 +00:00
parent::__construct();
$this->section = $section;
$this->priority = $priority;
$this->message = $message;
$this->time = time();
}
2009-05-08 10:52:29 +00:00
}
2019-11-03 18:28:38 +00:00
class DatabaseUpgradeEvent extends Event
{
}