Chatix Chatroom


Сейчас я покажу вам как можно сделать чат для команды/пользователей/друзей если у вас нет своего бэкенда или вы не хотите тратить время на его разработку. Мы сделаем простой текстовый чат и на это у нас уйдет около часа.


Написать работающий сетевой чат без бэкенда практически невозможно, он обязательно должен быть в том или ином виде. Мы будем использовать Chatix и его JavaScript SDK. Chatix и SDK будут заниматься хранением сообщений и сетевыми задачами, а мы займемся фронтендом.


Готовый код проекта доступен на GitHub
Demo


Структура проекта


  • App (корневой компонент приложения, выполняет роль хранителя состояния, т.к. в этом уроке мы не будем добавлять Redux или какой-либо другой state manager)
    • Header (шапка нашего приложения которая отображает логотип, название чата и позволяет пользователю написать свое имя)
    • LogoHeader
    • RoomTitle
    • Main
    • MemberList (список участников чата)
      • MemberItem[]
    • ChatField (контейнер для всего что связано с сообщениями чата)
      • MessageContainer
      • Message[] (представление сообщения; в данному уроке мы будем работать только с текстовыми сообщениями)
      • SendMessageForm (форма отправки нового сообщения в чат)
    • ChatixSDK (headless компонент отвечающий за работу с бэкендом)

Важное замечание насчет хранения состояния. Конечно было бы удобнее добавить сюда Redux и обрабатывать изменения состояния через него, но ради экономии времени мы будем хранить состояние в корневом компоненте App и будем пробрасывать данные в дочерние компоненты и вызывать из дочерних методы их родителей.
Например, когда мы получим название чата, мы сохраним его в состоянии компонента App и через props передадим его: App > Header > RoomTitle. Когда пользователь напишет сообщение мы передадим его из SendMessageForm до App: SendMessageForm > ChatField > Main > App.

Примерно так наш чат будет выглядеть в дизайне:


chatix.io


Взаимодействие между компонентами


Наши компоненты должны передавать друг другу данные и для того чтобы все работало как надо давайте уже сейчас определим как они будут друг с другом взаимодействовать.


Взаимодействие компонентов


Как видно на рисунке, основным компонентом у нас является App, который предоставляет данные в дочерние компоненты (благодаря реактивности мы просто назначим prop и дочерний компонент будет реагировать на изменения), а дочерние компоненты последовательно пробрасывают вызовы методов до App. Это не лучшая архитектура которую можно (и следует) сделать для продакшн-проекта, но для нашего урока сойдет.


Создание проекта


Создаем визуальные компоненты


Шапка


  1. Для начала нужно создать новый проект, для этого будем использовать create-react-app.


    npx create-react-app chatix-chatroom
    cd chatix-chatroom

    Запустите проект командой


    npm start

  2. Начнем с создания шапки.
    Сначала добавим в шапку логотип. Для этого Внутри папки src cоздаем папку components, а в ней папку logo_header. В эту папку загружаем логотип и создаем 2 файла LogoHeader.js и LogoHeader.css



LogoHeader.js


import React from 'react'
import logo from './chatix_logo.svg';
import './LogoHeader.css';

function LogoHeader(){
    return (
        <div className="LogoHeader">
            <img src={logo} className="App-logo" alt="Chatix logo" />
        </div>
    );
}

export default LogoHeader;

LogoHeader.css


.LogoHeader{
    flex-basis: 200px;
    flex-grow: 0;
    flex-shrink: 0;
}

Здесь все понятно, в этом компоненте просто импортируется файл с логотипом и стили.


Код файлов стилей здесь больше добавлять не буду, их вы можете посмотреть на странице готового проекта


Теперь выведем название чат-комнаты. Для этого создаем папку room-title а в ней компонент RoomTitle. Название в этот компонент мы будем прокидывать через props, поэтому пишем props.chatroomName и сейчас мы его сюда передадим.


RoomTitle


import React from 'react';
import './RoomTitle.css';

function RoomTitle(props){
    return (
        <div className="RoomTitle">
            <h1>{props.chatroomName}</h1>
        </div>
    );
}

