Реализация системы комментариев на сайте

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

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

Предлагаемые услуги
Показано 1 из 1 услугВсе 2065 услуг
Реализация системы комментариев на сайте
Средняя
~5 рабочих дней
Часто задаваемые вопросы
Наши компетенции:
Этапы разработки
Последние работы
  • 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

Реализация системы комментариев

Система комментариев позволяет пользователям обсуждать материалы. Ключевые решения: собственная реализация vs готовый виджет (Disqus, Commento), вложенность (flat vs nested), модерация.

Структура базы данных

CREATE TABLE comments (
    id          SERIAL PRIMARY KEY,
    entity_type VARCHAR(50)  NOT NULL,  -- 'article', 'product', 'post'
    entity_id   INTEGER      NOT NULL,
    parent_id   INTEGER REFERENCES comments(id) ON DELETE SET NULL,
    user_id     INTEGER REFERENCES users(id),
    author_name VARCHAR(100),           -- для гостей
    author_email VARCHAR(255),
    body        TEXT         NOT NULL,
    status      VARCHAR(20)  NOT NULL DEFAULT 'pending',  -- pending|approved|rejected|spam
    likes_count INTEGER      NOT NULL DEFAULT 0,
    created_at  TIMESTAMPTZ  NOT NULL DEFAULT NOW(),
    updated_at  TIMESTAMPTZ  NOT NULL DEFAULT NOW()
);

CREATE INDEX ON comments(entity_type, entity_id, status, created_at);
CREATE INDEX ON comments(parent_id);
CREATE INDEX ON comments(user_id);

Laravel API

class CommentController extends Controller
{
    // Получить комментарии к материалу
    public function index(Request $request, string $entityType, int $entityId): JsonResponse
    {
        $comments = Comment::where('entity_type', $entityType)
            ->where('entity_id', $entityId)
            ->where('status', 'approved')
            ->whereNull('parent_id')  // только корневые
            ->with(['user:id,name,avatar', 'replies' => fn($q) => $q->where('status', 'approved')->with('user:id,name,avatar')])
            ->latest()
            ->paginate(20);

        return response()->json($comments);
    }

    // Добавить комментарий
    public function store(StoreCommentRequest $request, string $entityType, int $entityId): JsonResponse
    {
        $this->throttle('comments', 10, 60);  // 10 комментариев в минуту

        $requiresModeration = !auth()->check()
            || auth()->user()->comments()->where('status', 'spam')->exists()
            || $this->containsSuspiciousLinks($request->body);

        $comment = Comment::create([
            'entity_type'  => $entityType,
            'entity_id'    => $entityId,
            'parent_id'    => $request->parent_id,
            'user_id'      => auth()->id(),
            'author_name'  => auth()->user()?->name ?? $request->author_name,
            'author_email' => auth()->user()?->email ?? $request->author_email,
            'body'         => $this->sanitize($request->body),
            'status'       => $requiresModeration ? 'pending' : 'approved',
        ]);

        if ($comment->status === 'approved') {
            $this->notifyParentAuthor($comment);
        } else {
            // Уведомить модераторов
            Notification::send(
                User::moderators()->get(),
                new CommentPendingNotification($comment)
            );
        }

        return response()->json(CommentResource::make($comment), 201);
    }

    private function sanitize(string $body): string
    {
        return strip_tags($body, '<b><i><em><strong><a><br><p>');
    }

    private function containsSuspiciousLinks(string $body): bool
    {
        preg_match_all('/<a[^>]+href=["\']?([^"\'> ]+)/i', $body, $matches);
        foreach ($matches[1] ?? [] as $url) {
            if (!str_contains($url, config('app.url'))) {
                return true;
            }
        }
        return false;
    }
}

React: дерево комментариев

interface Comment {
  id: number;
  user: { name: string; avatar: string } | null;
  author_name: string;
  body: string;
  likes_count: number;
  created_at: string;
  replies?: Comment[];
}

function CommentThread({ entityType, entityId }: { entityType: string; entityId: number }) {
  const { data, isLoading } = useQuery({
    queryKey: ['comments', entityType, entityId],
    queryFn: () => api.get(`/comments/${entityType}/${entityId}`),
  });

  return (
    <section aria-label="Комментарии">
      <h2>Комментарии ({data?.meta.total ?? 0})</h2>
      <CommentForm entityType={entityType} entityId={entityId} />

      {isLoading ? <CommentSkeleton /> : (
        <ul className="comment-list">
          {data?.data.map(comment => (
            <CommentItem key={comment.id} comment={comment} depth={0} />
          ))}
        </ul>
      )}
    </section>
  );
}

function CommentItem({ comment, depth }: { comment: Comment; depth: number }) {
  const [showReplyForm, setShowReplyForm] = useState(false);

  return (
    <li className={`comment depth-${depth}`}>
      <img
        src={comment.user?.avatar || '/default-avatar.png'}
        alt={comment.user?.name || comment.author_name}
        width={40} height={40}
      />
      <div className="comment__content">
        <header>
          <strong>{comment.user?.name || comment.author_name}</strong>
          <time dateTime={comment.created_at}>
            {new Date(comment.created_at).toLocaleDateString('ru-RU')}
          </time>
        </header>
        <p>{comment.body}</p>
        <footer>
          <LikeButton commentId={comment.id} count={comment.likes_count} />
          {depth < 3 && (
            <button onClick={() => setShowReplyForm(!showReplyForm)}>Ответить</button>
          )}
        </footer>

        {showReplyForm && (
          <CommentForm parentId={comment.id} onSubmit={() => setShowReplyForm(false)} />
        )}

        {comment.replies?.map(reply => (
          <ul key={reply.id}><CommentItem comment={reply} depth={depth + 1} /></ul>
        ))}
      </div>
    </li>
  );
}

Лайки комментариев

class CommentLikeController extends Controller
{
    public function toggle(Comment $comment): JsonResponse
    {
        $userId = auth()->id();
        $key    = "comment_like:{$comment->id}:{$userId}";

        if (Cache::has($key)) {
            Cache::forget($key);
            $comment->decrement('likes_count');
            return response()->json(['liked' => false, 'count' => $comment->likes_count]);
        }

        Cache::put($key, true, now()->addYears(1));
        $comment->increment('likes_count');

        return response()->json(['liked' => true, 'count' => $comment->likes_count]);
    }
}

Срок реализации

Система комментариев (flat) с модерацией для Laravel + React: 3–4 дня. С вложенностью, лайками и email-уведомлениями: 5–7 дней.