mirror of
https://github.com/flarum/framework.git
synced 2025-02-22 00:31:11 +08:00
Update for new extension API; implement l10n
This commit is contained in:
parent
fb9ed378e0
commit
9c384bee98
32
extensions/likes/.editorconfig
Normal file
32
extensions/likes/.editorconfig
Normal 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
|
5
extensions/likes/.eslintignore
Normal file
5
extensions/likes/.eslintignore
Normal file
@ -0,0 +1,5 @@
|
||||
**/bower_components/**/*
|
||||
**/node_modules/**/*
|
||||
vendor/**/*
|
||||
**/Gulpfile.js
|
||||
**/dist/**/*
|
171
extensions/likes/.eslintrc
Normal file
171
extensions/likes/.eslintrc
Normal 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
|
||||
}
|
||||
}
|
@ -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';
|
||||
|
@ -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"
|
||||
}
|
||||
}
|
||||
}
|
||||
|
109
extensions/likes/js/bootstrap.js
vendored
109
extensions/likes/js/bootstrap.js
vendored
@ -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']
|
||||
});
|
||||
});
|
||||
|
||||
});
|
@ -1,5 +1,5 @@
|
||||
var gulp = require('flarum-gulp');
|
||||
|
||||
gulp({
|
||||
modulePrefix: 'flarum-likes'
|
||||
modulePrefix: 'likes'
|
||||
});
|
41
extensions/likes/js/forum/src/addLikeAction.js
Normal file
41
extensions/likes/js/forum/src/addLikeAction.js
Normal 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()});
|
||||
}
|
||||
}
|
||||
})
|
||||
);
|
||||
});
|
||||
}
|
60
extensions/likes/js/forum/src/addLikesList.js
Normal file
60
extensions/likes/js/forum/src/addLikesList.js
Normal 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>
|
||||
));
|
||||
}
|
||||
});
|
||||
}
|
@ -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()
|
||||
});
|
||||
}
|
||||
}
|
30
extensions/likes/js/forum/src/components/PostLikesModal.js
Normal file
30
extensions/likes/js/forum/src/components/PostLikesModal.js
Normal 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>
|
||||
);
|
||||
}
|
||||
}
|
25
extensions/likes/js/forum/src/main.js
Normal file
25
extensions/likes/js/forum/src/main.js
Normal 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')
|
||||
});
|
||||
});
|
@ -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()]
|
||||
});
|
||||
}
|
||||
}
|
@ -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)
|
||||
]))
|
||||
)
|
||||
])
|
||||
]
|
||||
});
|
||||
}
|
||||
}
|
23
extensions/likes/less/forum/extension.less
Normal file
23
extensions/likes/less/forum/extension.less
Normal 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;
|
||||
}
|
||||
}
|
@ -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"
|
||||
|
@ -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)
|
||||
{
|
||||
|
@ -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)
|
||||
{
|
||||
|
16
extensions/likes/src/Extension.php
Normal file
16
extensions/likes/src/Extension.php
Normal 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');
|
||||
}
|
||||
}
|
@ -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();
|
||||
}
|
||||
}
|
@ -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
|
||||
);
|
||||
}
|
||||
}
|
@ -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()
|
||||
{
|
||||
//
|
||||
}
|
||||
}
|
50
extensions/likes/src/Listeners/AddApiAttributes.php
Executable file
50
extensions/likes/src/Listeners/AddApiAttributes.php
Executable 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');
|
||||
}
|
||||
}
|
||||
}
|
40
extensions/likes/src/Listeners/AddClientAssets.php
Executable file
40
extensions/likes/src/Listeners/AddClientAssets.php
Executable 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'
|
||||
]);
|
||||
}
|
||||
}
|
21
extensions/likes/src/Listeners/AddModelRelationship.php
Executable file
21
extensions/likes/src/Listeners/AddModelRelationship.php
Executable 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');
|
||||
}
|
||||
}
|
||||
}
|
56
extensions/likes/src/Listeners/NotifyPostLiked.php
Executable file
56
extensions/likes/src/Listeners/NotifyPostLiked.php
Executable 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
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
48
extensions/likes/src/Listeners/PersistData.php
Executable file
48
extensions/likes/src/Listeners/PersistData.php
Executable 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();
|
||||
}
|
||||
}
|
@ -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';
|
||||
}
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user