Update for new extension API; implement l10n

This commit is contained in:
Toby Zerner 2015-07-22 10:15:08 +09:30
parent fb9ed378e0
commit 9c384bee98
30 changed files with 690 additions and 340 deletions

View File

@ -0,0 +1,32 @@
# EditorConfig helps developers define and maintain consistent
# coding styles between different editors and IDEs
# editorconfig.org
root = true
[*]
end_of_line = lf
charset = utf-8
trim_trailing_whitespace = true
insert_final_newline = true
indent_style = space
indent_size = 2
[*.js]
indent_style = space
indent_size = 2
[*.{css,less}]
indent_style = space
indent_size = 2
[*.html]
indent_style = space
indent_size = 2
[*.{diff,md}]
trim_trailing_whitespace = false
[*.php]
indent_style = space
indent_size = 4

View File

@ -0,0 +1,5 @@
**/bower_components/**/*
**/node_modules/**/*
vendor/**/*
**/Gulpfile.js
**/dist/**/*

171
extensions/likes/.eslintrc Normal file
View File

@ -0,0 +1,171 @@
{
"parser": "babel-eslint", // https://github.com/babel/babel-eslint
"env": { // http://eslint.org/docs/user-guide/configuring.html#specifying-environments
"browser": true // browser global variables
},
"ecmaFeatures": {
"arrowFunctions": true,
"blockBindings": true,
"classes": true,
"defaultParams": true,
"destructuring": true,
"forOf": true,
"generators": false,
"modules": true,
"objectLiteralComputedProperties": true,
"objectLiteralDuplicateProperties": false,
"objectLiteralShorthandMethods": true,
"objectLiteralShorthandProperties": true,
"spread": true,
"superInFunctions": true,
"templateStrings": true,
"jsx": true
},
"globals": {
"m": true,
"app": true,
"$": true,
"moment": true
},
"rules": {
/**
* Strict mode
*/
// babel inserts "use strict"; for us
"strict": [2, "never"], // http://eslint.org/docs/rules/strict
/**
* ES6
*/
"no-var": 2, // http://eslint.org/docs/rules/no-var
"prefer-const": 2, // http://eslint.org/docs/rules/prefer-const
/**
* Variables
*/
"no-shadow": 2, // http://eslint.org/docs/rules/no-shadow
"no-shadow-restricted-names": 2, // http://eslint.org/docs/rules/no-shadow-restricted-names
"no-unused-vars": [2, { // http://eslint.org/docs/rules/no-unused-vars
"vars": "local",
"args": "after-used"
}],
"no-use-before-define": 2, // http://eslint.org/docs/rules/no-use-before-define
/**
* Possible errors
*/
"comma-dangle": [2, "never"], // http://eslint.org/docs/rules/comma-dangle
"no-cond-assign": [2, "always"], // http://eslint.org/docs/rules/no-cond-assign
"no-console": 1, // http://eslint.org/docs/rules/no-console
"no-debugger": 1, // http://eslint.org/docs/rules/no-debugger
"no-alert": 1, // http://eslint.org/docs/rules/no-alert
"no-constant-condition": 1, // http://eslint.org/docs/rules/no-constant-condition
"no-dupe-keys": 2, // http://eslint.org/docs/rules/no-dupe-keys
"no-duplicate-case": 2, // http://eslint.org/docs/rules/no-duplicate-case
"no-empty": 2, // http://eslint.org/docs/rules/no-empty
"no-ex-assign": 2, // http://eslint.org/docs/rules/no-ex-assign
"no-extra-boolean-cast": 0, // http://eslint.org/docs/rules/no-extra-boolean-cast
"no-extra-semi": 2, // http://eslint.org/docs/rules/no-extra-semi
"no-func-assign": 2, // http://eslint.org/docs/rules/no-func-assign
"no-inner-declarations": 2, // http://eslint.org/docs/rules/no-inner-declarations
"no-invalid-regexp": 2, // http://eslint.org/docs/rules/no-invalid-regexp
"no-irregular-whitespace": 2, // http://eslint.org/docs/rules/no-irregular-whitespace
"no-obj-calls": 2, // http://eslint.org/docs/rules/no-obj-calls
"no-reserved-keys": 2, // http://eslint.org/docs/rules/no-reserved-keys
"no-sparse-arrays": 2, // http://eslint.org/docs/rules/no-sparse-arrays
"no-unreachable": 2, // http://eslint.org/docs/rules/no-unreachable
"use-isnan": 2, // http://eslint.org/docs/rules/use-isnan
"block-scoped-var": 2, // http://eslint.org/docs/rules/block-scoped-var
/**
* Best practices
*/
"consistent-return": 2, // http://eslint.org/docs/rules/consistent-return
"curly": [2, "multi-line"], // http://eslint.org/docs/rules/curly
"default-case": 2, // http://eslint.org/docs/rules/default-case
"dot-notation": [2, { // http://eslint.org/docs/rules/dot-notation
"allowKeywords": true
}],
"eqeqeq": 2, // http://eslint.org/docs/rules/eqeqeq
"no-caller": 2, // http://eslint.org/docs/rules/no-caller
"no-else-return": 2, // http://eslint.org/docs/rules/no-else-return
"no-eq-null": 2, // http://eslint.org/docs/rules/no-eq-null
"no-eval": 2, // http://eslint.org/docs/rules/no-eval
"no-extend-native": 2, // http://eslint.org/docs/rules/no-extend-native
"no-extra-bind": 2, // http://eslint.org/docs/rules/no-extra-bind
"no-fallthrough": 2, // http://eslint.org/docs/rules/no-fallthrough
"no-floating-decimal": 2, // http://eslint.org/docs/rules/no-floating-decimal
"no-implied-eval": 2, // http://eslint.org/docs/rules/no-implied-eval
"no-lone-blocks": 2, // http://eslint.org/docs/rules/no-lone-blocks
"no-loop-func": 2, // http://eslint.org/docs/rules/no-loop-func
"no-multi-str": 2, // http://eslint.org/docs/rules/no-multi-str
"no-native-reassign": 2, // http://eslint.org/docs/rules/no-native-reassign
"no-new": 2, // http://eslint.org/docs/rules/no-new
"no-new-func": 2, // http://eslint.org/docs/rules/no-new-func
"no-new-wrappers": 2, // http://eslint.org/docs/rules/no-new-wrappers
"no-octal": 2, // http://eslint.org/docs/rules/no-octal
"no-octal-escape": 2, // http://eslint.org/docs/rules/no-octal-escape
"no-param-reassign": 2, // http://eslint.org/docs/rules/no-param-reassign
"no-proto": 2, // http://eslint.org/docs/rules/no-proto
"no-redeclare": 2, // http://eslint.org/docs/rules/no-redeclare
"no-return-assign": 2, // http://eslint.org/docs/rules/no-return-assign
"no-self-compare": 2, // http://eslint.org/docs/rules/no-self-compare
"no-sequences": 2, // http://eslint.org/docs/rules/no-sequences
"no-throw-literal": 2, // http://eslint.org/docs/rules/no-throw-literal
"no-with": 2, // http://eslint.org/docs/rules/no-with
"radix": 2, // http://eslint.org/docs/rules/radix
"vars-on-top": 2, // http://eslint.org/docs/rules/vars-on-top
"wrap-iife": [2, "any"], // http://eslint.org/docs/rules/wrap-iife
"yoda": 2, // http://eslint.org/docs/rules/yoda
/**
* Style
*/
"indent": [2, 2], // http://eslint.org/docs/rules/indent
"brace-style": [2, // http://eslint.org/docs/rules/brace-style
"1tbs", {
"allowSingleLine": true
}],
"quotes": [
2, "single", "avoid-escape" // http://eslint.org/docs/rules/quotes
],
"camelcase": [2, { // http://eslint.org/docs/rules/camelcase
"properties": "never"
}],
"comma-spacing": [2, { // http://eslint.org/docs/rules/comma-spacing
"before": false,
"after": true
}],
"comma-style": [2, "last"], // http://eslint.org/docs/rules/comma-style
"eol-last": 2, // http://eslint.org/docs/rules/eol-last
"func-names": 1, // http://eslint.org/docs/rules/func-names
"key-spacing": [2, { // http://eslint.org/docs/rules/key-spacing
"beforeColon": false,
"afterColon": true
}],
"new-cap": [2, { // http://eslint.org/docs/rules/new-cap
"newIsCap": true
}],
"no-multiple-empty-lines": [2, { // http://eslint.org/docs/rules/no-multiple-empty-lines
"max": 2
}],
"no-new-object": 2, // http://eslint.org/docs/rules/no-new-object
"no-spaced-func": 2, // http://eslint.org/docs/rules/no-spaced-func
"no-trailing-spaces": 2, // http://eslint.org/docs/rules/no-trailing-spaces
"no-wrap-func": 2, // http://eslint.org/docs/rules/no-wrap-func
"no-underscore-dangle": 0, // http://eslint.org/docs/rules/no-underscore-dangle
"one-var": [2, "never"], // http://eslint.org/docs/rules/one-var
"padded-blocks": [2, "never"], // http://eslint.org/docs/rules/padded-blocks
"semi": [2, "always"], // http://eslint.org/docs/rules/semi
"semi-spacing": [2, { // http://eslint.org/docs/rules/semi-spacing
"before": false,
"after": true
}],
"space-after-keywords": 2, // http://eslint.org/docs/rules/space-after-keywords
"space-before-blocks": 2, // http://eslint.org/docs/rules/space-before-blocks
"space-before-function-paren": [2, "never"], // http://eslint.org/docs/rules/space-before-function-paren
"space-infix-ops": 2, // http://eslint.org/docs/rules/space-infix-ops
"space-return-throw-case": 2, // http://eslint.org/docs/rules/space-return-throw-case
"spaced-line-comment": 2, // http://eslint.org/docs/rules/spaced-line-comment
}
}