export default RoomTitle;

Затем создаем сам компонент шапки и размещаем в нем логотип и название чат-комнаты. Сразу прокинем название чата в дочерний компонент через prop chatroomName.


Напоминаю что мы договорились о том все данные (состояние приложения) будет хранить корневой компонент App. Из него мы будeм передавать заголовок сначала в Header а из Header в RoomTitle.


components\header\Header.js


Header.js


import React from 'react';
import './Header.css'
import LogoHeader from '../logo_header/LogoHeader';
import RoomTitle from '../room-title/RoomTitle';

function Header(props) {
    return (
        <header>
            <LogoHeader/>
            <RoomTitle chatroomName={props.chatroomName} />
        </header>
    );
}

export default Header;

Далее открываем файл App.js и добавляем в него компонент Header.js.
Затем в стэйт добавляем название и через props пробрасываем его в шапку.
Еще в шапке нужно добавить имя текущего пользователя. Для этого в стэйт добавляем объект пользователя и аналогичным образом пробрасываем его в шапку


import React from 'react';
import './App.css';
import Header from './components/header/Header';

class App extends React.Component {
    constructor(props){
        super(props);
        chatroomName: 'Чат-комната',
        me: {
            is_online: true,
            name: "Алексей",
            uuid: "98s7dfh9a8s7dhf"
        }
    }
    render() {
        return (
          <div className="App">
            <Header 
                chatroomName={this.state.chatroomName}
                me={this.state.me}
            />
          </div>
        );
    };
}

export default App;

Теперь в шапке нужно добавить инпут с именем текущего пользователя и назначить обработчик на изменение чтобы мы могли передать новое имя пользователя в компонент App.


Для этого инпуту с именем добавляем функцию-обработчик handleChangeName и в ней вызываем callback-функцию props.updateVisitor в которую передаем объект пользователя с обновленным именем.


Header.js


function Header(props) {
    const [name, setName] = useState(props.me.name ? props.me.name : props.me.uuid.substr(-10))

    const handleChangeName = (e) => {
        setName(e.target.value)
        let visitor = {...props.me};
        visitor.name = e.target.value;
        props.updateVisitor(visitor)
    }

    return (
        <header>
            <LogoHeader/>
            <RoomTitle chatroomName={props.chatroomName}/>
            {
                props.me ? 
                    <input
                        className='name-input'
                        value={name}
                        placeholder='Ваше имя'
                        onChange={(e) => handleChangeName(e)}
                    />
                : null
            }
        </header>
    );
}

Теперь добавим в App эту функцию props.updateVisitor и пробросим ее в шапку. Пока что она просто обновляет объект пользователя в стэйте, но дальше мы через нее будем обновлять пользователя на сервере.


onUpdateVisitor = (visitor) => {
    this.setState({me: visitor})
 }

Итак, сейчас наше приложение выглядит вот так и пока что умеет только обновлять имя. Идем дальше


Chatroom header


Сайдбар


Теперь давайте займемся созданием сайдбара.
Сайдбар будет находиться внутри основного компонента на странице Main.js.
Создаем его components\main\Main.js, затем создаем компонент со списком пользователей components\member-list\MemberList.js и сразу создаем компонент который будет отображать самих пользователей components\member-item\MemberItem.js.


Что бы стало понятней как связаны эти 3 компонента взгляните на схему проекта в начале статьи.


Компоненты созданы, теперь пойдем по порядку.
Для начала добавим в стэйт компонента App массив пользователей и добавим компонент Main. Затем пробросим в него этих пользователей.


App


class App extends React.Component {

