Реализация roadmap-страницы (публичная дорожная карта) на сайте

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

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

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

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

Предлагаемые услуги
Показано 1 из 1 услугВсе 2065 услуг
Реализация roadmap-страницы (публичная дорожная карта) на сайте
Средняя
~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

Разработка публичной страницы roadmap

Публичный roadmap показывает пользователям, что команда планирует сделать, что в работе и что уже выполнено. Это инструмент доверия и удержания — пользователи видят, что их запросы услышаны. Реализуется как статичная страница с данными из Notion, Linear, Jira или собственной БД.

Источники данных

Вариант 1: Notion как CMS для roadmap

// lib/roadmap.ts
import { Client } from '@notionhq/client';

const notion = new Client({ auth: process.env.NOTION_TOKEN });

export interface RoadmapItem {
  id:          string;
  title:       string;
  status:      'planned' | 'in_progress' | 'done';
  quarter:     string;  // 'Q1 2025'
  category:    string;
  votes:       number;
  description: string;
}

export async function getRoadmap(): Promise<RoadmapItem[]> {
  const response = await notion.databases.query({
    database_id: process.env.NOTION_ROADMAP_DB_ID!,
    filter: {
      property: 'Public',
      checkbox: { equals: true },
    },
    sorts: [{ property: 'Quarter', direction: 'ascending' }],
  });

  return response.results.map(page => ({
    id:          page.id,
    title:       page.properties.Name.title[0]?.plain_text ?? '',
    status:      page.properties.Status.select?.name?.toLowerCase().replace(' ', '_') as any,
    quarter:     page.properties.Quarter.select?.name ?? '',
    category:    page.properties.Category.select?.name ?? '',
    votes:       page.properties.Votes.number ?? 0,
    description: page.properties.Description.rich_text[0]?.plain_text ?? '',
  }));
}

Вариант 2: собственная таблица

// Модель RoadmapItem
Schema::create('roadmap_items', function (Blueprint $table) {
    $table->id();
    $table->string('title');
    $table->text('description')->nullable();
    $table->enum('status', ['planned', 'in_progress', 'done', 'cancelled'])->default('planned');
    $table->string('quarter')->nullable();   // 'Q2 2025'
    $table->string('category')->nullable();  // 'Performance', 'Mobile', 'API'
    $table->integer('sort_order')->default(0);
    $table->boolean('is_public')->default(true);
    $table->timestamps();
});

Frontend: Kanban-стиль roadmap

// components/Roadmap.tsx
const STATUS_COLUMNS = [
  { key: 'planned',     label: 'Запланировано', color: 'bg-gray-100' },
  { key: 'in_progress', label: 'В работе',       color: 'bg-blue-100' },
  { key: 'done',        label: 'Готово',          color: 'bg-green-100' },
];

export function RoadmapBoard({ items }: { items: RoadmapItem[] }) {
  const grouped = STATUS_COLUMNS.map(col => ({
    ...col,
    items: items.filter(i => i.status === col.key),
  }));

  return (
    <div className="grid grid-cols-1 md:grid-cols-3 gap-6">
      {grouped.map(col => (
        <div key={col.key}>
          <h3 className={`font-semibold px-3 py-2 rounded-t ${col.color}`}>
            {col.label} <span className="text-gray-500 font-normal">({col.items.length})</span>
          </h3>
          <div className="space-y-3 mt-3">
            {col.items.map(item => (
              <RoadmapCard key={item.id} item={item} />
            ))}
          </div>
        </div>
      ))}
    </div>
  );
}

function RoadmapCard({ item }: { item: RoadmapItem }) {
  return (
    <div className="bg-white border rounded-lg p-4 shadow-sm">
      {item.category && (
        <span className="text-xs bg-indigo-100 text-indigo-700 px-2 py-0.5 rounded-full">{item.category}</span>
      )}
      <h4 className="font-medium mt-2">{item.title}</h4>
      {item.description && <p className="text-sm text-gray-500 mt-1">{item.description}</p>}
      {item.quarter && <p className="text-xs text-gray-400 mt-2">{item.quarter}</p>}
    </div>
  );
}

Фильтрация и поиск

// Клиентский фильтр без перезагрузки страницы
const [filter, setFilter] = useState<string>('all');
const [search, setSearch]  = useState('');

const visible = items.filter(item =>
  (filter === 'all' || item.category === filter) &&
  (search === '' || item.title.toLowerCase().includes(search.toLowerCase()))
);

ISR-кэширование для Next.js

// pages/roadmap.tsx
export const getStaticProps: GetStaticProps = async () => {
  const items = await getRoadmap();
  return {
    props: { items },
    revalidate: 3600,  // Перегенерация раз в час
  };
};

Уведомление подписчиков об обновлениях

При смене статуса на done можно автоматически отправлять письма пользователям, подписавшимся на эту фичу:

public function handle(RoadmapItemStatusChanged $event): void
{
    if ($event->item->status !== 'done') return;

    $subscribers = $event->item->subscribers()->with('user')->get();
    foreach ($subscribers as $sub) {
        Mail::to($sub->user->email)->queue(new RoadmapItemCompletedMail($event->item));
    }
}

Сроки

Публичный roadmap с Notion-бэкендом или собственной БД, фильтрами и ISR-кэшем: 3–4 рабочих дня.