如何在Yii2中以一种形式使用多个模型?
在我的创建操作中,我可以保存到 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]) ?>
有什么问题吗?
编辑(完整错误):
最佳答案
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/