  constructor(props) {
    super(props);
    this.state = {
      chatroomName: 'Чат-комната',
      members: [
        {
          is_online: true,
          name: "Алексей",
          uuid: "98s7dfh9a8s7dhf"
        },
        {
          is_online: true,
          name: "Дмитрий",
          uuid: "mnzxcv97zx6chvo"
        },
        {
          is_online: false,
          name: "Андрей",
          uuid: "kjuhv987ashdfoua"
        },
        {
          is_online: false,
          name: "Владимир",
          uuid: "jdhnf978WEHJSNDL"
        },
      ],
      me: {
        is_online: true,
        name: "Алексей",
        uuid: "98s7dfh9a8s7dhf"
      }
    };
  }
  render() {
    return (
      <div className="App">
        <Header 
            chatroomName={this.state.chatroomName} 
            me={this.state.me}    
        />
        <Main 
            members={this.state.members}
            me={this.state.me}
        />
      </div>
    );
  };
}

В компоненте Main добавляем компонент MemberList и пробрасываем массив пользователей в него.


Main.js


function Main(props) {
    return(
        <section className="Main">
            <MemberList members={props.members} />
        </section>
    );
}

А в компоненте MemberList мы в цикле перебираем всех пользователей и для каждого возвращаем компонент MemberItem и передаем в него объект пользователя.


MemberList.js


function MemberList(props) {
    const members = props.members.map((member) => 
        <MemberItem key={member.uuid} member={member} />
    );

    return (
        <section className="MemberList">
            {members}
        </section>
    );
}

Компонент MemberItem занимается уже непосредственного отображением пользователя в сайдбаре. В нем мы проверяем наличие имени у пользователя, если оно не установлено, то отображаем первые 10 символов идентификатора. Так же проверяем статус онлайн/оффлайн и сравниваем идентификатор с идентификатором текущего пользователя, что бы напротив него отобразить пометку "(Вы)".


function MemberItem(props) {

    function getName(){
        let name = ''
        if (props.member.uuid === props.me.uuid) {
            if(props.me.name) {
                name = props.me.name
            }
            else {
                name = props.me.uuid.substring(props.me.uuid.length-10, props.me.uuid.length);
            }
        }
        else {
            if(props.member.name){
                name = props.member.name
            }
            else {
                name = props.member.uuid.substring(props.member.uuid.length-10, props.member.uuid.length);
            }
        }
        return name;
    }

    return(
        <div className="MemberItem">
            <img src={ icon } alt={ props.member.name }/>
            <span>
                { getName() }
                {
                    props.member.uuid === props.me.uuid && " (Вы) "
                }
            </span>
            {
                props.member.is_online && <span className="online">•</span>
            }
        </div>
    );
}

Готово. Сейчас приложение выглядит уже вот так


Chatroom header


Список сообщений и форма отправки


Теперь займемся список сообщений и формой отправки.
Для начала в стэйт компонента App добавим массив с сообщениями.


App


this.state = {
      chatroomName: 'Чат-комната',
      messages: [
        {
          content: "Сообщение 1",
          sender_id: "mnzxcv97zx6chvo",
          uuid: "dg897sdfg"
        },
        {
          content: "Сообщение 2",
          sender_id: "98s7dfh9a8s7dhf",
          uuid: "8723hernm"
        },
        {
          content: "Еще одно сообщение",
          sender_id: "mnzxcv97zx6chvo",
          uuid: "435nbcv98234"
        }
      ],
      members: [
        {
          is_online: true,
          name: "Алексей",
          uuid: "98s7dfh9a8s7dhf"
        },
        {
          is_online: true,
          name: "Дмитрий",
          uuid: "mnzxcv97zx6chvo"
        },
        {
          is_online: false,
          name: "Андрей",
          uuid: "kjuhv987ashdfoua"
        },
        {
          is_online: false,
          name: "Владимир",
          uuid: "jdhnf978WEHJSNDL"
        },
      ],
      me: {
        is_online: true,
        name: "Алексей",
        uuid: "98s7dfh9a8s7dhf"
      }
    };

И пробросим их в компонент Main


App


 <Main
    members={this.state.members}
    messages={this.state.messages}
    me={this.state.me}
/>

Теперь создадим компонент conponents/chat-field/ChatField.js
Подключим его в Main и пробросим сообщения в него.


Main


function Main(props) {
    return(
        <section className="Main">
            <MemberList 
                me={props.me} 
                members={props.members} />
            <ChatField messages={props.messages} />
        </section>
    );
}