View File

@ -1,9 +1,5 @@
<?php
// Require the extension's composer autoload file. This will enable all of our
// classes in the src directory to be autoloaded.
require __DIR__.'/vendor/autoload.php';
// Register our service provider with the Flarum application. In here we can
// register bindings and execute code when the application boots.
return $this->app->register('Flarum\Likes\LikesServiceProvider');
return 'Flarum\Likes\Extension';

View File

@ -1,16 +1,21 @@
{
"name": "flarum-likes",
"name": "likes",
"title": "Likes",
"description": "Allows users to like posts.",
"tags": [],
"keywords": ["discussions"],
"version": "0.1.0",
"author": {
"name": "Toby Zerner",
"email": "toby@flarum.org'"
"email": "toby@flarum.org",
"homepage": "http://tobyzerner.com"
},
"license": "MIT",
"require": {
"php": ">=5.4.0",
"flarum": ">0.1.0"
},
"support": {
"source": "https://github.com/flarum/likes",
"issues": "https://github.com/flarum/likes/issues"
}
}
}

View File

@ -1,109 +0,0 @@
import { extend, override } from 'flarum/extension-utils';
import app from 'flarum/app';
import Post from 'flarum/models/post';
import Model from 'flarum/model';
import DiscussionPage from 'flarum/components/discussion-page';
import SettingsPage from 'flarum/components/settings-page';
import ActionButton from 'flarum/components/action-button';
import CommentPost from 'flarum/components/comment-post';
import punctuate from 'flarum/helpers/punctuate';
import username from 'flarum/helpers/username';
import icon from 'flarum/helpers/icon';
import PostLikedNotification from 'flarum-likes/components/post-liked-notification';
import PostLikesModal from 'flarum-likes/components/post-likes-modal';
app.initializers.add('flarum-likes', function() {
app.notificationComponentRegistry['postLiked'] = PostLikedNotification;
Post.prototype.canLike = Model.prop('canLike');
Post.prototype.likes = Model.many('likes');
extend(DiscussionPage.prototype, 'params', function(params) {
params.include.push('posts.likes');
});
extend(CommentPost.prototype, 'footerItems', function(items) {
var post = this.props.post;
var likes = post.likes();
if (likes && likes.length) {
var limit = 3;
var names = likes.slice(0, limit).map(user => {
return m('a', {
href: app.route.user(user),
config: m.route
}, [
app.session.user() && user === app.session.user() ? 'You' : username(user)
])
});
if (likes.length > limit + 1) {
names.push(
m('a', {
href: '#',
onclick: function(e) {
e.preventDefault();
app.modal.show(new PostLikesModal({ post }));
}
}, (likes.length - limit)+' others')
);
}
items.add('liked',
m('div.liked-by', [
icon('thumbs-o-up icon'),
punctuate(names),
names.length === 1 && (!app.session.user() || likes[0] !== app.session.user()) ? ' likes this.' : ' like this.'
]),
{before: 'replies'}
);
}
});
extend(CommentPost.prototype, 'actionItems', function(items) {
var post = this.props.post;
if (post.isHidden() || !post.canLike()) return;
var isLiked = app.session.user() && post.likes().some(user => user === app.session.user());
items.add('like',
ActionButton.component({
icon: 'thumbs-o-up',
label: isLiked ? 'Unlike' : 'Like',
onclick: () => {
isLiked = !isLiked;
post.save({ isLiked });
var linkage = post.data().links.likes.linkage;
linkage.some((like, i) => {
if (like.id == app.session.user().id()) {
linkage.splice(i, 1);
return true;
}
});
if (isLiked) {
linkage.unshift({ type: 'users', id: app.session.user().id() });
}
m.redraw();
}
}),
{before: 'reply'}
);
});
// Add a notification preference.
extend(SettingsPage.prototype, 'notificationTypes', function(items) {
items.add('postLiked', {
name: 'postLiked',
label: [icon('thumbs-o-up'), ' Someone likes my post']
});
});
});

