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

Наша компанія займається розробкою, підтримкою та обслуговуванням сайтів будь-якої складності. Від простих односторінкових сайтів до масштабних кластерних систем, побудованих на мікро сервісах. Досвід розробників підтверджено сертифікатами від вендорів.
Розробка та обслуговування будь-яких видів сайтів:
Інформаційні сайти або веб-програми
Сайти візитки, 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('uk-UA')}
          </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 днів.