Zend Validation Db_NoRecordExists и исключить опцию - PullRequest
7 голосов
/ 21 марта 2011

Я пытаюсь использовать опцию «exclude» для валидатора Db_NoRecordExists, потому что, когда я «редактирую» элемент, он всегда возвращает мне «дублированную» ошибку, как обычно.

То, к чему я стремлюсь, это сказать форме, чтобы она хранила значение, переданное самой форме из контроллера ...

Это контроллер:

public function editAction()
{
$id = $this->getRequest()->getParam('id');
$pagesMapper = new Application_Model_PagesMapper();
$form = new Application_Form_PageEdit();
$form->populate($pagesMapper->fetchId($id, true));
if ($this->getRequest()->isPost()) {
    if ($form->isValid($this->getRequest()->getPost())) {
        //... cut ...
    }
}
$this->view->form = $form;
}

Это форма:

class Application_Form_PageEdit extends Zend_Form
{
public function init()
{
$commonFilters      = array('StringTrim');
$commonValidators = array('NotEmpty');
    $this->setMethod('post')->setAction('/admin-page/edit');

$id = new Zend_Form_Element_Hidden('id');
$pid = new Zend_Form_Element_Hidden('pid');

$keyname = new Zend_Form_Element_Text('keyname');
$keyname->setLabel('Keyname')
    ->setRequired(true)
    ->addFilters($commonFilters)
    ->addFilter('StringToLower')
    ->addFilter('Word_SeparatorToDash')
    ->addValidator('Db_NoRecordExists', false, array(
        'table'     => 'pages',
        'field'     => 'keyname',
        'exclude'   => array(
            'field' => 'id',
            'value' => $this->getValue('id)
        )
        )
    );

// ... вырезать ...

Несколько советов?

Ответы [ 6 ]

17 голосов
/ 21 марта 2011

У меня была похожая проблема. Моим решением было перевести валидатор из функции init в функцию isValid.

public function isValid($data)
{
  $this->getElement('keyname')
       ->addValidator(
           'Db_NoRecordExists',
           false,
           array(
               'table'     => 'pages',
               'field'     => 'keyname',
               'exclude'   => array(
                   'field' => 'id',
                   'value' => $this->getValue('id')
               )
           )
       );
  return parent::isValid($data);
}
2 голосов
/ 28 октября 2011

У меня отлично работает только это решение:

public function isValid($data)
{
    $clause    = $this->_dbAdapter->quoteInto('id = ?', $this->getValue('id'));
    $this->getElement('keyname')
        ->addValidator('Db_NoRecordExists', false, array(
            'adapter'   =>  $this->_dbAdapter,
            'table'     => 'employee_name',
            'field'     => 'name',
            'exclude'   => $clause
        )
    );
    return parent::isValid($data);
}
1 голос
/ 25 сентября 2016

Мне не понравилось решение переопределить функцию isValid () . Такое ощущение, что мы не понимаем, как работает валидатор Db_NoRecordExists . Вам просто нужно предоставить валидатору фактический идентификатор записи, который необходимо исключить до вызова isValid () . Валидатор даже предоставляет средство доступа для установки этого значения прямо внутри себя! Инструкции Zend действительно помогают не так хорошо, как могли бы, поэтому неудивительно, что люди борются с этим.

Вот более элегантный и ориентированный на рабочий процесс способ установки исключения без взлома, расширения или переопределения формы:

// $post is assumed to have our posted form variables // 
// $form is assumed to be our form that is already setup with validators. //

$form->getElement('username')->
        getValidator('Db_NoRecordExists')->
        setExclude([
            'field' => 'user_id',
            'value' => $post['user_id'],
        ]);

if ( $form->isValid( $post ) ) {
    // do valid stuff
    // ...
}
else {
    // do invalid stuff
    // ...
}

Исключением является массив наших полей и значений. Мы просто сообщаем валидатору, что нужно исключить непосредственно перед вызовом isValid () . Готово. Простой.

1 голос
/ 31 января 2014

Что я делаю, перед логикой проверки в контроллере я добавляю следующий код для проверки поля с текущими данными пользователя. Если это совпадение, я просто удаляю валидатор из элемента следующим образом:

    if(($this->getRequest()->isPost()) && $request->getParam('email')==$this->user->getEmail()) {
        $form_preferences->getElement('email')->removeValidator('Db_NoRecordExists');
    }

В этот момент логика isValid пройдет без проблем.

if(($this->getRequest()->isPost()) && $form_preferences->isValid($_POST)) {

Наслаждайтесь!

0 голосов
/ 29 января 2013

Нет необходимости создавать валидатор в функции isValid, просто обновите предложение exclude:

public function isValid($data)
{
    $this->getElement('name')->getValidator('Db_NoRecordExists')->setExclude('id != ' . $data['id']);
    return parent::isValid($data);
}
0 голосов
/ 10 августа 2012

Внесите изменения в форму:

class Application_Form_PageEdit extends Zend_Form
{
    public function getExcludeFromQuery($elementName)
    {
        $element = $this->getElement($elementName);
        if ($element)
        {
            return $element->getValidator('Db_NoRecordExists')->getExclude();
        } else
        {
            return NULL;
        }
    }

    public function setExcludeFromQuery($elementName, $excludeFromQuery)
    {
        $element = $this->getElement($elementName);
        if ($element)
        {
            $element->getValidator('Db_NoRecordExists')->setExclude($excludeFromQuery);
        }
        return $this;
    }


    public function init()
    {
    $commonFilters      = array('StringTrim');
    $commonValidators = array('NotEmpty');
        $this->setMethod('post')->setAction('/admin-page/edit');

    $id = new Zend_Form_Element_Hidden('id');
    $pid = new Zend_Form_Element_Hidden('pid');

    $keyname = new Zend_Form_Element_Text('keyname');
    $keyname->setLabel('Keyname')
        ->setRequired(true)
        ->addFilters($commonFilters)
        ->addFilter('StringToLower')
        ->addFilter('Word_SeparatorToDash')
        ->addValidator('Db_NoRecordExists', false, array(
            'table'     => 'pages',
            'field'     => 'keyname',
            'exclude'   => array(
                'field' => 'id',
                'value' => $this->getValue('id)
            )
            )
        );

И в действие редактирования:

public function editAction()
{
$duplicateElementName = 'whatever';
$duplicateElementValue = 'value from db';

$id = $this->getRequest()->getParam('id');
$pagesMapper = new Application_Model_PagesMapper();
$form = new Application_Form_PageEdit();
$form->populate($pagesMapper->fetchId($id, true));

$form->setExcludeFromQuery($duplicateElementName, $duplicateElementValue);

if ($this->getRequest()->isPost()) {
    if ($form->isValid($this->getRequest()->getPost())) {
        //... cut ...
    }
}
$this->view->form = $form;
}
...