如何在Reactjs中正确实现对div消息的scrollHeight或scrollTop?

3

我正在尝试在Reactjs中实现聊天消息。聊天消息被正确地追加,并且功能良好。

我的问题是:当聊天消息到达底部时,它不会向上滚动。

我想要实现的是当消息到达底部时,自动将聊天div向上滚动。

scrollToBottom() {
  chat_messages.scrollTop = chat_messages.scrollHeight;
}

有没有解决办法?

这是代码:

<!DOCTYPE html>
<html>
   <head>

<style>

.chat_container{

background:blue;
height:40%;
max-height:40%;

width: 60%;
max-width: 60%;
 overflow-y: auto;

}


.chat_message{

background:green;
color:white;
padding:10px;
border:none;

}

.myFooter{
bottom:0px;
position:fixed;
background:red;
height:10%;
max-height:10%;

width: 60%;
max-width: 60%;


}
</style>
   </head>
   <body>
<script src="build/react.min.js"></script>
<script src="build/react-dom.min.js"></script>
<script src="build/browser.min.js"></script>
<script src="build/jquery.min.js"></script>

<div id="app"></div>

<script type="text/babel">


class Application extends React.Component {

//function Application() {
  constructor(props) {
    super(props);

const sender = 'John';

    this.state = {
message: '',
value: '',
sname: sender,
date: new Date(),
messages: []
};

this.handleChange = this.handleChange.bind(this);
    this.handleSubmit = this.handleSubmit.bind(this);

  }

handleChange(event) {
    this.setState({message: event.target.value});
  }

  handleSubmit(event) {
   // alert('A name was submitted: ' + this.state.message);
    event.preventDefault();

const data = {'message': this.state.message}
 this.setState({
            messages : this.state.messages.concat([data])
        });
//$(".chat_container").animate({ scrollTop: 20000000 }, "slow");
 this.scrollToBottom();

  }



scrollToBottom() {
  chat_messages.scrollTop = chat_messages.scrollHeight;
}

componentDidMount() {
// fetch content axio or ajax
  }


  render() {

    return (
      <div>
        <div className="chat_container">


  <h2>Welcome to Chat: Time is: {this.state.date.toLocaleTimeString()}</h2>

      

 {this.state.messages.map((m, e) => (


              <div key={e}>


               
<div className="chat_message">chat: {m.message} ---{this.state.date.toLocaleTimeString()}</div> 

              </div>
            ))}

       
        </div>

<div className="myFooter">
 <form onSubmit={this.handleSubmit}>
        <label>
          Name:
          <input type="text" value={this.state.message} onChange={this.handleChange} />
        </label>
        <input type="submit" value="Submit" />
      </form>
</div>


      </div>
    );
  }
}


ReactDOM.render(<Application />, document.getElementById('app'));
</script>

<script src="https://cdnjs.cloudflare.com/ajax/libs/react/15.1.0/react.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/react/15.1.0/react-dom.min.js"></script>


   </body>
</html>
1个回答

3

我已经成功运行了你的示例:

bottomRef.current.scrollIntoView({ behavior: "smooth" });

以下是完整的代码;我已将其转换为可用的React组件,并使用了 useRef 钩子(每当更新消息状态时都会滚动):

import React, { useState, useRef, useEffect } from "react";
import "./styles.css";

export default function App(props) {
  const [message, setMessage] = useState("");
  const [messages, setMessages] = useState([]);

  const currentDate = new Date();
  const bottomRef = useRef();

  useEffect(() => {
    // fetch content axio or ajax
  }, []);

  useEffect(() => {
    bottomRef.current.scrollIntoView({ behavior: "smooth" });
  }, [messages]);

  function handleChange(event) {
    setMessage(event.target.value);
  }

  function handleSubmit(event) {
    // alert('A name was submitted: ' + this.state.message);
    event.preventDefault();
    const newMessages = messages;
    const data = { message: message, date: new Date() };
    setMessages(newMessages.concat([data]));
  }

  return (
    <React.Fragment>
      <h2>Welcome to Chat: Time is: {currentDate.toLocaleTimeString()}</h2>

      <div className="chat_container">
        {messages.map((m, e) => (
          <div key={e}>
            <div className="chat_message">
              chat: {m.message} ---{m.date.toLocaleTimeString()}
            </div>
          </div>
        ))}

        <div id="bottom-reference" ref={bottomRef} />
      </div>

      <div className="myFooter">
        <form onSubmit={handleSubmit}>
          <label>
            Name:
            <input type="text" value={message} onChange={handleChange} />
          </label>
          <input type="submit" value="Submit" />
        </form>
      </div>
    </React.Fragment>
  );
}

我使用这些样式来测试滚动:
.chat_container {
  height: 200px;
  overflow: scroll;
}

不错的方法。我的所有代码都已经采用了类的方式。我会模仿你的函数式方法,看看效果如何。顺便感谢你的努力。 - Nancy Moore

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接