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/ext/index/main.php

408 lines
15 KiB
PHP
Raw Normal View History

<?php
2010-01-04 12:41:04 +00:00
/**
* Name: Image List
2010-01-04 12:41:04 +00:00
* Author: Shish <webmaster@shishnet.org>
2010-01-04 12:44:20 +00:00
* Link: http://code.shishnet.org/shimmie2/
2010-01-04 12:41:04 +00:00
* License: GPLv2
* Description: Show a list of uploaded images
* Documentation:
* Here is a list of the search methods available out of the box;
* Shimmie extensions may provide other filters:
* <ul>
* <li>by tag, eg
* <ul>
* <li>cat
* <li>pie
* <li>somethi* -- wildcards are supported
* </ul>
* <li>size (=, &lt;, &gt;, &lt;=, &gt;=) width x height, eg
* <ul>
* <li>size=1024x768 -- a specific wallpaper size
* <li>size&gt;=500x500 -- no small images
* <li>size&lt;1000x1000 -- no large images
* </ul>
2014-01-02 14:10:08 +00:00
* <li>width (=, &lt;, &gt;, &lt;=, &gt;=) width, eg
* <ul>
* <li>width=1024 -- find images with 1024 width
* <li>width>2000 -- find images bigger than 2000 width
* </ul>
* <li>height (=, &lt;, &gt;, &lt;=, &gt;=) height, eg
* <ul>
* <li>height=768 -- find images with 768 height
* <li>height>1000 -- find images bigger than 1000 height
* </ul>
2010-01-04 12:41:04 +00:00
* <li>ratio (=, &lt;, &gt;, &lt;=, &gt;=) width : height, eg
* <ul>
* <li>ratio=4:3, ratio=16:9 -- standard wallpaper
* <li>ratio=1:1 -- square images
* <li>ratio<1:1 -- tall images
* <li>ratio>1:1 -- wide images
* </ul>
* <li>filesize (=, &lt;, &gt;, &lt;=, &gt;=) size, eg
* <ul>
2012-02-17 01:45:55 +00:00
* <li>filesize&gt;1024 -- no images under 1KB
* <li>filesize&lt=3MB -- shorthand filesizes are supported too
2010-01-04 12:41:04 +00:00
* </ul>
* <li>id (=, &lt;, &gt;, &lt;=, &gt;=) number, eg
* <ul>
* <li>id<20 -- search only the first few images
* <li>id>=500 -- search later images
* </ul>
* <li>user=Username & poster=Username, eg
2010-01-04 12:41:04 +00:00
* <ul>
* <li>user=Shish -- find all of Shish's posts
* <li>poster=Shish -- same as above
2010-01-04 12:41:04 +00:00
* </ul>
* <li>user_id=userID & poster_id=userID, eg
* <ul>
* <li>user_id=2 -- find all posts by user id 2
* <li>poster_id=2 -- same as above
* </ul>
* <li>hash=md5sum & md5=md5sum, eg
2010-01-04 12:41:04 +00:00
* <ul>
* <li>hash=bf5b59173f16b6937a4021713dbfaa72 -- find the "Taiga want up!" image
* <li>md5=bf5b59173f16b6937a4021713dbfaa72 -- same as above
2010-01-04 12:41:04 +00:00
* </ul>
* <li>filetype=type & ext=type, eg
2010-01-04 12:41:04 +00:00
* <ul>
* <li>filetype=png -- find all PNG images
* <li>ext=png -- same as above
2010-01-04 12:41:04 +00:00
* </ul>
* <li>filename=blah & name=blah, eg
2010-01-04 12:41:04 +00:00
* <ul>
2010-02-27 02:09:10 +00:00
* <li>filename=kitten -- find all images with "kitten" in the original filename
* <li>name=kitten -- same as above
2010-01-04 12:41:04 +00:00
* </ul>
2012-02-17 01:45:55 +00:00
* <li>posted (=, &lt;, &gt;, &lt;=, &gt;=) date, eg
2010-01-04 12:41:04 +00:00
* <ul>
2012-02-17 01:45:55 +00:00
* <li>posted&gt;=2009-12-25 posted&lt;=2010-01-01 -- find images posted between christmas and new year
2010-01-04 12:41:04 +00:00
* </ul>
* <li>tags (=, &lt;, &gt;, &lt;=, &gt;=) count, eg
* <ul>
* <li>tags=1 -- search for images with only 1 tag
* <li>tags>=10 -- search for images with 10 or more tags
* <li>tags<25 -- search for images with less than 25 tags
* </ul>
* <li>source=(URL, any, none) eg
* <ul>
* <li>source=http://example.com -- find all images with "http://example.com" in the source
* <li>source=any -- find all images with a source
* <li>source=none -- find all images without a source
* </ul>
2014-01-05 11:52:09 +00:00
* <li>order=(id, width, height, filesize, filename)_(ASC, DESC), eg
* <ul>
* <li>order=width -- find all images sorted from highest > lowest width
* <li>order=filesize_asc -- find all images sorted from lowest > highest filesize
* </ul>
* <li>order=random_####, eg
* <ul>
* <li>order=random_8547 -- find all images sorted randomly using 8547 as a seed
* </ul>
2010-01-05 09:40:26 +00:00
* </ul>
2010-01-04 12:41:04 +00:00
* <p>Search items can be combined to search for images which match both,
* or you can stick "-" in front of an item to search for things that don't
* match it.
* <p>Metatags can be followed by ":" rather than "=" if you prefer.
* <br />I.E: "posted:2014-01-01", "id:>=500" etc.
2010-01-05 09:40:26 +00:00
* <p>Some search methods provided by extensions:
* <ul>
* <li>Numeric Score
* <ul>
* <li>score (=, &lt;, &gt;, &lt;=, &gt;=) number -- seach by score
* <li>upvoted_by=Username -- search for a user's likes
* <li>downvoted_by=Username -- search for a user's dislikes
* <li>upvoted_by_id=UserID -- search for a user's likes by user ID
* <li>downvoted_by_id=UserID -- search for a user's dislikes by user ID
* <li>order=score_(ASC, DESC) -- find all images sorted from by score
2010-01-05 09:40:26 +00:00
* </ul>
* <li>Image Rating
* <ul>
* <li>rating=se -- find safe and explicit images, ignore questionable and unknown
* </ul>
* <li>Favorites
* <ul>
* <li>favorites (=, &lt;, &gt;, &lt;=, &gt;=) number -- search for images favourited a certain number of times
* <li>favourited_by=Username -- search for a user's choices by username
* <li>favorited_by_userno=UserID -- search for a user's choice by userID
2010-01-05 09:40:26 +00:00
* </ul>
* <li>Notes
* <ul>
* <li>notes (=, &lt;, &gt;, &lt;=, &gt;=) number -- search by the number of notes an image has
* <li>notes_by=Username -- search for images containing notes created by username
* <li>notes_by_userno=UserID -- search for images containing notes created by userID
2010-01-05 09:40:26 +00:00
* </ul>
* <li>Artists
* <ul>
* <li>author=ArtistName -- search for images by artist
* </ul>
* <li>Image Comments
* <ul>
* <li>comments (=, &lt;, &gt;, &lt;=, &gt;=) number -- search for images by number of comments
* <li>commented_by=Username -- search for images containing user's comments by username
* <li>commented_by_userno=UserID -- search for images containing user's comments by userID
* </ul>
* <li>Pools
* <ul>
* <li>pool=(PoolID, any, none) -- search for images in a pool by PoolID.
* <li>pool_by_name=PoolName -- search for images in a pool by PoolName. underscores are replaced with spaces
* </ul>
2014-02-03 16:27:05 +00:00
* <li>Post Relationships
* <ul>
* <li>parent=(parentID, any, none) -- search for images by parentID / if they have, do not have a parent
* <li>child=(any, none) -- search for images which have, or do not have children
* </ul>
2010-01-05 09:40:26 +00:00
* </ul>
2010-01-04 12:41:04 +00:00
*/
/*
* SearchTermParseEvent:
* Signal that a search term needs parsing
*/
class SearchTermParseEvent extends Event {
2014-04-27 19:29:36 -04:00
/** @var null|string */
public $term = null;
/** @var null */
public $context = array();
2014-04-27 19:29:36 -04:00
/** @var \Querylet[] */
public $querylets = array();
2014-04-27 19:29:36 -04:00
/**
* @param string|null $term
* @param string[] $context
2014-04-27 19:29:36 -04:00
*/
public function __construct($term, array $context) {
$this->term = $term;
$this->context = $context;
}
2014-04-27 19:29:36 -04:00
/**
* @return bool
*/
public function is_querylet_set() {
return (count($this->querylets) > 0);
}
2014-04-27 19:29:36 -04:00
/**
* @return \Querylet[]
*/
public function get_querylets() {
return $this->querylets;
}
2014-04-27 19:29:36 -04:00
/**
* @param \Querylet $q
*/
public function add_querylet($q) {
$this->querylets[] = $q;
}
}
class SearchTermParseException extends SCoreException {
}
class PostListBuildingEvent extends Event {
/** @var array */
public $search_terms = array();
2014-04-27 19:29:36 -04:00
/** @var array */
public $parts = array();
/**
* @param string[] $search
2014-04-27 19:29:36 -04:00
*/
public function __construct(array $search) {
$this->search_terms = $search;
}
2012-08-18 19:23:11 +01:00
2014-04-27 19:29:36 -04:00
/**
* @param string $html
* @param int $position
*/
2012-08-18 19:23:11 +01:00
public function add_control(/*string*/ $html, /*int*/ $position=50) {
while(isset($this->parts[$position])) $position++;
$this->parts[$position] = $html;
}
}
class Index extends Extension {
2017-03-12 00:18:26 -08:00
/** @var int */
2016-06-19 23:05:57 +01:00
private $stpen = 0; // search term parse event number
2012-02-02 08:07:57 +00:00
public function onInitExt(InitExtEvent $event) {
2009-05-11 14:09:24 -07:00
global $config;
$config->set_default_int("index_images", 24);
2009-05-11 14:09:24 -07:00
$config->set_default_bool("index_tips", true);
2014-01-03 08:24:47 +00:00
$config->set_default_string("index_order", "id DESC");
2009-05-11 14:09:24 -07:00
}
2012-02-02 08:07:57 +00:00
public function onPageRequest(PageRequestEvent $event) {
2015-09-12 11:43:28 +01:00
global $database, $page;
2009-05-11 14:09:24 -07:00
if($event->page_matches("post/list")) {
if(isset($_GET['search'])) {
// implode(explode()) to resolve aliases and sanitise
$search = url_escape(Tag::implode(Tag::explode($_GET['search'], false)));
if(empty($search)) {
$page->set_mode("redirect");
$page->set_redirect(make_link("post/list/1"));
}
else {
$page->set_mode("redirect");
2012-01-15 23:03:27 -05:00
$page->set_redirect(make_link('post/list/'.$search.'/1'));
}
return;
}
2010-04-26 02:36:05 +01:00
$search_terms = $event->get_search_terms();
$page_number = $event->get_page_number();
$page_size = $event->get_page_size();
2013-10-04 17:17:42 -04:00
$count_search_terms = count($search_terms);
2010-05-04 20:10:37 +01:00
try {
2013-08-30 00:19:38 +01:00
#log_debug("index", "Search for ".implode(" ", $search_terms), false, array("terms"=>$search_terms));
2010-05-04 20:10:37 +01:00
$total_pages = Image::count_pages($search_terms);
2013-10-04 17:17:42 -04:00
if(SPEED_HAX && $count_search_terms === 0 && ($page_number < 10)) { // extra caching for the first few post/list pages
$images = $database->cache->get("post-list:$page_number");
if(!$images) {
$images = Image::find_images(($page_number-1)*$page_size, $page_size, $search_terms);
$database->cache->set("post-list:$page_number", $images, 600);
}
}
else {
$images = Image::find_images(($page_number-1)*$page_size, $page_size, $search_terms);
}
2010-05-04 20:10:37 +01:00
}
catch(SearchTermParseException $stpe) {
// FIXME: display the error somewhere
$total_pages = 0;
$images = array();
}
2013-10-04 17:17:42 -04:00
$count_images = count($images);
2013-10-04 17:17:42 -04:00
if($count_search_terms === 0 && $count_images === 0 && $page_number === 1) {
$this->theme->display_intro($page);
2009-07-07 08:52:09 -07:00
send_event(new PostListBuildingEvent($search_terms));
}
2013-10-04 17:17:42 -04:00
else if($count_search_terms > 0 && $count_images === 1 && $page_number === 1) {
$page->set_mode("redirect");
2012-01-15 23:03:27 -05:00
$page->set_redirect(make_link('post/view/'.$images[0]->id));
2009-01-22 01:39:44 -08:00
}
else {
2012-08-18 19:23:11 +01:00
$plbe = new PostListBuildingEvent($search_terms);
send_event($plbe);
$this->theme->set_page($page_number, $total_pages, $search_terms);
$this->theme->display_page($page, $images);
2012-08-18 19:23:11 +01:00
if(count($plbe->parts) > 0) {
$this->theme->display_admin_block($plbe->parts);
}
2009-01-22 01:39:44 -08:00
}
}
2009-05-11 14:09:24 -07:00
}
2012-02-02 08:07:57 +00:00
public function onSetupBuilding(SetupBuildingEvent $event) {
2009-05-11 14:09:24 -07:00
$sb = new SetupBlock("Index Options");
$sb->position = 20;
2009-01-04 11:18:37 -08:00
$sb->add_label("Show ");
$sb->add_int_option("index_images");
$sb->add_label(" images on the post list");
2009-05-11 14:09:24 -07:00
$event->panel->add_block($sb);
}
public function onImageInfoSet(ImageInfoSetEvent $event) {
global $database;
if(SPEED_HAX) {
$database->cache->delete("thumb-block:{$event->image->id}");
}
}
2012-02-02 08:07:57 +00:00
public function onSearchTermParse(SearchTermParseEvent $event) {
2009-05-11 14:09:24 -07:00
$matches = array();
2012-01-15 23:03:27 -05:00
// check for tags first as tag based searches are more common.
if(preg_match("/^tags([:]?<|[:]?>|[:]?<=|[:]?>=|[:|=])(\d+)$/i", $event->term, $matches)) {
$cmp = ltrim($matches[1], ":") ?: "=";
$count = $matches[2];
$event->add_querylet(
new Querylet("EXISTS (
SELECT 1
FROM image_tags it
LEFT JOIN tags t ON it.tag_id = t.id
WHERE images.id = it.image_id
GROUP BY image_id
HAVING COUNT(*) $cmp $count
)")
);
2009-05-11 14:09:24 -07:00
}
else if(preg_match("/^ratio([:]?<|[:]?>|[:]?<=|[:]?>=|[:|=])(\d+):(\d+)$/i", $event->term, $matches)) {
$cmp = preg_replace('/^:/', '=', $matches[1]);
2013-06-16 12:20:12 +01:00
$args = array("width{$this->stpen}"=>int_escape($matches[2]), "height{$this->stpen}"=>int_escape($matches[3]));
$event->add_querylet(new Querylet("width / height $cmp :width{$this->stpen} / :height{$this->stpen}", $args));
2009-05-11 14:09:24 -07:00
}
else if(preg_match("/^(filesize|id)([:]?<|[:]?>|[:]?<=|[:]?>=|[:|=])(\d+[kmg]?b?)$/i", $event->term, $matches)) {
2009-05-11 14:09:24 -07:00
$col = $matches[1];
$cmp = ltrim($matches[2], ":") ?: "=";
2009-05-11 14:09:24 -07:00
$val = parse_shorthand_int($matches[3]);
2013-06-16 12:20:12 +01:00
$event->add_querylet(new Querylet("images.$col $cmp :val{$this->stpen}", array("val{$this->stpen}"=>$val)));
2009-05-11 14:09:24 -07:00
}
else if(preg_match("/^(hash|md5)[=|:]([0-9a-fA-F]*)$/i", $event->term, $matches)) {
2009-08-02 08:57:30 +01:00
$hash = strtolower($matches[2]);
2013-06-16 12:20:12 +01:00
$event->add_querylet(new Querylet('images.hash = :hash', array("hash" => $hash)));
2009-05-11 14:09:24 -07:00
}
else if(preg_match("/^(filetype|ext)[=|:]([a-zA-Z0-9]*)$/i", $event->term, $matches)) {
2009-05-11 14:09:24 -07:00
$ext = strtolower($matches[2]);
2013-06-16 12:20:12 +01:00
$event->add_querylet(new Querylet('images.ext = :ext', array("ext" => $ext)));
2009-05-11 14:09:24 -07:00
}
else if(preg_match("/^(filename|name)[=|:]([a-zA-Z0-9]*)$/i", $event->term, $matches)) {
2009-05-11 14:09:24 -07:00
$filename = strtolower($matches[2]);
2013-06-16 12:20:12 +01:00
$event->add_querylet(new Querylet("images.filename LIKE :filename{$this->stpen}", array("filename{$this->stpen}"=>"%$filename%")));
2012-04-05 17:33:50 +01:00
}
else if(preg_match("/^(source)[=|:](.*)$/i", $event->term, $matches)) {
$source = strtolower($matches[2]);
if(preg_match("/^(any|none)$/i", $source)){
$not = ($source == "any" ? "NOT" : "");
$event->add_querylet(new Querylet("images.source IS $not NULL"));
}else{
$event->add_querylet(new Querylet('images.source LIKE :src', array("src"=>"%$source%")));
}
2009-05-11 14:09:24 -07:00
}
else if(preg_match("/^posted([:]?<|[:]?>|[:]?<=|[:]?>=|[:|=])([0-9-]*)$/i", $event->term, $matches)) {
$cmp = ltrim($matches[1], ":") ?: "=";
2012-02-17 01:43:49 +00:00
$val = $matches[2];
2013-06-16 12:20:12 +01:00
$event->add_querylet(new Querylet("images.posted $cmp :posted{$this->stpen}", array("posted{$this->stpen}"=>$val)));
}
else if(preg_match("/^size([:]?<|[:]?>|[:]?<=|[:]?>=|[:|=])(\d+)x(\d+)$/i", $event->term, $matches)) {
$cmp = ltrim($matches[1], ":") ?: "=";
2013-06-16 12:20:12 +01:00
$args = array("width{$this->stpen}"=>int_escape($matches[2]), "height{$this->stpen}"=>int_escape($matches[3]));
$event->add_querylet(new Querylet("width $cmp :width{$this->stpen} AND height $cmp :height{$this->stpen}", $args));
2010-01-05 09:05:16 +00:00
}
else if(preg_match("/^width([:]?<|[:]?>|[:]?<=|[:]?>=|[:|=])(\d+)$/i", $event->term, $matches)) {
2014-01-02 14:10:08 +00:00
$cmp = ltrim($matches[1], ":") ?: "=";
$event->add_querylet(new Querylet("width $cmp :width{$this->stpen}", array("width{$this->stpen}"=>int_escape($matches[2]))));
}
else if(preg_match("/^height([:]?<|[:]?>|[:]?<=|[:]?>=|[:|=])(\d+)$/i", $event->term, $matches)) {
2014-01-02 14:10:08 +00:00
$cmp = ltrim($matches[1], ":") ?: "=";
$event->add_querylet(new Querylet("height $cmp :height{$this->stpen}",array("height{$this->stpen}"=>int_escape($matches[2]))));
}
else if(preg_match("/^order[=|:](id|width|height|filesize|filename)[_]?(desc|asc)?$/i", $event->term, $matches)){
2014-01-03 08:24:47 +00:00
$ord = strtolower($matches[1]);
2014-01-05 11:52:09 +00:00
$default_order_for_column = preg_match("/^(id|filename)$/", $matches[1]) ? "ASC" : "DESC";
$sort = isset($matches[2]) ? strtoupper($matches[2]) : $default_order_for_column;
2015-08-02 20:39:41 +01:00
Image::$order_sql = "images.$ord $sort";
$event->add_querylet(new Querylet("1=1")); //small hack to avoid metatag being treated as normal tag
}
else if(preg_match("/^order[=|:]random[_]([0-9]{1,4})$/i", $event->term, $matches)){
//order[=|:]random requires a seed to avoid duplicates
//since the tag can't be changed during the parseevent, we instead generate the seed during submit using js
$seed = $matches[1];
2015-08-02 20:39:41 +01:00
Image::$order_sql = "RAND($seed)";
$event->add_querylet(new Querylet("1=1")); //small hack to avoid metatag being treated as normal tag
}
2013-06-16 12:20:12 +01:00
$this->stpen++;
}
}