我需要在我的symfony2应用程序中实现syliusOrderBundle,我已经从他们的官方网站http://docs.sylius.org/en/latest/bundles/SyliusOrderBundle/installation.html一遍又一遍地阅读了文档 我最终安装并启用了以下捆绑包
new Sylius\Bundle\ResourceBundle\SyliusResourceBundle(), new Sylius\Bundle\MoneyBundle\SyliusMoneyBundle(), new Sylius\Bundle\OrderBundle\SyliusOrderBundle(), new Sylius\Bundle\CartBundle\SyliusCartBundle(), new Sylius\Bundle\ProductBundle\SyliusProductBundle(), new Sylius\Bundle\ArchetypeBundle\SyliusArchetypeBundle(), new Sylius\Bundle\AttributeBundle\SyliusAttributeBundle(), new Sylius\Bundle\AssociationBundle\SyliusAssociationBundle(), new Sylius\Bundle\VariationBundle\SyliusVariationBundle(),
问题是由于安全原因安装不必要的软件包并不舒服,并且sylius doc没有帮助。我需要的是能够将产品添加到订单中,如果您已经使用过它可以帮助您。致谢
答案 0 :(得分:4)
首先,您需要的只是sylius/order-bundle
版本0.17(在撰写本文时)
$ composer require sylius/order-bundle "^0.17"
然后将必要的捆绑包添加到app/AppKernel.php
:
public function registerBundles()
{
$bundles = [
// Bundles you've already registered go here.
// The following bundles are dependencies of Sylius ResourceBundle.
new FOS\RestBundle\FOSRestBundle(),
new JMS\SerializerBundle\JMSSerializerBundle($this),
new Stof\DoctrineExtensionsBundle\StofDoctrineExtensionsBundle(),
new WhiteOctober\PagerfantaBundle\WhiteOctoberPagerfantaBundle(),
// The following Sylius bundles are dependencies of Sylius OrderBundle
new Sylius\Bundle\ResourceBundle\SyliusResourceBundle(),
new Sylius\Bundle\MoneyBundle\SyliusMoneyBundle(),
new Sylius\Bundle\SequenceBundle\SyliusSequenceBundle(),
new Sylius\Bundle\OrderBundle\SyliusOrderBundle(),
// Doctrine bundle MUST be the last bundle registered.
new Doctrine\Bundle\DoctrineBundle\DoctrineBundle(),
];
//....
return $bundles;
}
现在要将OrderBundle与您自己的实体一起使用,您需要创建一个 Order 和 OrderItem 实体,以扩展Sylius提供的实体。
对于Order
,我们会添加一个字段来捕获访问者的电子邮件地址。您可以捕获当前用户或您喜欢的任何内容,以确定谁下订单。
<?php
namespace AppBundle\Entity;
use Sylius\Component\Order\Model\Order as SyliusOrder;
class Order extends SyliusOrder
{
/**
* @var string
*/
private $email;
/**
* @return string
*/
public function getEmail()
{
return $this->email;
}
/**
* @param string $email
*/
public function setEmail($email)
{
$this->email = $email;
}
}
出于演示目的,我们假设访问者希望订购音乐曲目以供下载。
否则,这可以是任何事情。在Sylius应用程序中,它是一个产品。在某些应用中,它可能不是产品,而是订阅或服务等。
因此对于OrderItem
,我们会添加一个字段来捕获访问者要下载的下载内容。
<?php
namespace AppBundle\Entity;
use Sylius\Component\Order\Model\OrderItem as SyliusOrderItem;
class OrderItem extends SyliusOrderItem
{
/**
* @var Download
*/
private $download;
/**
* @return Download
*/
public function getDownload()
{
return $this->download;
}
/**
* @param Download $download
*/
public function setDownload(Download $download)
{
$this->download = $download;
}
}
现在您拥有实体,可以添加Doctrine映射。在这个例子中,我们使用XML(丑陋但它验证配置)但它可以是YAML或注释。
在src/AppBundle/Resources/config/doctrine/Order.orm.xml
:
<?xml version="1.0" encoding="UTF-8"?>
<doctrine-mapping xmlns="http://doctrine-project.org/schemas/orm/doctrine-mapping"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://doctrine-project.org/schemas/orm/doctrine-mapping
http://doctrine-project.org/schemas/orm/doctrine-mapping.xsd">
<entity name="AppBundle\Entity\Order" table="app_order">
<field name="email" column="email" type="string" nullable="true" />
<one-to-many field="items" target-entity="Sylius\Component\Order\Model\OrderItemInterface" mapped-by="order" orphan-removal="true">
<cascade>
<cascade-all/>
</cascade>
</one-to-many>
<one-to-many field="adjustments" target-entity="Sylius\Component\Order\Model\AdjustmentInterface" mapped-by="order" orphan-removal="true">
<cascade>
<cascade-all/>
</cascade>
</one-to-many>
</entity>
在src/AppBundle/Resources/config/doctrine/OrderItem.orm.xml
:
<?xml version="1.0" encoding="UTF-8"?>
<doctrine-mapping xmlns="http://doctrine-project.org/schemas/orm/doctrine-mapping"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://doctrine-project.org/schemas/orm/doctrine-mapping
http://doctrine-project.org/schemas/orm/doctrine-mapping.xsd">
<entity name="AppBundle\Entity\OrderItem" table="app_order_item">
<many-to-one field="download" target-entity="AppBundle\Entity\Download">
<join-column name="download_id" referenced-column-name="id" nullable="false" />
</many-to-one>
</entity>
</doctrine-mapping>
然后我们需要让Sylius OrderBundle知道我们的新实体,以便它不会使用它的默认实体。
我们还希望为Order
实体生成订单号,这是由Sylius SequenceBundle完成的。
在app/config/config.yml
中,添加以下配置:
sylius_sequence:
generators:
AppBundle\Entity\Order: sylius.sequence.sequential_number_generator
sylius_order:
resources:
order:
classes:
model: AppBundle\Entity\Order
order_item:
classes:
model: AppBundle\Entity\OrderItem
然后更新数据库架构:
$ php app/console doctrine:schema:update --dump-sql --force
现在,当持久保存新实体时,为了实际生成订单号并将其分配给Order
,我们需要注册一个监听器。
在您的论坛app/config/services.yml
中的Resources/config/services.yml
或,添加此配置:
parameters:
sylius.model.sequence.class: Sylius\Component\Sequence\Model\Sequence
services:
app.order_number_listener:
class: Sylius\Bundle\OrderBundle\EventListener\OrderNumberListener
arguments:
- "@sylius.sequence.doctrine_number_listener"
tags:
- { name: kernel.event_listener, event: app.download_ordered, method: generateOrderNumber }
事件app.download_ordered
的名称在这里很重要,您可以根据需要命名,但必须在创建新订单时调度。
以下是创建新订单的示例,我们将在其中发送app.download_ordered
事件。
use AppBundle\Entity\Download;
use AppBundle\Entity\OrderItem;
use AppBundle\Form\OrderType;
use Sensio\Bundle\FrameworkExtraBundle\Configuration as Framework;
use AppBundle\Entity\Order;
use Symfony\Bundle\FrameworkBundle\Controller\Controller;
use Symfony\Component\EventDispatcher\GenericEvent;
use Symfony\Component\HttpFoundation\Request;
const STATE_BEGIN = 'begin';
const STATE_COMPLETE = 'complete';
/**
* @Framework\Route("/begin-order-for-download/{id}", name="begin_order_for_download")
*/
public function beginOrderForDownloadAction(Request $request, $id)
{
$download = $this->getDoctrine()->getRepository(Download::class)->findOneBy(['id' => $id]);
$order = new Order();
$form = $this->createForm(new OrderType(), $order);
if ('POST' === $request->getMethod()) {
$order->setState(self::STATE_BEGIN);
$orderItem = new OrderItem();
$orderItem->setDownload($download);
$orderItem->setOrder($order);
$orderItem->setUnitPrice(59);
// $orderItem->setImmutable(true); // Need to verify how this affects behavior.
$this->get('event_dispatcher')->dispatch('app.download_ordered', new GenericEvent($order));
$form->handleRequest($request);
$em = $this->getDoctrine()->getManager();
if ($form->isValid()) {
$order->setState(self::STATE_COMPLETE);
$this->addFlash('order.state', self::STATE_COMPLETE);
$em->persist($order);
$em->flush();
return $this->redirectToRoute('complete_order', [
'id' => $order->getId(),
]);
}
}
return $this->render('AppBundle::begin_order_for_download.html.twig', [
'form' => $form->createView(),
'download' => $download,
]);
}
OrderType
表单看起来像这样,为了演示目的而保持简单。根据您的需求进行编辑。
<?php
namespace AppBundle\Form;
use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\FormBuilderInterface;
class OrderType extends AbstractType
{
/**
* {@inheritdoc}
*/
public function getName()
{
return 'order';
}
/**
* {@inheritdoc}
*/
public function buildForm(FormBuilderInterface $builder, array $options)
{
$builder->add('email', 'repeated', [
'type' => 'email',
'first_options' => ['label' => 'Email'],
'second_options' => ['label' => 'Repeat Email'],
'invalid_message' => 'The email fields must match.',
]);
$builder->add('proceed', 'submit', ['attr' => ['class' => 'button']]);
}
}
如果表单有效,我们的订单将被保留,并且听众会为其分配一个订单号,然后我们将重新设置complete_order
路由。在这里,您可以执行支付付款等操作,或者在这种情况下,提供下载歌曲的链接。
那就是它。您可以在https://github.com/adamelso/orda
查看完整工作示例的代码