Далее создадим компонент conponents/message-container/MessageContainer.js
Подключим его в ChatField и так же пробрасываем сообщения в него.


ChatField


function Main(props) {
    return(
        <section className="Main">
            <MemberList 
                me={props.me} 
                members={props.members} />
            <ChatField messages={props.messages} />
        </section>
    );
}

Дальше мы циклом будем перебирать все сообщения и для каждого возвращать компонент который будет его показывать.
Давайте создадим его conponents/message/Message.js. В нем мы отображаем иконку посетителя, его имя или идентификатор если имя не указано и сам текст сообщения.


Message


function Message(props) {

    const getSenderName = () => {
        if (props.sender) {
            return props.sender.name ? props.sender.name : props.sender.uuid.substr(-10);
        }
        return "Unknown sender";
    };

    return(
        <div className="Message">
            <div className="message-sender-icon">
                <img src={icon} alt="visitor icon"/>
            </div>
            <div className="message-bubble">
                <div className="message-sender-name">{getSenderName()}</div>
                <div className="message-content">{props.message.content}</div>
            </div>
        </div>
    );
}

Теперь в MessageContainer циклом перебираем все сообщения и для каждого возвращаем компонент Message, в который передаем объект сообщения


MessageContainer


function MessageContainer(props) {
     const messageList = props.messages.map(message => 
        <Message 
            key={message.uuid}
            sender={props.members.find((member) => member.uuid === message.sender_id)} 
            message={message} />
        );

    return (
        <section className="MessageContainer" ref={messagesContainer}>
            {messageList}
        </section>
    );
}

Сейчас проект выглядит вот так:


Chatroom Messages


Теперь создадим компонент с формой для отправки сообщений components/send-message-form/SendMessageForm.js. В нем создадим инпут и кнопку для отправки. При изменении инпута текст из него записываем в стэйт, а при клике на кнопку вызываем callback-функцию onSendNewMessage и передаем в нее сообщение из стэйта. Функцию onSendNewMessage создадим чуть позже в компоненте App и пробросим ее через props.


SendMessageForm


class SendMessageForm extends React.Component {
    constructor(props) {
        super(props);
        this.state = {
            message: ''
        };
    }

    currentMessageChanged = (e) => {
        this.setState({message: e.target.value });
    }

    sendMessageClicked = async (e) => {
        e.preventDefault();
        if (this.state.message.length > 0) {
            await this.props.onSendNewMessage(this.state.message);
            this.setState({...this.state, ...{message : ''}});
        }
    }

    render(){
        return (
            <section className="SendMessageForm">
                <form>
                    <input 
                        type="text" 
                        value={this.state.message} 
                        onChange={this.currentMessageChanged} 
                        placeholder="Type message to send"/>
                    <button 
                        type="submit" 
                        onClick={this.sendMessageClicked}
                    >
                        Send
                    </button>
                </form>
            </section>
        );
    }
}

Теперь разместим компонент SendMessageForm внутри ChatField.


ChatField


function ChatField(props) {
    return(
        <section className="ChatField">
            <MessageContainer 
                members={props.members}
                messages={props.messages} 
            />
            <SendMessageForm onSendNewMessage={props.onSendNewMessage}/>
        </section>
    );
}

В компоненте Main также пробросим функцию onSendNewMessage в ChatField.


Main


<ChatField
    members={props.members}
    messages={props.messages}
    onSendNewMessage={props.onSendNewMessage} 
/>

Теперь создадим эту функцию в App и пробросим ее в Main.


App


onSendNewMessage = async (message) => {
    console.log(message)
 }

App


<Main
    members={this.state.members}
    messages={this.state.messages}
    onSendNewMessage={this.onSendNewMessage}
    me={this.state.me}
/>

Готово. Теперь при клике на кнопку отправки сообщения оно будет передаваться в компонент App.
Сейчас приложение выглядит вот так:


Chatroom Final


Итак, сейчас у нас в приложении все отображается и все работает как нужно, но пока что со статичными данными, а что бы оживить наш чат, нужно связать его с бэкендом.


