Разработка кастомного модуля PrestaShop

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

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

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

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

Предлагаемые услуги
Показано 1 из 1 услугВсе 2065 услуг
Разработка кастомного модуля PrestaShop
Средняя
~3-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

Разработка кастомного модуля PrestaShop

Модуль — основная единица расширения PrestaShop. Через модули реализуется любая кастомная логика: от кастомных полей заказа до интеграций с внешними API. Модуль может регистрировать хуки, добавлять контроллеры, расширять схему БД, добавлять виджеты в back-office через Symfony DI.

Структура модуля

modules/mymodule/
├── mymodule.php             # Главный класс модуля
├── config.xml               # Метаданные (опционально)
├── composer.json            # Зависимости (опционально)
├── logo.png                 # Иконка 32x32
├── controllers/
│   └── front/
│       └── callback.php     # FrontController: /module/mymodule/callback
├── src/                     # PSR-4 namespace классы
│   ├── Entity/
│   ├── Repository/
│   └── Service/
├── views/
│   └── templates/
│       ├── admin/
│       │   └── configure.html.twig
│       └── hook/
│           └── display_banner.tpl
├── translations/            # Переводы модуля
└── upgrade/                 # SQL-миграции при обновлении
    └── upgrade-1.1.0.php

Главный класс и жизненный цикл

<?php
declare(strict_types=1);

use PrestaShop\PrestaShop\Adapter\SymfonyContainer;

if (!defined('_PS_VERSION_')) {
    exit;
}

class MyModule extends Module
{
    public function __construct()
    {
        $this->name    = 'mymodule';
        $this->tab     = 'other';
        $this->version = '1.2.0';
        $this->author  = 'Company Name';
        $this->need_instance     = 0;
        $this->bootstrap         = true;
        $this->ps_versions_compliancy = ['min' => '8.0.0', 'max' => _PS_VERSION_];

        parent::__construct();

        $this->displayName = $this->trans('My Module', [], 'Modules.Mymodule.Admin');
        $this->description = $this->trans('Module description', [], 'Modules.Mymodule.Admin');
    }

    public function install(): bool
    {
        return parent::install()
            && $this->installDb()
            && $this->registerHook('actionOrderStatusPostUpdate')
            && $this->registerHook('displayCustomerAccount')
            && $this->registerHook('displayHeader');
    }

    public function uninstall(): bool
    {
        return parent::uninstall() && $this->uninstallDb();
    }

    private function installDb(): bool
    {
        $sql = 'CREATE TABLE IF NOT EXISTS `' . _DB_PREFIX_ . 'mymodule_data` (
            `id_mymodule` INT UNSIGNED NOT NULL AUTO_INCREMENT,
            `id_order`    INT UNSIGNED NOT NULL,
            `payload`     TEXT,
            `status`      TINYINT(1) NOT NULL DEFAULT 0,
            `date_add`    DATETIME NOT NULL,
            PRIMARY KEY (`id_mymodule`),
            KEY `id_order` (`id_order`)
        ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;';

        return Db::getInstance()->execute($sql);
    }
}

Работа с хуками

Хуки бывают двух типов: action (изменяют поведение, возвращают данные) и display (возвращают HTML). Важно различать:

// Display hook — возвращает HTML строку
public function hookDisplayCustomerAccount(array $params): string
{
    $customerId = (int) $this->context->customer->id;
    $data = $this->getCustomerData($customerId);

    if (empty($data)) {
        return '';
    }

    $this->smarty->assign([
        'mymodule_data' => $data,
        'moduleUrl'     => $this->context->link->getModuleLink('mymodule', 'account'),
    ]);

    return $this->display(__FILE__, 'views/templates/hook/customer_account.tpl');
}

// Action hook — модифицирует данные или выполняет side-effect
public function hookActionOrderStatusPostUpdate(array $params): void
{
    /** @var Order $order */
    $order  = $params['order'];
    $newStatus = $params['newOrderStatus'];

    if ((int) $newStatus->id === (int) Configuration::get('PS_OS_PAYMENT')) {
        $this->notifyExternalSystem($order);
    }
}

Symfony DI и современный back-office

PrestaShop 8.x позволяет использовать Symfony DI в модулях через config/services.yml:

# modules/mymodule/config/services.yml
services:
  _defaults:
    autowire: true
    autoconfigure: true
    public: true

  MyModule\Repository\OrderDataRepository:
    arguments:
      $connection: '@doctrine.dbal.default_connection'

  MyModule\Service\ExternalApiService:
    arguments:
      $apiKey: '%env(MYMODULE_API_KEY)%'
      $logger: '@logger'

Использование в контроллере back-office (Symfony-style):

// src/Controller/Admin/OrderDataController.php
namespace MyModule\Controller\Admin;

use PrestaShopBundle\Controller\Admin\FrameworkBundleAdminController;
use Symfony\Component\HttpFoundation\JsonResponse;
use MyModule\Repository\OrderDataRepository;

class OrderDataController extends FrameworkBundleAdminController
{
    public function __construct(
        private readonly OrderDataRepository $repository
    ) {}

    public function indexAction(int $orderId): JsonResponse
    {
        $data = $this->repository->findByOrderId($orderId);

        return $this->json([
            'success' => true,
            'data'    => $data,
        ]);
    }
}

Миграции схемы БД

При обновлении модуля используйте upgrade-скрипты:

// modules/mymodule/upgrade/upgrade-1.2.0.php
function upgrade_module_1_2_0(Module $module): bool
{
    $result = Db::getInstance()->execute(
        'ALTER TABLE `' . _DB_PREFIX_ . 'mymodule_data`
         ADD COLUMN `external_id` VARCHAR(128) NULL AFTER `id_order`,
         ADD INDEX `external_id` (`external_id`)'
    );

    if ($result) {
        // Миграция данных
        Db::getInstance()->execute(
            'UPDATE `' . _DB_PREFIX_ . 'mymodule_data` SET `status` = 1 WHERE `status` = 0 AND `date_add` < NOW()'
        );
    }

    return (bool) $result;
}

Локализация модуля

PrestaShop использует Symfony Translation для back-office и собственный механизм для front-office:

// Back-office (Symfony Translation)
$this->trans('Order processed', [], 'Modules.Mymodule.Admin');

// Front-office (Smarty)
// В .tpl файле:
// {l s='Order processed' mod='mymodule'}

// Генерация файлов переводов
php bin/console prestashop:generate:translations --module=mymodule

Тестирование модуля

// tests/Unit/Service/ExternalApiServiceTest.php
use PHPUnit\Framework\TestCase;
use MyModule\Service\ExternalApiService;

class ExternalApiServiceTest extends TestCase
{
    public function testNotifyOrder(): void
    {
        $httpClient = $this->createMock(HttpClientInterface::class);
        $httpClient->expects($this->once())
            ->method('request')
            ->with('POST', 'https://api.example.com/orders')
            ->willReturn(new MockResponse('{"status":"ok"}', ['http_code' => 200]));

        $service = new ExternalApiService($httpClient, 'test-api-key');
        $result  = $service->notifyOrderPaid(12345);

        $this->assertTrue($result);
    }
}

Сроки разработки

  • Простой модуль: хуки, конфигурация, вывод данных: 2–4 дня
  • Интеграция с внешним API (платёжная система, CRM): 5–10 дней
  • Модуль с кастомными сущностями, CRUD в back-office, миграциями: 7–14 дней
  • Сложный модуль (кастомный checkout, loyalty-программа): 3–6 недель