在SilverStripe中,我有两个管理模型:服务和位置。地点是地点列表。服务是这些位置可以提供的服务列表。但是,只有某些地点可以提供某些服务。我想要做的是,每当添加一个新的位置(或者如果删除或重命名一个),我想将它添加到每个服务下的复选框字段(这样,内容管理员可以轻松控制哪些服务提供在哪里在任何时候)。
这是服务数据对象代码(注意:我还没有在复选框字段中添加):
class Services extends DataObject {
private static $db = array(
'Name' => 'varchar',
);
public static $summary_fields = array(
'Name' => 'Title',
);
}
以下是Locations数据对象代码:
class Locations extends DataObject {
private static $db = array(
'Name' => 'varchar',
'AcceptingAppointments' => 'boolean',
);
public static $summary_fields = array(
'Name' => 'Title',
'AcceptingAppointments' => 'Accepting Appointments?',
);
public function AcceptingAppointments() {
return ($this->AcceptingAppointments==true ? 'Yes':'No');
}
}
到目前为止一切正常。我可以添加,编辑和删除位置和服务。但是,我不知道如何处理编码服务,以便每当内容管理员添加,删除或重命名位置条目时,每个服务都会使用所有位置的完整列表进行更新。
在这方面使用$has_many
关系会有效吗?
答案 0 :(得分:3)
我需要的是$many_many
和Service
之间的Location
关系。我们需要多对多的关系,因为服务可以有许多位置,并且位置可以有许多服务。要了解有关使用$many_many
关系的详情,请阅读this lesson或this documentation。
一旦我们定义了$many_many
关系,我们就可以添加CheckboxSetField
来控制这种关系。
注意,我已将Locations
课程重命名为Location
,将Services
重命名为Service
,因为我认为课程名称应为单数。
<强>服务强>
class Service extends DataObject {
private static $db = array(
'Name' => 'Varchar'
);
private static $belongs_many_many = array(
'Locations' => 'Location'
);
private static $summary_fields = array(
'Name' => 'Title'
);
private static $field_labels = array(
'Name'
);
public function getCMSFields() {
$fields = parent::getCMSFields();
if ($this->ID) {
$fields->addFieldToTab('Root.Locations', CheckboxSetField::create(
'Locations',
'Locations',
Location::get()->map()
));
}
return $fields;
}
}
在Location
我们将$belongs_many_many
关系添加回Service
,以便我们可以控制双方的这种关系。
位置强>
class Location extends DataObject {
private static $db = array(
'Name' => 'Varchar',
'AcceptingAppointments' => 'Boolean'
);
private static $many_many = array(
'Services' => 'Service'
);
private static $summary_fields = array(
'Name',
'AcceptingAppointmentsNice'
);
private static $field_labels = array(
'Name' => 'Title',
'AcceptingAppointmentsNice' => 'Accepting Appointments?'
);
public function AcceptingAppointmentsNice() {
return $this->AcceptingAppointments ? 'Yes' : 'No';
}
public function getCMSFields() {
$fields = parent::getCMSFields();
if ($this->ID) {
$fields->addFieldToTab('Root.Services', CheckboxSetField::create(
'Services',
'Services',
Service::get()->map()
));
}
return $fields;
}
}
CheckboxSetField
的替代方案是ListboxField
。 ListboxField
是一个整齐的下拉多选字段。如果选项列表很长,这很有用。
class Service extends DataObject {
// ...
public function getCMSFields() {
$fields = parent::getCMSFields();
if ($this->ID) {
$fields->addFieldToTab('Root.Locations', ListboxField::create(
'Locations',
'Locations',
Location::get()->map('ID', 'Name')->toArray(),
array(),
null,
true
));
}
return $fields;
}
}
另请注意,$summary_fields
必须声明为private
变量,而不是public
变量。