Реализация формы запроса функции (Feature Request) на сайте

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

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

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

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

Предлагаемые услуги
Показано 1 из 1 услугВсе 2065 услуг
Реализация формы запроса функции (Feature Request) на сайте
Средняя
~2-3 рабочих дня
Часто задаваемые вопросы

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

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

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

  • 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

Реализация формы запроса функции (Feature Request) на сайте

Feature Request форма — это не просто «поле для текста». Хорошо реализованная форма структурирует запросы пользователей: отделяет описание проблемы от предложенного решения, собирает контекст (кто просит, насколько часто сталкивается с проблемой), и позволяет команде приоритизировать беклог без телефонных звонков.

Структура формы

Минимальный набор полей, который даёт полезный сигнал:

  • Заголовок запроса (коротко, суть)
  • Описание проблемы (какую задачу пытаетесь решить — не «добавьте кнопку», а зачем)
  • Предложенное решение (опционально)
  • Категория / область продукта
  • Оценка важности (насколько часто встречаете проблему)
// FeatureRequestForm.tsx (React + React Hook Form + Zod)
import { useForm, Controller } from 'react-hook-form';
import { zodResolver } from '@hookform/resolvers/zod';
import { z } from 'zod';

const schema = z.object({
  title: z
    .string()
    .min(10, 'Минимум 10 символов')
    .max(120, 'Максимум 120 символов'),
  problem: z
    .string()
    .min(30, 'Опишите проблему подробнее')
    .max(2000),
  solution: z.string().max(2000).optional(),
  category: z.enum(['ui-ux', 'performance', 'integrations', 'api', 'other']),
  importance: z.enum(['critical', 'high', 'medium', 'low']),
  email: z.string().email().optional().or(z.literal('')),
});

type FormData = z.infer<typeof schema>;

const CATEGORIES = [
  { value: 'ui-ux', label: 'Интерфейс / UX' },
  { value: 'performance', label: 'Производительность' },
  { value: 'integrations', label: 'Интеграции' },
  { value: 'api', label: 'API / разработчикам' },
  { value: 'other', label: 'Другое' },
] as const;

const IMPORTANCE = [
  { value: 'critical', label: 'Критично — не могу работать без этого' },
  { value: 'high', label: 'Высокая — сталкиваюсь каждый день' },
  { value: 'medium', label: 'Средняя — неудобно, но терпимо' },
  { value: 'low', label: 'Низкая — было бы приятно иметь' },
] as const;

