Реализация оценки статьи/страницы (Was this helpful?) на сайте

Наша компания занимается разработкой, поддержкой и обслуживанием сайтов любой сложности. От простых одностраничных сайтов до масштабных кластерных систем построенных на микро сервисах. Опыт разработчиков подтвержден сертификатами от вендоров.

Разработка и обслуживание любых видов сайтов:

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

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

Предлагаемые услуги
Показано 1 из 1 услугВсе 2065 услуг
Реализация оценки статьи/страницы (Was this helpful?) на сайте
Простая
~1 рабочий день
Часто задаваемые вопросы

Наши компетенции:

Этапы разработки

Последние работы

  • 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

Виджет оценки полезности статьи

«Была ли эта статья полезной?» — простейший инструмент обратной связи для базы знаний, блога и документации. Помогает выявить страницы с плохим контентом без анализа аналитики вручную.

Backend

Schema::create('article_ratings', function (Blueprint $table) {
    $table->id();
    $table->foreignId('article_id')->constrained()->cascadeOnDelete();
    $table->boolean('helpful');
    $table->text('comment')->nullable();
    $table->string('session_id');
    $table->foreignId('user_id')->nullable()->constrained()->nullOnDelete();
    $table->timestamps();

    $table->unique(['article_id', 'session_id']);  // Один голос с сессии
});

// ArticleRatingController
public function store(Request $request, Article $article): JsonResponse
{
    $request->validate(['helpful' => 'required|boolean', 'comment' => 'nullable|string|max:500']);

    ArticleRating::updateOrCreate(
        ['article_id' => $article->id, 'session_id' => session()->getId()],
        ['helpful' => $request->helpful, 'comment' => $request->comment, 'user_id' => auth()->id()]
    );

    return response()->json(['success' => true]);
}

// Агрегат для отображения статистики
public function stats(Article $article): JsonResponse
{
    return response()->json([
        'helpful'     => $article->ratings()->where('helpful', true)->count(),
        'not_helpful' => $article->ratings()->where('helpful', false)->count(),
    ]);
}

Frontend

export function ArticleRating({ articleId }: { articleId: number }) {
  const [voted,   setVoted]   = useState<boolean | null>(null);
  const [comment, setComment] = useState('');
  const [showBox, setShowBox] = useState(false);

  const vote = async (helpful: boolean) => {
    setVoted(helpful);
    if (!helpful) setShowBox(true);  // Показываем поле комментария для негативных
    await fetch(`/api/articles/${articleId}/rating`, {
      method: 'POST',
      headers: { 'Content-Type': 'application/json' },
      body: JSON.stringify({ helpful }),
    });
  };

  const submitComment = async () => {
    await fetch(`/api/articles/${articleId}/rating`, {
      method: 'POST',
      headers: { 'Content-Type': 'application/json' },
      body: JSON.stringify({ helpful: false, comment }),
    });
    setShowBox(false);
  };

  if (voted === true) return <p className="text-sm text-green-600">Рады, что помогли!</p>;

  return (
    <div className="border-t pt-6 mt-8">
      {voted === null ? (
        <div className="flex items-center gap-4">
          <span className="text-sm text-gray-600">Была ли статья полезной?</span>
          <button onClick={() => vote(true)} className="text-sm px-3 py-1 rounded border hover:bg-green-50">👍 Да</button>
          <button onClick={() => vote(false)} className="text-sm px-3 py-1 rounded border hover:bg-red-50">👎 Нет</button>
        </div>
      ) : showBox ? (
        <div>
          <p className="text-sm mb-2">Что можно улучшить?</p>
          <textarea value={comment} onChange={e => setComment(e.target.value)}
            className="w-full border rounded p-2 text-sm h-24 resize-none" placeholder="Необязательно..." />
          <button onClick={submitComment} className="mt-2 text-sm bg-gray-800 text-white px-4 py-1.5 rounded">
            Отправить
          </button>
        </div>
      ) : null}
    </div>
  );
}

Сроки

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