Подключение бэкенда


Для этого первым делом нужно установить пакет chatix-core.


npm i chatix-core

Затем создать аккаунт в chatix и создать чат-комнату. Для этого переходим на chatix.io и регистрируемся.
После регистрации вы можете посмотреть идентификатор сайта websiteId в интерфейсе администратора на странице настроек чата.


Теперь создаем новую чат-комнату с которой и будем работать.


Создание чат-комнаты


Возвращаемся в наш проект и создаем новый компонент через который мы будем работать с сервером.
components\chatix\ChatixSDK.js


В нем импортируем ChatixCore


import ChatixCore from 'chatix-core';

В компоненте ChatixSDK создаем экземпляр класса ChatixCore и в качестве аргумента передаем websiteId.


const websiteId = "ВАШ_WEBSITE_ID";
this.sdk = new ChatixCore(websiteId);

Теперь в this.sdk вам доступны методы для работы с чат-комнатой. Посмотреть список методов можно на странице проекта chatix-core


Далее нам нужно подключиться к серверу и получить данные о созданной ранее чат-комнате. Для этого есть асинхронные методы start() и getChatroom().


После того как получили объект чат-комнаты, давайте сразу возьмем его название и передадим его в App. Для этого в App добавим callback-функцию updateChatroomTitle(chatroom.title) и вызовем ее в ChatixSDK.


ChatixSDK


class ChatixSDK extends React.Component {
    constructor(props){
        super(props);
        const websiteId = "ВАШ_WEBSITE_ID";
        this.chatroomId = "ВАШ_CHATROOM_ID";
        this.sdk = new ChatixCore(websiteId);
        this.sdk.start()
            .then( async () => {
                try {
                    // refresh information about chatroom and call passed handler
                    const chatroom = await this.sdk.getChatroom(this.chatroomId);
                    if (props.updateChatroomTitle) {
                        props.updateChatroomTitle(chatroom.title);
                    }
                } catch (err) {
                    console.error(err);
                }
            })
            .catch((e) => {
                console.error(e);
            });
    }
    render(){
        return null;
    }
}

this.chatroomId вы можете посмотреть в интерфейсе менеджера открыв нужную чат-комнату.


ChatroomID


Теперь в App подключим компонент ChatixSDK и прокинем в него функцию updateChatroomTitle которая будем обновлять название чата. Так же добавим ему ref-ссылку, что бы могли обращаться к этому компоненту.


App


this.chatixSDK = React.createRef();

setChatroomTitle = (newName) => {
    const newStateFragment = { chatroomName: newName};
    this.setState({...this.state, ...newStateFragment});
};

App


render() {
    return (
        <div className="App">
            <Header 
                chatroomName={this.state.chatroomName}
                me={this.state.me}
                updateVisitor={this.onUpdateVisitor}
            />
            <Main
                members={this.state.members}
                messages={this.state.messages}
                onSendNewMessage={this.onSendNewMessage}
                me={this.state.me}
            />
            <ChatixSDK 
                ref={this.chatixSDK}
                updateChatroomTitle={this.setChatroomTitle}
            />
        </div>
    );
  };

Готово. Теперь сразу после подключения к серверу мы запрашиваем данные о чате, получаем его название и записываем его в стэйт компонента App, а поскольку изменения в стэйте вызывают повторный рендер компонента, то название в шапке обновится автоматически. Сейчас название по умолчанию в стэйте можно заменить пустой строкой.


App


chatroomName: ''

Теперь давайте заполним боковую панель настоящими пользователями.
Но прежде чем получить список пользователей вы должны подключиться к чату, для этого в ChatixSDK внутри функции this.sdk.start() получаем список всех чат-комнат пользователя, проверяем подключен ли он к текущей и если нет, то подключаем его.


ChatixSDK


const myChatrooms = await this.sdk.getMyChatrooms();
if (myChatrooms.filter(x => x.id===this.chatroomId).length === 0) {
    await this.sdk.connectToChatroom(this.chatroomId);
}

