forked from GNUsocial/gnu-social
[COMPONENTS][Tag] Refactor Tag and add self tag stream
This commit is contained in:
parent
6680772e47
commit
5c3d561a67
@ -12,7 +12,7 @@ use Functional as F;
|
|||||||
|
|
||||||
class Tag extends Controller
|
class Tag extends Controller
|
||||||
{
|
{
|
||||||
private function process(string|array $tag_or_tags, callable $key, string $query)
|
private function process(string|array $tag_or_tags, callable $key, string $query, string $template)
|
||||||
{
|
{
|
||||||
$actor = Common::actor();
|
$actor = Common::actor();
|
||||||
$page = $this->int('page') ?: 1;
|
$page = $this->int('page') ?: 1;
|
||||||
@ -26,7 +26,7 @@ class Tag extends Controller
|
|||||||
} else {
|
} else {
|
||||||
$canonical = F\map($tag_or_tags, fn ($t) => CompTag::canonicalTag($t, $lang));
|
$canonical = F\map($tag_or_tags, fn ($t) => CompTag::canonicalTag($t, $lang));
|
||||||
}
|
}
|
||||||
$notes = Cache::pagedStream(
|
$results = Cache::pagedStream(
|
||||||
key: $key($canonical),
|
key: $key($canonical),
|
||||||
query: $query,
|
query: $query,
|
||||||
query_args: ['canon' => $canonical],
|
query_args: ['canon' => $canonical],
|
||||||
@ -35,28 +35,51 @@ class Tag extends Controller
|
|||||||
);
|
);
|
||||||
|
|
||||||
return [
|
return [
|
||||||
'_template' => 'tag_stream.html.twig',
|
'_template' => $template,
|
||||||
'notes' => $notes,
|
'results' => $results,
|
||||||
'page' => $page,
|
'page' => $page,
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
public function single_tag(string $tag)
|
public function single_note_tag(string $tag)
|
||||||
{
|
{
|
||||||
return $this->process(
|
return $this->process(
|
||||||
tag_or_tags: $tag,
|
tag_or_tags: $tag,
|
||||||
key: fn ($canonical) => "tag-{$canonical}",
|
key: fn ($canonical) => "note-tag-feed-{$canonical}",
|
||||||
query: 'select n from note n join note_tag nt with n.id = nt.note_id where nt.canonical = :canon order by nt.created DESC, nt.note_id DESC',
|
query: 'select n from note n join note_tag nt with n.id = nt.note_id where nt.canonical = :canon order by nt.created DESC, nt.note_id DESC',
|
||||||
|
template: 'note_tag_feed.html.twig',
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function multi_tags(string $tags)
|
public function multi_note_tags(string $tags)
|
||||||
{
|
{
|
||||||
$tags = explode(',', $tags);
|
$tags = explode(',', $tags);
|
||||||
return $this->process(
|
return $this->process(
|
||||||
tag_or_tags: $tags,
|
tag_or_tags: $tags,
|
||||||
key: fn ($canonical) => 'tags-' . implode('-', $canonical),
|
key: fn ($canonical) => 'note-tags-feed-' . implode('-', $canonical),
|
||||||
query: 'select n from note n join note_tag nt with n.id = nt.note_id where nt.canonical in (:canon) order by nt.created DESC, nt.note_id DESC',
|
query: 'select n from note n join note_tag nt with n.id = nt.note_id where nt.canonical in (:canon) order by nt.created DESC, nt.note_id DESC',
|
||||||
|
template: 'note_tag_feed.html.twig',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function single_actor_tag(string $tag)
|
||||||
|
{
|
||||||
|
return $this->process(
|
||||||
|
tag_or_tags: $tag,
|
||||||
|
key: fn ($canonical) => "actor-tag-feed-{$canonical}",
|
||||||
|
query: 'select a from actor a join actor_tag at with a.id = at.tagged where at.canonical = :canon order by at.modified DESC',
|
||||||
|
template: 'actor_tag_feed.html.twig',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function multi_actor_tag(string $tag)
|
||||||
|
{
|
||||||
|
$tags = explode(',', $tags);
|
||||||
|
return $this->process(
|
||||||
|
tag_or_tags: $tag,
|
||||||
|
key: fn ($canonical) => 'actor-tags-feed-' . implode('-', $canonical),
|
||||||
|
query: 'select a from actor a join actor_tag at with a.id = at.tagged where at.canonical = :canon order by at.modified DESC',
|
||||||
|
template: 'actor_tag_feed.html.twig',
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -52,8 +52,10 @@ class Tag extends Component
|
|||||||
|
|
||||||
public function onAddRoute($r): bool
|
public function onAddRoute($r): bool
|
||||||
{
|
{
|
||||||
$r->connect('single_tag', '/tag/{tag<' . self::TAG_SLUG_REGEX . '>}', [Controller\Tag::class, 'single_tag']);
|
$r->connect('single_note_tag', '/note-tag/{tag<' . self::TAG_SLUG_REGEX . '>}', [Controller\Tag::class, 'single_note_tag']);
|
||||||
$r->connect('multiple_tags', '/tags/{tags<(' . self::TAG_SLUG_REGEX . ',)+' . self::TAG_SLUG_REGEX . '>}', [Controller\Tag::class, 'multi_tags']);
|
$r->connect('multiple_note_tags', '/note-tags/{tags<(' . self::TAG_SLUG_REGEX . ',)+' . self::TAG_SLUG_REGEX . '>}', [Controller\Tag::class, 'multi_note_tags']);
|
||||||
|
$r->connect('single_actor_tag', '/actor-tag/{tag<' . self::TAG_SLUG_REGEX . '>}', [Controller\Tag::class, 'single_actor_tag']);
|
||||||
|
$r->connect('multiple_actor_tags', '/actor-tags/{tags<(' . self::TAG_SLUG_REGEX . ',)+' . self::TAG_SLUG_REGEX . '>}', [Controller\Tag::class, 'multi_actor_tags']);
|
||||||
return Event::next;
|
return Event::next;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -80,17 +82,15 @@ class Tag extends Component
|
|||||||
|
|
||||||
public function onRenderPlainTextNoteContent(string &$text, ?string $language = null): bool
|
public function onRenderPlainTextNoteContent(string &$text, ?string $language = null): bool
|
||||||
{
|
{
|
||||||
if (!is_null($language)) {
|
|
||||||
$text = preg_replace_callback(self::TAG_REGEX, fn ($m) => $m[1] . self::tagLink($m[2], $language), $text);
|
$text = preg_replace_callback(self::TAG_REGEX, fn ($m) => $m[1] . self::tagLink($m[2], $language), $text);
|
||||||
}
|
|
||||||
return Event::next;
|
return Event::next;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static function tagLink(string $tag, string $language): string
|
private static function tagLink(string $tag, ?string $language): string
|
||||||
{
|
{
|
||||||
$tag = self::ensureLength($tag);
|
$tag = self::ensureLength($tag);
|
||||||
$canonical = self::canonicalTag($tag, $language);
|
$canonical = self::canonicalTag($tag, $language);
|
||||||
$url = Router::url('tag', ['tag' => $canonical, 'lang' => $language]);
|
$url = Router::url('single_note_tag', !\is_null($language) ? ['tag' => $canonical, 'lang' => $language] : ['tag' => $canonical]);
|
||||||
return HTML::html(['a' => ['attrs' => ['href' => $url, 'title' => $tag, 'rel' => 'tag'], $tag]], options: ['indent' => false]);
|
return HTML::html(['a' => ['attrs' => ['href' => $url, 'title' => $tag, 'rel' => 'tag'], $tag]], options: ['indent' => false]);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -99,12 +99,12 @@ class Tag extends Component
|
|||||||
return mb_substr($tag, 0, self::MAX_TAG_LENGTH);
|
return mb_substr($tag, 0, self::MAX_TAG_LENGTH);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function canonicalTag(string $tag, string $language): string
|
public static function canonicalTag(string $tag, ?string $language): string
|
||||||
{
|
{
|
||||||
$result = '';
|
$result = '';
|
||||||
foreach (Formatting::splitWords(str_replace('#', '', $tag)) as $word) {
|
foreach (Formatting::splitWords(str_replace('#', '', $tag)) as $word) {
|
||||||
$temp_res = null;
|
$temp_res = null;
|
||||||
if (Event::handle('StemWord', [$language, $word, &$temp_res]) !== Event::stop) {
|
if (\is_null($language) || Event::handle('StemWord', [$language, $word, &$temp_res]) !== Event::stop) {
|
||||||
$temp_res = $word;
|
$temp_res = $word;
|
||||||
}
|
}
|
||||||
$result .= Formatting::slugify($temp_res);
|
$result .= Formatting::slugify($temp_res);
|
||||||
|
9
components/Tag/templates/actor_tag_feed.html.twig
Normal file
9
components/Tag/templates/actor_tag_feed.html.twig
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
{% extends 'base.html.twig' %}
|
||||||
|
|
||||||
|
{% block body %}
|
||||||
|
{% for actor in results %}
|
||||||
|
{% block profile_view %}{% include 'cards/profile/view.html.twig' %}{% endblock profile_view %}
|
||||||
|
{% endfor %}
|
||||||
|
|
||||||
|
{{ "Page: " ~ page }}
|
||||||
|
{% endblock %}
|
@ -2,7 +2,7 @@
|
|||||||
{% import '/cards/note/view.html.twig' as noteView %}
|
{% import '/cards/note/view.html.twig' as noteView %}
|
||||||
|
|
||||||
{% block body %}
|
{% block body %}
|
||||||
{% for note in notes %}
|
{% for note in results %}
|
||||||
{% block current_note %}
|
{% block current_note %}
|
||||||
{{ noteView.macro_note(note) }}
|
{{ noteView.macro_note(note) }}
|
||||||
{% endblock current_note %}
|
{% endblock current_note %}
|
@ -372,7 +372,7 @@ abstract class Cache
|
|||||||
$per_page = Common::config('streams', 'notes_per_page');
|
$per_page = Common::config('streams', 'notes_per_page');
|
||||||
}
|
}
|
||||||
|
|
||||||
$filter_scope = fn (Note $n) => $n->isVisibleTo($actor);
|
$filter_scope = fn (Note|Actor $o) => $o->isVisibleTo($actor);
|
||||||
|
|
||||||
$getter = fn (int $offset, int $length) => DB::dql($query, $query_args, options: ['offset' => $offset, 'limit' => $length]);
|
$getter = fn (int $offset, int $length) => DB::dql($query, $query_args, options: ['offset' => $offset, 'limit' => $length]);
|
||||||
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types = 1);
|
||||||
|
|
||||||
// {{{ License
|
// {{{ License
|
||||||
|
|
||||||
@ -29,15 +29,13 @@ use App\Core\Entity;
|
|||||||
use App\Core\Event;
|
use App\Core\Event;
|
||||||
use App\Core\Router\Router;
|
use App\Core\Router\Router;
|
||||||
use App\Core\UserRoles;
|
use App\Core\UserRoles;
|
||||||
use App\Util\Common;
|
|
||||||
use App\Util\Exception\NicknameException;
|
use App\Util\Exception\NicknameException;
|
||||||
use App\Util\Exception\NotFoundException;
|
use App\Util\Exception\NotFoundException;
|
||||||
use App\Util\Nickname;
|
use App\Util\Nickname;
|
||||||
use Component\Avatar\Avatar;
|
use Component\Avatar\Avatar;
|
||||||
|
use Component\Tag\Tag as TagComponent;
|
||||||
use DateTimeInterface;
|
use DateTimeInterface;
|
||||||
use Functional as F;
|
use Functional as F;
|
||||||
use function in_array;
|
|
||||||
use function is_null;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Entity for actors
|
* Entity for actors
|
||||||
@ -102,7 +100,7 @@ class Actor extends Entity
|
|||||||
|
|
||||||
public function getFullname(): ?string
|
public function getFullname(): ?string
|
||||||
{
|
{
|
||||||
if (is_null($this->fullname)) {
|
if (\is_null($this->fullname)) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
return $this->fullname;
|
return $this->fullname;
|
||||||
@ -254,17 +252,17 @@ class Actor extends Entity
|
|||||||
|
|
||||||
public static function getById(int $id): ?self
|
public static function getById(int $id): ?self
|
||||||
{
|
{
|
||||||
return Cache::get('actor-id-' . $id, fn() => DB::find('actor', ['id' => $id]));
|
return Cache::get('actor-id-' . $id, fn () => DB::find('actor', ['id' => $id]));
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function getNicknameById(int $id): string
|
public static function getNicknameById(int $id): string
|
||||||
{
|
{
|
||||||
return Cache::get('actor-nickname-id-' . $id, fn() => self::getById($id)->getNickname());
|
return Cache::get('actor-nickname-id-' . $id, fn () => self::getById($id)->getNickname());
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function getFullnameById(int $id): ?string
|
public static function getFullnameById(int $id): ?string
|
||||||
{
|
{
|
||||||
return Cache::get('actor-fullname-id-' . $id, fn() => self::getById($id)->getFullname());
|
return Cache::get('actor-fullname-id-' . $id, fn () => self::getById($id)->getFullname());
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -280,22 +278,22 @@ class Actor extends Entity
|
|||||||
/**
|
/**
|
||||||
* Get tags that other people put on this actor, in reverse-chron order
|
* Get tags that other people put on this actor, in reverse-chron order
|
||||||
*
|
*
|
||||||
* @param Actor|int|null $scoped Actor we are requesting as:
|
* @param null|Actor|int $scoped Actor we are requesting as:
|
||||||
* - If null = All tags attributed to self by other actors (excludes self tags)
|
* - If null = All tags attributed to self by other actors (excludes self tags)
|
||||||
* - If self = Same as getSelfTags
|
* - If self = Same as getSelfTags
|
||||||
* - otherwise = Tags that $scoped attributed to $this
|
* - otherwise = Tags that $scoped attributed to $this
|
||||||
* @param int|null $offset Offset from latest
|
* @param null|int $offset Offset from latest
|
||||||
* @param int|null $limit Max number to get
|
* @param null|int $limit Max number to get
|
||||||
* @param bool $_test_force_recompute
|
*
|
||||||
* @return [ActorCircle] resulting lists
|
* @return [ActorCircle] resulting lists
|
||||||
*/
|
*/
|
||||||
public function getOtherTags(Actor|int|null $scoped = null, ?int $offset = null, ?int $limit = null, bool $_test_force_recompute = false): array
|
public function getOtherTags(self|int|null $scoped = null, ?int $offset = null, ?int $limit = null, bool $_test_force_recompute = false): array
|
||||||
{
|
{
|
||||||
if (is_null($scoped)) {
|
if (\is_null($scoped)) {
|
||||||
return Cache::get(
|
return Cache::get(
|
||||||
"othertags-{$this->getId()}",
|
"othertags-{$this->getId()}",
|
||||||
fn() => DB::dql(
|
fn () => DB::dql(
|
||||||
<<< EOQ
|
<<< 'EOQ'
|
||||||
SELECT circle
|
SELECT circle
|
||||||
FROM App\Entity\ActorTag tag
|
FROM App\Entity\ActorTag tag
|
||||||
JOIN App\Entity\ActorCircle circle
|
JOIN App\Entity\ActorCircle circle
|
||||||
@ -307,14 +305,15 @@ class Actor extends Entity
|
|||||||
EOQ,
|
EOQ,
|
||||||
['id' => $this->getId()],
|
['id' => $this->getId()],
|
||||||
['offset' => $offset,
|
['offset' => $offset,
|
||||||
'limit' => $limit])
|
'limit' => $limit, ],
|
||||||
|
),
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
$scoped_id = is_int($scoped) ? $scoped : $scoped->getId();
|
$scoped_id = \is_int($scoped) ? $scoped : $scoped->getId();
|
||||||
return Cache::get(
|
return Cache::get(
|
||||||
"othertags-{$this->getId()}-by-{$scoped_id}",
|
"othertags-{$this->getId()}-by-{$scoped_id}",
|
||||||
fn() => DB::dql(
|
fn () => DB::dql(
|
||||||
<<< EOQ
|
<<< 'EOQ'
|
||||||
SELECT circle
|
SELECT circle
|
||||||
FROM App\Entity\ActorTag tag
|
FROM App\Entity\ActorTag tag
|
||||||
JOIN App\Entity\ActorCircle circle
|
JOIN App\Entity\ActorCircle circle
|
||||||
@ -331,40 +330,40 @@ class Actor extends Entity
|
|||||||
ORDER BY tag.modified DESC, tag.tagged DESC
|
ORDER BY tag.modified DESC, tag.tagged DESC
|
||||||
EOQ,
|
EOQ,
|
||||||
['id' => $this->getId(),
|
['id' => $this->getId(),
|
||||||
'scoped' => $scoped_id],
|
'scoped' => $scoped_id, ],
|
||||||
['offset' => $offset,
|
['offset' => $offset,
|
||||||
'limit' => $limit]
|
'limit' => $limit, ],
|
||||||
)
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param array $tags array of strings to become self tags
|
* @param array $tags array of strings to become self tags
|
||||||
* @param array|null $existing array of existing self tags (actor_circle[])
|
* @param null|array $existing array of existing self tags (actor_circle[])
|
||||||
* @return $this
|
*
|
||||||
* @throws NotFoundException
|
|
||||||
* @throws \App\Util\Exception\DuplicateFoundException
|
* @throws \App\Util\Exception\DuplicateFoundException
|
||||||
|
* @throws NotFoundException
|
||||||
|
*
|
||||||
|
* @return $this
|
||||||
*/
|
*/
|
||||||
public function setSelfTags(array $tags, ?array $existing = null): self
|
public function setSelfTags(array $tags, ?array $existing = null): self
|
||||||
{
|
{
|
||||||
if (is_null($existing)) {
|
if (\is_null($existing)) {
|
||||||
$existing = $this->getSelfTags();
|
$existing = $this->getSelfTags();
|
||||||
}
|
}
|
||||||
$existing_actor_circles = F\map($existing, fn($actor_circle) => $actor_circle->getTag());
|
$existing_actor_circles = F\map($existing, fn ($actor_circle) => $actor_circle->getTag());
|
||||||
$tags_to_add = array_diff($tags, $existing_actor_circles);
|
$tags_to_add = array_diff($tags, $existing_actor_circles);
|
||||||
$tags_to_remove = array_diff($existing_actor_circles, $tags);
|
$tags_to_remove = array_diff($existing_actor_circles, $tags);
|
||||||
$actor_circles_to_remove = F\filter($existing, fn($actor_circle) => in_array($actor_circle->getTag(), $tags_to_remove));
|
$actor_circles_to_remove = F\filter($existing, fn ($actor_circle) => \in_array($actor_circle->getTag(), $tags_to_remove));
|
||||||
foreach ($tags_to_add as $tag) {
|
foreach ($tags_to_add as $tag) {
|
||||||
$actor_circle = ActorCircle::create(['tagger' => $this->getId(), 'tag' => $tag, 'private' => false]);
|
$canonical_tag = TagComponent::canonicalTag($tag, $this->getTopLanguage()->getLocale());
|
||||||
$actor_tag = ActorTag::create(['tagger' => $this->id, 'tagged' => $this->id, 'tag' => $tag]);
|
DB::persist(ActorCircle::create(['tagger' => $this->getId(), 'tag' => $canonical_tag, 'private' => false]));
|
||||||
DB::persist($actor_circle);
|
DB::persist(ActorTag::create(['tagger' => $this->id, 'tagged' => $this->id, 'tag' => $tag, 'canonical' => $canonical_tag]));
|
||||||
DB::persist($actor_tag);
|
|
||||||
}
|
}
|
||||||
foreach ($actor_circles_to_remove as $actor_circle) {
|
foreach ($actor_circles_to_remove as $actor_circle) {
|
||||||
$actor_tag = DB::findOneBy('actor_tag', ['tagger' => $this->getId(), 'tagged' => $this->getId(), 'tag' => $actor_circle->getTag()]);
|
$canonical_tag = TagComponent::canonicalTag($actor_circle->getTag(), $this->getTopLanguage()->getLocale());
|
||||||
DB::persist($actor_tag);
|
DB::removeBy('actor_tag', ['tagger' => $this->getId(), 'tagged' => $this->getId(), 'canonical' => $canonical_tag]);
|
||||||
DB::remove($actor_tag);
|
|
||||||
DB::removeBy('actor_circle', ['id' => $actor_circle->getId()]);
|
DB::removeBy('actor_circle', ['id' => $actor_circle->getId()]);
|
||||||
}
|
}
|
||||||
Cache::delete("selftags-{$this->getId()}");
|
Cache::delete("selftags-{$this->getId()}");
|
||||||
@ -419,7 +418,7 @@ class Actor extends Entity
|
|||||||
$nickname = Nickname::normalize($nickname, check_already_used: false);
|
$nickname = Nickname::normalize($nickname, check_already_used: false);
|
||||||
return Cache::get(
|
return Cache::get(
|
||||||
'relative-nickname-' . $nickname . '-' . $this->getId(),
|
'relative-nickname-' . $nickname . '-' . $this->getId(),
|
||||||
fn() => DB::dql(
|
fn () => DB::dql(
|
||||||
<<<'EOF'
|
<<<'EOF'
|
||||||
select a from actor a where
|
select a from actor a where
|
||||||
a.id in (select fa.subscribed from subscription fa join actor aa with fa.subscribed = aa.id where fa.subscriber = :actor_id and aa.nickname = :nickname) or
|
a.id in (select fa.subscribed from subscription fa join actor aa with fa.subscribed = aa.id where fa.subscriber = :actor_id and aa.nickname = :nickname) or
|
||||||
@ -471,6 +470,11 @@ class Actor extends Entity
|
|||||||
return $aliases;
|
return $aliases;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getTopLanguage(): Language
|
||||||
|
{
|
||||||
|
return ActorLanguage::getActorLanguages($this, context: null)[0];
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the most appropriate language for $this to use when
|
* Get the most appropriate language for $this to use when
|
||||||
* referring to $context (a reply or a group, for instance)
|
* referring to $context (a reply or a group, for instance)
|
||||||
@ -479,18 +483,13 @@ class Actor extends Entity
|
|||||||
*/
|
*/
|
||||||
public function getPreferredLanguageChoices(?self $context = null): array
|
public function getPreferredLanguageChoices(?self $context = null): array
|
||||||
{
|
{
|
||||||
$id = $context?->getId() ?? $this->getId();
|
$langs = ActorLanguage::getActorLanguages($this, context: $context);
|
||||||
$key = ActorLanguage::collectionCacheKey($this, $context);
|
return array_merge(...F\map($langs, fn ($l) => $l->toChoiceFormat()));
|
||||||
$langs = Cache::getList(
|
}
|
||||||
$key,
|
|
||||||
fn() => DB::dql(
|
public function isVisibleTo(self $other): bool
|
||||||
'select l from actor_language al join language l with al.language_id = l.id where al.actor_id = :id order by al.ordering ASC',
|
{
|
||||||
['id' => $id],
|
return true; // TODO
|
||||||
),
|
|
||||||
) ?: [
|
|
||||||
Language::getFromLocale(Common::config('site', 'language')),
|
|
||||||
];
|
|
||||||
return array_merge(...F\map($langs, fn($l) => $l->toChoiceFormat()));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function schemaDef(): array
|
public static function schemaDef(): array
|
||||||
|
@ -53,7 +53,7 @@ class ActorCircle extends Entity
|
|||||||
private DateTimeInterface $created;
|
private DateTimeInterface $created;
|
||||||
private DateTimeInterface $modified;
|
private DateTimeInterface $modified;
|
||||||
|
|
||||||
public function setId(int $id): ActorCircle
|
public function setId(int $id): self
|
||||||
{
|
{
|
||||||
$this->id = $id;
|
$this->id = $id;
|
||||||
return $this;
|
return $this;
|
||||||
@ -133,22 +133,29 @@ class ActorCircle extends Entity
|
|||||||
// @codeCoverageIgnoreEnd
|
// @codeCoverageIgnoreEnd
|
||||||
// }}} Autocode
|
// }}} Autocode
|
||||||
|
|
||||||
|
public function getActorTag()
|
||||||
|
{
|
||||||
|
return Cache::get(
|
||||||
|
"actor-tag-{$this->getTag()}",
|
||||||
|
fn () => DB::findBy('actor_tag', ['tagger' => $this->getTagger(), 'canonical' => $this->getTag()], limit: 1)[0], // TODO jank
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
public function getSubscribedActors(?int $offset = null, ?int $limit = null): array
|
public function getSubscribedActors(?int $offset = null, ?int $limit = null): array
|
||||||
{
|
{
|
||||||
return Cache::get(
|
return Cache::get(
|
||||||
"circle-{$this->getId()}",
|
"circle-{$this->getId()}",
|
||||||
fn() => DB::dql(
|
fn () => DB::dql(
|
||||||
<<< EOQ
|
<<< 'EOQ'
|
||||||
SELECT a
|
SELECT a
|
||||||
FROM App\Entity\Actor a
|
FROM App\Entity\Actor a
|
||||||
JOIN App\Entity\ActorCircleSubscription s
|
JOIN App\Entity\ActorCircleSubscription s
|
||||||
WITH a.id = s.actor_id
|
WITH a.id = s.actor_id
|
||||||
ORDER BY s.created DESC, a.id DESC
|
ORDER BY s.created DESC, a.id DESC
|
||||||
EOQ,
|
EOQ,
|
||||||
options:
|
options: ['offset' => $offset,
|
||||||
['offset' => $offset,
|
'limit' => $limit, ],
|
||||||
'limit' => $limit]
|
),
|
||||||
)
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -160,7 +167,7 @@ class ActorCircle extends Entity
|
|||||||
'fields' => [
|
'fields' => [
|
||||||
'id' => ['type' => 'serial', 'not null' => true, 'description' => 'unique identifier'],
|
'id' => ['type' => 'serial', 'not null' => true, 'description' => 'unique identifier'],
|
||||||
'tagger' => ['type' => 'int', 'foreign key' => true, 'target' => 'Actor.id', 'multiplicity' => 'many to one', 'name' => 'actor_list_tagger_fkey', 'not null' => true, 'description' => 'user making the tag'],
|
'tagger' => ['type' => 'int', 'foreign key' => true, 'target' => 'Actor.id', 'multiplicity' => 'many to one', 'name' => 'actor_list_tagger_fkey', 'not null' => true, 'description' => 'user making the tag'],
|
||||||
'tag' => ['type' => 'varchar', 'length' => 64, 'foreign key' => true, 'target' => 'ActorTag.tag', 'multiplicity' => 'many to one', 'not null' => true, 'description' => 'actor tag'], // Join with ActorTag // // so, Doctrine doesn't like that the target is not unique, even though the pair is
|
'tag' => ['type' => 'varchar', 'length' => 64, 'foreign key' => true, 'target' => 'ActorTag.canonical', 'multiplicity' => 'many to one', 'not null' => true, 'description' => 'actor tag'], // Join with ActorTag // // so, Doctrine doesn't like that the target is not unique, even though the pair is
|
||||||
'description' => ['type' => 'text', 'description' => 'description of the people tag'],
|
'description' => ['type' => 'text', 'description' => 'description of the people tag'],
|
||||||
'private' => ['type' => 'bool', 'default' => false, 'description' => 'is this tag private'],
|
'private' => ['type' => 'bool', 'default' => false, 'description' => 'is this tag private'],
|
||||||
'created' => ['type' => 'datetime', 'not null' => true, 'default' => 'CURRENT_TIMESTAMP', 'description' => 'date this record was created'],
|
'created' => ['type' => 'datetime', 'not null' => true, 'default' => 'CURRENT_TIMESTAMP', 'description' => 'date this record was created'],
|
||||||
|
@ -23,8 +23,10 @@ declare(strict_types = 1);
|
|||||||
|
|
||||||
namespace App\Entity;
|
namespace App\Entity;
|
||||||
|
|
||||||
|
use App\Core\Cache;
|
||||||
use App\Core\DB\DB;
|
use App\Core\DB\DB;
|
||||||
use App\Core\Entity;
|
use App\Core\Entity;
|
||||||
|
use App\Util\Common;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Entity for actor languages
|
* Entity for actor languages
|
||||||
@ -79,9 +81,9 @@ class ActorLanguage extends Entity
|
|||||||
// @codeCoverageIgnoreEnd
|
// @codeCoverageIgnoreEnd
|
||||||
// }}} Autocode
|
// }}} Autocode
|
||||||
|
|
||||||
public static function collectionCacheKey(LocalUser|Actor $actor, ?Actor $content = null)
|
public static function collectionCacheKey(LocalUser|Actor $actor, ?Actor $context = null)
|
||||||
{
|
{
|
||||||
return 'actor-' . $actor->getId() . '-langs' . (!\is_null($content) ? '-cxt-' . $content->getId() : '');
|
return 'actor-' . $actor->getId() . '-langs' . (!\is_null($context) ? '-cxt-' . $context->getId() : '');
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function normalizeOrdering(LocalUser|Actor $actor)
|
public static function normalizeOrdering(LocalUser|Actor $actor)
|
||||||
@ -94,6 +96,21 @@ class ActorLanguage extends Entity
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return self[]
|
||||||
|
*/
|
||||||
|
public static function getActorLanguages(LocalUser|Actor $actor, ?Actor $context): array
|
||||||
|
{
|
||||||
|
$id = $context?->getId() ?? $actor->getId();
|
||||||
|
return Cache::getList(
|
||||||
|
self::collectionCacheKey($actor, context: $context),
|
||||||
|
fn () => DB::dql(
|
||||||
|
'select l from actor_language al join language l with al.language_id = l.id where al.actor_id = :id order by al.ordering ASC',
|
||||||
|
['id' => $id],
|
||||||
|
),
|
||||||
|
) ?: [Language::getFromLocale(Common::config('site', 'language'))];
|
||||||
|
}
|
||||||
|
|
||||||
public static function schemaDef(): array
|
public static function schemaDef(): array
|
||||||
{
|
{
|
||||||
return [
|
return [
|
||||||
|
@ -23,6 +23,7 @@ namespace App\Entity;
|
|||||||
|
|
||||||
use App\Core\DB\DB;
|
use App\Core\DB\DB;
|
||||||
use App\Core\Entity;
|
use App\Core\Entity;
|
||||||
|
use App\Core\Router\Router;
|
||||||
use Component\Tag\Tag;
|
use Component\Tag\Tag;
|
||||||
use DateTimeInterface;
|
use DateTimeInterface;
|
||||||
|
|
||||||
@ -108,6 +109,15 @@ class ActorTag extends Entity
|
|||||||
// @codeCoverageIgnoreEnd
|
// @codeCoverageIgnoreEnd
|
||||||
// }}} Autocode
|
// }}} Autocode
|
||||||
|
|
||||||
|
public function getUrl(?Actor $actor = null): string
|
||||||
|
{
|
||||||
|
$params = ['tag' => $this->getCanonical()];
|
||||||
|
if (!\is_null($actor)) {
|
||||||
|
$params['lang'] = $actor->getTopLanguage()->getLocale();
|
||||||
|
}
|
||||||
|
return Router::url('single_actor_tag', $params);
|
||||||
|
}
|
||||||
|
|
||||||
public static function schemaDef(): array
|
public static function schemaDef(): array
|
||||||
{
|
{
|
||||||
return [
|
return [
|
||||||
@ -119,10 +129,10 @@ class ActorTag extends Entity
|
|||||||
'canonical' => ['type' => 'varchar', 'length' => Tag::MAX_TAG_LENGTH, 'not null' => true, 'description' => 'ascii slug of tag'],
|
'canonical' => ['type' => 'varchar', 'length' => Tag::MAX_TAG_LENGTH, 'not null' => true, 'description' => 'ascii slug of tag'],
|
||||||
'modified' => ['type' => 'timestamp', 'not null' => true, 'default' => 'CURRENT_TIMESTAMP', 'description' => 'date this record was modified'],
|
'modified' => ['type' => 'timestamp', 'not null' => true, 'default' => 'CURRENT_TIMESTAMP', 'description' => 'date this record was modified'],
|
||||||
],
|
],
|
||||||
'primary key' => ['tagger', 'tagged', 'tag'],
|
'primary key' => ['tagger', 'tagged', 'canonical'],
|
||||||
'indexes' => [
|
'indexes' => [
|
||||||
'actor_tag_modified_idx' => ['modified'],
|
'actor_tag_modified_idx' => ['modified'],
|
||||||
'actor_tag_tagger_tag_idx' => ['tagger', 'tag'], // For Circles
|
'actor_tag_tagger_canonical_idx' => ['tagger', 'canonical'], // For Circles
|
||||||
'actor_tag_tagged_idx' => ['tagged'],
|
'actor_tag_tagged_idx' => ['tagged'],
|
||||||
],
|
],
|
||||||
];
|
];
|
||||||
|
@ -29,7 +29,7 @@
|
|||||||
<nav class="profile-info-tags">
|
<nav class="profile-info-tags">
|
||||||
{% if actor_tags %}
|
{% if actor_tags %}
|
||||||
{% for tag in actor_tags %}
|
{% for tag in actor_tags %}
|
||||||
<a href='#'><em>#{{ tag.getTag() }}</em></a>
|
<a href="{{ tag.getActorTag().getUrl(actor) }}"><em>#{{ tag.getTag() }}</em></a>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
{% else %}
|
{% else %}
|
||||||
{{ '(No tags)' | trans }}
|
{{ '(No tags)' | trans }}
|
||||||
|
Loading…
Reference in New Issue
Block a user