View File

@ -1,5 +1,5 @@
var gulp = require('flarum-gulp');
gulp({
modulePrefix: 'flarum-likes'
modulePrefix: 'likes'
});

View File

@ -0,0 +1,41 @@
import { extend } from 'flarum/extend';
import app from 'flarum/app';
import Button from 'flarum/components/Button';
import CommentPost from 'flarum/components/CommentPost';
export default function() {
extend(CommentPost.prototype, 'actionItems', function(items) {
const post = this.props.post;
if (post.isHidden() || !post.canLike()) return;
let isLiked = app.session.user && post.likes().some(user => user === app.session.user);
items.add('like',
Button.component({
children: app.trans(isLiked ? 'likes.unlike_action' : 'likes.like_action'),
className: 'Button Button--text',
onclick: () => {
isLiked = !isLiked;
post.save({isLiked});
// We've saved the fact that we do or don't like the post, but in order
// to provide instantaneous feedback to the user, we'll need to add or
// remove the like from the relationship data manually.
const data = post.data.relationships.likes.data;
data.some((like, i) => {
if (like.id === app.session.user.id()) {
data.splice(i, 1);
return true;
}
});
if (isLiked) {
data.unshift({type: 'users', id: app.session.user.id()});
}
}
})
);
});
}

View File

