mirror of
https://github.com/silverstripe/silverstripe-framework
synced 2024-10-22 14:05:37 +02:00
API CHANGE Deprecated DataObject->fieldExists() - please use hasField(), hasDatabaseField(), hasOwnTableDatabaseField()
ENHANCEMENT Added DataObject->hasOwnTableDatabaseField(), replaced legacy usage of fieldExists() ENHANCEMENT Renamed cached static "fieldExists" to "_cache_hasOwnTableDatabaseField" ENHANCEMENT Added DataObjectTest test cases for checking various field existence levels API CHANGE Deprecated DataObject->listOfFields() - use custom code instead git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@63337 467b73ca-7a2a-4603-9d3b-597d59a354a9
This commit is contained in:
parent
2af039785c
commit
bf896a2cfd
@ -701,7 +701,7 @@ class DataObject extends ViewableData implements DataObjectInterface {
|
|||||||
foreach($ancestry as $idx => $class) {
|
foreach($ancestry as $idx => $class) {
|
||||||
$classSingleton = singleton($class);
|
$classSingleton = singleton($class);
|
||||||
foreach($this->record as $fieldName => $fieldValue) {
|
foreach($this->record as $fieldName => $fieldValue) {
|
||||||
if(isset($this->changed[$fieldName]) && $this->changed[$fieldName] && $fieldType = $classSingleton->fieldExists($fieldName)) {
|
if(isset($this->changed[$fieldName]) && $this->changed[$fieldName] && $fieldType = $classSingleton->hasOwnTableDatabaseField($fieldName)) {
|
||||||
$fieldObj = $this->obj($fieldName);
|
$fieldObj = $this->obj($fieldName);
|
||||||
if(!isset($manipulation[$class])) $manipulation[$class] = array();
|
if(!isset($manipulation[$class])) $manipulation[$class] = array();
|
||||||
|
|
||||||
@ -1699,7 +1699,7 @@ class DataObject extends ViewableData implements DataObjectInterface {
|
|||||||
* @return boolean True if the given field exists
|
* @return boolean True if the given field exists
|
||||||
*/
|
*/
|
||||||
public function hasField($field) {
|
public function hasField($field) {
|
||||||
return array_key_exists($field, $this->record) || $this->fieldExists($field);
|
return array_key_exists($field, $this->record) || $this->hasOwnTableDatabaseField($field);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -1713,6 +1713,46 @@ class DataObject extends ViewableData implements DataObjectInterface {
|
|||||||
return array_key_exists($field, $this->databaseFields());
|
return array_key_exists($field, $this->databaseFields());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the field type of the given field, if it belongs to this class, and not a parent.
|
||||||
|
* Note that the field type will not include constructor arguments in round brackets, only the classname.
|
||||||
|
*
|
||||||
|
* @param string $field Name of the field
|
||||||
|
* @return string The field type of the given field
|
||||||
|
*/
|
||||||
|
public function hasOwnTableDatabaseField($field) {
|
||||||
|
// Add base fields which are not defined in static $db
|
||||||
|
if($field == "ID") return "Int";
|
||||||
|
if($field == "ClassName" && get_parent_class($this) == "DataObject") return "Enum";
|
||||||
|
if($field == "LastEdited" && get_parent_class($this) == "DataObject") return "Datetime";
|
||||||
|
if($field == "Created" && get_parent_class($this) == "DataObject") return "Datetime";
|
||||||
|
|
||||||
|
// Add fields from Versioned decorator
|
||||||
|
if($field == "Version") return $this->hasExtension('Versioned') ? "Int" : false;
|
||||||
|
|
||||||
|
// get cached fieldmap
|
||||||
|
$fieldMap = $this->uninherited('_cache_hasOwnTableDatabaseField');
|
||||||
|
|
||||||
|
// if no fieldmap is cached, get all fields
|
||||||
|
if(!$fieldMap) {
|
||||||
|
// all $db fields on this specific class (no parents)
|
||||||
|
$fieldMap = $this->uninherited('db', true);
|
||||||
|
|
||||||
|
// all has_one relations on this specific class,
|
||||||
|
// add foreign key
|
||||||
|
$hasOne = $this->uninherited('has_one', true);
|
||||||
|
if($hasOne) foreach($hasOne as $fieldName => $fieldSchema) {
|
||||||
|
$fieldMap[$fieldName . 'ID'] = "Int";
|
||||||
|
}
|
||||||
|
|
||||||
|
// set cached fieldmap
|
||||||
|
$this->set_uninherited('_cache_hasOwnTableDatabaseField', $fieldMap);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove string-based "constructor-arguments" from the DBField definition
|
||||||
|
return isset($fieldMap[$field]) ? strtok($fieldMap[$field],'(') : null;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns true if the member is allowed to do the given action.
|
* Returns true if the member is allowed to do the given action.
|
||||||
*
|
*
|
||||||
@ -1833,32 +1873,13 @@ class DataObject extends ViewableData implements DataObjectInterface {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the field type of the given field, if it belongs to this class, and not a parent.
|
* @deprecated 2.3 (For external use) Please use hasField(), hasDatabaseField(), hasOwnTableDatabaseField() instead
|
||||||
* Can be used to detect whether the given field exists.
|
|
||||||
* Note that the field type will not include constructor arguments; only the classname.
|
|
||||||
*
|
*
|
||||||
* @param string $field Name of the field
|
* @param string $field Name of the field
|
||||||
*
|
|
||||||
* @return string The field type of the given field
|
* @return string The field type of the given field
|
||||||
*/
|
*/
|
||||||
public function fieldExists($field) {
|
public function fieldExists($field) {
|
||||||
if($field == "ID") return "Int";
|
return $this->hasOwnTableDatabaseField($field);
|
||||||
if($field == "ClassName" && get_parent_class($this) == "DataObject") return "Enum";
|
|
||||||
if($field == "LastEdited" && get_parent_class($this) == "DataObject") return "Datetime";
|
|
||||||
if($field == "Created" && get_parent_class($this) == "DataObject") return "Datetime";
|
|
||||||
|
|
||||||
if($field == "Version") return $this->hasExtension('Versioned') ? "Int" : false;
|
|
||||||
$fieldMap = $this->uninherited('fieldExists');
|
|
||||||
if(!$fieldMap) {
|
|
||||||
$fieldMap = $this->uninherited('db', true);
|
|
||||||
$has = $this->uninherited('has_one', true);
|
|
||||||
if($has) foreach($has as $fieldName => $fieldSchema) {
|
|
||||||
$fieldMap[$fieldName . 'ID'] = "Int";
|
|
||||||
}
|
|
||||||
$this->set_uninherited('fieldExists', $fieldMap);
|
|
||||||
}
|
|
||||||
|
|
||||||
return isset($fieldMap[$field]) ? strtok($fieldMap[$field],'(') : null;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -2057,6 +2078,7 @@ class DataObject extends ViewableData implements DataObjectInterface {
|
|||||||
* - name: value
|
* - name: value
|
||||||
* - name: value
|
* - name: value
|
||||||
*
|
*
|
||||||
|
* @deprecated 2.3 Use custom code
|
||||||
* @return string The fields as an HTML unordered list
|
* @return string The fields as an HTML unordered list
|
||||||
*/
|
*/
|
||||||
function listOfFields() {
|
function listOfFields() {
|
||||||
@ -2390,7 +2412,10 @@ class DataObject extends ViewableData implements DataObjectInterface {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the custom database fields for this object, from self::$db and self::$has_one
|
* Get the custom database fields for this object, from self::$db and self::$has_one,
|
||||||
|
* but not built-in fields like ID, ClassName, Created, LastEdited.
|
||||||
|
*
|
||||||
|
* @return array
|
||||||
*/
|
*/
|
||||||
public function customDatabaseFields() {
|
public function customDatabaseFields() {
|
||||||
$db = $this->uninherited('db',true);
|
$db = $this->uninherited('db',true);
|
||||||
@ -2437,15 +2462,22 @@ class DataObject extends ViewableData implements DataObjectInterface {
|
|||||||
public function searchableFields() {
|
public function searchableFields() {
|
||||||
// can have mixed format, need to make consistent in most verbose form
|
// can have mixed format, need to make consistent in most verbose form
|
||||||
$fields = $this->stat('searchable_fields');
|
$fields = $this->stat('searchable_fields');
|
||||||
|
|
||||||
$labels = $this->fieldLabels();
|
$labels = $this->fieldLabels();
|
||||||
|
|
||||||
// fallback to summary fields
|
// fallback to summary fields
|
||||||
if(!$fields) $fields = array_keys($this->summaryFields());
|
if(!$fields) $fields = array_keys($this->summaryFields());
|
||||||
|
|
||||||
|
// we need to make sure the format is unified before
|
||||||
|
// augumenting fields, so decorators can apply consistent checks
|
||||||
|
// but also after augumenting fields, because the decorator
|
||||||
|
// might use the shorthand notation as well
|
||||||
|
|
||||||
// rewrite array, if it is using shorthand syntax
|
// rewrite array, if it is using shorthand syntax
|
||||||
$rewrite = array();
|
$rewrite = array();
|
||||||
foreach($fields as $name => $specOrName) {
|
foreach($fields as $name => $specOrName) {
|
||||||
$identifer = (is_int($name)) ? $specOrName : $name;
|
$identifer = (is_int($name)) ? $specOrName : $name;
|
||||||
|
|
||||||
if(is_int($name)) {
|
if(is_int($name)) {
|
||||||
// Format: array('MyFieldName')
|
// Format: array('MyFieldName')
|
||||||
$rewrite[$identifer] = array();
|
$rewrite[$identifer] = array();
|
||||||
@ -2454,7 +2486,7 @@ class DataObject extends ViewableData implements DataObjectInterface {
|
|||||||
// 'filter => 'ExactMatchFilter',
|
// 'filter => 'ExactMatchFilter',
|
||||||
// 'field' => 'NumericField', // optional
|
// 'field' => 'NumericField', // optional
|
||||||
// 'title' => 'My Title', // optiona.
|
// 'title' => 'My Title', // optiona.
|
||||||
// )
|
// ))
|
||||||
$rewrite[$identifer] = array_merge(
|
$rewrite[$identifer] = array_merge(
|
||||||
array('filter' => $this->relObject($identifer)->stat('default_search_filter_class')),
|
array('filter' => $this->relObject($identifer)->stat('default_search_filter_class')),
|
||||||
(array)$specOrName
|
(array)$specOrName
|
||||||
@ -2472,8 +2504,10 @@ class DataObject extends ViewableData implements DataObjectInterface {
|
|||||||
$rewrite[$identifer]['filter'] = 'PartialMatchFilter';
|
$rewrite[$identifer]['filter'] = 'PartialMatchFilter';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$fields = $rewrite;
|
$fields = $rewrite;
|
||||||
|
|
||||||
|
// apply DataObjectDecorators if present
|
||||||
$this->extend('updateSearchableFields', $fields);
|
$this->extend('updateSearchableFields', $fields);
|
||||||
|
|
||||||
return $fields;
|
return $fields;
|
||||||
|
@ -45,7 +45,9 @@ abstract class DataObjectDecorator extends Extension {
|
|||||||
eval("$className::\$$relationType = array_merge((array){$className}::\$$relationType, (array)\$fields);");
|
eval("$className::\$$relationType = array_merge((array){$className}::\$$relationType, (array)\$fields);");
|
||||||
$this->owner->set_stat($relationType, eval("return $className::\$$relationType;"));
|
$this->owner->set_stat($relationType, eval("return $className::\$$relationType;"));
|
||||||
}
|
}
|
||||||
$this->owner->set_uninherited('fieldExists', null);
|
|
||||||
|
// clear previously set caches from DataObject->hasOwnTableDatabaseField()
|
||||||
|
$this->owner->set_uninherited('_cache_hasOwnTableDatabaseField', null);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -288,6 +288,126 @@ class DataObjectTest extends SapphireTest {
|
|||||||
$existingTeam->write();
|
$existingTeam->write();
|
||||||
$this->assertEquals(0, DB::query("SELECT CaptainID FROM DataObjectTest_Team WHERE ID = $existingTeam->ID")->value());
|
$this->assertEquals(0, DB::query("SELECT CaptainID FROM DataObjectTest_Team WHERE ID = $existingTeam->ID")->value());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function testFieldExistence() {
|
||||||
|
$teamInstance = $this->objFromFixture('DataObjectTest_Team', 'team1');
|
||||||
|
$teamSingleton = singleton('DataObjectTest_Team');
|
||||||
|
|
||||||
|
$subteamInstance = $this->objFromFixture('DataObjectTest_SubTeam', 'subteam1');
|
||||||
|
$subteamSingleton = singleton('DataObjectTest_SubTeam');
|
||||||
|
|
||||||
|
/* hasField() singleton checks */
|
||||||
|
$this->assertTrue($teamSingleton->hasField('ID'), 'hasField() finds built-in fields in singletons');
|
||||||
|
$this->assertTrue($teamSingleton->hasField('Title'), 'hasField() finds custom fields in singletons');
|
||||||
|
|
||||||
|
/* hasField() instance checks */
|
||||||
|
$this->assertFalse($teamInstance->hasField('NonExistingField'), 'hasField() doesnt find non-existing fields in instances');
|
||||||
|
$this->assertTrue($teamInstance->hasField('ID'), 'hasField() finds built-in fields in instances');
|
||||||
|
$this->assertTrue($teamInstance->hasField('Created'), 'hasField() finds built-in fields in instances');
|
||||||
|
$this->assertTrue($teamInstance->hasField('DatabaseField'), 'hasField() finds custom fields in instances');
|
||||||
|
//$this->assertFalse($teamInstance->hasField('SubclassDatabaseField'), 'hasField() doesnt find subclass fields in parentclass instances');
|
||||||
|
//$this->assertTrue($teamInstance->hasField('DynamicField'), 'hasField() finds dynamic getters in instances');
|
||||||
|
$this->assertTrue($teamInstance->hasField('HasOneRelationshipID'), 'hasField() finds foreign keys in instances');
|
||||||
|
$this->assertTrue($teamInstance->hasField('DecoratedDatabaseField'), 'hasField() finds decorated fields in instances');
|
||||||
|
$this->assertTrue($teamInstance->hasField('DecoratedHasOneRelationshipID'), 'hasField() finds decorated foreign keys in instances');
|
||||||
|
//$this->assertTrue($teamInstance->hasField('DecoratedDynamicField'), 'hasField() includes decorated dynamic getters in instances');
|
||||||
|
|
||||||
|
/* hasField() subclass checks */
|
||||||
|
$this->assertTrue($subteamInstance->hasField('ID'), 'hasField() finds built-in fields in subclass instances');
|
||||||
|
$this->assertTrue($subteamInstance->hasField('Created'), 'hasField() finds built-in fields in subclass instances');
|
||||||
|
$this->assertTrue($subteamInstance->hasField('DatabaseField'), 'hasField() finds custom fields in subclass instances');
|
||||||
|
$this->assertTrue($subteamInstance->hasField('SubclassDatabaseField'), 'hasField() finds custom fields in subclass instances');
|
||||||
|
//$this->assertTrue($subteamInstance->hasField('DynamicField'), 'hasField() finds dynamic getters in subclass instances');
|
||||||
|
$this->assertTrue($subteamInstance->hasField('HasOneRelationshipID'), 'hasField() finds foreign keys in subclass instances');
|
||||||
|
$this->assertTrue($subteamInstance->hasField('DecoratedDatabaseField'), 'hasField() finds decorated fields in subclass instances');
|
||||||
|
$this->assertTrue($subteamInstance->hasField('DecoratedHasOneRelationshipID'), 'hasField() finds decorated foreign keys in subclass instances');
|
||||||
|
|
||||||
|
/* hasDatabaseField() singleton checks */
|
||||||
|
//$this->assertTrue($teamSingleton->hasDatabaseField('ID'), 'hasDatabaseField() finds built-in fields in singletons');
|
||||||
|
$this->assertTrue($teamSingleton->hasDatabaseField('Title'), 'hasDatabaseField() finds custom fields in singletons');
|
||||||
|
|
||||||
|
/* hasDatabaseField() instance checks */
|
||||||
|
$this->assertFalse($teamInstance->hasDatabaseField('NonExistingField'), 'hasDatabaseField() doesnt find non-existing fields in instances');
|
||||||
|
//$this->assertTrue($teamInstance->hasDatabaseField('ID'), 'hasDatabaseField() finds built-in fields in instances');
|
||||||
|
$this->assertTrue($teamInstance->hasDatabaseField('Created'), 'hasDatabaseField() finds built-in fields in instances');
|
||||||
|
$this->assertTrue($teamInstance->hasDatabaseField('DatabaseField'), 'hasDatabaseField() finds custom fields in instances');
|
||||||
|
$this->assertFalse($teamInstance->hasDatabaseField('SubclassDatabaseField'), 'hasDatabaseField() doesnt find subclass fields in parentclass instances');
|
||||||
|
//$this->assertFalse($teamInstance->hasDatabaseField('DynamicField'), 'hasDatabaseField() doesnt dynamic getters in instances');
|
||||||
|
$this->assertTrue($teamInstance->hasDatabaseField('HasOneRelationshipID'), 'hasDatabaseField() finds foreign keys in instances');
|
||||||
|
$this->assertTrue($teamInstance->hasDatabaseField('DecoratedDatabaseField'), 'hasDatabaseField() finds decorated fields in instances');
|
||||||
|
$this->assertTrue($teamInstance->hasDatabaseField('DecoratedHasOneRelationshipID'), 'hasDatabaseField() finds decorated foreign keys in instances');
|
||||||
|
$this->assertFalse($teamInstance->hasDatabaseField('DecoratedDynamicField'), 'hasDatabaseField() doesnt include decorated dynamic getters in instances');
|
||||||
|
|
||||||
|
/* hasDatabaseField() subclass checks */
|
||||||
|
$this->assertTrue($subteamInstance->hasField('DatabaseField'), 'hasField() finds custom fields in subclass instances');
|
||||||
|
$this->assertTrue($subteamInstance->hasField('SubclassDatabaseField'), 'hasField() finds custom fields in subclass instances');
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
function testFieldInheritance() {
|
||||||
|
$teamInstance = $this->objFromFixture('DataObjectTest_Team', 'team1');
|
||||||
|
$subteamInstance = $this->objFromFixture('DataObjectTest_SubTeam', 'subteam1');
|
||||||
|
|
||||||
|
$this->assertEquals(
|
||||||
|
array_keys($teamInstance->inheritedDatabaseFields()),
|
||||||
|
array(
|
||||||
|
//'ID',
|
||||||
|
//'ClassName',
|
||||||
|
//'Created',
|
||||||
|
//'LastEdited',
|
||||||
|
'Title',
|
||||||
|
'DatabaseField',
|
||||||
|
'DecoratedDatabaseField',
|
||||||
|
'CaptainID',
|
||||||
|
'HasOneRelationshipID',
|
||||||
|
'DecoratedHasOneRelationshipID'
|
||||||
|
),
|
||||||
|
'inheritedDatabaseFields() contains all fields defined on instance, including base fields, decorated fields and foreign keys'
|
||||||
|
);
|
||||||
|
|
||||||
|
$this->assertEquals(
|
||||||
|
array_keys($teamInstance->databaseFields()),
|
||||||
|
array(
|
||||||
|
//'ID',
|
||||||
|
'ClassName',
|
||||||
|
'Created',
|
||||||
|
'LastEdited',
|
||||||
|
'Title',
|
||||||
|
'DatabaseField',
|
||||||
|
'DecoratedDatabaseField',
|
||||||
|
'CaptainID',
|
||||||
|
'HasOneRelationshipID',
|
||||||
|
'DecoratedHasOneRelationshipID'
|
||||||
|
),
|
||||||
|
'databaseFields() contains only fields defined on instance, including base fields, decorated fields and foreign keys'
|
||||||
|
);
|
||||||
|
|
||||||
|
$this->assertEquals(
|
||||||
|
array_keys($subteamInstance->inheritedDatabaseFields()),
|
||||||
|
array(
|
||||||
|
//'ID',
|
||||||
|
//'ClassName',
|
||||||
|
//'Created',
|
||||||
|
//'LastEdited',
|
||||||
|
'SubclassDatabaseField',
|
||||||
|
'Title',
|
||||||
|
'DatabaseField',
|
||||||
|
'DecoratedDatabaseField',
|
||||||
|
'CaptainID',
|
||||||
|
'HasOneRelationshipID',
|
||||||
|
'DecoratedHasOneRelationshipID',
|
||||||
|
),
|
||||||
|
'inheritedDatabaseFields() on subclass contains all fields defined on instance, including base fields, decorated fields and foreign keys'
|
||||||
|
);
|
||||||
|
|
||||||
|
$this->assertEquals(
|
||||||
|
array_keys($subteamInstance->databaseFields()),
|
||||||
|
array(
|
||||||
|
'SubclassDatabaseField',
|
||||||
|
),
|
||||||
|
'databaseFields() on subclass contains only fields defined on instance'
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class DataObjectTest_Player extends Member implements TestOnly {
|
class DataObjectTest_Player extends Member implements TestOnly {
|
||||||
@ -302,16 +422,49 @@ class DataObjectTest_Team extends DataObject implements TestOnly {
|
|||||||
|
|
||||||
static $db = array(
|
static $db = array(
|
||||||
'Title' => 'Text',
|
'Title' => 'Text',
|
||||||
|
'DatabaseField' => 'Text'
|
||||||
);
|
);
|
||||||
|
|
||||||
static $has_one = array(
|
static $has_one = array(
|
||||||
"Captain" => 'DataObjectTest_Player',
|
"Captain" => 'DataObjectTest_Player',
|
||||||
|
'HasOneRelationship' => 'DataObjectTest_Player',
|
||||||
);
|
);
|
||||||
|
|
||||||
static $many_many = array(
|
static $many_many = array(
|
||||||
'Players' => 'DataObjectTest_Player'
|
'Players' => 'DataObjectTest_Player'
|
||||||
);
|
);
|
||||||
|
|
||||||
|
function getDynamicField() {
|
||||||
|
return 'dynamicfield';
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class DataObjectTest_SubTeam extends DataObjectTest_Team implements TestOnly {
|
||||||
|
static $db = array(
|
||||||
|
'SubclassDatabaseField' => 'Text'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
class DataObjectTest_Team_Decorator extends DataObjectDecorator implements TestOnly {
|
||||||
|
|
||||||
|
function extraDBFields() {
|
||||||
|
return array(
|
||||||
|
'db' => array(
|
||||||
|
'DecoratedDatabaseField' => 'Text'
|
||||||
|
),
|
||||||
|
'has_one' => array(
|
||||||
|
'DecoratedHasOneRelationship' => 'DataObjectTest_Player'
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getDecoratedDynamicField() {
|
||||||
|
return "decorated dynamic field";
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
DataObject::add_extension('DataObjectTest_Team', 'DataObjectTest_Team_Decorator');
|
||||||
|
|
||||||
?>
|
?>
|
||||||
|
@ -47,3 +47,6 @@ DataObjectTest_Player:
|
|||||||
player2:
|
player2:
|
||||||
FirstName: Player 2
|
FirstName: Player 2
|
||||||
Teams: =>DataObjectTest_Team.team1,=>DataObjectTest_Team.team2
|
Teams: =>DataObjectTest_Team.team1,=>DataObjectTest_Team.team2
|
||||||
|
DataObjectTest_SubTeam:
|
||||||
|
subteam1:
|
||||||
|
Title: Subteam 1
|
Loading…
Reference in New Issue
Block a user