React.js 小书 Lesson27 - 实战分析:评论功能(六)

前端之家收集整理的这篇文章主要介绍了React.js 小书 Lesson27 - 实战分析:评论功能(六)前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。

React.js 小书 Lesson27 - 实战分析:评论功能(六)

本文作者:胡子大哈
本文原文:http://huziketang.com/books/react/lesson27

转载请注明出处,保留原文链接以及作者信息

在线阅读:http://huziketang.com/books/react


{%raw%}

删除评论

现在发布评论评论不会消失,评论越来越多并不是什么好事。所以我们给评论组件加上删除评论功能,这样就可以删除不想要的评论了。修改 src/Comment.jsrender 方法,新增一个删除按钮:

  1. ...
  2. render () {
  3. const { comment } = this.props
  4. return (
  5. <div className='comment'>
  6. <div className='comment-user'>
  7. <span className='comment-username'>
  8. {comment.username}
  9. </span>:
  10. </div>
  11. <p>{comment.content}</p>
  12. <span className='comment-createdtime'>
  13. {this.state.timeString}
  14. </span>
  15. <span className='comment-delete'>
  16. 删除
  17. </span>
  18. </div>
  19. )
  20. }
  21. ...

我们在后面加了一个删除按钮,因为 index.css 定义了样式,所以鼠标放到特定的评论上才会显示删除按钮,让用户体验好一些。

我们知道评论列表数据是放在 CommentApp 当中的,而这个删除按钮是在 Comment 当中的,现在我们要做的事情是用户点击某条评论删除按钮,然后在 CommentApp 中把相应的数据删除。但是 CommentAppComment 的关系是这样的:

CommentCommentApp 之间隔了一个 CommentListComment 无法直接跟 CommentApp 打交道,只能通过 CommentList 来转发这种删除评论的消息。修改 Comment 组件,让它可以把删除的消息传递到上一层:

  1. class Comment extends Component {
  2. static propTypes = {
  3. comment: PropTypes.object.isrequired,onDeleteComment: PropTypes.func,index: PropTypes.number
  4. }
  5. ...
  6. handleDeleteComment () {
  7. if (this.props.onDeleteComment) {
  8. this.props.onDeleteComment(this.props.index)
  9. }
  10. }
  11.  
  12. render () {
  13. ...
  14. <span
  15. onClick={this.handleDeleteComment.bind(this)}
  16. className='comment-delete'>
  17. 删除
  18. </span>
  19. </div>
  20. )
  21. }

现在在使用 Comment 的时候,可以传入 onDeleteCommentindex 两个参数。index 用来标志这个评论在列表的下标,这样点击删除按钮的时候我们才能知道你点击的是哪个评论,才能知道怎么从列表数据中删除用户点击删除调用 handleDeleteComment ,它会调用从上层传入的 props. onDeleteComment 函数告知上一层组件删除的消息,并且把评论下标传出去。现在修改 src/CommentList.js 让它把这两个参数传进来:

  1. class CommentList extends Component {
  2. static propTypes = {
  3. comments: PropTypes.array,onDeleteComment: PropTypes.func
  4. }
  5.  
  6. static defaultProps = {
  7. comments: []
  8. }
  9.  
  10. handleDeleteComment (index) {
  11. if (this.props.onDeleteComment) {
  12. this.props.onDeleteComment(index)
  13. }
  14. }
  15.  
  16. render() {
  17. return (
  18. <div>
  19. {this.props.comments.map((comment,i) =>
  20. <Comment
  21. comment={comment}
  22. key={i}
  23. index={i}
  24. onDeleteComment={this.handleDeleteComment.bind(this)} />
  25. )}
  26. </div>
  27. )
  28. }
  29. }

用户点击按钮的时候,Comment 组件会调用 props.onDeleteComment,也就是 CommentListhandleDeleteComment 方法。而 handleDeleteComment调用 CommentList 所接受的配置参数中的 props.onDeleteComment,并且把下标传出去。