После того как мы убедимся что наш пользователь подключен к чат-комнате, мы можем получить список участников этого чата.


ChatixSDK


// lets get all chatroom members using infinite loop with break on empty server response
let membersPage = 1;
let allChatroomMembers = [];
while(true) {
    let pagedMembers = await this.sdk.getChatroomMembers(this.chatroomId, membersPage++, 10);
    allChatroomMembers = [...allChatroomMembers, ...pagedMembers];
    if (pagedMembers.length === 0) {
        break;
    }
}

Здесь мы в бесконечном цикле постранично запрашиваем пользователей пока не получим всех, как только получили всех — прерываем цикл. После этого так же как и название чат-комнаты пробрасываем в родительский компонент использую callback функцию.


ChatixSDK


if (props.setChatroomMembers) {
    props.setChatroomMembers(allChatroomMembers);
}

Теперь в компоненте App создадим эту callback функцию setChatroomMembers которая будет сортировать пользователей по статусу в сети\не в сети и по алфавиту и записывать их в state.


App.js


setChatroomMembers = (members) => {
    members.sort(this.sortMembers);
    const newStateFragment = { members: members};
    this.setState({...this.state, ...newStateFragment});
}

Добавим функцию сортировки sortMembers. Она сортирует пользователей по статусу и по алфавиту.


App.js


sortMembers(a, b) {
    if (a.is_online === true && b.is_online === false) {
      return -1;
    } else if (b.is_online === true && a.is_online === false) {
      return 1;
    } else {
      if (a.name && b.name) {
        if (a.name.toLocaleUpperCase() > b.name.toLocaleUpperCase()) {
          return 1;
        } else if (a.name.toLocaleUpperCase() < b.name.toLocaleUpperCase()) {
          return -1;
        }
      } else if (a.name && !b.name) {
        return -1;
      } else if (!a.name && b.name) {
        return 1;
      } 
      if (a.uuid > b.uuid) {
        return -1;
      } else {
        return 1;
      }
    }
  }

Далее пробрасываем функцию setChatroomMembers в ChatixSDK.


App


render() {
    return (
        <div className="App">
            <Header 
                chatroomName={this.state.chatroomName}
                me={this.state.me}
                updateVisitor={this.onUpdateVisitor}
            />
            <Main
                members={this.state.members}
                messages={this.state.messages}
                onSendNewMessage={this.onSendNewMessage}
                me={this.state.me}
            />
            <ChatixSDK 
                ref={this.chatixSDK}
                updateChatroomTitle={this.setChatroomTitle}
                setChatroomMembers={this.setChatroomMembers}
            />
        </div>
    );
  };

Теперь сразу после подключения к серверу мы так же как и с заголовком запрашиваем список всех подключенных пользователей и записываем его в стэйт компонента App. И так же меняем дефолтное значение списка пользователей в стэйте.


App


members: []

Теперь точно по такому же принципу получаем объект текущего пользователя и массив сообщений и также записываем их в стэйт App


ChatixSDK


// lets load 100 last messages from current chatroom
const lastMessages = await this.sdk.getChatroomMessages(this.chatroomId, null, 100);
if (props.setChatroomMessages) {
    props.setChatroomMessages(lastMessages);
}

if (props.setMe) {
    const me = this.sdk.getVisitor();
    this.props.setMe(me);
}

App


<ChatixSDK 
    ref={this.chatixSDK}
    setMe={this.setMe}
    updateChatroomTitle={this.setChatroomTitle}
    setChatroomMembers={this.setChatroomMembers}
    setChatroomMessages={this.setChatroomMessages}
/>

Дальше займемся отправкой сообщений.


У нас в App уже есть функция onSendNewMessage которая выводит отправляем сообщение в консоль. Вместо этого мы просто будем вызывать метод sendChatroomMessage для отправки сообщения из ChatixSDK.
Это асинхронный метод и он возвращает в ответе объект отправленного сообщения, которые мы тут же добавляем в массив сообщений в стйэте. Кстати обратите внимание, что к chatixSDK мы обращаемся по созданной ранее ссылке this.chatixSDK.


