[PLUGIN][Pinned Notes] Allow user to pin his notes

This commit is contained in:
Phablulo Joel 2022-01-16 13:04:56 -03:00 committed by Hugo Sales
parent f7cbfbff8c
commit 21c7912702
Signed by untrusted user: someonewithpc
GPG Key ID: 7D0C7EAFC9D835A0
7 changed files with 336 additions and 0 deletions

View File

@ -0,0 +1,91 @@
<?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 Plugin\PinnedNotes\Controller;
use App\Core\DB\DB;
use App\Core\Form;
use function App\Core\I18n\_m;
use App\Core\Router\Router;
use App\Util\Common;
use App\Util\Exception\ClientException;
use App\Util\Exception\NoSuchNoteException;
use App\Util\Exception\RedirectException;
use Component\Collection\Util\Controller\FeedController;
use Plugin\PinnedNotes\Entity as E;
use Symfony\Component\Form\Extension\Core\Type\SubmitType;
use Symfony\Component\HttpFoundation\Request;
class PinnedNotes extends FeedController
{
public function togglePin(Request $request, int $id)
{
$user = Common::ensureLoggedIn();
$note = DB::findOneBy('note', ['id' => $id]);
if ($user->getId() !== $note?->getActorId()) {
throw new NoSuchNoteException();
}
$opts = ['note_id' => $id, 'actor_id' => $user->getId()];
$is_pinned = !\is_null(DB::findOneBy(E\PinnedNotes::class, $opts, return_null: true));
$form = Form::create([
['toggle_pin', SubmitType::class, [
'label' => _m(($is_pinned ? 'Unpin' : 'Pin') . ' this note'),
'attr' => [
'title' => _m(($is_pinned ? 'Unpin' : 'Pin') . ' this note'),
],
]],
]);
$form->handleRequest($request);
if ($form->isSubmitted()) {
$opts = ['note_id' => $id, 'actor_id' => $user->getId()];
if ($is_pinned) {
$pinned = DB::findOneBy(E\PinnedNotes::class, $opts);
DB::remove($pinned);
} else {
DB::persist(E\PinnedNotes::create($opts));
}
DB::flush();
// redirect user to where they came from, but prevent open redirect
if (!\is_null($from = $this->string('from'))) {
if (Router::isAbsolute($from)) {
throw new ClientException(_m('Can not redirect to outside the website from here'), 400); // 400 Bad request (deceptive)
} else {
throw new RedirectException(url: $from);
}
} else {
// If we don't have a URL to return to, go to the instance root
throw new RedirectException('root');
}
}
return [
'_template' => 'PinnedNotes/toggle.html.twig',
'note' => $note,
'title' => _m($is_pinned ? 'Unpin note' : 'Pin note'),
'toggle_form' => $form->createView(),
];
}
}

View File

@ -0,0 +1,57 @@
<?php
declare(strict_types = 1);
namespace Plugin\PinnedNotes\Entity;
use App\Core\Entity;
class PinnedNotes extends Entity
{
private int $id;
private int $actor_id;
private int $note_id;
public function getId()
{
return $this->id;
}
public function setId($id)
{
$this->id = $id;
return $this;
}
public function getActorId()
{
return $this->actor_id;
}
public function setActorId($actor_id)
{
$this->actor_id = $actor_id;
return $this;
}
public function getNoteId()
{
return $this->note_id;
}
public function setNoteId($note_id)
{
$this->note_id = $note_id;
return $this;
}
public static function schemaDef()
{
return [
'name' => 'pinned_notes',
'fields' => [
'id' => ['type' => 'serial', 'not null' => true, 'description' => 'unique identifier'],
'actor_id' => ['type' => 'int', 'not null' => true, 'foreign key' => true, 'target' => 'Actor.id', 'multiplicity' => 'many to one', 'description' => 'Actor who pinned the note'],
'note_id' => ['type' => 'int', 'not null' => true, 'foreign key' => true, 'target' => 'Note.id', 'multiplicity' => 'many to one', 'description' => 'Pinned note'],
],
'primary key' => ['id'],
];
}
}

View File

