1
0
mirror of https://github.com/flarum/core.git synced 2025-08-03 15:07:53 +02:00

feat(pm): delete own messages (#4180)

This commit is contained in:
Sami Mazouz
2025-02-08 19:04:20 +01:00
committed by GitHub
parent ce5feca140
commit 975c2c936f
14 changed files with 186 additions and 26 deletions

View File

@@ -52,6 +52,8 @@ return [
->fields(fn () => [
Schema\Boolean::make('canSendAnyMessage')
->get(fn (User $user, Context $context) => $user->can('sendAnyMessage')),
Schema\Boolean::make('canDeleteOwnMessages')
->visible(fn (User $user, Context $context) => $context->getActor()->is($user)),
Schema\Integer::make('messageCount')
->get(function (object $model, Context $context) {
return Dialog::whereVisibleTo($context->getActor())

View File

@@ -1,18 +0,0 @@
import Extend from 'flarum/common/extenders';
import commonExtend from '../common/extend';
import app from 'flarum/admin/app';
export default [
...commonExtend,
new Extend.Admin().permission(
() => ({
icon: 'fas fa-envelope-open-text',
label: app.translator.trans('flarum-messages.admin.permissions.send_messages'),
permission: 'dialog.sendMessage',
allowGuest: false,
}),
'start',
95
),
];

View File

@@ -0,0 +1,45 @@
import Extend from 'flarum/common/extenders';
import commonExtend from '../common/extend';
import app from 'flarum/admin/app';
import SettingDropdown from 'flarum/admin/components/SettingDropdown';
export default [
...commonExtend,
new Extend.Admin()
.permission(
() => ({
icon: 'fas fa-envelope-open-text',
label: app.translator.trans('flarum-messages.admin.permissions.send_messages_label'),
permission: 'dialog.sendMessage',
allowGuest: false,
}),
'start',
95
)
.permission(
() => ({
icon: 'far fa-trash-alt',
label: app.translator.trans('flarum-messages.admin.permissions.delete_own_messages_label'),
id: 'flarum-messages.allow_delete_own_messages',
setting: () => {
const minutes = parseInt(app.data.settings['flarum-messages.allow_delete_own_messages'], 10);
return (
<SettingDropdown
default={'0'}
key="flarum-messages.allow_delete_own_messages"
options={[
{ value: '-1', label: app.translator.trans('core.admin.permissions_controls.allow_indefinitely_button') },
{ value: '10', label: app.translator.trans('core.admin.permissions_controls.allow_ten_minutes_button') },
{ value: 'reply', label: app.translator.trans('core.admin.permissions_controls.allow_until_reply_button') },
{ value: '0', label: app.translator.trans('core.admin.permissions_controls.allow_never_button') },
]}
/>
);
},
}),
'reply',
80
),
];

View File

@@ -9,5 +9,6 @@ export default [
.add('dialog-messages', DialogMessage), //
new Extend.Model(User) //
.attribute<boolean>('canSendAnyMessage'),
.attribute<boolean>('canSendAnyMessage')
.attribute<boolean>('canDeleteOwnMessage'),
];

View File

@@ -36,4 +36,8 @@ export default class DialogMessage extends Model {
user() {
return Model.hasOne<User>('user').call(this);
}
canDelete() {
return Model.attribute<boolean>('canDelete').call(this);
}
}

View File

@@ -9,9 +9,12 @@ import Comment from 'flarum/forum/components/Comment';
import PostUser from 'flarum/forum/components/PostUser';
import PostMeta from 'flarum/forum/components/PostMeta';
import classList from 'flarum/common/utils/classList';
import MessageControls from '../utils/MessageControls';
import type MessageStreamState from '../states/MessageStreamState';
export interface IMessageAttrs extends IAbstractPostAttrs {
message: DialogMessage;
state: MessageStreamState;
}
/**
@@ -29,7 +32,7 @@ export default abstract class Message<CustomAttrs extends IMessageAttrs = IMessa
}
controls(): Mithril.Children[] {
return [];
return MessageControls.controls(this.attrs.message, this).toArray();
}
freshness(): Date {

View File

@@ -161,7 +161,7 @@ export default class MessageStream<CustomAttrs extends IDialogStreamAttrs = IDia
return (
<div className="MessageStream-item" key={index} data-id={message.id()} data-number={message.number()}>
{this.timeGap(message)}
<Message message={message} />
<Message message={message} state={this.attrs.state} />
</div>
);
}

View File

@@ -0,0 +1,67 @@
import ItemList from 'flarum/common/utils/ItemList';
import Separator from 'flarum/common/components/Separator';
import type Mithril from 'mithril';
import type DialogMessage from '../../common/models/DialogMessage';
import type Message from '../components/Message';
import Button from 'flarum/common/components/Button';
import app from 'flarum/forum/app';
import extractText from 'flarum/common/utils/extractText';
const MessageControls = {
controls(message: DialogMessage, context: Message<any>) {
const items = new ItemList<Mithril.Children>();
Object.entries(this.sections()).forEach(([section, method]) => {
const controls = method.call(this, message, context).toArray();
if (controls.length) {
controls.forEach((item) => items.add(item.itemName, item));
items.add(section + 'Separator', <Separator />);
}
});
return items;
},
sections() {
return {
user: this.userControls,
moderation: this.moderationControls,
destructive: this.destructiveControls,
};
},
userControls(message: DialogMessage, context: Message) {
return new ItemList<Mithril.Children>();
},
moderationControls(message: DialogMessage, context: Message) {
return new ItemList<Mithril.Children>();
},
destructiveControls(message: DialogMessage, context: Message) {
const items = new ItemList<Mithril.Children>();
if (message.canDelete()) {
items.add(
'delete',
<Button icon="far fa-trash-alt" onclick={() => this.deleteAction(message, context)}>
{app.translator.trans('flarum-messages.forum.message_controls.delete_button')}
</Button>
);
}
return items;
},
deleteAction(message: DialogMessage, context: Message) {
if (!confirm(extractText(app.translator.trans('flarum-messages.forum.message_controls.delete_confirmation')))) return;
return message.delete().then(() => {
context.attrs.state.remove(message);
m.redraw();
});
},
};
export default MessageControls;

View File

@@ -3,7 +3,8 @@ flarum-messages:
# Translations in this namespace are used by the admin interface.
admin:
permissions:
send_messages: Send private messages
send_messages_label: Send private messages
delete_own_messages_label: Delete own messages
# Translations in this namespace are used by the forum user interface.
forum:
@@ -42,6 +43,10 @@ flarum-messages:
newest_button: Newest
oldest_button: Oldest
message_controls:
delete_button: Delete
delete_confirmation: Are you sure you want to delete this message? This action cannot be undone.
messages_page:
cannot_send_message_button: Can't Send a Message
empty_text: No new messages

View File

@@ -9,14 +9,36 @@
namespace Flarum\Messages\Access;
use Carbon\Carbon;
use Flarum\Messages\DialogMessage;
use Flarum\Settings\SettingsRepositoryInterface;
use Flarum\User\Access\AbstractPolicy;
use Flarum\User\User;
class DialogMessagePolicy extends AbstractPolicy
{
public function update(User $actor, DialogMessage $dialogMessage): bool
public function __construct(
protected SettingsRepositoryInterface $settings
) {
}
public function update(User $actor, DialogMessage $message): ?bool
{
return null;
}
public function delete(User $actor, DialogMessage $message): bool|null|string
{
if ($message->user_id === $actor->id) {
$allowHiding = $this->settings->get('flarum-messages.allow_delete_own_messages');
if ($allowHiding === '-1'
|| ($allowHiding === 'reply' && $message->number >= $message->dialog->lastMessage->number)
|| (is_numeric($allowHiding) && $message->created_at->diffInMinutes(new Carbon, true) < $allowHiding)) {
return $this->allow();
}
}
return false;
}
}

View File

@@ -78,6 +78,11 @@ class DialogMessageResource extends Resource\AbstractDatabaseResource
return $actor->can('sendAnyMessage');
}
}),
Endpoint\Delete::make()
->authenticated()
->visible(function (DialogMessage $message, Context $context): bool {
return $context->getActor()->can('delete', $message);
}),
Endpoint\Index::make()
->authenticated()
->defaultInclude([
@@ -166,6 +171,12 @@ class DialogMessageResource extends Resource\AbstractDatabaseResource
->items(1)
->set(fn () => null),
// Read-only.
Schema\Boolean::make('canDelete')
->get(function (DialogMessage $message, Context $context) {
return $context->getActor()->can('delete', $message);
}),
Schema\Relationship\ToOne::make('user')
->type('users')
->includable(),

View File

@@ -12,6 +12,7 @@ export type SettingDropdownOption = {
export interface ISettingDropdownAttrs extends ISelectDropdownAttrs {
setting?: string;
options: Array<SettingDropdownOption>;
default: any;
}
export default class SettingDropdown<CustomAttrs extends ISettingDropdownAttrs = ISettingDropdownAttrs> extends SelectDropdown<CustomAttrs> {
@@ -33,7 +34,7 @@ export default class SettingDropdown<CustomAttrs extends ISettingDropdownAttrs =
return super.view({
...vnode,
children: this.attrs.options.map(({ value, label }) => {
const active = app.data.settings[this.attrs.setting!] === value;
const active = (app.data.settings[this.attrs.setting!] ?? this.attrs.default) === value;
return (
<Button icon={active ? 'fas fa-check' : true} onclick={saveSettings.bind(this, { [this.attrs.setting!]: value })} active={active}>

View File

@@ -391,4 +391,12 @@ export default abstract class PaginatedListState<T extends Model, P extends Pagi
1
);
}
remove(model: T): void {
const page = this.pages.find((pg) => pg.items.includes(model));
if (page) {
page.items = page.items.filter((item) => item !== model);
}
}
}

View File

@@ -20,8 +20,9 @@ const PostControls = {
controls(post, context) {
const items = new ItemList();
['user', 'moderation', 'destructive'].forEach((section) => {
const controls = this[section + 'Controls'](post, context).toArray();
Object.entries(this.sections()).forEach(([section, method]) => {
const controls = method.call(this, post, context).toArray();
if (controls.length) {
controls.forEach((item) => items.add(item.itemName, item));
items.add(section + 'Separator', <Separator />);
@@ -31,6 +32,14 @@ const PostControls = {
return items;
},
sections() {
return {
user: this.userControls,
moderation: this.moderationControls,
destructive: this.destructiveControls,
};
},
/**
* Get controls for a post pertaining to the current user (e.g. report).
*