App


onSendNewMessage = async (message) => {
    let receivedMsg = await this.chatixSDK.current.sendChatroomMessage(message);

    const currentMessages = this.state.messages;
    currentMessages.push(receivedMsg);
    const newStateFragment = {messages: currentMessages};
    this.setState({...this.state, ...newStateFragment});
  }

Поскольку изменение в стэйте вызывает его повторный рендер, то список сообщений у нас обновится автоматически. Но нам нужно сделать что бы при добавлении сообщений скролл в блоке с сообщениями опускался вниз.


Для этого открываем компонент MessageContainer и используя хук useEffect следим за изменением массива с сообщениями, и как только он изменился и сообщений добавилось мы получаем scrollHeight блока с сообщениями и скроллим его на эту же величину


function MessageContainer(props) {
    const messagesContainer = React.createRef();

    useEffect(() => {
        messagesContainer.current.scrollTop = messagesContainer.current.scrollHeight
    }, [props, messagesContainer]);

    const messageList = props.messages.map(message => 
        <Message 
            key={message.uuid}
            sender={props.members.find((member) => member.uuid === message.sender_id)} 
            message={message} />
        );

    return (
        <section className="MessageContainer" ref={messagesContainer}>
            {messageList}
        </section>
    );
}

Сейчас давайте доделаем обновление имени пользователя. Мы уже создали инпут в шапке и при его изменении пробрасываем обновленный объект пользователя в компонент App и там выводим его в консоль. Давайте доделаем эту функцию. Для этого добавим в нее вызов метода this.chatixSDK.current.updateVisitor(user), это обновит данные на сервере. И так же обновим данные в локальном стэйте, для этого обновим объект this.state.me и в массиве this.state.members так же найдем текущего пользователя и обновим его. Это нужно что бы обновилось имя текущего пользователя в отправленных им сообщениях.


App


onUpdateVisitor = (user) => {
    this.chatixSDK.current.updateVisitor(user)
    this.setMe(user)
    let currentUser = this.state.members.find((member) => (member.uuid === user.uuid))
    let currentUserIndex = this.state.members.indexOf(currentUser)
    let newMembers = [...this.state.members]
    newMembers[currentUserIndex] = user;
    this.setState({
      members: newMembers
    })
}

Теперь нам нужно научиться реагировать на входящие сообщения, подключение/отключение пользователей и изменение информации и подключенных пользователях.


Для этого в файле ChatixSDK.js в конструкторе нам нужно переопределить callback функции. Полный список функций и аргументов вы можете посмотреть на странице проекта chatix-core.


Сейчас нас интересуют onChatroomMessageReceived, onMemberConnectedToChatroom, onMemberDisconnectedFromChatroom и onApplyVisitorInfo.


Переопределяем их и на каждую функцию вызываем свой callback который создадим в App.


this.sdk.onChatroomMessageReceived = (chatroomId, message) => {
    if (chatroomId === this.chatroomId) {
        this.props.onNewMessageReceived(message);
    }
};
this.sdk.onMemberConnectedToChatroom = (chatroomId, member) => {
    if (chatroomId === this.chatroomId && props.addChatroomMember) {
        this.props.addChatroomMember(member);
    }
};
this.sdk.onMemberDisconnectedFromChatroom = (chatroomId, member) => {
    if (chatroomId === this.chatroomId && props.removeChatroomMember) {
        this.props.removeChatroomMember(member);
    }
};
this.sdk.onApplyVisitorInfo = (visitor) => {
    this.props.onMemberUpdated(visitor)
}

Далее идем в App и создаем эти функции.


onNewMessageReceived(message)
Эта функция принимает объект сообщения и просто добавляет его в state к остальным сообщениям. После этого компонент повторно отрендерится и оно отобразится в списке, так же как во время отправки исходящего сообщения.


App


onNewMessageReceived = (message) => {
    const currentMessages = this.state.messages;
    currentMessages.push(message);
    const newStateFragment = {messages: currentMessages};
    this.setState({...this.state, ...newStateFragment});
  }

