CakePHP HABTM数据未保存到数据库

时间:2013-08-26 15:05:53

标签: php validation cakephp cakephp-2.3

我有一个模型GenForm,它与另一个模型PdfFile有HABTM关系。我使用它来生成GenForm索引视图中的复选框列表。在GenForm模型中,我添加了:

public $hasAndBelongsToMany = array(
    'PdfFile' => array(
        'className' => 'PdfFile',
        'joinTable' => 'gen_forms_x_pdf_files'
    )

以下是GenForm index.ctp视图中的一个片段:

<?php 
echo $this->Form->input( 'PdfFile', array('label' => 'Select some PDF files', 'multiple' => 'checkbox') );
echo $this->Form->input( 'first_name' );
echo $this->Form->input( 'last_name' );
?>

在控制器中,我有一个基本保存:

    if ($this->request->is('post')) { // form was submitted
        $this->GenForm->create();
        if ($this->GenForm->save($this->request->data)) {
            return $this->redirect(array('action' => 'generate', $this->GenForm->id)); // assemble the PDF for this record
        } else {
            $this->Session->setFlash(__('Log entry not saved.'));
        }
    }

现在$this->data看起来就像这样debug()

array(
    'PdfFile' => array(
        'PdfFile' => array(
            (int) 0 => '1',
            (int) 1 => '5'
        )
    ),
    'GenForm' => array(
        'first_name' => 'xxx',
        'last_name' => 'xxx',
        'association_id' => '1',
        'email' => ''
    )
)

一切都很完美,但我无法验证复选框(必须至少检查一个)。所以,按照this answer,我做了一些改动。

index.ctp视图变为:

<?php 
echo $this->Form->input( 'GenForm.PdfFile', array('label' => 'Select some PDF files', 'multiple' => 'checkbox') );
echo $this->Form->input( 'first_name' );
echo $this->Form->input( 'last_name' );
?>

这是我的验证规则:

public $validate = array(
    'PdfFile' => array(
        'rule' => array(
            'multiple', array('min' => 1)
        ), 
        'message' => 'Please select one or more PDFs'
    )
)

这就是$this->data现在的样子:

array(
    'GenForm' => array(
        'PdfFile' => array(
            (int) 0 => '1',
            (int) 1 => '5'
        ),
        'first_name' => 'xxx',
        'last_name' => 'xxx',
        'association_id' => '1',
        'email' => ''
    )
)

现在PdfFile的复选框已验证,但PdfFile数据未保存 - 尽管GenForm的其他字段已正确保存到自己的表中。

有人能告诉我我缺少什么,以便PdfFile自动保存获得验证吗?

2 个答案:

答案 0 :(得分:2)

第一种形式是正确的

陈述显而易见的,但有效的形式是使用的形式,即:

echo $this->Form->input('PdfFile', array(
    'label' => 'Select some PDF files', 
    'multiple' => 'checkbox'
));

将表单更改为具有名为“PdfFile”的“字段”将无法正常工作 - 因为模型图层将删除不存在的字段的任何数据 - 并且在此表单中它将检查gen_forms.PdfFile ,找到没有字段并忽略数据。

验证

要处理验证错误 - 使用在模型上运行的验证规则来检查要保存的habtm记录的数量。用于验证的字段的名称是什么并不重要:

<?php
class GenForm extends AppModel {

    public $validate = array(
        'dummy' => array( // <- name this whatever you like
            'atLeastOne' => array(
                'required' => true, // run always
                'rule' => array('validateAtLeastOne')
            )
        )
    );

    function validateAtLeastOne() {
        if (!isset($this->data['PdfFile'])) {
            // there is no pdf data at all, ignore this rule
            // allow other save operations to work
            return true;
        }

        $return = count(array_filter($this->data['PdfFile']['PdfFile']));
        if (!$return) {
            $this->PdfFile->invalidate('PdfFile', 'Please upload a file');
        }
        return $return;
    }

}

因为如果没有记录,验证规则将返回false,它将暂停保存。通过使用表单助手将查找的相同“字段”名称调用HABTM关联上的invalidate,将显示错误消息。

替代地

您可以在问题中使用第二种方法:

echo $this->Form->input('GenForm.PdfFile', array(
    'label' => 'Select some PDF files', 
    'multiple' => 'checkbox'
));

完全了解这是蛋糕如何接收数据,然后将其操作为beforeValidate中的正确格式:

<?php
class GenForm extends AppModel {

    public $validate = array(
        'PdfFile' => array( // existing rule
            ...
        )
    );

    function beforeValidate() {
        if (isset($this->data[$this->alias]['PdfFile'])) {
            // keep the existing data as that's what validation will check against
            // copy to the right location so Cake will process it
            $this->data['PdfFile']['PdfFile'] = $this->data[$this->alias]['PdfFile'];
        }
        return true;
    }

    ...
}

答案 1 :(得分:0)

如果我没记错的话,可能从cake manual错误地格式化数据。

尝试$this->data看起来像这样:

array(
    'GenForm' => array(
        'first_name' => 'xxx',
        'last_name' => 'xxx',
        'association_id' => '1',
        'email' => ''
    ),
    'PdfFile' => array(
        (int) 0 => '1',
        (int) 1 => '5'
    ),
)