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

218 lines
7.6 KiB
PHP
Raw Normal View History

2009-07-28 11:20:50 +01:00
<?php
2009-08-20 23:37:17 +01:00
/*
2009-07-28 11:20:50 +01:00
* Name: Favorites
* Author: Daniel Marschall <info@daniel-marschall.de>
* License: GPLv2
* Description: Allow users to favorite images
2009-08-20 23:37:17 +01:00
* Documentation:
* Gives users a "favorite this image" button that they can press
* <p>Favorites for a user can then be retrieved by searching for
* "favorited_by=UserName"
* <p>Popular images can be searched for by eg. "favorites>5"
* <p>Favorite info can be added to an image's filename or tooltip
* using the $favorites placeholder
2009-07-28 11:20:50 +01:00
*/
class FavoriteSetEvent extends Event {
/** @var int */
public $image_id;
/** @var \User */
public $user;
/** @var bool */
public $do_set;
/**
* @param int $image_id
* @param User $user
* @param bool $do_set
*/
2014-03-22 09:00:59 +00:00
public function __construct(/*int*/ $image_id, User $user, /*boolean*/ $do_set) {
2014-12-14 16:06:31 -08:00
assert(is_int($image_id));
2010-01-03 10:05:27 +00:00
assert(is_bool($do_set));
2009-07-28 11:20:50 +01:00
$this->image_id = $image_id;
$this->user = $user;
$this->do_set = $do_set;
}
}
class Favorites extends Extension {
public function onInitExt(InitExtEvent $event) {
2009-07-28 11:21:41 +01:00
global $config;
if($config->get_int("ext_favorites_version", 0) < 1) {
$this->install();
}
}
2009-07-28 11:20:50 +01:00
public function onImageAdminBlockBuilding(ImageAdminBlockBuildingEvent $event) {
global $database, $user;
2009-07-28 11:21:41 +01:00
if(!$user->is_anonymous()) {
$user_id = $user->id;
$image_id = $event->image->id;
2009-07-28 11:20:50 +01:00
2011-01-01 16:28:04 +00:00
$is_favorited = $database->get_one(
"SELECT COUNT(*) AS ct FROM user_favorites WHERE user_id = :user_id AND image_id = :image_id",
array("user_id"=>$user_id, "image_id"=>$image_id)) > 0;
2009-07-28 11:20:50 +01:00
2009-07-28 11:21:41 +01:00
$event->add_part($this->theme->get_voter_html($event->image, $is_favorited));
2009-07-28 11:20:50 +01:00
}
2009-07-28 11:21:41 +01:00
}
public function onDisplayingImage(DisplayingImageEvent $event) {
2009-07-28 11:21:41 +01:00
$people = $this->list_persons_who_have_favorited($event->image);
if(count($people) > 0) {
2014-03-29 11:44:34 +00:00
$this->theme->display_people($people);
2009-07-28 11:20:50 +01:00
}
2009-07-28 11:21:41 +01:00
}
public function onPageRequest(PageRequestEvent $event) {
2009-07-28 11:21:41 +01:00
global $page, $user;
2010-05-28 14:26:46 +01:00
if($event->page_matches("change_favorite") && !$user->is_anonymous() && $user->check_auth_token()) {
2009-07-28 11:21:41 +01:00
$image_id = int_escape($_POST['image_id']);
2010-07-26 22:59:18 +01:00
if((($_POST['favorite_action'] == "set") || ($_POST['favorite_action'] == "unset")) && ($image_id > 0)) {
2012-06-10 04:21:03 +01:00
if($_POST['favorite_action'] == "set") {
send_event(new FavoriteSetEvent($image_id, $user, true));
log_debug("favourite", "Favourite set for $image_id", "Favourite added");
}
else {
send_event(new FavoriteSetEvent($image_id, $user, false));
log_debug("favourite", "Favourite removed for $image_id", "Favourite removed");
}
2009-07-28 11:20:50 +01:00
}
2009-07-28 11:21:41 +01:00
$page->set_mode("redirect");
$page->set_redirect(make_link("post/view/$image_id"));
2009-07-28 11:20:50 +01:00
}
2009-07-28 11:21:41 +01:00
}
2009-07-28 11:20:50 +01:00
public function onUserPageBuilding(UserPageBuildingEvent $event) {
$i_favorites_count = Image::count_images(array("favorited_by={$event->display_user->name}"));
$i_days_old = ((time() - strtotime($event->display_user->join_date)) / 86400) + 1;
$h_favorites_rate = sprintf("%.1f", ($i_favorites_count / $i_days_old));
$favorites_link = make_link("post/list/favorited_by={$event->display_user->name}/1");
$event->add_stats("<a href='$favorites_link'>Images favorited</a>: $i_favorites_count, $h_favorites_rate per day");
}
public function onImageInfoSet(ImageInfoSetEvent $event) {
2009-07-28 11:21:41 +01:00
global $user;
if(
in_array('favorite_action', $_POST) &&
(($_POST['favorite_action'] == "set") || ($_POST['favorite_action'] == "unset"))
) {
send_event(new FavoriteSetEvent($event->image->id, $user, ($_POST['favorite_action'] == "set")));
2009-07-28 11:20:50 +01:00
}
2009-07-28 11:21:41 +01:00
}
public function onFavoriteSet(FavoriteSetEvent $event) {
2009-07-28 11:21:41 +01:00
global $user;
$this->add_vote($event->image_id, $user->id, $event->do_set);
}
// FIXME: this should be handled by the foreign key. Check that it
// is, and then remove this
public function onImageDeletion(ImageDeletionEvent $event) {
2009-07-28 11:21:41 +01:00
global $database;
$database->execute("DELETE FROM user_favorites WHERE image_id=:image_id", array("image_id"=>$event->image->id));
2009-07-28 11:21:41 +01:00
}
public function onParseLinkTemplate(ParseLinkTemplateEvent $event) {
2009-07-28 11:21:41 +01:00
$event->replace('$favorites', $event->image->favorites);
}
2009-07-28 11:20:50 +01:00
public function onUserBlockBuilding(UserBlockBuildingEvent $event) {
2009-07-28 11:21:41 +01:00
global $user;
2010-08-02 12:14:03 -03:00
$username = url_escape($user->name);
$event->add_link("My Favorites", make_link("post/list/favorited_by=$username/1"), 20);
2009-07-28 11:21:41 +01:00
}
2009-07-28 11:20:50 +01:00
public function onSearchTermParse(SearchTermParseEvent $event) {
2009-07-28 11:21:41 +01:00
$matches = array();
if(preg_match("/^favorites([:]?<|[:]?>|[:]?<=|[:]?>=|[:|=])(\d+)$/i", $event->term, $matches)) {
$cmp = ltrim($matches[1], ":") ?: "=";
2009-07-28 11:21:41 +01:00
$favorites = $matches[2];
2009-07-30 03:27:40 +01:00
$event->add_querylet(new Querylet("images.id IN (SELECT id FROM images WHERE favorites $cmp $favorites)"));
2009-07-28 11:21:41 +01:00
}
else if(preg_match("/^favorited_by[=|:](.*)$/i", $event->term, $matches)) {
2009-07-28 11:21:41 +01:00
$user = User::by_name($matches[1]);
if(!is_null($user)) {
$user_id = $user->id;
2009-07-28 11:20:50 +01:00
}
2009-07-28 11:21:41 +01:00
else {
$user_id = -1;
2009-07-28 11:20:50 +01:00
}
2009-07-28 11:21:41 +01:00
$event->add_querylet(new Querylet("images.id IN (SELECT image_id FROM user_favorites WHERE user_id = $user_id)"));
}
else if(preg_match("/^favorited_by_userno[=|:](\d+)$/i", $event->term, $matches)) {
2009-07-28 11:21:41 +01:00
$user_id = int_escape($matches[1]);
$event->add_querylet(new Querylet("images.id IN (SELECT image_id FROM user_favorites WHERE user_id = $user_id)"));
2009-07-28 11:20:50 +01:00
}
}
2009-07-28 11:21:41 +01:00
2009-07-28 11:20:50 +01:00
private function install() {
global $database;
global $config;
if($config->get_int("ext_favorites_version") < 1) {
$database->Execute("ALTER TABLE images ADD COLUMN favorites INTEGER NOT NULL DEFAULT 0");
$database->Execute("CREATE INDEX images__favorites ON images(favorites)");
2014-03-02 18:50:46 +00:00
$database->create_table("user_favorites", "
2009-07-28 11:20:50 +01:00
image_id INTEGER NOT NULL,
user_id INTEGER NOT NULL,
2014-03-02 18:50:46 +00:00
created_at SCORE_DATETIME NOT NULL,
2009-07-28 11:20:50 +01:00
UNIQUE(image_id, user_id),
FOREIGN KEY (user_id) REFERENCES users(id) ON DELETE CASCADE,
FOREIGN KEY (image_id) REFERENCES images(id) ON DELETE CASCADE
2014-03-02 18:50:46 +00:00
");
$database->execute("CREATE INDEX user_favorites_image_id_idx ON user_favorites(image_id)", array());
2015-08-08 17:21:37 +01:00
$config->set_int("ext_favorites_version", 2);
2009-07-28 11:20:50 +01:00
}
if($config->get_int("ext_favorites_version") < 2) {
log_info("favorites", "Cleaning user favourites");
$database->Execute("DELETE FROM user_favorites WHERE user_id NOT IN (SELECT id FROM users)");
$database->Execute("DELETE FROM user_favorites WHERE image_id NOT IN (SELECT id FROM images)");
log_info("favorites", "Adding foreign keys to user favourites");
$database->Execute("ALTER TABLE user_favorites ADD FOREIGN KEY (user_id) REFERENCES users(id) ON DELETE CASCADE;");
$database->Execute("ALTER TABLE user_favorites ADD FOREIGN KEY (image_id) REFERENCES images(id) ON DELETE CASCADE;");
$config->set_int("ext_favorites_version", 2);
}
2009-07-28 11:20:50 +01:00
}
/**
* @param int $image_id
* @param int $user_id
* @param bool $do_set
*/
2012-02-09 23:04:37 -05:00
private function add_vote(/*int*/ $image_id, /*int*/ $user_id, /*bool*/ $do_set) {
2009-07-28 11:20:50 +01:00
global $database;
if ($do_set) {
$database->Execute(
"INSERT INTO user_favorites(image_id, user_id, created_at) VALUES(:image_id, :user_id, NOW())",
array("image_id"=>$image_id, "user_id"=>$user_id));
2009-07-28 11:20:50 +01:00
} else {
$database->Execute(
"DELETE FROM user_favorites WHERE image_id = :image_id AND user_id = :user_id",
array("image_id"=>$image_id, "user_id"=>$user_id));
2009-07-28 11:20:50 +01:00
}
$database->Execute(
"UPDATE images SET favorites=(SELECT COUNT(*) FROM user_favorites WHERE image_id=:image_id) WHERE id=:user_id",
array("image_id"=>$image_id, "user_id"=>$user_id));
2009-07-28 11:20:50 +01:00
}
/**
* @param Image $image
2015-09-27 12:38:48 +01:00
* @return string[]
*/
private function list_persons_who_have_favorited(Image $image) {
2009-07-28 11:20:50 +01:00
global $database;
return $database->get_col(
"SELECT name FROM users WHERE id IN (SELECT user_id FROM user_favorites WHERE image_id = :image_id) ORDER BY name",
array("image_id"=>$image->id));
2009-07-28 11:20:50 +01:00
}
}