From 573303de39331c25068f441022b530f1b41f885b Mon Sep 17 00:00:00 2001 From: Enrico Fasoli Date: Mon, 23 Nov 2015 18:26:34 +0100 Subject: [PATCH] I guess comments are kinda implemented --- lib/boards-api.js | 115 +++++++++++++++++++++++++-------- webapp/components/comment.jsx | 39 +++++++++++ webapp/components/comments.jsx | 30 +++++++++ webapp/components/post.jsx | 3 +- webapp/components/postpage.jsx | 5 +- webapp/style.css | 11 +++- 6 files changed, 173 insertions(+), 30 deletions(-) create mode 100644 webapp/components/comment.jsx create mode 100644 webapp/components/comments.jsx diff --git a/lib/boards-api.js b/lib/boards-api.js index b1693a1..5c7637f 100644 --- a/lib/boards-api.js +++ b/lib/boards-api.js @@ -11,6 +11,7 @@ var EventEmitter = require('wolfy87-eventemitter') var asyncjs = require('async') function asObj(str,done){ + if(str.toString) str = str.toString() if(typeof str === 'string'){ var obj try { @@ -43,8 +44,9 @@ function replyAsObj(res,isJson,done){ done(null,data) } }) - } else { - console.log('got string:',res) + } else if(res.split || res.toString){ + console.log('got string or buffer:',res) + if(res.toString) res = res.toString() // Is a string if(isJson){ asObj(res,done) @@ -88,7 +90,7 @@ BoardsAPI.prototype.resolveIPNS = function(n,handler){ var cached = this.users[n] if(cached){ this.ee.emit(n,cached) - console.log(n,'was cached',cached) + //console.log(n,'was cached',cached) } else { console.log(n,'not cached') } @@ -178,7 +180,6 @@ BoardsAPI.prototype.searchUsers = function(){ } BoardsAPI.prototype.getProfile = function(userID,done){ - console.log('profile requested for',userID) this.resolveIPNS(userID,(url,err) => { if(err){ this.ee.emit('error',err) @@ -282,41 +283,55 @@ BoardsAPI.prototype.downloadPost = function(hash,adminID,board,op,done){ return this.ee } -BoardsAPI.prototype.getPostsInBoard = function(adminID,board){ - this.getBoardSettings(adminID,board) +BoardsAPI.prototype.retrieveListOfApproved = function(what,addr,adminID,board){ + var a = addr+this.baseurl+'boards/'+board+'/approved/'+what+'/' + this.ipfs.ls(a,(err,res) => { + if(err){ + this.ee.emit('error',err) + } else { + // Send approved posts list + var ret = res.Objects[0].Links.map(item => { + return { date: item.Name, hash: item.Hash } + }) + this.emit('approved '+what+' for '+board+'@'+adminID,ret) + } + }) +} + +BoardsAPI.prototype.getAllowedContentProducers = function(adminID,board,options){ + if(!options) return this.ee.on('settings for'+board+'@'+adminID,function(settings,addr){ - // Download posts based on settings + // Get stuff based on settings if(settings.approval_required == true){ // Get approved posts list - var a = addr+this.baseurl+'boards/'+board+'/approved/posts/' - this.ipfs.ls(a,(err,res) => { - if(err){ - this.ee.emit('error',err) - } else { - // Send approved posts list - var ret = res.Objects[0].Links.map(item => { - return { date: item.Name, hash: item.Hash } - }) - this.emit('approved posts for '+board+'@'+adminID,ret) - // Automatically download approved posts - ret.forEach(item => this.downloadPost(item.hash,adminID,board)) - } - }) + if(options.posts) this.retrieveListOfApproved('posts',addr,adminID,board) + // Get approved comments list + if(options.comments) this.retrieveListOfApproved('comments',addr,adminID,board) if(settings.whitelist == true){ - // TODO: Download all posts from whitelisted users + // TODO: emit all whitelisted users } } else if(settings.whitelist_only == true){ - // TODO: download all posts from whitelisted users + // TODO: emit all whitelisted users } else if(settings.blacklist == true){ - // TODO: get the blacklist, then start downloading posts from everyone not in the blacklist + // TODO: emit all users not in the blacklist } }) + this.getBoardSettings(adminID,board) + return this.ee +} + +BoardsAPI.prototype.getPostsInBoard = function(adminID,board){ + this.ee.on('approved posts for '+board+'@'+adminID,ret => { + // Automatically download approved posts + ret.forEach(item => this.downloadPost(item.hash,adminID,board)) + }) // Get the admin's posts this.getUserPostListInBoard(adminID,board,(err,res) => { if(err){ console.log(err) } else res.forEach(item => this.downloadPost(item.hash,adminID,board,adminID)) }) + this.getAllowedContentProducers(adminID,board,{ posts: true }) return this.ee } @@ -343,9 +358,57 @@ BoardsAPI.prototype.getUserPostListInBoard = function(user,board,done){ return this.ee } +BoardsAPI.prototype.downloadComment = function(hash,adminID,board,done){ + this.ipfs.cat(hash,(err2,r) => { + if(err2){ + this.ee.emit('error',err2) + console.log('Could not download comment',hash,'of',board+'@'+adminID) + if(done && done.apply) done(err2) + } else { + // TODO: add JSON parsing error handling + var cmnt = JSON.parse(r.toString()) + cmnt.hash = hash + this.ee.emit(hash,cmnt,adminID,board) + this.ee.emit('comment for '+cmnt.parent,cmnt) + if(done && done.apply) done(null,cmnt) + } + }) + return this.ee +} + BoardsAPI.prototype.getCommentsFor = function(parent,board,adminID){ - // Create an EventEmitter, start looking and emit an event for every new comment - // Consider the rules of @adminID#board + this.ee.on('approved comments for '+board+'@'+adminID,ret => { + ret.forEach(item => this.downloadComment(item.hash,adminID,board)) + }) + // get the admin's comments + this.getUserCommentList(parent,adminID,(err,res) => { + if(err){ + console.log(err) + } else res.forEach(item => this.downloadComment(item.hash,adminID,board)) + }) + this.getAllowedContentProducers(adminID,board,{ comments: true }) +} + +BoardsAPI.prototype.getUserCommentList = function(parent,user,done){ + this.resolveIPNS(user,(url,err) => { + if(err){ + this.ee.emit('error',err) + done(err) + } else this.ipfs.ls(url+this.baseurl+'comments/'+parent,(e,r) => { + if(e){ + this.ee.emit('error',e) + done(e) + } else if(r && !r.split){ + console.log('Found',r.Objects[0].Links.length,'comments for',parent,'at',user) + var l = r.Objects[0].Links.map(i => { + return { date: i.Name, hash: i.Hash } + }) + done(null,l) + } + }) + return true // remove myself from listeners + }) + return this.ee } // API for publishing content and managing to be done later... diff --git a/webapp/components/comment.jsx b/webapp/components/comment.jsx new file mode 100644 index 0000000..bc6767a --- /dev/null +++ b/webapp/components/comment.jsx @@ -0,0 +1,39 @@ +var React = require('react') +var Markdown = require('markdown.jsx') +var Icon = require('icon.jsx') + +module.exports = function(boardsAPI){ + var UserID = require('userID.jsx')(boardsAPI) + return React.createClass({ + getInitialState: function(){ + return { moment: false } + }, + componentDidMount: function(){ + require.ensure(['moment'],_ => { + if(this.isMounted()) this.setState({ moment: require('moment') }) + }) + }, + getDate: function(){ + if(this.props.comment.date){ + if(this.state.moment) + return this.state.moment.unix(this.props.comment.date).fromNow() + else return '...' + } else { + return 'Unknown Date' + } + }, + render: function(){ + if(this.props.comment){ + return

+
+ + {this.getDate()} +
+ +
+ } else { + return

Invalid Comment
+ } + } + }) +} diff --git a/webapp/components/comments.jsx b/webapp/components/comments.jsx new file mode 100644 index 0000000..b8e8635 --- /dev/null +++ b/webapp/components/comments.jsx @@ -0,0 +1,30 @@ +var React = require('react') + +module.exports = function(boardsAPI){ + var Comment = require('comment.jsx')(boardsAPI) + return React.createClass({ + getInitialState: function(){ + return { comments: [] } + }, + componentDidMount: function(){ + boardsAPI.use(boards => { + boards.getEventEmitter().on('comment for '+this.props.parent,cmnt => { + if(this.isMounted()) this.setState({ comments: this.state.comments.concat(cmnt) }) + }) + if(boards.isInit && this.isMounted()){ + boards.getCommentsFor(this.props.parent,this.props.board,this.props.adminID) + } + boards.getEventEmitter().on('init', err => { + if(!err && this.isMounted()) + boards.getCommentsFor(this.props.parent,this.props.board,this.props.adminID) + }) + }) + }, + getComments: function(){ + return this.state.comments.map(cmnt => () ) + }, + render: function(){ + return
{this.getComments()}
+ } + }) +} diff --git a/webapp/components/post.jsx b/webapp/components/post.jsx index 60d8881..4d147d3 100644 --- a/webapp/components/post.jsx +++ b/webapp/components/post.jsx @@ -24,7 +24,6 @@ module.exports = function(boardsAPI){ }) }, postLink: function(){ - console.log('op',this.props.post.op,'board',this.props.board) if(this.props.post.op){ if(this.props.board){ return '/@'+this.props.post.op+'/'+this.props.board+'/'+this.props.post.hash @@ -39,7 +38,7 @@ module.exports = function(boardsAPI){ return
{this.props.post.title}

-
+
{this.getDate()} diff --git a/webapp/components/postpage.jsx b/webapp/components/postpage.jsx index b2347d0..eea44ab 100644 --- a/webapp/components/postpage.jsx +++ b/webapp/components/postpage.jsx @@ -5,6 +5,8 @@ module.exports = function(boardsAPI){ var UserID = require('userID.jsx')(boardsAPI) var GetIPFS = require('getipfs.jsx')(boardsAPI) var Post = require('post.jsx')(boardsAPI) + var Comments = require('comments.jsx')(boardsAPI) + return React.createClass({ getInitialState: function(){ return { post: { title: '...', text: '...' }, api: false } @@ -36,7 +38,7 @@ module.exports = function(boardsAPI){ getContext(){ if(this.props.params.userid){ if(this.props.params.boardname) - return
Posted by in #{this.props.params.boardname}
+ return
Posted by in #{this.props.params.boardname}
else return
Posted by
} else return
You are viewing a single post
@@ -48,6 +50,7 @@ module.exports = function(boardsAPI){ {this.getContext()}
+
else return } diff --git a/webapp/style.css b/webapp/style.css index 488a09d..653c24d 100644 --- a/webapp/style.css +++ b/webapp/style.css @@ -75,7 +75,7 @@ a:hover { margin: 2rem; } -.post .icons .fa { +.post .icons .fa, .comment .icons .fa { color: #707070; } @@ -95,6 +95,15 @@ a:hover { display: inline } +.comment .icons { + margin-bottom: 1rem; +} + +.comment .icons .user-id { + margin-right: 1rem; +} + + .navbar { border-bottom: 1px solid #eee; display: block;