Реалізація онлайн-чату підтримки на сайті

Наша компанія займається розробкою, підтримкою та обслуговуванням сайтів будь-якої складності. Від простих односторінкових сайтів до масштабних кластерних систем, побудованих на мікро сервісах. Досвід розробників підтверджено сертифікатами від вендорів.
Розробка та обслуговування будь-яких видів сайтів:
Інформаційні сайти або веб-програми
Сайти візитки, landing page, корпоративні сайти, онлайн каталоги, квіз, промо-сайти, блоги, ресурси новин, інформаційні портали, форуми, агрегатори
Сайти або веб-програми електронної комерції
Інтернет-магазини, B2B-портали, маркетплейси, онлайн-обмінники, кешбек-сайти, біржі, дропшиппінг-платформи, парсери товарів
Веб-програми для управління бізнес-процесами
CRM-системи, ERP-системи, корпоративні портали, системи управління виробництвом, парсери інформації
Сайти або веб-програми електронних послуг
Дошки оголошень, онлайн-школи, онлайн-кінотеатри, конструктори сайтів, портали надання електронних послуг, відеохостинги, тематичні портали

Це лише деякі з технічних типів сайтів, з якими ми працюємо, і кожен із них може мати свої специфічні особливості та функціональність, а також бути адаптованим під конкретні потреби та цілі клієнта.

Пропоновані послуги
Показано 1 з 1 послугУсі 2065 послуг
Реалізація онлайн-чату підтримки на сайті
Складна
~1-2 тижні
Часті питання
Наші компетенції:
Етапи розробки
Останні роботи
  • image_website-b2b-advance_0.png
    Розробка сайту компанії B2B ADVANCE
    1262
  • image_web-applications_feedme_466_0.webp
    Розробка веб-додатків для компанії FEEDME
    1171
  • image_websites_belfingroup_462_0.webp
    Розробка веб-сайту для компанії БЕЛФІНГРУП
    874
  • image_ecommerce_furnoro_435_0.webp
    Розробка інтернет магазину для компанії FURNORO
    1094
  • image_crm_enviok_479_0.webp
    Розробка веб-додатків для компанії Enviok
    831
  • image_bitrix-bitrix-24-1c_fixper_448_0.png
    Розробка веб-сайту для компанії ФІКСПЕР
    851

Реалізація live-чату підтримки

Live-чат дозволяє користувачам спілкуватися з операторами підтримки в реальному часі. Архітектура: WebSocket-сервер (Socket.IO або Laravel Reverb), база для зберігання історії, інтерфейс оператора з чергою діалогів.

Архітектура

[Visitor widget] ←──WebSocket──→ [Chat Server]
[Operator panel] ←──WebSocket──→    ↕
                                [Database: chats, messages]
                                    ↕
                              [Queue: offline messages → email]

Backend: Node.js + Socket.IO

import { Server } from 'socket.io';
import { createServer } from 'http';

const io = new Server(createServer(), {
  cors: { origin: process.env.FRONTEND_URL, credentials: true },
});

// Авторизація операторів
io.use(async (socket, next) => {
  const token = socket.handshake.auth.token;
  if (token) {
    const user = await verifyToken(token);
    if (user?.role === 'operator') {
      socket.data.user = user;
      socket.data.isOperator = true;
    }
  }
  next();
});