@ -0,0 +1,60 @@
import { extend } from 'flarum/extend';
import app from 'flarum/app';
import DiscussionPage from 'flarum/components/DiscussionPage';
import CommentPost from 'flarum/components/CommentPost';
import punctuate from 'flarum/helpers/punctuate';
import username from 'flarum/helpers/username';
import icon from 'flarum/helpers/icon';
import PostLikesModal from 'likes/components/PostLikesModal';
export default function() {
extend(DiscussionPage.prototype, 'params', function(params) {
params.include.push('posts.likes');
});
extend(CommentPost.prototype, 'footerItems', function(items) {
const post = this.props.post;
const likes = post.likes();
if (likes && likes.length) {
const limit = 3;
// Construct a list of names of users who have like this post. Make sure the
// current user is first in the list, and cap a maximum of 3 names.
const names = likes.sort(a => a === app.session.user ? -1 : 1)
.slice(0, limit)
.map(user => {
return (
<a href={app.route.user(user)} config={m.route}>
{user === app.session.user ? 'You' : username(user)}
</a>
);
});
// If there are more users that we've run out of room to display, add a "x
// others" name to the end of the list. Clicking on it will display a modal
// with a full list of names.
if (likes.length > limit + 1) {
names.push(
<a href="#" onclick={e => {
e.preventDefault();
app.modal.show(new PostLikesModal({post}));
}}>
{app.trans('likes.others', {count: likes.length - limit})}
</a>
);
}
items.add('liked', (
<div className="Post-likedBy">
{icon('thumbs-o-up')}
{app.trans('likes.post_liked_by' + (likes[0] === app.session.user ? '_self' : ''), {
count: names.length,
users: punctuate(names)
})}
</div>
));
}
});
}

View File

@ -0,0 +1,27 @@
import Notification from 'flarum/components/Notification';
export default class PostLikedNotification extends Notification {
icon() {
return 'thumbs-o-up';
}
href() {
return app.route.post(this.props.notification.subject());
}
content() {
const notification = this.props.notification;
const post = notification.subject();
const user = notification.sender();
const auc = notification.additionalUnreadCount();
return app.trans('likes.post_liked_notification', {
user,
username: auc ? punctuate([
username(user),
app.trans('core.others', {count: auc})
]) : undefined,
number: post.number()
});
}
}

View File

