123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171 |
- <?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;
- use App\Core\Event;
- use App\Core\GSFile;
- use App\Core\Modules\Component;
- use App\Core\Router;
- use App\Util\Common;
- use Component\Attachment\Entity\Attachment;
- use Component\Attachment\Entity\AttachmentThumbnail;
- use Component\Avatar\Controller as C;
- use Component\Avatar\Exception\NoAvatarException;
- use EventResult;
- use Symfony\Component\HttpFoundation\Request;
- class Avatar extends Component
- {
- public function onInitializeComponent(): EventResult
- {
- return EventResult::next;
- }
- public function onAddRoute(Router $r): EventResult
- {
- $r->connect('avatar_actor', '/actor/{actor_id<\d+>}/avatar/{size<full|big|medium|small>?medium}', [Controller\Avatar::class, 'avatar_view']);
- $r->connect('avatar_default', '/avatar/default/{size<full|big|medium|small>?medium}', [Controller\Avatar::class, 'default_avatar_view']);
- $r->connect('avatar_settings', '/settings/avatar', [Controller\Avatar::class, 'settings_avatar']);
- return Event::next;
- }
- /**
- * @param SettingsTabsType $tabs
- *
- * @throws \App\Util\Exception\ClientException
- */
- public function onPopulateSettingsTabs(Request $request, string $section, &$tabs): EventResult
- {
- if ($section === 'profile') {
- $tabs[] = [
- 'title' => 'Avatar',
- 'desc' => 'Change your avatar.',
- 'id' => 'settings-avatar',
- 'controller' => C\Avatar::settings_avatar($request),
- ];
- }
- return Event::next;
- }
- public function onAvatarUpdate(int $actor_id): EventResult
- {
- 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 = 'medium', 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 = 'medium')
- {
- try {
- $attachment = self::getAvatar($actor_id)->getAttachment();
- return ['width' => (int) $attachment->getWidth(), 'height' => (int) $attachment->getHeight()];
- } catch (NoAvatarException) {
- return ['width' => (int) (Common::config('thumbnail', 'small')), 'height' => (int) (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
- *
- * @return array{id: null|int, filename: null|string, title: string, mimetype: string, filepath?: string}
- */
- public static function getAvatarFileInfo(int $actor_id, string $size = 'medium'): 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.
- if ($size === 'full') {
- $res['filepath'] = Attachment::getByPK(['id' => $res['id']])->getPath();
- } else {
- $res['filepath'] = AttachmentThumbnail::getOrCreate(Attachment::getByPK(['id' => $res['id']]), $size)->getPath();
- }
- return $res;
- }
- }
- }
|