也就是说,我们可以在 CommentAppCommentList 传入一个 onDeleteComment 的配置参数来接受这个删除评论的消息,修改 CommentApp.js

  1. ...
  2. handleDeleteComment (index) {
  3. console.log(index)
  4. }
  5.  
  6. render() {
  7. return (
  8. <div className='wrapper'>
  9. <CommentInput onSubmit={this.handleSubmitComment.bind(this)} />
  10. <CommentList
  11. comments={this.state.comments}
  12. onDeleteComment={this.handleDeleteComment.bind(this)} />
  13. </div>
  14. )
  15. }
  16. }
  17. ...

现在点击删除按钮,可以在控制台看到评论对应的下标打印了出来。其实这就是这么一个过程:CommentList 把下标 index 传给 Comment。点击删除按钮的时候,Commentindex 传给了 CommentListCommentList 再把它传给 CommentApp。现在可以在 CommentApp删除评论了:

  1. ...
  2. handleDeleteComment (index) {
  3. const comments = this.state.comments
  4. comments.splice(index,1)
  5. this.setState({ comments })
  6. this._saveComments(comments)
  7. }
  8. ...

我们通过 comments.splice 删除特定下标的评论,并且通过 setState 重新渲染整个评论列表;当然了,还需要把最新的评论列表数据更新到 LocalStorage 中,所以我们在删除、更新以后调用_saveComments 方法把数据同步到 LocalStorage 中。

现在就可以愉快地删除评论了。但是,你删除评论以后 5 秒钟后就会在控制台中看到报错了:

这是因为我们忘了清除评论的定时器,修改 src/Comment.js,新增生命周期 commentWillUnmount评论组件销毁的时候清除定时器:

  1. ...
  2. componentWillUnmount () {
  3. clearInterval(this._timer)
  4. }
  5. ...

这才算完成了第 5 个需求。

显示代码

用户在的输入内容中任何以 `` 包含的内容都会用 <code> 包含起来显示页面上。<code> 这是一个 HTML 结构,需要往页面动态插入 HTML 结构我们只能用 dangerouslySetInnerHTML 了,修改 src/Comment.js,把原来 render() 函数中的:

  1. <p>{comment.content}</p>

修改成:

  1. <p dangerouslySetInnerHTML={{
  2. __html: this._getProcessedContent(comment.content)
  3. }} />

我们把经过 this._getProcessedContent 处理的评论内容以 HTML 的方式插入到 <p> 元素中,this._getProcessedContent 是这样实现的:

  1. ...
  2. _getProcessedContent (content) {
  3. return content
  4. .replace(/&/g,"&amp;")
  5. .replace(/</g,"&lt;")
  6. .replace(/>/g,"&gt;")
  7. .replace(/"/g,"&quot;")
  8. .replace(/'/g,"&#039;")
  9. .replace(/`([\S\s]+?)`/g,'<code>$1</code>')
  10. }
  11. ...

看起来很复杂,其实前 5 行是用来处理 HTML 内容转义的,最后一行是用来插入 <code> 标签的。如果我们把用户输入的内容全部以 HTML 显示页面上,那么就会造成跨站脚本攻击。所以前 5 个 replace 实际上是把类似于 <> 这种内容替换转义一下。而最后一行才是真正实现需求的代码,把 `` 包含的内容<code> 包裹起来。

输入:

这是代码块 `console.log`,这是 <h1>正常内容</h1>。

看看效果

我们安全地完成了第 6 个需求。到目前为止,第二阶段的实战已经全部完成,你可以在这里找到完整的代码

总结

到这里第二阶段已经全部结束,我们已经掌握了全部 React.js 实战需要的入门知识。接下来我们会学习两个相对比较高级的 React.js 的概念,然后进入 React-router 和 Redux 的世界,让它们配合 React.js 来构建更成熟的前端页面
{%endraw%}

下一节中我们将介绍《React.js 小书 Lesson28 - 高阶组件(Higher-Order Components)》

猜你在找的React相关文章