App
addChatroomMember(member)
Эта функция принимает объект посетителя и так же добавляем его в state в уже имеющейся массив members. После этого компонент так же повторно отрендерится и пользователи добавится в список подключенных пользователей.


App


addChatroomMember = (member) => {
    const newStateFragment = {};
    const currentMembers = this.state.members;
    currentMembers.push(member);
    currentMembers.sort(this.sortMembers);
    newStateFragment.members = currentMembers;
    this.setState({...this.state, ...newStateFragment});
  }

App
removeChatroomMember(memberId)
Эта функция принимает идентификатор посетителя и удаляет из state посетителя с таким идентификатором members в state.


removeChatroomMember = (memberId) => {
    const currentMembers = this.state.members;
    const filteredMembers = currentMembers.filter(x=> x.uuid !== memberId);
    const newStateFragment = {members: filteredMembers};
    this.setState({...this.state, ...newStateFragment});
  }

onMemberUpdated(updatedMember)
Эта функция обновляет информацию и посетителе. Например если у него изменилось имя или статус. По идентификатору ищем этого пользователя в state и заменяем его на обновленного.


App


onMemberUpdated = (updatedMember) => {
    let oldMember = this.state.members.find(member => member.uuid === updatedMember.uuid);
    oldMember = this.state.members.indexOf(oldMember);
    let newStateMembers = this.state.members;
    newStateMembers[oldMember] = updatedMember;

    this.setState({
      members: newStateMembers
    })
}

И пробрасываем их все в ChatixSDK


ChatixSDK


<ChatixSDK 
    ref={this.chatixSDK}
    setMe={this.setMe}
    updateChatroomTitle={this.setChatroomTitle}
    setChatroomMembers={this.setChatroomMembers}
    addChatroomMember={this.addChatroomMember} 
    removeChatroomMember={this.removeChatroomMember}
    setChatroomMessages={this.setChatroomMessages}
    onNewMessageReceived={this.onNewMessageReceived}
    onMemberUpdated={this.onMemberUpdated} 
/>

Готово! Мы создали просто чат который умеет отправлять\принимать сообщения, показывать список пользователей, реагировать на их подключение/отключение и обновление информации.


Статью подготовил alekseyso
Дополнительные материалы:
SDK Chatix (документация)
SDK Chatix (npm)
192 выпуск подкаста Вэб-стандарты

Комментарии (8)


  1. mayorovp
    11.11.2019 13:02
    +3

    От заголовка "Делаем многопользовательский чат только на фронтенде" я ожидал увидеть каких-то хитростей с использованием WebRTC, а не "возьмём чужой бэк и напишем к нему свой фронт".


    1. zeolant
      11.11.2019 15:03

      В принципе, чисто теоретически, можно даже не зарываться в WebRTC а использовать какой-нибудь PubNub, но не знаю будет ли это считаться за «чат только на фронтенде» ))


  1. Rebeiro
    11.11.2019 17:29

    а есть что нибудь типа такого, установил и пользуйся, а то пока до конца домотал, чуть не уснул) а это еще и ставить надо


  1. igormich88
    12.11.2019 21:15

    А Chatix это ваша разработка?


    1. undestroyer Автор
      12.11.2019 03:45

      Да, chatix это наш продукт


      1. igormich88
        12.11.2019 16:13

        А он коммерческий?
        Текстовые сообщения 0.003 руб за шт. в день
        Сообщения с картинками 0.0035 руб за шт. в день
        Сообщения с файлами 0.0035 руб за шт. в день
        Сообщения с шепотом (бесплатно)

        Я кстати так и не смог понять что значит «Сообщения с шепотом» и чем они отличаются от обычных?


        1. undestroyer Автор
          13.11.2019 02:16

          Да, продукт коммерческий с оплатой по мере использования.
          Шепот — это острый вид текстовых сообщений, которые доступны только администраторам. Их удобно использовать если нужно что-то обсудить в чате в тайне от обычных пользователей.


          1. undestroyer Автор
            13.11.2019 02:17

            *Шепот — это особый вид текстовых сообщений