php - Yii2 一种形式的多个模型

标签 php yii2 yii2-model

如何在Yii2中以一种形式使用多个模型?

我的情况: ER diagram

在我的创建操作中,我可以保存到 agenda_fiscalizacao 表中,但在更新中,当我尝试加载表单时收到此错误:

Call to a member function formName() on array    

我的更新操作:

public function actionUpdate($id)
{
    $model = $this->findModel($id);
    $modelAgenda = AgendaFiscalizacao::findAll(['fiscalizacao_id' => $id]);

    if ($model->load(Yii::$app->request->post()) && Model::loadMultiple($modelAgenda, Yii::$app->request->post())) {
        $valid = $model->validate();
        $valid = $modelAgenda->validade() && $valid;

        if ($valid) {
            $model->save(false);
            $modelAgenda->save(false);
            return $this->redirect(['view', 'id' => $model->id]);
        }
    }

    return $this->render('update', [
        'model' => $model,
        'modelAgenda' => $modelAgenda
    ]);
}

我的表单 View

<?= $form->field($modelAgenda, 'agenda_id')->checkboxList(Agenda::combo(), ['class' => 'checkbox']) ?>
<?= $form->field($model, 'bioma_id')->dropDownList(Bioma::combo(), ['prompt' => $prompt]) ?>
<?= $form->field($model, 'nome')->textInput(['maxlength' => true]) ?>
<?= $form->field($model, 'tipo_doc')->radioList(['CPF'=>'CPF', 'CNPJ'=>'CNPJ'], ['class' => 'radio']) ?>
<?= $form->field($model, 'n_doc')->widget(MaskedInput::className(), ['mask' => ['999.999.999-99', '99.999.999/9999-99']]) ?>
<?= $form->field($model, 'observacao')->textarea(['rows' => 7]) ?>    

有什么问题吗?

编辑(完整错误):

enter image description here

最佳答案

1) 如果您的意思是使用同一类型的多个模型,则错误在这一行:

$valid = $modelAgenda->validade() && $valid;

第一个应该是$modelAgenda->validate(),第二个$modelAgenda包含模型数组,validate()方法可以仅在单个模型上调用。

为了验证多个模型,Yii2 建议使用内置方法 validateMultiple():

use yii\base\Model;

...

$valid = Model::validateMultiple($modelAgenda) && $valid;

官方文档 (Collecting Tabular Input) 详细介绍了使用多个模型。

请注意,他们建议在像这样之前通过 id 索引模型数组:

$models = YourModel::find()->index('id')->all();

2) 如果您只需要两个不同类型的模型,请不要使用 findAll() 因为它用于查找多个模型并且总是返回数组(即使是空的结果)。使用 new 进行 create 操作,使用 findOne() 进行update 操作来初始化模型。假设您初始化了两个模型,$firstModel$secondModel,然后您可以像这样加载和保存它们:

$isSuccess = false;

Yii::$app->db->transaction(function () use ($isSuccess) {
    $areLoaded = $firstModel->load(Yii::$app->request->post()) && $secondModel->load(Yii::$app->request->post();
    $areSaved = $firstModel->save() && $secondModel->save();
    $isSuccess = $areLoaded && $areSaved;
});

if ($isSuccess) {
    return $this->redirect(['view', 'id' => $model->id]);
}

在保存第二个模型失败的情况下添加事务(因此第一个模型也不会保存)。

或者,您可以声明 transactions在你的模型中,例如:

return [
    'admin' => self::OP_INSERT,
    'api' => self::OP_INSERT | self::OP_UPDATE | self::OP_DELETE,
    // the above is equivalent to the following:
    // 'api' => self::OP_ALL,
];

然后只需使用:

$firstModel->scenario = 'scenarioForTransaction';
$secondModel->scenario = 'scenarioForTransaction';
$areLoaded = $firstModel->load(Yii::$app->request->post()) && $secondModel->load(Yii::$app->request->post();
$areSaved = $firstModel->save() && $secondModel->save();

if ($areLoaded && $areSaved) {
    return $this->redirect(['view', 'id' => $model->id]);
}

对于两个以上的模型,最好使用循环。

P.S. 我建议将保存分别保存到不同的 Controller /操作并通过 AJAX 调用它,这样会更加用户友好。

要保存关系,请阅读 - Saving Relations .

关于php - Yii2 一种形式的多个模型,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35018220/

相关文章:

php - 使用mySql根据条件检索特定值

php - 我如何在 Yii2 中处理多对多关系

php - 使用 Yii2 中的关系连接多个表

Yii2 保存后。使用changedAttributes显示错误

php - 正则表达式 - 如何让它接受阿拉伯语和英语 alpha

php - ORDER BY 最小数字

php - Yii2 从名称动态加载类

module - Yii2 禁用供应商模块的 Assets

php - 如何在 Yii2 的动态模型中添加自定义验证功能?

javascript - 如何将点击图像的名称传输到另一个 php 页面?