Я написав більш ретельне та оновлене повідомлення в блозі на цю тему: http://elnur.pro/symfony-without-bundles/
Ні, не все повинно бути в комплекті. У вас може бути така структура:
src/Vendor/Model
- для моделей,
src/Vendor/Controller
- для контролерів,
src/Vendor/Service
- за послуги,
src/Vendor/Bundle
- для пачок, як src/Vendor/Bundle/AppBundle
,
- тощо.
Таким чином, ви б помістили AppBundle
єдиний матеріал, який дійсно є специфічним для Symfony2. Якщо ви вирішите пізніше перейти на інший фреймворк, ви позбудетесь Bundle
простору імен і заміните його вибраним фреймворком.
Зверніть увагу на те , що я пропоную тут для додатка певного коду. Для пакетів для багаторазового використання я все ж пропоную використовувати найкращі практики .
Збереження сутностей поза пакетами
Щоб зберегти об'єкти в src/Vendor/Model
зовнішній будь зв'язці, я змінив doctrine
розділ config.yml
з
doctrine:
# ...
orm:
# ...
auto_mapping: true
до
doctrine:
# ...
orm:
# ...
mappings:
model:
type: annotation
dir: %kernel.root_dir%/../src/Vendor/Model
prefix: Vendor\Model
alias: Model
is_bundle: false
Імена Entities s - для доступу з Доктрини сховищ - Почнемо з того, Model
в цьому випадку, наприклад, Model:User
.
Ви можете використовувати subnamespaces для пов'язаних груп осіб разом, наприклад, src/Vendor/User/Group.php
. У цьому випадку назва підприємства Model:User\Group
.
Збереження контролерів від пачок
Спочатку потрібно сказати JMSDiExtraBundle просканувати src
папку для служб, додавши це до config.yml
:
jms_di_extra:
locations:
directories: %kernel.root_dir%/../src
Потім ви визначаєте контролери як служби та розміщуєте їх під Controller
простором імен:
<?php
namespace Vendor\Controller;
use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\HttpFoundation\RedirectResponse;
use Sensio\Bundle\FrameworkExtraBundle\Configuration\Route;
use Sensio\Bundle\FrameworkExtraBundle\Configuration\Template;
use JMS\DiExtraBundle\Annotation\Service;
use JMS\DiExtraBundle\Annotation\InjectParams;
use JMS\SecurityExtraBundle\Annotation\Secure;
use Elnur\AbstractControllerBundle\AbstractController;
use Vendor\Service\UserService;
use Vendor\Model\User;
/**
* @Service("user_controller", parent="elnur.controller.abstract")
* @Route(service="user_controller")
*/
class UserController extends AbstractController
{
/**
* @var UserService
*/
private $userService;
/**
* @InjectParams
*
* @param UserService $userService
*/
public function __construct(UserService $userService)
{
$this->userService = $userService;
}
/**
* @Route("/user/add", name="user.add")
* @Template
* @Secure("ROLE_ADMIN")
*
* @param Request $request
* @return array
*/
public function addAction(Request $request)
{
$user = new User;
$form = $this->formFactory->create('user', $user);
if ($request->getMethod() == 'POST') {
$form->bind($request);
if ($form->isValid()) {
$this->userService->save($user);
$request->getSession()->getFlashBag()->add('success', 'user.add.success');
return new RedirectResponse($this->router->generate('user.list'));
}
}
return ['form' => $form->createView()];
}
/**
* @Route("/user/profile", name="user.profile")
* @Template
* @Secure("ROLE_USER")
*
* @param Request $request
* @return array
*/
public function profileAction(Request $request)
{
$user = $this->getCurrentUser();
$form = $this->formFactory->create('user_profile', $user);
if ($request->getMethod() == 'POST') {
$form->bind($request);
if ($form->isValid()) {
$this->userService->save($user);
$request->getSession()->getFlashBag()->add('success', 'user.profile.edit.success');
return new RedirectResponse($this->router->generate('user.view', [
'username' => $user->getUsername()
]));
}
}
return [
'form' => $form->createView(),
'user' => $user
];
}
}
Зауважте, що я використовую свій ElnurAb абстрактControllerBundle для спрощення визначення контролерів як служб.
Останнє, що залишилося - сказати Symfony шукати шаблони без розшарувань. Я роблю це, переосмислюючи службу відгадування шаблонів, але оскільки підхід відрізняється між Symfony 2.0 та 2.1, я надаю версії для обох.
Переосмислення відгадки шаблонів Symfony 2.1+
Я створив пакет, який робить це для вас.
Перезапис слухача шаблонів Symfony 2.0
Спочатку визначте клас:
<?php
namespace Vendor\Listener;
use InvalidArgumentException;
use Symfony\Bundle\FrameworkBundle\Templating\TemplateReference;
use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\HttpKernel\Bundle\Bundle;
use Sensio\Bundle\FrameworkExtraBundle\EventListener\TemplateListener as FrameworkExtraTemplateListener;
use JMS\DiExtraBundle\Annotation\Service;
class TemplateListener extends FrameworkExtraTemplateListener
{
/**
* @param array $controller
* @param Request $request
* @param string $engine
* @throws InvalidArgumentException
* @return TemplateReference
*/
public function guessTemplateName($controller, Request $request, $engine = 'twig')
{
if (!preg_match('/Controller\\\(.+)Controller$/', get_class($controller[0]), $matchController)) {
throw new InvalidArgumentException(sprintf('The "%s" class does not look like a controller class (it must be in a "Controller" sub-namespace and the class name must end with "Controller")', get_class($controller[0])));
}
if (!preg_match('/^(.+)Action$/', $controller[1], $matchAction)) {
throw new InvalidArgumentException(sprintf('The "%s" method does not look like an action method (it does not end with Action)', $controller[1]));
}
$bundle = $this->getBundleForClass(get_class($controller[0]));
return new TemplateReference(
$bundle ? $bundle->getName() : null,
$matchController[1],
$matchAction[1],
$request->getRequestFormat(),
$engine
);
}
/**
* @param string $class
* @return Bundle
*/
protected function getBundleForClass($class)
{
try {
return parent::getBundleForClass($class);
} catch (InvalidArgumentException $e) {
return null;
}
}
}
А потім скажіть Symfony використовувати його, додавши це до config.yml
:
parameters:
jms_di_extra.template_listener.class: Vendor\Listener\TemplateListener
Використання шаблонів без розшарувань
Тепер ви можете використовувати шаблони з пачок. Зберігайте їх під app/Resources/views
папкою. Наприклад, шаблони для цих двох дій з контролера прикладу, розташовані вище:
app/Resources/views/User/add.html.twig
app/Resources/views/User/profile.html.twig
Коли ви посилаєтесь на шаблон, просто опустіть частину групи:
{% include ':Controller:view.html.twig' %}