@ -0,0 +1,30 @@
import Modal from 'flarum/components/Modal';
import avatar from 'flarum/helpers/avatar';
import username from 'flarum/helpers/username';
export default class PostLikesModal extends Modal {
className() {
return 'PostLikesModal Modal--small';
}
title() {
return app.trans('likes.post_likes_modal_title');
}
content() {
return (
<div className="Modal-body">
<ul className="PostLikesModal-list">
{this.props.post.likes().map(user => (
<li>
<a href={app.route.user(user)} config={m.route}>
{avatar(user)}{' '}
{username(user)}
</a>
</li>
))}
</ul>
</div>
);
}
}

View File

@ -0,0 +1,25 @@
import { extend } from 'flarum/extend';
import app from 'flarum/app';
import Post from 'flarum/models/Post';
import Model from 'flarum/Model';
import NotificationGrid from 'flarum/components/NotificationGrid';
import addLikeAction from 'likes/addLikeAction';
import addLikesList from 'likes/addLikesList';
import PostLikedNotification from 'likes/components/PostLikedNotification';
app.notificationComponents.postLiked = PostLikedNotification;
Post.prototype.canLike = Model.attribute('canLike');
Post.prototype.likes = Model.hasMany('likes');
addLikeAction();
addLikesList();
extend(NotificationGrid.prototype, 'notificationTypes', function(items) {
items.add('postLiked', {
name: 'postLiked',
icon: 'thumbs-o-up',
label: app.trans('likes.notify_post_liked')
});
});

View File

@ -1,16 +0,0 @@
import Notification from 'flarum/components/notification';
import username from 'flarum/helpers/username';
export default class PostLikedNotification extends Notification {
view() {
var notification = this.props.notification;
var post = notification.subject();
var auc = notification.additionalUnreadCount();
return super.view({
href: app.route.post(post),
icon: 'thumbs-o-up',
content: [username(notification.sender()), auc ? ' and '+auc+' others' : '', ' liked your post #', post.number()]
});
}
}

View File

@ -1,24 +0,0 @@
import FormModal from 'flarum/components/form-modal';
import avatar from 'flarum/helpers/avatar';
import username from 'flarum/helpers/username';
export default class PostLikesModal extends FormModal {
view() {
var post = this.props.post;
return super.view({
className: 'post-likes-modal',
title: 'Users Who Like This',
body: [
m('ul.post-likes-list', [
post.likes().map(user =>
m('li', m('a', {href: app.route.user(user), config: m.route}, [
avatar(user),
username(user)
]))
)
])
]
});
}
}

View File

@ -0,0 +1,23 @@
.PostLikesModal-list {
list-style: none;
padding: 0;
margin: 0;
a {
color: @text-color;
font-size: 15px;
font-weight: bold;
display: block;
margin-bottom: 10px;
text-decoration: none;
&:hover .username {
text-decoration: underline;
}
}
.Avatar {
.Avatar--size(32px);
vertical-align: middle;
margin-right: 5px;
}
}

View File

@ -1,2 +1,11 @@
flarum-likes:
# hello_world: Hello, world!
likes:
post_liked_notification: "{username} liked your post #{number}"
post_likes_modal_title: Users Who Like This
post_liked_by_self: "{users} like this."
post_liked_by:
one: "{users} likes this."
other: "{users} like this."
unlike_action: Unlike
like_action: Like
notify_post_liked: Someone likes my post
others: "{count} others"

View File

