<?php declare(strict_types = 1); // {{{ License // This file is part of GNU social - https://www.gnu.org/software/social // // GNU social 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. // // GNU social 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 GNU social. If not, see <http://www.gnu.org/licenses/>. // }}} namespace Component\Avatar; use App\Core\Cache; use App\Core\DB\DB; use App\Core\Event; use App\Core\GSFile; use App\Core\Modules\Component; use App\Core\Router\RouteLoader; use App\Core\Router\Router; use App\Util\Common; use Component\Avatar\Controller as C; use Component\Avatar\Exception\NoAvatarException; use Symfony\Component\HttpFoundation\Request; class Avatar extends Component { public function onInitializeComponent() { } public function onAddRoute(RouteLoader $r): bool { $r->connect('avatar_actor', '/actor/{actor_id<\d+>}/avatar/{size<full|big|medium|small>?full}', [Controller\Avatar::class, 'avatar_view']); $r->connect('avatar_default', '/avatar/default/{size<full|big|medium|small>?full}', [Controller\Avatar::class, 'default_avatar_view']); $r->connect('avatar_settings', '/settings/avatar', [Controller\Avatar::class, 'settings_avatar']); return Event::next; } /** * @throws \App\Util\Exception\ClientException */ public function onPopulateProfileSettingsTabs(Request $request, &$tabs): bool { // TODO avatar template shouldn't be on settings folder $tabs[] = [ 'title' => 'Avatar', 'desc' => 'Change your avatar.', 'controller' => C\Avatar::settings_avatar($request), ]; return Event::next; } public function onAvatarUpdate(int $actor_id): bool { Cache::delete("avatar-{$actor_id}"); foreach (['full', 'big', 'medium', 'small'] as $size) { foreach ([Router::ABSOLUTE_PATH, Router::ABSOLUTE_URL] as $type) { Cache::delete("avatar-url-{$actor_id}-{$size}-{$type}"); } Cache::delete("avatar-file-info-{$actor_id}-{$size}"); } return Event::next; } // UTILS ---------------------------------- /** * Get the avatar associated with the given Actor id */ public static function getAvatar(?int $actor_id = null): Entity\Avatar { $actor_id = $actor_id ?: Common::userId(); return GSFile::error( NoAvatarException::class, $actor_id, Cache::get( "avatar-{$actor_id}", function () use ($actor_id) { return DB::dql( 'select a from Component\Avatar\Entity\Avatar a ' . 'where a.actor_id = :actor_id', ['actor_id' => $actor_id], ); }, ), ); } /** * Get the cached avatar associated with the given Actor id, or the current user if not given */ public static function getUrl(int $actor_id, string $size = 'full', int $type = Router::ABSOLUTE_PATH): string { try { return self::getAvatar($actor_id)->getUrl($size, $type); } catch (NoAvatarException) { return Router::url('avatar_default', ['size' => $size], $type); } } public static function getDimensions(int $actor_id, string $size = 'full') { try { $attachment = self::getAvatar($actor_id)->getAttachment(); return ['width' => $attachment->getWidth(), 'height' => $attachment->getHeight()]; } catch (NoAvatarException) { return ['width' => Common::config('thumbnail', 'small'), 'height' => Common::config('thumbnail', 'small')]; } } /** * Get the cached avatar file info associated with the given Actor id * * Returns the avatar file's hash, mimetype, title and path. * Ensures exactly one cached value exists */ public static function getAvatarFileInfo(int $actor_id, string $size = 'full'): array { $res = Cache::get( "avatar-file-info-{$actor_id}-{$size}", function () use ($actor_id) { return DB::dql( 'select f.id, f.filename, a.title, f.mimetype ' . 'from Component\Attachment\Entity\Attachment f ' . 'join Component\Avatar\Entity\Avatar a with f.id = a.attachment_id ' . 'where a.actor_id = :actor_id', ['actor_id' => $actor_id], ); }, ); if ($res === []) { // Avatar not found $filepath = INSTALLDIR . '/public/assets/default-avatar.svg'; return [ 'id' => null, 'filepath' => $filepath, 'mimetype' => 'image/svg+xml', 'filename' => null, 'title' => 'default_avatar.svg', ]; } else { $res = $res[0]; // A user must always only have one avatar. $res['filepath'] = DB::findOneBy('attachment', ['id' => $res['id']])->getPath(); return $res; } } }