mirror of https://gitlab.com/ecentrics/concordia
Apostolos Fanakis
7 years ago
12 changed files with 525 additions and 544 deletions
@ -1,110 +1,35 @@ |
|||
import { drizzleConnect } from 'drizzle-react'; |
|||
import React, { Component } from 'react'; |
|||
import PropTypes from 'prop-types'; |
|||
import React from 'react'; |
|||
|
|||
import Post from './Post'; |
|||
|
|||
import epochTimeConverter from '../helpers/EpochTimeConverter' |
|||
|
|||
const contract = "Forum"; |
|||
const contractMethod = "getPost"; |
|||
|
|||
class PostList extends Component { |
|||
constructor(props, context) { |
|||
super(props); |
|||
|
|||
this.fetchPost = this.fetchPost.bind(this); |
|||
|
|||
this.drizzle = context.drizzle; |
|||
this.dataKeys = []; |
|||
this.postsData = new Array(parseInt(this.props.postIDs.length, 10)).fill(undefined); |
|||
this.orbitPostsData = new Array(parseInt(this.props.postIDs.length, 10)).fill(undefined); |
|||
this.orbitPostsDataFetchStatus = new Array(parseInt(this.props.postIDs.length, 10)).fill("pending"); |
|||
|
|||
for (var i = 0; i < this.props.postIDs.length; ++i){ |
|||
this.dataKeys[i] = this.drizzle.contracts[contract].methods[contractMethod] |
|||
.cacheCall(this.props.postIDs[i]); |
|||
} |
|||
} |
|||
import WithBlockchainData from './WithBlockchainData'; |
|||
|
|||
async fetchPost(index) { |
|||
/*const fullAddress = this.postsData[postID][1]; |
|||
const store = await this.props.orbitDB.orbitdb.keyvalue(JSON.stringify(fullAddress)); |
|||
await store.load(); |
|||
var som = store.get(JSON.stringify(postID)); |
|||
this.orbitPostsData[postID] = som['subject']; |
|||
this.orbitPostsDataFetchStatus[postID] = "fetched";*/ |
|||
|
|||
var som = this.props.orbitDB.postsDB.get(this.props.postIDs[index]); |
|||
this.orbitPostsData[index] = som; |
|||
this.orbitPostsDataFetchStatus[index] = "fetched"; |
|||
} |
|||
import Post from './Post'; |
|||
|
|||
render (){ |
|||
const posts = this.postsData.map((post, index) => { |
|||
if (post) { |
|||
return ( |
|||
<div onClick={() => { |
|||
this.context.router.push("/topic/" + post[4] + "/" + |
|||
((this.orbitPostsData[index] !== undefined) |
|||
? this.orbitPostsData[index].subject + "/" + this.props.postIDs[index] |
|||
: ""))}} |
|||
key={index}> |
|||
<Post post={{ |
|||
avatarUrl: post.avatarUrl, |
|||
userAddress: post[1], |
|||
username: post[2], |
|||
subject: (this.orbitPostsData[index] !== undefined) && this.orbitPostsData[index].subject, |
|||
date: epochTimeConverter(post[3]), |
|||
postIndex: index, |
|||
postContent: (this.orbitPostsData[index] !== undefined) && this.orbitPostsData[index].content |
|||
}} |
|||
id={index} |
|||
key={index}/> |
|||
</div> |
|||
); |
|||
} else { |
|||
const PostList = (props) => { |
|||
const posts = props.postIDs.map((postID, index) => { |
|||
return ( |
|||
<Post post={null} |
|||
id={index} |
|||
key={index}/> |
|||
<WithBlockchainData |
|||
component={Post} |
|||
callsInfo={[{ |
|||
contract: 'Forum', |
|||
method: 'getPost', |
|||
params: [postID] |
|||
}]} |
|||
avatarUrl={""} |
|||
postIndex={index} |
|||
postID={postID} |
|||
key={postID} |
|||
/> |
|||
); |
|||
} |
|||
}); |
|||
|
|||
return ( |
|||
<div className="posts-list"> |
|||
{posts} |
|||
{props.recentToTheTop |
|||
?posts.slice(0).reverse() |
|||
:posts |
|||
} |
|||
</div> |
|||
); |
|||
} |
|||
|
|||
componentWillReceiveProps() { |
|||
for (var i = 0; i < this.props.postIDs.length; ++i){ |
|||
if (this.postsData[i] === undefined) { |
|||
let currentDrizzleState = this.drizzle.store.getState(); |
|||
let dataFetched = (currentDrizzleState.contracts[contract][contractMethod])[this.dataKeys[i]]; |
|||
if (dataFetched){ |
|||
this.postsData[i] = dataFetched.value; |
|||
} |
|||
} else if (!this.orbitPostsData[i] && this.orbitPostsDataFetchStatus[i] === "pending") { |
|||
this.orbitPostsDataFetchStatus[i] = "fetching"; |
|||
this.fetchPost(i); |
|||
} |
|||
} |
|||
} |
|||
}; |
|||
|
|||
PostList.contextTypes = { |
|||
drizzle: PropTypes.object, |
|||
router: PropTypes.object |
|||
}; |
|||
|
|||
const mapStateToProps = state => { |
|||
return { |
|||
user: state.user, //Needed!!
|
|||
orbitDB: state.orbitDB |
|||
} |
|||
}; |
|||
|
|||
export default drizzleConnect(PostList, mapStateToProps); |
|||
export default PostList; |
@ -1,30 +1,91 @@ |
|||
import React from 'react'; |
|||
import React, { Component } from 'react'; |
|||
import { Link } from 'react-router'; |
|||
import { drizzleConnect } from 'drizzle-react'; |
|||
|
|||
import TimeAgo from 'react-timeago'; |
|||
import epochTimeConverter from '../helpers/EpochTimeConverter' |
|||
|
|||
class Topic extends Component { |
|||
constructor(props){ |
|||
super(props); |
|||
|
|||
this.fetchSubject = this.fetchSubject.bind(this); |
|||
|
|||
this.topicSubject = null; |
|||
this.topicSubjectFetchStatus = "pending"; |
|||
} |
|||
|
|||
async fetchSubject(topicID) { |
|||
this.topicSubjectFetchStatus = "fetching"; |
|||
|
|||
const Topic = (props) => { |
|||
if (this.props.blockchainData[0].returnData[1] === this.props.user.address) { |
|||
let som = this.props.orbitDB.topicsDB.get(topicID); |
|||
this.topicSubject = som['subject']; |
|||
this.topicSubjectFetchStatus = "fetched"; |
|||
} else { |
|||
const fullAddress = "/orbitdb" + this.props.blockchainData[0].returnData[0] + "/topics"; |
|||
const store = await this.props.orbitDB.orbitdb.keyvalue(fullAddress); |
|||
|
|||
/*store.events.on('replicated', () => { |
|||
const result = store.iterator({ limit: -1 }).collect().map(e => e.payload.value) |
|||
console.log(result.join('\n')) |
|||
})*/ |
|||
|
|||
await store.load(); |
|||
let som = store.get(topicID); |
|||
this.topicSubject = som['subject']; |
|||
this.topicSubjectFetchStatus = "fetched"; |
|||
} |
|||
} |
|||
|
|||
render(){ |
|||
return ( |
|||
props.topic !== null |
|||
? <div className={"pure-u-1-1 topic card"}> |
|||
<p className="topic-subject" style={{color: props.topic.topicSubject ? "" : "grey"}}> |
|||
<strong>{props.topic.topicSubject ? props.topic.topicSubject : "Subject"}</strong> |
|||
<Link to={"/topic/" + this.props.topicID}> |
|||
<div className={"pure-u-1-1 topic card"}> |
|||
<p className="topic-subject" style={{color: this.topicSubject ? "" : "grey"}}> |
|||
<strong>{this.topicSubject !== null ? this.topicSubject : "Subject"}</strong> |
|||
</p> |
|||
<hr/> |
|||
<div className="topic-meta"> |
|||
<p className="no-margin">{props.topic.topicStarter}</p> |
|||
<p className="no-margin">Number of replies: {props.topic.numberOfReplies}</p> |
|||
<p className="topic-date">Started <TimeAgo date={props.topic.date}/></p> |
|||
</div> |
|||
</div> |
|||
: <div className={"pure-u-1-1 topic card"} style={{color: 'grey'}}> |
|||
<p className="topic-subject"><strong>Subject</strong></p> |
|||
<hr/> |
|||
<div className="topic-meta"> |
|||
<p className="no-margin">Username</p> |
|||
<p className="no-margin">Number of replies: </p> |
|||
<p className="topic-date">Started </p> |
|||
<p className="no-margin" style={{ |
|||
color: this.props.blockchainData[0].returnData !== null ? "" : "grey" |
|||
}}> |
|||
{this.props.blockchainData[0].returnData !== null |
|||
?this.props.blockchainData[0].returnData[2] |
|||
:"Username" |
|||
} |
|||
</p> |
|||
<p className="no-margin" style={{ |
|||
color: this.props.blockchainData[0].returnData !== null ? "" : "grey" |
|||
}}> |
|||
{"Number of replies: " + (this.props.blockchainData[0].returnData !== null |
|||
?this.props.blockchainData[0].returnData[4].length |
|||
:"") |
|||
} |
|||
</p> |
|||
<p className="topic-date"> |
|||
Started {this.props.blockchainData[0].returnData !== null && |
|||
<TimeAgo date={epochTimeConverter(this.props.blockchainData[0].returnData[3])}/> |
|||
} |
|||
</p> |
|||
</div> |
|||
</div> |
|||
</Link> |
|||
); |
|||
} |
|||
|
|||
componentDidUpdate(){ |
|||
if (this.props.blockchainData[0].returnData !== null && this.topicSubjectFetchStatus === "pending") { |
|||
this.fetchSubject(this.props.topicID); |
|||
} |
|||
} |
|||
}; |
|||
|
|||
export default Topic; |
|||
const mapStateToProps = state => { |
|||
return { |
|||
user: state.user, |
|||
orbitDB: state.orbitDB |
|||
} |
|||
} |
|||
|
|||
export default drizzleConnect(Topic, mapStateToProps); |
@ -0,0 +1,81 @@ |
|||
import React, { Component } from 'react'; |
|||
import PropTypes from 'prop-types'; |
|||
|
|||
class WithBlockchainData extends Component { |
|||
constructor(props, context) { |
|||
super(props); |
|||
|
|||
{ |
|||
let {component, callsInfo, ...rest } = this.props; |
|||
this.component = component; |
|||
this.callsInfo = callsInfo; |
|||
this.forwardedProps = rest; |
|||
} |
|||
|
|||
this.checkContractUpdates = this.checkContractUpdates.bind(this); |
|||
|
|||
this.drizzle = context.drizzle; |
|||
this.dataKeys = []; |
|||
this.blockchainData = this.callsInfo.map((call) => { |
|||
return ({ |
|||
callInfo: call, |
|||
status: "initialized", |
|||
returnData: null |
|||
}); |
|||
}); |
|||
|
|||
for (var i = 0; i < this.callsInfo.length; ++i){ |
|||
this.dataKeys[i] = this.drizzle |
|||
.contracts[this.callsInfo[i].contract] |
|||
.methods[this.callsInfo[i].method] |
|||
.cacheCall(...(this.callsInfo[i].params)); |
|||
this.blockchainData[i].status = "pending"; |
|||
} |
|||
|
|||
this.state = { |
|||
transactionsState: new Array(this.callsInfo.length).fill("pending") |
|||
} |
|||
} |
|||
|
|||
render() { |
|||
let {component, callsInfo, ...rest } = this.props; |
|||
return ( |
|||
<this.component blockchainData={this.blockchainData} {...rest}/> |
|||
); |
|||
} |
|||
|
|||
componentDidMount() { |
|||
this.intervalChecker = setInterval(this.checkContractUpdates, 10); //HOWMUCHMUCHACHO???
|
|||
} |
|||
|
|||
componentWillUnmount() { |
|||
clearInterval(this.intervalChecker); |
|||
} |
|||
|
|||
checkContractUpdates() { |
|||
for (var i = 0; i < this.callsInfo.length; ++i){ |
|||
let currentDrizzleState = this.drizzle.store.getState(); |
|||
if (this.state.transactionsState[i] === "pending") { |
|||
let dataFetched = (currentDrizzleState |
|||
.contracts[this.callsInfo[i].contract][this.callsInfo[i].method][this.dataKeys[i]]); |
|||
if (dataFetched){ |
|||
this.blockchainData[i].returnData = dataFetched.value; |
|||
this.blockchainData[i].status = "success"; |
|||
this.setState((prevState) => ({ |
|||
transactionsState: [ |
|||
...prevState.transactionsState.slice(0, i), |
|||
"success", |
|||
...prevState.transactionsState.slice(i) |
|||
] |
|||
})); |
|||
} |
|||
} //TODO cover errors!!
|
|||
} |
|||
} |
|||
} |
|||
|
|||
WithBlockchainData.contextTypes = { |
|||
drizzle: PropTypes.object |
|||
}; |
|||
|
|||
export default WithBlockchainData; |
Loading…
Reference in new issue