您好,我使用的是Symfony CMF 1.2,liip / imag-bundle 1.3,symfony-cmf / media-bundle 1.2。我想在我的块中添加2个额外的图像字段来扩展ImagineBlock,因为我上传的每个图像都会有一个移动和平板电脑版本的图像,这不是一个简单的调整大小,宽高比或什么都不相似。我不能在不影响图像质量的情况下裁剪/调整大小。
我的阻止
namespace xx\BlockBundle\Document;
use Doctrine\ODM\PHPCR\Mapping\Annotations as PHPCR;
use Symfony\Cmf\Bundle\BlockBundle\Doctrine\Phpcr\ImagineBlock;
use Symfony\Cmf\Bundle\MediaBundle\Doctrine\Phpcr\Image;
use Symfony\Cmf\Bundle\MediaBundle\ImageInterface;
use Symfony\Component\HttpFoundation\File\UploadedFile;
/**
* Class ClickableBlock
* @package xx\BlockBundle\Document
* @PHPCR\Document(referenceable=true)
*/
class ClickableBlock extends ImagineBlock
{
/**
* @PHPCR\Child(nodeName="image-mobile", cascade={"persist"})
* @var Image
*/
protected $imageMobile;
/**
* @PHPCR\Child(nodeName="image-tablet", cascade={"persist"})
* @var Image
*/
protected $imageTablet;
public function setIsPublishable($publishable)
{
$this->setPublishable($publishable);
}
/**
* @return Image
*/
public function getImageMobile()
{
return $this->imageMobile;
}
/**
* @return Image
*/
public function getImageTablet()
{
return $this->imageTablet;
}
/**
* Set the imageMobile for this block.
*
* @param ImageInterface|UploadedFile|null $image optional the imageMobile to update
* @return $this
* @throws \InvalidArgumentException If the $image parameter can not be handled.
*/
public function setImageMobile($image = null)
{
return $this->processImage($image, 'image-mobile', $this->imageMobile);
}
/**
* Set the imageTablet for this block.
*
* @param ImageInterface|UploadedFile|null $image optional the imageTablet to update
* @return $this
* @throws \InvalidArgumentException If the $image parameter can not be handled.
*/
public function setImageTablet($image = null)
{
return $this->processImage($image, 'image-tablet', $this->imageTablet);
}
/**
* @param ImageInterface|UploadedFile|null $image
* @param string $imageName
* @param Image $imageRef
* @return $this
*/
protected function processImage($image, $imageName, $imageRef)
{
if (!$image) {
return $this;
}
if (!$image instanceof ImageInterface && !$image instanceof UploadedFile) {
$type = is_object($image) ? get_class($image) : gettype($image);
throw new \InvalidArgumentException(sprintf(
'Image is not a valid type, "%s" given.',
$type
));
}
if ($imageRef) {
// existing imageTablet, only update content
$imageRef->copyContentFromFile($image);
} elseif ($image instanceof ImageInterface) {
$image->setName($imageName); // ensure document has right name
$imageRef = $image;
} else {
$imageRef = new Image();
$imageRef->copyContentFromFile($image);
}
return $this;
}
}
管理
namespace xx\BlockBundle\Admin;
use xx\BlockBundle\Document\ClickableBlock;
use xx\MainBundle\Form\Common\FormMapper as CommonFormMapper;
use Cocur\Slugify\Slugify;
use Sonata\AdminBundle\Datagrid\ListMapper;
use Sonata\AdminBundle\Form\FormMapper;
use Symfony\Cmf\Bundle\BlockBundle\Admin\Imagine\ImagineBlockAdmin;
class ClickableBlockAdmin extends ImagineBlockAdmin
{
/**
* {@inheritdoc}
*/
public function toString($object)
{
return $object instanceof ClickableBlock && $object->getLabel()
? $object->getLabel()
: parent::toString($object);
}
/**
* {@inheritdoc}
*/
public function prePersist($document)
{
parent::prePersist($document);
$this->InitialiseDocument($document);
}
/**
* @param $document
*/
private function InitialiseDocument(&$document)
{
$documentManager = $this->getModelManager();
$parentDocument = $documentManager->find(null, '/cms/xx/block');
$document->setParentDocument($parentDocument);
$slugifier = new Slugify();
$document->setName($slugifier->slugify($document->getLabel()));
}
/**
* {@inheritdoc}
*/
public function preUpdate($document)
{
parent::preUpdate($document);
$this->InitialiseDocument($document);
}
/**
* {@inheritdoc}
*/
protected function configureFormFields(FormMapper $formMapper)
{
parent::configureFormFields($formMapper);
if (null === $this->getParentFieldDescription()) {
$imageRequired = ($this->getSubject() && $this->getSubject()->getParentDocument()) ? false : true;
$formMapper
->with('form.group_general')
->remove('parentDocument')
->remove('filter')
->add('parentDocument', 'hidden', ['required' => false, 'data' => 'filler'])
->add('name', 'hidden', ['required' => false, 'data' => 'filler'])
->add('imageMobile', 'cmf_media_image', array('required' => $imageRequired))
->add('imageTablet', 'cmf_media_image', array('required' => $imageRequired))
->end();
// Append common fields to FormMapper
$commonFormMapper = new CommonFormMapper($formMapper);
$formMapper = $commonFormMapper->getPublishingFields();
}
}
}
注意我无法将服务容器注入此类(通过构造函数/方法),这就是为什么我现在使用硬编码的节点路径和实例化的Slugify类而不是使用它的服务。我也全力以赴解决这个问题。参考 -
xx.main.admin.pageadmin.container:
class: xx\MainBundle\Admin\PageAdmin
calls:
- [setContainer,[ @service_container ]]
# arguments: ["@service_container"]
图像字段上的注释基于我找到的以下配置
\vendor\symfony-cmf\block-bundle\Resources\config\doctrine-phpcr\ImagineBlock.phpcr.xml
:
<doctrine-mapping
xmlns="http://doctrine-project.org/schemas/phpcr-odm/phpcr-mapping"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://doctrine-project.org/schemas/phpcr-odm/phpcr-mapping
https://github.com/doctrine/phpcr-odm/raw/master/doctrine-phpcr-odm-mapping.xsd"
>
<document
name="Symfony\Cmf\Bundle\BlockBundle\Doctrine\Phpcr\ImagineBlock"
referenceable="true"
translator="attribute"
>
<node name="node"/>
<locale name="locale"/>
<field name="label" type="string" translated="true" nullable="true"/>
<field name="linkUrl" type="string" translated="true" nullable="true"/>
<field name="filter" type="string" nullable="true"/>
<child name="image" node-name="image">
<cascade>
<cascade-persist/>
</cascade>
</child>
</document>
</doctrine-mapping>
虽然默认的“图像”字段仍然正常,但是其他两个添加的图像字段没有被考虑,因为当我在prePersist上调试时,我看到两个字段都为空,而图像字段包含其上传的文件。
我尝试添加一个正常的文本字段,该字段在我的页面上正常保存和显示。
我在我的项目中使用YAML,所以我不确定给定的XML是如何转换的,如果它是正确的定义映射。
请帮忙。 :)
答案 0 :(得分:1)
一位同事发现了以下问题:
protected function processImage($image, $imageName, $imageRef)
应该是
protected function processImage($image, $imageName, &$imageRef)
$ imageRef未通过引用传递,使其始终为null。傻我。让我们希望这段代码至少能帮助其他人。 :)
答案 1 :(得分:0)
对于管理员问题:phpcr-odm管理员有一个value.parseHtml().select("p:contains(xyz)")
,正是为了你正在做的目的。您可以像这样添加到服务定义中:
rootPath
然后你做$ this-&gt; getRootPath()