@ -1,23 +1,23 @@
<?php namespace Flarum\Likes\Events;
use Flarum\Core\Models\Post;
use Flarum\Core\Models\User;
use Flarum\Core\Posts\Post;
use Flarum\Core\Users\User;
class PostWasLiked
{
/**
* @var \Flarum\Core\Models\Post
* @var Post
*/
public $post;
/**
* @var \Flarum\Core\Models\User
* @var User
*/
public $user;
/**
* @param \Flarum\Core\Models\Post $post
* @param \Flarum\Core\Models\User $user
* @param Post $post
* @param User $user
*/
public function __construct(Post $post, User $user)
{

View File

@ -1,23 +1,23 @@
<?php namespace Flarum\Likes\Events;
use Flarum\Core\Models\Post;
use Flarum\Core\Models\User;
use Flarum\Core\Posts\Post;
use Flarum\Core\Users\User;
class PostWasUnliked
{
/**
* @var \Flarum\Core\Models\Post
* @var Post
*/
public $post;
/**
* @var \Flarum\Core\Models\User
* @var User
*/
public $user;
/**
* @param \Flarum\Core\Models\Post $post
* @param \Flarum\Core\Models\User $user
* @param Post $post
* @param User $user
*/
public function __construct(Post $post, User $user)
{

View File

@ -0,0 +1,16 @@
<?php namespace Flarum\Likes;
use Flarum\Support\Extension as BaseExtension;
use Illuminate\Events\Dispatcher;
class Extension extends BaseExtension
{
public function boot(Dispatcher $events)
{
$events->subscribe('Flarum\Likes\Listeners\AddClientAssets');
$events->subscribe('Flarum\Likes\Listeners\AddModelRelationship');
$events->subscribe('Flarum\Likes\Listeners\AddApiAttributes');
$events->subscribe('Flarum\Likes\Listeners\PersistData');
$events->subscribe('Flarum\Likes\Listeners\NotifyPostLiked');
}
}

View File

@ -1,47 +0,0 @@
<?php namespace Flarum\Likes\Handlers;
use Flarum\Likes\Events\PostWasLiked;
use Flarum\Likes\Events\PostWasUnliked;
use Flarum\Core\Events\PostWillBeSaved;
use Flarum\Core\Events\PostWasDeleted;
use Flarum\Core\Models\Post;
use Flarum\Core\Exceptions\PermissionDeniedException;
class LikedSaver
{
public function subscribe($events)
{
$events->listen('Flarum\Core\Events\PostWillBeSaved', __CLASS__.'@whenPostWillBeSaved');
$events->listen('Flarum\Core\Events\PostWasDeleted', __CLASS__.'@whenPostWasDeleted');
}
public function whenPostWillBeSaved(PostWillBeSaved $event)
{
$post = $event->post;
$data = $event->command->data;
if ($post->exists && isset($data['isLiked'])) {
$user = $event->command->user;
$liked = (bool) $data['isLiked'];
if (! $post->can($user, 'like')) {
throw new PermissionDeniedException;
}
if ($liked) {
$post->likes()->attach($user->id);
$post->raise(new PostWasLiked($post, $user));
} else {
$post->likes()->detach($user->id);
$post->raise(new PostWasUnliked($post, $user));
}
}
}
public function whenPostWasDeleted(PostWasDeleted $event)
{
$event->post->likes()->detach();
}
}

View File

@ -1,50 +0,0 @@
<?php namespace Flarum\Likes\Handlers;
use Flarum\Likes\PostLikedNotification;
use Flarum\Likes\Events\PostWasLiked;
use Flarum\Likes\Events\PostWasUnliked;
use Flarum\Core\Notifications\NotificationSyncer;
use Illuminate\Contracts\Events\Dispatcher;
class PostLikedNotifier
{
protected $notifications;
public function __construct(NotificationSyncer $notifications)
{
$this->notifications = $notifications;
}
/**
* Register the listeners for the subscriber.
*
* @param \Illuminate\Contracts\Events\Dispatcher $events
*/
public function subscribe(Dispatcher $events)
{
$events->listen('Flarum\Likes\Events\PostWasLiked', __CLASS__.'@whenPostWasLiked');
$events->listen('Flarum\Likes\Events\PostWasUnliked', __CLASS__.'@whenPostWasUnliked');
}
public function whenPostWasLiked(PostWasLiked $event)
{
if ($event->post->user->id != $event->user->id) {
$this->sync($event->post, $event->user, [$event->post->user]);
}
}
public function whenPostWasUnliked(PostWasUnliked $event)
{
if ($event->post->user->id != $event->user->id) {
$this->sync($event->post, $event->user, []);
}
}
public function sync($post, $user, array $recipients)
{
$this->notifications->sync(
new PostLikedNotification($post, $user),
$recipients
);
}
}

View File

@ -1,64 +0,0 @@
<?php namespace Flarum\Likes;
use Flarum\Support\ServiceProvider;
use Flarum\Extend;
class LikesServiceProvider extends ServiceProvider
{
/**
* Bootstrap the application events.
*
* @return void
*/
public function boot()
{
$this->extend([
(new Extend\Locale('en'))->translations(__DIR__.'/../locale/en.yml'),
(new Extend\ForumClient)
->assets([
__DIR__.'/../js/dist/extension.js',
__DIR__.'/../less/extension.less'
]),
(new Extend\Model('Flarum\Core\Models\Post'))
->belongsToMany('likes', 'Flarum\Core\Models\User', 'posts_likes', 'post_id', 'user_id'),
(new Extend\ApiSerializer('Flarum\Api\Serializers\PostSerializer'))
->hasMany('likes', 'Flarum\Api\Serializers\UserBasicSerializer')
->attributes(function (&$attributes, $post, $user) {
$attributes['canLike'] = $post->can($user, 'like');
}),
(new Extend\ApiAction('Flarum\Api\Actions\Discussions\ShowAction'))
->addInclude('posts.likes'),
(new Extend\ApiAction([
'Flarum\Api\Actions\Posts\IndexAction',
'Flarum\Api\Actions\Posts\ShowAction',
'Flarum\Api\Actions\Posts\CreateAction',
'Flarum\Api\Actions\Posts\UpdateAction'
]))
->addInclude('likes'),
new Extend\EventSubscriber('Flarum\Likes\Handlers\LikedSaver'),
new Extend\EventSubscriber('Flarum\Likes\Handlers\PostLikedNotifier'),
(new Extend\NotificationType(
'Flarum\Likes\PostLikedNotification',
'Flarum\Api\Serializers\PostBasicSerializer'
))
->enableByDefault('alert')
]);
}
/**
* Register the service provider.
*
* @return void
*/
public function register()
{
//
}
}

View File

@ -0,0 +1,50 @@
<?php namespace Flarum\Likes\Listeners;
use Flarum\Events\ApiAttributes;
use Flarum\Events\ApiRelationship;
use Flarum\Events\BuildApiAction;
use Illuminate\Contracts\Events\Dispatcher;
use Flarum\Api\Serializers\PostSerializer;
use Flarum\Api\Actions\Discussions;
use Flarum\Api\Actions\Posts;
class AddApiAttributes
{
public function subscribe(Dispatcher $events)
{
$events->listen(ApiAttributes::class, __CLASS__.'@addAttributes');
$events->listen(ApiRelationship::class, __CLASS__.'@addRelationship');
$events->listen(BuildApiAction::class, __CLASS__.'@includeLikes');
}
public function addAttributes(ApiAttributes $event)
{
if ($event->serializer instanceof PostSerializer) {
$event->attributes['canLike'] = (bool) $event->model->can($event->actor, 'like');
}
}
public function addRelationship(ApiRelationship $event)
{
if ($event->serializer instanceof PostSerializer &&
$event->relationship === 'likes') {
return $event->serializer->hasMany('Flarum\Api\Serializers\UserBasicSerializer', 'likes');
}
}
public function includeLikes(BuildApiAction $event)
{
$action = $event->action;
if ($action instanceof Discussions\ShowAction) {
$event->addInclude('posts.likes');
}
if ($action instanceof Posts\IndexAction ||
$action instanceof Posts\ShowAction ||
$action instanceof Posts\CreateAction ||
$action instanceof Posts\UpdateAction) {
$event->addInclude('likes');
}
}
}

View File

@ -0,0 +1,40 @@
<?php namespace Flarum\Likes\Listeners;
use Flarum\Events\RegisterLocales;
use Flarum\Events\BuildClientView;
use Illuminate\Contracts\Events\Dispatcher;
class AddClientAssets
{
public function subscribe(Dispatcher $events)
{
$events->listen(RegisterLocales::class, __CLASS__.'@addLocale');
$events->listen(BuildClientView::class, __CLASS__.'@addAssets');
}
public function addLocale(RegisterLocales $event)
{
$event->addTranslations('en', __DIR__.'/../../locale/en.yml');
}
public function addAssets(BuildClientView $event)
{
$event->forumAssets([
__DIR__.'/../../js/forum/dist/extension.js',
__DIR__.'/../../less/forum/extension.less'
]);
$event->forumBootstrapper('likes/main');
$event->forumTranslations([
'likes.post_liked_notification',
'likes.post_likes_modal_title',
'likes.post_liked_by_self',
'likes.post_liked_by',
'likes.unlike_action',
'likes.like_action',
'likes.notify_post_liked',
'likes.others'
]);
}
}

View File

@ -0,0 +1,21 @@
<?php namespace Flarum\Likes\Listeners;
use Flarum\Events\ModelRelationship;
use Flarum\Core\Posts\Post;
use Illuminate\Contracts\Events\Dispatcher;
class AddModelRelationship
{
public function subscribe(Dispatcher $events)
{
$events->listen(ModelRelationship::class, __CLASS__.'@addRelationship');
}
public function addRelationship(ModelRelationship $event)
{
if ($event->model instanceof Post &&
$event->relationship === 'likes') {
return $event->model->belongsToMany('Flarum\Core\Users\User', 'posts_likes', 'post_id', 'user_id', 'likes');
}
}
}

View File

@ -0,0 +1,56 @@
<?php namespace Flarum\Likes\Listeners;
use Flarum\Likes\Notifications\PostLikedBlueprint;
use Flarum\Events\RegisterNotificationTypes;
use Flarum\Likes\Events\PostWasLiked;
use Flarum\Likes\Events\PostWasUnliked;
use Flarum\Core\Posts\Post;
use Flarum\Core\Users\User;
use Flarum\Core\Notifications\NotificationSyncer;
use Illuminate\Contracts\Events\Dispatcher;
class NotifyPostLiked
{
protected $notifications;
public function __construct(NotificationSyncer $notifications)
{
$this->notifications = $notifications;
}
public function subscribe(Dispatcher $events)
{
$events->listen(RegisterNotificationTypes::class, __CLASS__.'@registerNotificationType');
$events->listen(PostWasLiked::class, __CLASS__.'@whenPostWasLiked');
$events->listen(PostWasUnliked::class, __CLASS__.'@whenPostWasUnliked');
}
public function registerNotificationType(RegisterNotificationTypes $event)
{
$event->register(
'Flarum\Likes\Notifications\PostLikedBlueprint',
'Flarum\Api\Serializers\PostBasicSerializer',
['alert']
);
}
public function whenPostWasLiked(PostWasLiked $event)
{
$this->sync($event->post, $event->user, [$event->post->user]);
}
public function whenPostWasUnliked(PostWasUnliked $event)
{
$this->sync($event->post, $event->user, []);
}
public function sync(Post $post, User $user, array $recipients)
{
if ($post->user->id != $user->id) {
$this->notifications->sync(
new PostLikedBlueprint($post, $user),
$recipients
);
}
}
}

View File

@ -0,0 +1,48 @@
<?php namespace Flarum\Likes\Listeners;
use Flarum\Likes\Events\PostWasLiked;
use Flarum\Likes\Events\PostWasUnliked;
use Flarum\Events\PostWillBeSaved;
use Flarum\Events\PostWasDeleted;
use Flarum\Core\Posts\Post;
use Flarum\Core\Exceptions\PermissionDeniedException;
use Illuminate\Contracts\Events\Dispatcher;
class PersistData
{
public function subscribe(Dispatcher $events)
{
$events->listen(PostWillBeSaved::class, __CLASS__.'@whenPostWillBeSaved');
$events->listen(PostWasDeleted::class, __CLASS__.'@whenPostWasDeleted');
}
public function whenPostWillBeSaved(PostWillBeSaved $event)
{
$post = $event->post;
$data = $event->data;
if ($post->exists && isset($data['attributes']['isLiked'])) {
$actor = $event->actor;
$liked = (bool) $data['attributes']['isLiked'];
if (! $post->can($actor, 'like')) {
throw new PermissionDeniedException;
}
if ($liked) {
$post->likes()->attach($actor->id);
$post->raise(new PostWasLiked($post, $actor));
} else {
$post->likes()->detach($actor->id);
$post->raise(new PostWasUnliked($post, $actor));
}
}
}
public function whenPostWasDeleted(PostWasDeleted $event)
{
$event->post->likes()->detach();
}
}

View File

@ -1,10 +1,10 @@
<?php namespace Flarum\Likes;
<?php namespace Flarum\Likes\Notifications;
use Flarum\Core\Models\Post;
use Flarum\Core\Models\User;
use Flarum\Core\Notifications\NotificationAbstract;
use Flarum\Core\Posts\Post;
use Flarum\Core\Users\User;
use Flarum\Core\Notifications\Blueprint;
class PostLikedNotification extends NotificationAbstract
class PostLikedBlueprint implements Blueprint
{
public $post;
@ -26,6 +26,11 @@ class PostLikedNotification extends NotificationAbstract
return $this->user;
}
public function getData()
{
return null;
}
public static function getType()
{
return 'postLiked';
@ -33,6 +38,6 @@ class PostLikedNotification extends NotificationAbstract
public static function getSubjectModel()
{
return 'Flarum\Core\Models\Post';
return 'Flarum\Core\Posts\Post';
}
}