export function FeatureRequestForm() {
  const {
    register,
    control,
    handleSubmit,
    formState: { errors, isSubmitting, isSubmitSuccessful },
    reset,
    watch,
  } = useForm<FormData>({
    resolver: zodResolver(schema),
    defaultValues: {
      category: 'other',
      importance: 'medium',
    },
  });

  const titleValue = watch('title', '');

  const onSubmit = async (data: FormData) => {
    const res = await fetch('/api/feature-requests', {
      method: 'POST',
      headers: { 'Content-Type': 'application/json' },
      body: JSON.stringify({
        ...data,
        submittedAt: new Date().toISOString(),
        pageUrl: window.location.href,
      }),
    });

    if (!res.ok) {
      const err = await res.json();
      throw new Error(err.message ?? 'Ошибка при отправке');
    }
  };

  if (isSubmitSuccessful) {
    return (
      <div className="rounded-lg border border-green-200 bg-green-50 p-6 text-center">
        <p className="text-lg font-semibold text-green-800">Запрос отправлен</p>
        <p className="mt-2 text-sm text-green-700">
          Мы рассмотрим его при планировании следующего релиза.
        </p>
        <button
          onClick={() => reset()}
          className="mt-4 text-sm text-green-700 underline"
        >
          Отправить ещё один
        </button>
      </div>
    );
  }

  return (
    <form onSubmit={handleSubmit(onSubmit)} className="space-y-5 max-w-xl">
      {/* Заголовок */}
      <div>
        <label className="block text-sm font-medium mb-1">
          Кратко опишите запрос
          <span className="text-gray-400 ml-1 font-normal">
            ({titleValue.length}/120)
          </span>
        </label>
        <input
          {...register('title')}
          type="text"
          placeholder="Например: Экспорт данных в CSV"
          className="w-full border rounded-md px-3 py-2 text-sm focus:outline-none focus:ring-2 focus:ring-blue-500"
        />
        {errors.title && (
          <p className="mt-1 text-xs text-red-600">{errors.title.message}</p>
        )}
      </div>

      {/* Описание проблемы */}
      <div>
        <label className="block text-sm font-medium mb-1">
          Какую проблему это решает?
        </label>
        <p className="text-xs text-gray-500 mb-1">
          Опишите ситуацию, а не конкретное решение — это поможет нам найти лучший подход
        </p>
        <textarea
          {...register('problem')}
          rows={4}
          placeholder="Когда я пытаюсь делать X, мне приходится Y, что неудобно потому что..."
          className="w-full border rounded-md px-3 py-2 text-sm resize-y focus:outline-none focus:ring-2 focus:ring-blue-500"
        />
        {errors.problem && (
          <p className="mt-1 text-xs text-red-600">{errors.problem.message}</p>
        )}
      </div>

      {/* Предложенное решение */}
      <div>
        <label className="block text-sm font-medium mb-1">
          Как бы вы это реализовали? <span className="text-gray-400">(опционально)</span>
        </label>
        <textarea
          {...register('solution')}
          rows={3}
          placeholder="Добавьте кнопку «Экспорт» в меню таблицы, которая скачивает..."
          className="w-full border rounded-md px-3 py-2 text-sm resize-y focus:outline-none focus:ring-2 focus:ring-blue-500"
        />
      </div>

      {/* Категория */}
      <div>
        <label className="block text-sm font-medium mb-2">Область продукта</label>
        <Controller
          control={control}
          name="category"
          render={({ field }) => (
            <div className="flex flex-wrap gap-2">
              {CATEGORIES.map(cat => (
                <button
                  key={cat.value}
                  type="button"
                  onClick={() => field.onChange(cat.value)}
                  className={`px-3 py-1.5 rounded-full text-xs border transition-colors ${
                    field.value === cat.value
                      ? 'bg-blue-600 border-blue-600 text-white'
                      : 'border-gray-300 hover:border-blue-400'
                  }`}
                >
                  {cat.label}
                </button>
              ))}
            </div>
          )}
        />
      </div>

      {/* Важность */}
      <div>
        <label className="block text-sm font-medium mb-2">Насколько это важно для вас?</label>
        <div className="space-y-2">
          {IMPORTANCE.map(item => (
            <label key={item.value} className="flex items-start gap-2 cursor-pointer">
              <input
                {...register('importance')}
                type="radio"
                value={item.value}
                className="mt-0.5"
              />
              <span className="text-sm">{item.label}</span>
            </label>
          ))}
        </div>
      </div>

      {/* Email */}
      <div>
        <label className="block text-sm font-medium mb-1">
          Email <span className="text-gray-400">(чтобы уведомить вас о реализации)</span>
        </label>
        <input
          {...register('email')}
          type="email"
          placeholder="[email protected]"
          className="w-full border rounded-md px-3 py-2 text-sm focus:outline-none focus:ring-2 focus:ring-blue-500"
        />
        {errors.email && (
          <p className="mt-1 text-xs text-red-600">{errors.email.message}</p>
        )}
      </div>

      <button
        type="submit"
        disabled={isSubmitting}
        className="w-full bg-blue-600 hover:bg-blue-700 disabled:opacity-50 text-white font-medium py-2.5 rounded-md text-sm transition-colors"
      >
        {isSubmitting ? 'Отправляю...' : 'Отправить запрос'}
      </button>
    </form>
  );
}

API endpoint

// pages/api/feature-requests.ts (Next.js) или routes/feature-requests.ts (Express)
import type { NextApiRequest, NextApiResponse } from 'next';

export default async function handler(req: NextApiRequest, res: NextApiResponse) {
  if (req.method !== 'POST') return res.status(405).end();

  const { title, problem, solution, category, importance, email, pageUrl } = req.body;

  // Базовая валидация
  if (!title || !problem || !category || !importance) {
    return res.status(400).json({ message: 'Обязательные поля не заполнены' });
  }

  const record = await db.featureRequest.create({
    data: {
      title,
      problem,
      solution: solution || null,
      category,
      importance,
      email: email || null,
      pageUrl,
      status: 'new',
      votes: 0,
    },
  });

  // Уведомление в Linear/Jira/Notion через webhook
  if (process.env.LINEAR_WEBHOOK_URL) {
    await fetch(process.env.LINEAR_WEBHOOK_URL, {
      method: 'POST',
      headers: { 'Content-Type': 'application/json' },
      body: JSON.stringify({
        title: `[Feature] ${title}`,
        description: `**Проблема:**\n${problem}\n\n**Решение:**\n${solution ?? 'не указано'}`,
        priority: importance === 'critical' ? 1 : importance === 'high' ? 2 : 3,
        labelIds: [CATEGORY_LABEL_MAP[category]],
      }),
    });
  }

  return res.status(201).json({ id: record.id });
}

Интеграция с системой голосования

Если планируется добавить упдоуты к запросам — форму стоит сразу проектировать с уникальным ID записи и страницей /roadmap или /feature-requests, где пользователи видят и голосуют за уже существующие запросы. Это позволяет избежать дублей и собирать реальные сигналы приоритетности.

Сроки

Форма с валидацией, API и уведомлениями — два-три дня. Добавление дедупликации (поиск похожих запросов перед отправкой через простой text search), страница со списком запросов и публичное голосование — ещё три-пять дней.