[Favourite][TWIG][CSS] Favourite now works.
This commit is contained in:
parent
e15044fe36
commit
1647c5391f
@ -28,6 +28,7 @@ use App\Core\Modules\Plugin;
|
|||||||
use App\Core\Router\RouteLoader;
|
use App\Core\Router\RouteLoader;
|
||||||
use App\Entity\Note;
|
use App\Entity\Note;
|
||||||
use App\Util\Common;
|
use App\Util\Common;
|
||||||
|
use App\Util\Exception\RedirectException;
|
||||||
use App\Util\Formatting;
|
use App\Util\Formatting;
|
||||||
use App\Util\Nickname;
|
use App\Util\Nickname;
|
||||||
use Symfony\Component\Form\Extension\Core\Type\HiddenType;
|
use Symfony\Component\Form\Extension\Core\Type\HiddenType;
|
||||||
@ -39,31 +40,56 @@ class Favourite extends Plugin
|
|||||||
/**
|
/**
|
||||||
* HTML rendering event that adds the favourite form as a note
|
* HTML rendering event that adds the favourite form as a note
|
||||||
* action, if a user is logged in
|
* action, if a user is logged in
|
||||||
|
*
|
||||||
|
* @param Request $request
|
||||||
|
* @param Note $note
|
||||||
|
* @param array $actions
|
||||||
|
*
|
||||||
|
* @throws RedirectException
|
||||||
|
* @throws \App\Util\Exception\InvalidFormException
|
||||||
|
* @throws \App\Util\Exception\NoSuchNoteException
|
||||||
|
*
|
||||||
|
* @return bool Event hook
|
||||||
*/
|
*/
|
||||||
public function onAddNoteActions(Request $request, Note $note, array &$actions): bool
|
public function onAddNoteActions(Request $request, Note $note, array &$actions): bool
|
||||||
{
|
{
|
||||||
if (($user = Common::user()) == null) {
|
if (($user = Common::user()) === null) {
|
||||||
return Event::next;
|
return Event::next;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// if note is favourited, "is_set" is 1
|
||||||
$opts = ['note_id' => $note->getId(), 'gsactor_id' => $user->getId()];
|
$opts = ['note_id' => $note->getId(), 'gsactor_id' => $user->getId()];
|
||||||
$is_set = DB::find('favourite', $opts) != null;
|
$is_set = DB::find('favourite', $opts) != null;
|
||||||
$form = Form::create([
|
$form = Form::create([
|
||||||
['is_set', HiddenType::class, ['data' => $is_set ? '1' : '0']],
|
['is_set', HiddenType::class, ['data' => $is_set ? '1' : '0']],
|
||||||
['note_id', HiddenType::class, ['data' => $note->getId()]],
|
['note_id', HiddenType::class, ['data' => $note->getId()]],
|
||||||
['favourite', SubmitType::class, ['label' => ' ']],
|
["favourite-{$note->getId()}", SubmitType::class, ['label' => ' ', 'attr' => ['class' => $is_set ? 'favourite-button-on' : 'favourite-button-off']]],
|
||||||
]);
|
]);
|
||||||
|
|
||||||
// Form handler
|
// Form handler
|
||||||
$ret = self::noteActionHandle($request, $form, $note, 'favourite', function ($note, $data) use ($opts) {
|
$ret = self::noteActionHandle(
|
||||||
|
$request, $form, $note, "favourite-{$note->getId()}", /**
|
||||||
|
* Called from form handler
|
||||||
|
*
|
||||||
|
* @param $note Note to be favourited
|
||||||
|
* @param $data Form input
|
||||||
|
*
|
||||||
|
* @throws RedirectException Always thrown in order to prevent accidental form re-submit from browser
|
||||||
|
*/ function ($note, $data) use ($opts, $request) {
|
||||||
$fave = DB::find('favourite', $opts);
|
$fave = DB::find('favourite', $opts);
|
||||||
if (!$data['is_set'] && ($fave == null)) {
|
if ($data['is_set'] === '0' && $fave === null) {
|
||||||
DB::persist(Entity\Favourite::create($opts));
|
DB::persist(Entity\Favourite::create($opts));
|
||||||
DB::flush();
|
DB::flush();
|
||||||
} else {
|
} else {
|
||||||
|
if ($data['is_set'] === '1' && $fave !== null) {
|
||||||
DB::remove($fave);
|
DB::remove($fave);
|
||||||
DB::flush();
|
DB::flush();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Prevent accidental refreshes from resubmitting the form
|
||||||
|
throw new RedirectException();
|
||||||
|
|
||||||
return Event::stop;
|
return Event::stop;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -190,6 +190,34 @@
|
|||||||
height: var(--main-size);
|
height: var(--main-size);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* TODO: icons */
|
||||||
|
.note-actions {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
.favourite-button-off {
|
||||||
|
all: unset;
|
||||||
|
|
||||||
|
cursor: pointer !important;
|
||||||
|
border: none !important;
|
||||||
|
width: 1em !important;
|
||||||
|
height: 1em !important;
|
||||||
|
background-size: cover !important;
|
||||||
|
background-image: url(../icons/heart.svg) !important;
|
||||||
|
filter: invert(26%) sepia(21%) saturate(580%) hue-rotate(201deg) brightness(96%) contrast(93%);
|
||||||
|
}
|
||||||
|
.favourite-button-on {
|
||||||
|
all: unset;
|
||||||
|
|
||||||
|
cursor: pointer !important;
|
||||||
|
border: none !important;
|
||||||
|
width: 1em !important;
|
||||||
|
height: 1em !important;
|
||||||
|
background-size: cover !important;
|
||||||
|
background-image: url(../icons/heart.svg) !important;
|
||||||
|
filter: invert(100%);
|
||||||
|
}
|
||||||
|
|
||||||
.note-content {
|
.note-content {
|
||||||
padding: var(--small-size);
|
padding: var(--small-size);
|
||||||
}
|
}
|
||||||
|
@ -37,7 +37,7 @@
|
|||||||
|
|
||||||
#user-avatar img {
|
#user-avatar img {
|
||||||
border-radius: 50%;
|
border-radius: 50%;
|
||||||
max-width: calc(3 * var(--main-size));
|
width: calc(3 * var(--main-size));
|
||||||
height: auto;
|
height: auto;
|
||||||
|
|
||||||
margin-right: var(--unit-size);
|
margin-right: var(--unit-size);
|
||||||
|
@ -81,6 +81,7 @@ html {
|
|||||||
-webkit-font-smoothing: antialiased;
|
-webkit-font-smoothing: antialiased;
|
||||||
-moz-osx-font-smoothing: grayscale;
|
-moz-osx-font-smoothing: grayscale;
|
||||||
text-rendering: optimizeLegibility;
|
text-rendering: optimizeLegibility;
|
||||||
|
scroll-behavior: smooth;
|
||||||
|
|
||||||
background-attachment: fixed;
|
background-attachment: fixed;
|
||||||
color: var(--white);
|
color: var(--white);
|
||||||
|
9
public/assets/icons/heart.svg
Normal file
9
public/assets/icons/heart.svg
Normal file
File diff suppressed because one or more lines are too long
After Width: | Height: | Size: 8.9 KiB |
File diff suppressed because one or more lines are too long
Before Width: | Height: | Size: 8.8 KiB |
@ -19,8 +19,12 @@
|
|||||||
|
|
||||||
namespace App\Core\Modules;
|
namespace App\Core\Modules;
|
||||||
|
|
||||||
|
use App\Core\Event;
|
||||||
|
use App\Core\Log;
|
||||||
use App\Entity\Note;
|
use App\Entity\Note;
|
||||||
use App\Util\Common;
|
use App\Util\Common;
|
||||||
|
use App\Util\Exception\InvalidFormException;
|
||||||
|
use App\Util\Exception\NoSuchNoteException;
|
||||||
use Symfony\Component\Form\Form;
|
use Symfony\Component\Form\Form;
|
||||||
use Symfony\Component\HttpFoundation\Request;
|
use Symfony\Component\HttpFoundation\Request;
|
||||||
|
|
||||||
@ -51,6 +55,17 @@ abstract class Module
|
|||||||
* This function is called when a user interacts with a note, such as through favouriting or commenting
|
* This function is called when a user interacts with a note, such as through favouriting or commenting
|
||||||
*
|
*
|
||||||
* @codeCoverageIgnore
|
* @codeCoverageIgnore
|
||||||
|
*
|
||||||
|
* @param Request $request
|
||||||
|
* @param Form $form
|
||||||
|
* @param Note $note
|
||||||
|
* @param string $form_name
|
||||||
|
* @param callable $handle
|
||||||
|
*
|
||||||
|
* @throws InvalidFormException
|
||||||
|
* @throws NoSuchNoteException
|
||||||
|
*
|
||||||
|
* @return bool|void
|
||||||
*/
|
*/
|
||||||
public static function noteActionHandle(Request $request, Form $form, Note $note, string $form_name, callable $handle)
|
public static function noteActionHandle(Request $request, Form $form, Note $note, string $form_name, callable $handle)
|
||||||
{
|
{
|
||||||
@ -65,7 +80,7 @@ abstract class Module
|
|||||||
$user = Common::user();
|
$user = Common::user();
|
||||||
if (!$note->isVisibleTo($user)) {
|
if (!$note->isVisibleTo($user)) {
|
||||||
// ^ Ensure user isn't trying to trip us up
|
// ^ Ensure user isn't trying to trip us up
|
||||||
Log::error('Suspicious activity: user ' . $user->getNickname() .
|
Log::warning('Suspicious activity: user ' . $user->getNickname() .
|
||||||
' tried to interact with note ' . $note->getId() .
|
' tried to interact with note ' . $note->getId() .
|
||||||
', but they shouldn\'t have access to it');
|
', but they shouldn\'t have access to it');
|
||||||
throw new NoSuchNoteException();
|
throw new NoSuchNoteException();
|
||||||
|
@ -29,7 +29,7 @@
|
|||||||
</head>
|
</head>
|
||||||
|
|
||||||
<body class="bg">
|
<body class="bg">
|
||||||
{% block header %}
|
{% block header %}
|
||||||
<div id='header'>
|
<div id='header'>
|
||||||
<details class="panel" id="left-container">
|
<details class="panel" id="left-container">
|
||||||
<summary tabindex="1">
|
<summary tabindex="1">
|
||||||
@ -59,7 +59,6 @@
|
|||||||
{% block javascripts %}{% endblock javascripts %}
|
{% block javascripts %}{% endblock javascripts %}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
|
||||||
|
@ -12,6 +12,16 @@
|
|||||||
{% if reply_to is not null and not skip_reply_to is defined %}
|
{% if reply_to is not null and not skip_reply_to is defined %}
|
||||||
{% trans with {'%name%': reply_to} %} in reply to %name% {% endtrans %}
|
{% trans with {'%name%': reply_to} %} in reply to %name% {% endtrans %}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
|
<div class="note-actions">
|
||||||
|
|
||||||
|
{% if have_user %}
|
||||||
|
{% for act in get_note_actions(note) %}
|
||||||
|
{{ form(act) }}
|
||||||
|
{% endfor %}
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
<div class="e-content entry-content note-content">
|
<div class="e-content entry-content note-content">
|
||||||
{% block markdown %}
|
{% block markdown %}
|
||||||
@ -32,13 +42,7 @@
|
|||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</div>
|
</div>
|
||||||
<div class="note-actions">
|
|
||||||
{# {% if have_user %} #}
|
|
||||||
{# {\#{% for act in get_note_actions(note) %}#\} #}
|
|
||||||
{# {\#{{ form(act) }}#\} #}
|
|
||||||
{# {\#{% endfor %}#\} #}
|
|
||||||
{# {% endif %} #}
|
|
||||||
</div>
|
|
||||||
{% if replies is defined %}
|
{% if replies is defined %}
|
||||||
<div class="u-in-reply-to replies">
|
<div class="u-in-reply-to replies">
|
||||||
{% for conversation in replies %}
|
{% for conversation in replies %}
|
||||||
|
Loading…
Reference in New Issue
Block a user