@ -0,0 +1,132 @@
<?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/>.
// }}}
/**
* WebMonetization for GNU social
*
* @package GNUsocial
* @category Plugin
*
* @author Phablulo <phablulo@gmail.com>
* @copyright 2018-2019, 2021 Free Software Foundation, Inc http://www.fsf.org
* @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
*/
namespace Plugin\PinnedNotes;
use App\Core\DB\DB;
use App\Core\Event;
use App\Core\Modules\Plugin;
use App\Core\Router\RouteLoader;
use App\Core\Router\Router;
use App\Entity\Actor;
use App\Entity\LocalUser;
use App\Entity\Note;
use App\Util\Common;
use App\Util\Formatting;
use Component\Collection\Collection;
use Doctrine\Common\Collections\ExpressionBuilder;
use Doctrine\ORM\Query\Expr;
use Doctrine\ORM\QueryBuilder;
use Plugin\PinnedNotes\Controller as C;
use Plugin\PinnedNotes\Entity as E;
use Symfony\Component\HttpFoundation\Request;
class PinnedNotes extends Plugin
{
public function onAddRoute(RouteLoader $r): bool
{
// Pin and unpin notes
$r->connect(id: 'toggle_note_pin', uri_path: '/object/note/{id<\d+>}/pin', target: [C\PinnedNotes::class, 'togglePin']);
return Event::next;
}
public function onBeforeFeed(Request $request, &$res): bool
{
$path = $request->attributes->get('_route');
if ($path === 'actor_view_nickname') {
$actor = LocalUser::getByNickname($request->attributes->get('nickname'))->getActor();
} elseif ($path === 'actor_view_id') {
$actor = DB::findOneBy(Actor::class, ['id' => $request->attributes->get('id')]);
} else {
return Event::next;
}
$locale = Common::currentLanguage()->getLocale();
$notes = Collection::query('pinned:true actor:' . $actor->getId(), 1, $locale, $actor);
$res[] = Formatting::twigRenderFile('PinnedNotes/notes.html.twig', ['pinnednotes' => $notes['notes']]);
return Event::next;
}
public function onCollectionQueryAddJoins(QueryBuilder &$note_qb, QueryBuilder &$actor_qb): bool
{
$note_qb->leftJoin(E\PinnedNotes::class, 'pinned', Expr\Join::WITH, 'note.id = pinned.note_id');
return Event::next;
}
public function onCollectionQueryCreateExpression(ExpressionBuilder $eb, string $term, ?string $language, ?Actor $actor, &$note_expr, &$actor_expr): bool
{
if ($term === 'pinned:true') {
$note_expr = $eb->neq('pinned', null);
return Event::stop;
}
if (str_starts_with($term, 'actor:')) {
$actor_id = (int) mb_substr($term, 6);
$actor_expr = $eb->eq('actor.id', $actor_id);
return Event::stop;
}
return Event::next;
}
public function onAddNoteActions(Request $request, Note $note, array &$actions): bool
{
$user = Common::user();
if ($user->getId() !== $note->getActorId()) {
return Event::next;
}
$opts = ['note_id' => $note->getId(), 'actor_id' => $user->getId()];
$is_pinned = !\is_null(DB::findOneBy(E\PinnedNotes::class, $opts, return_null: true));
$router_args = ['id' => $note->getId()];
$router_type = Router::ABSOLUTE_PATH;
$action_url = Router::url('toggle_note_pin', $router_args, $router_type);
$action_url .= '?from=' . urlencode($request->getRequestUri()); // so we can go back
$actions[] = [
'url' => $action_url,
'title' => ($is_pinned ? 'Unpin' : 'Pin') . ' this note',
'classes' => 'button-container pin-button-container ' . ($is_pinned ? 'pinned' : ''),
'id' => 'pin-button-container-' . $note->getId(),
];
return Event::next;
}
public function onEndShowStyles(array &$styles, string $route): bool
{
$styles[] = 'plugins/PinnedNotes/assets/css/pinned-notes.css';
return Event::next;
}
}

View File

@ -0,0 +1,20 @@
{% import '/cards/note/view.html.twig' as noteView %}
{# Backwards compatibility with hAtom 0.1 #}
{% if pinnednotes is not empty %}
<main class="feed pinned" tabindex="0" role="feed">
<h2>Pinned Notes</h2>
<div class="h-feed hfeed notes">
{% for conversation in pinnednotes %}
{% block current_note %}
{% if conversation is instanceof('array') %}
{{ noteView.macro_note(conversation['note'], conversation['replies']) }}
{% else %}
{{ noteView.macro_note(conversation) }}
{% endif %}
<hr tabindex="0" title="{{ 'End of note and replies.' | trans }}">
{% endblock current_note %}
{% endfor %}
</div>
</main>
{% endif %}

View File

@ -0,0 +1,19 @@
{% extends 'stdgrid.html.twig' %}
{% import "/cards/note/view.html.twig" as noteView %}
{% block title %}{{ title }}{% endblock %}
{% block stylesheets %}
{{ parent() }}
<link rel="stylesheet" href="{{ asset('assets/default_theme/css/pages/feeds.css') }}" type="text/css">
{% endblock stylesheets %}
{% block body %}
{{ parent() }}
<div class="page">
<div class="main">
{{ noteView.macro_note_minimal(note) }}
{{ form(toggle_form) }}
</div>
</div>
{% endblock body %}

View File

@ -0,0 +1,13 @@
.pin-button-container {
-o-mask-image: url("/plugins/PinnedNotes/assets/icons/pin.svg");
-moz-mask-image: url("/plugins/PinnedNotes/assets/icons/pin.svg");
-webkit-mask-image: url("/plugins/PinnedNotes/assets/icons/pin.svg");
mask-image: url("/plugins/PinnedNotes/assets/icons/pin.svg");
}
.pin-button-container.pinned {
opacity: 1 !important;
}
.feed.pinned {
margin-top: 2em;
margin-bottom: 6em;
}

View File

@ -0,0 +1,4 @@
<svg width="16" height="16" version="1.1" viewBox="0 0 16 16" xmlns="http://www.w3.org/2000/svg">
<path d="m1.0747 6.0949c3.3791 1.7463 4.05 1.6143 5.6765 7.3569l6.7287-6.7695c-5.6096-1.7525-5.8268-2.3806-7.3809-5.6197z" stroke="#000" stroke-linecap="round" stroke-linejoin="round" stroke-width="2" style="paint-order:normal"/>
<path d="m10.358 10.209 4.5675 4.7288" fill="none" stroke="#000" stroke-linecap="round" stroke-linejoin="round" stroke-width="2"/>
</svg>

After

Width:  |  Height:  |  Size: 468 B