io.on('connection', (socket) => {
  // Відвідувач починає чат
  socket.on('chat:start', async (data) => {
    const chat = await Chat.create({
      visitor_name:  data.name,
      visitor_email: data.email,
      page_url:      data.pageUrl,
      status:        'waiting',
    });

    socket.join(`chat:${chat.id}`);
    socket.data.chatId = chat.id;

    // Сповіщаємо всіх операторів
    io.to('operators').emit('chat:new', {
      id: chat.id,
      visitor_name: chat.visitor_name,
      page_url: chat.page_url,
      started_at: chat.created_at,
    });

    socket.emit('chat:started', { chatId: chat.id });
  });

  // Оператор приймає чат
  socket.on('chat:accept', async ({ chatId }) => {
    const chat = await Chat.findByPk(chatId);
    if (!chat || chat.status !== 'waiting') return;

    await chat.update({ operator_id: socket.data.user.id, status: 'active', accepted_at: new Date() });
    socket.join(`chat:${chatId}`);

    // Сповіщаємо відвідувача
    io.to(`chat:${chatId}`).emit('chat:accepted', {
      operator: { name: socket.data.user.name, avatar: socket.data.user.avatar },
    });
  });

  // Повідомлення
  socket.on('chat:message', async ({ chatId, text }) => {
    const message = await Message.create({
      chat_id:     chatId,
      sender_type: socket.data.isOperator ? 'operator' : 'visitor',
      sender_id:   socket.data.user?.id ?? null,
      text:        sanitize(text),
    });

    io.to(`chat:${chatId}`).emit('chat:message', {
      id:          message.id,
      text:        message.text,
      sender_type: message.sender_type,
      sent_at:     message.created_at,
    });

    await chat.update({ last_message_at: new Date() });
  });

  // Індикатор друкування
  socket.on('chat:typing', ({ chatId }) => {
    socket.to(`chat:${chatId}`).emit('chat:typing', {
      sender_type: socket.data.isOperator ? 'operator' : 'visitor',
    });
  });

  // Закрити чат
  socket.on('chat:close', async ({ chatId }) => {
    await Chat.update({ status: 'closed', closed_at: new Date() }, { where: { id: chatId } });
    io.to(`chat:${chatId}`).emit('chat:closed');

    const chat = await Chat.findByPk(chatId, { include: [Message] });
    if (chat.visitor_email) {
      await emailService.sendTranscript(chat);
    }
  });

  if (socket.data.isOperator) {
    socket.join('operators');
    io.to('operators').emit('operator:online', { id: socket.data.user.id });
  }

  socket.on('disconnect', async () => {
    if (socket.data.isOperator) {
      io.to('operators').emit('operator:offline', { id: socket.data.user?.id });
    } else if (socket.data.chatId) {
      io.to(`chat:${socket.data.chatId}`).emit('visitor:left');
    }
  });
});

React: Віджет для відвідувача

export function ChatWidget() {
  const [isOpen, setIsOpen] = useState(false);
  const [status, setStatus] = useState<'idle' | 'starting' | 'waiting' | 'active' | 'closed'>('idle');
  const [messages, setMessages] = useState<Message[]>([]);
  const [input, setInput] = useState('');
  const [chatId, setChatId] = useState<string | null>(null);
  const [operator, setOperator] = useState<{ name: string } | null>(null);
  const socketRef = useRef<Socket | null>(null);

  useEffect(() => {
    socketRef.current = io(import.meta.env.VITE_CHAT_SERVER);
    const socket = socketRef.current;

    socket.on('chat:started', ({ chatId }) => {
      setChatId(chatId);
      setStatus('waiting');
    });

    socket.on('chat:accepted', ({ operator }) => {
      setOperator(operator);
      setStatus('active');
    });

    socket.on('chat:message', (message) => {
      setMessages(prev => [...prev, message]);
    });

    socket.on('chat:closed', () => setStatus('closed'));

    return () => { socket.disconnect(); };
  }, []);

  const startChat = () => {
    setStatus('starting');
    socketRef.current?.emit('chat:start', {
      name: 'Відвідувач',
      pageUrl: window.location.href,
    });
  };

  const sendMessage = () => {
    if (!input.trim() || !chatId) return;
    socketRef.current?.emit('chat:message', { chatId, text: input });
    setInput('');
  };

  if (!isOpen) {
    return (
      <button className="chat-trigger" onClick={() => setIsOpen(true)} aria-label="Відкрити чат">
        💬
      </button>
    );
  }

  return (
    <div className="chat-widget" role="dialog" aria-label="Чат підтримки">
      <header>
        <h2>{operator ? `Чат з ${operator.name}` : 'Чат підтримки'}</h2>
        <button onClick={() => setIsOpen(false)} aria-label="Закрити">×</button>
      </header>

      <div className="messages" aria-live="polite">
        {status === 'idle' && (
          <div className="start-screen">
            <p>Привіт! Чим можемо допомогти?</p>
            <button onClick={startChat}>Почати чат</button>
          </div>
        )}
        {status === 'waiting' && <p>Шукаємо вільного оператора...</p>}
        {messages.map(msg => (
          <div key={msg.id} className={`message message--${msg.sender_type}`}>
            <p>{msg.text}</p>
          </div>
        ))}
      </div>

      {status === 'active' && (
        <footer>
          <input
            value={input}
            onChange={e => {
              setInput(e.target.value);
              socketRef.current?.emit('chat:typing', { chatId });
            }}
            onKeyDown={e => e.key === 'Enter' && !e.shiftKey && (e.preventDefault(), sendMessage())}
            placeholder="Напишіть повідомлення..."
          />
          <button onClick={sendMessage} disabled={!input.trim()}>Відправити</button>
        </footer>
      )}
    </div>
  );
}

Терміни реалізації

Задача Термін
Socket.IO сервер + базові события 2–3 дні
Віджет відвідувача (React) 2–3 дні
Панель оператора 3–4 дні
Історія чатів + транскрипти на email +1–2 дні
Повна система з чергою та моніторингом 8–12 днів