mgrechanik/yii2-materialized-path

Materialized Path Behavior for Yii2 Framework

1.0.0 2019-06-28 07:02 UTC

This package is auto-updated.

Last update: 2025-01-15 23:53:03 UTC


README

This extension allows to organize Active Record models into a tree according to Materialized Path algorithm

Русская версия

Table of contents

Features

  • Allows to organize ActiveRecord models into a tree
  • Every tree has only one Root node
  • It is possible to keep many independent trees in one database table, for example to keep menu items for many menus
  • A lot of ways to navigate among the nodes of the tree and question properties of the node
  • Tree modification operations: inserting new nodes, moving existed ones. They run in transaction if asked
  • Two modes when deleting a node: when descendants are deleted along with it or when they are moved to it's parent
  • Service of managing trees allows:
    • Query a tree (or subtree) with one query to database
    • Tree is formed it two formats: nested structure useful to be displayed in a form of <ul>-<li> lists or "flat" presentation of a tree - simple php array useful to be displayed in <select> list or to be used for Data Provider
    • Cloning subtrees
    • Getting id ranges of descendants of a node, useful for validation rules

Installation

The preferred way to install this extension is through composer.:

Either run

composer require --prefer-dist mgrechanik/yii2-materialized-path

or add

"mgrechanik/yii2-materialized-path" : "^1.0"

to the require section of your composer.json

Migrations

This extension expects additional columns in the database table which are responsible to keep a tree.

Example of migration for a table with many trees look here

And here is example of migration for a table with only one tree:

use yii\db\Migration;

/**
 * Handles the creation of table `animal`.
 */
class m170208_094404_create_animal_table extends Migration
{
    /**
     * @inheritdoc
     */
    public function up()
    {
        $this->createTable('animal', [
            'id' => $this->primaryKey(),
            'path' => $this->string(255)->notNull()->defaultValue('')->comment('Path to parent node'),
            'level' => $this->integer(4)->notNull()->defaultValue(1)->comment('Level of the node in the tree'),
            'weight' => $this->integer(11)->notNull()->defaultValue(1)->comment('Weight among siblings'),
            'name' => $this->string()->notNull()->comment('Name'),
        ]);
        
        $this->createIndex('animal_path_index', 'animal', 'path');
        $this->createIndex('animal_level_index', 'animal', 'level');
        $this->createIndex('animal_weight_index', 'animal', 'weight');        
        
    }

    /**
     * @inheritdoc
     */
    public function down()
    {
        $this->dropTable('animal');
    }
}

Important in the code above:

  1. Explanation of field's roles will be given in explanation of data structure
  2. For path we set field's length in 255 symbols which is optimal for mysql and allows to keep the trees with huge nesting but you can put any value wanted
  3. defaultValue for fields path, weight and level was set up just in case so even rows added not with this extension's api but manually (using phpmyadmin for example) could take their starting position in the tree
  4. For SQLite database take off ->comment()'s from migration above

Settings

To turn Active Record model into a tree node you need to set up the next behavior for it:

use mgrechanik\yiimaterializedpath\MaterializedPathBehavior;

class Animal extends \yii\db\ActiveRecord
{
    public static function tableName()
    {
        return 'animal';
    }
	
    public function behaviors()
    {
        return [
            'materializedpath' => [
                'class' => MaterializedPathBehavior::class,
                // behavior settings
            ],
        ];
    } 
	
	// ...
}	

This MaterializedPathBehavior behavior has next settings:

  1. treeIdentityFields - array with field names which holds the unique identifier of the tree (when there are many).
  2. mpFieldNames - a map of field names used in this extension to the ones in your model class. Use it when your names are different from id, path, level, weight
  3. modelScenarioForAffectedModelsForSavingProcess is explained here
  4. About moveChildrenWnenDeletingParent and modelScenarioForChildrenNodesWhenTheyDeletedAfterParent look in deleting of a node.
  5. maxPathLength can be arbitrary set to some value of limit for path field so when it becomes longer exception will be thrown

Explanation of data structure

Data in the database looks like this:

When table holds only one tree: Tree presentation in the database

When table holds many trees: Tree presentation in the database

For following examples we will use the trees above.
For the first table there is Animal ActiveRecord model.
For the second table there is Menuitem ActiveRecord model.

Common things we see:

  • If table holds many trees, like Menuitem table, we use additional column(s) for unique tree identificator. Like treeid in example. And all future manipulations with the tree will be isolated, they would concern only this concrete tree, other trees are out of concern
  • Every tree has only one Root node who is not presented in the table, being some virtual node, but you can get it and work with it the way you work with any other node. More details
  • Node Animal(1) - cat is the first child of this Root node
  • id column - it holds unique numeric identifier of the node. This extension demands using only positive integer numbers as id of nodes.
  • path column - it holds path to parent of this node. In the form of node's id separated by /. Empty string for root's children.
  • level column - it holds level of this node, 0 for Root and so on
  • weight column - it holds "weight" of the node among its siblings. Neighbors are sorted by this column

So this architecture guarantees effective and sufficient structure to save trees into database table:

  • We keep, if need to, a sign to which tree the node belongs to
  • Every node thanks to path knows its parent
  • Between siblings node is positioned according to weight

Differences from other popular implementations of this algorithm

  1. In other extensions such nodes as Animal(1) - cat are often treated as Root nodes. And it becomes to look like one tree has many root nodes.
    In our extension every tree has only one root node (which is not saved in the database).
    This way all tree nodes are organized only in ONE tree
  2. Also in path column full path is often saved - the path with the id of the very same node at the end.
    In this extension we save only path to node's parent. This value for sibling nodes will be the same

Extension structure

This extension gives two main things:

  • mgrechanik\yiimaterializedpath\MaterializedPathBehavior - this behavior connected to ActiveRecord model turns it into tree node by adding necessary functionality to it
  • mgrechanik\yiimaterializedpath\Service - this service is meant to give additional operations to manage trees: building and outputting trees (subtrees), getting root nodes, cloning and other.

Working with a tree

Root node of the tree

Common

Every tree has only one root node (futher just "root") - node who stays at the very top of the tree and has no parent.

We does not save a row in the database for this root node because we already know that every tree has it. So we do not need to additionally create it in the table to begin fill a tree with nodes.
We consider root node already exists.
And in the database we keep only data really added in the tree. Starting with the first level nodes - cat, dog, snake, bear.

However with this root node we can work the way we work with any other tree nodes:

  • we can ask for it's descendants, getting this way all our tree
  • we can add nodes to it, they will become nodes of first level
  • but in the situation with the root node only logical things work. We can add nodes to root, but we cannot insert before/after root node

Technically root node is represented by object of mgrechanik\yiimaterializedpath\tools\RootNode class - some virtual AR model which you are not supposed to try to save in the database (exception will be risen) but who also configured with MaterializedPathBehavior, which allows to work with it the way like with any other node.

Id of the root node

As we said earlier in path field we do not keep id of the root (because it does not exist in the table) but still root node has it's id - it is controlled in RootNode::getId(). We need it mostly for edition html forms when we may choose the root node in the form and we need a way to distinguish it from other nodes by identifier.

This id of the root is formed according to a simple algorithm:

  • it will be negative integer number
  • if there is only one tree in the table it's value will be -100
  • if there are many trees in the table it's value is calculated by next formula - -100 * (treeField1 + treeField1 + treeFieldi), so for ['treeid' => 2] id will be -200

Work with the root node

To work with the root node we need at first to get it's object.
It could be done in several ways:

  1. By means of AR model name (and arbitrary tree condition)
use mgrechanik\yiimaterializedpath\ServiceInterface;
// AR class
use mgrechanik\yiimaterializedpath\tests\models\Animal;
// service of managing the trees
$service = \Yii::createObject(ServiceInterface::class);
// getting
// the root of the tree
$root = $service->getRoot(Animal::class);  

If your table holds many trees to get the root of needed tree you are expected to give tree condition of this tree:

use mgrechanik\yiimaterializedpath\tests\models\Menuitem;
// ...
// the root of the first tree
$root1 = $service->getRoot(Menuitem::class, ['treeid' => 1]);
// the root of the second tree
$root2 = $service->getRoot(Menuitem::class, ['treeid' => 2]);
  1. By means of any AR model (not new record) we can get the root of the tree this node belongs to:
$model7 = Animal::findOne(7); // 'stag'
// the root of the tree to which $model7 belongs to
$root = $model7->getRoot();

For all nodes of the tree the cache will be returning the same root object (you can compare them using === operator).

  1. By means of his negative id
$root = $service->getModelById(Animal::class, -100); 

Use this method of getting a node when you get id values from html form in which both root node and common nodes could be choosen.

Examples: 1, 2.

Queries

After you choose any node, including root, you can ask for the next information:

  1. Getting descendants of the node

Get the query object for descendants of the node:

               $node->getDescendantsQuery($exceptIds = [], $depth = null)
  • Separately by means of $exceptIds you can inform which subtrees you want to exclude from query, it's format is like [1, 2, 3 => false] meaning subtrees of nodes 1, 2 exclude fully , and leave node 3 but exclude all it's descendants
  • $depth shows how many level of descendants to query
    null means to query for all descendants,
    1 means to query only 1 level deep, another word query only for children
    and so on.

Example:

$model = Animal::findOne(1);
$query = $model->getDescendantsQuery();
$result = $query->asArray()->all();

The result we get will be sorted like level ASC, weight ASC so it will not be ready for any output in the form of a tree.
For building php trees have a look here.

  1. Getting query object

When you need to build your own query to tree nodes instead of working with the AR model like ClassName::find() you would rather start with this query object:

               $node->getQuery()
  • This object holds tree condition for the tree this $node belongs to
  • All queries of this extention start with this query object
  • You can get it from root node either
  • Technically you can see it's implementation in RootNode::getQuery()
  • Your new conditions you need to start now with andWhere()

Navigation

  1. Get the Root node of the tree
               $node->getRoot()

It works only for existed in database models because new model does not belong to any tree yet.
It returns the same object for every tree node because tree has only one root.

  1. Work with the children of the node

Get all node's children:

               $node->children()

We will get the array of AR models - direct children of the node.

Or you can use more common query object:

               $node->getChildrenQuery($sortAsc = true)

Example:

$query = $model->getChildrenQuery();
$result = $query->asArray()->all();

Get the first child of the node:

               $node->firstChild()

Get the last child of the node:

               $node->lastChild()
  1. Work with the parents of the node

Get the parent:

               $node->parent()

Get the parents:

               $node->parents($orderFromRootToCurrent = true, $includeRootNode = false, $indexResultBy = false)

Here:

  • $orderFromRootToCurrent - sort parents from root to the current node or vise versa
  • $includeRootNode - include or not the root node in the result
  • $indexResultBy - whether result should be indexed by id of the models

Getting parents ids:

               $node->getParentIds($includeRootNode = false)

Here:

  • Tre result will be node's ids in the order from root to current node's parent
  • $includeRootNode - include or not the id of the root node in the result
  1. Work with the siblings of the node

Get:

All siblings:

               $node->siblings($withCurrent = false, $indexResultBy = false)

Here:

  • $withCurrent - include or not current node in the result
  • $indexResultBy - whether result should be indexed by id of the models

One next:

               $node->next()

One previous:

               $node->prev()

All next:

               $node->nextAll()

All previous:

               $node->prevAll()

Get the position of current node among siblings (starting with 0):

               $node->index()	

Node properties

Over node you can perform the next checks:

Check whether the node is the root one:

               $node->isRoot()

Check whether the node is a leaf meaning it does not have any descendants:

               $node->isLeaf()

Check whether our node is the descendant of another node:

               $node->isDescendantOf($node)

      ,$node argument here might be ActiveRecord object or number - primary key of the node.

Check whether our node is the child of another node:

               $node->isChildOf($node)

      , argument $node as above

Check whether our node is the sibling to another one:

               $node->isSiblingOf($node)

      , argument $node could be only ActiveRecord model

About node you can get the next information too:

The full path to the node:

               $node->getFullPath()

      , this path includes what we hold in path field concatinated with the id
of the current node. For example for Animal(5) node this value will be '1/5'

Id of the node:

               $node->getId()

      , this wrapper is useful because it works for Root node also (gives negative id)

Level of the node:

               $node->getLevel()

Condition of the tree this node belongs to:

               $node->getTreeCondition()

      , it will return an array like ['treeid' => 2] for Menuitem table

Get the name of the fields this extension uses:

               $node->getIdFieldName()
               $node->getPathFieldName()
               $node->getLevelFieldName()
               $node->getWeightFieldName()

Inserting new and moving existed nodes

Common information

The common information you need to know about inserting/moving nodes:

  1. The same methods work both for inserting new nodes and for moving existed ones
  2. The signature of every method of modification is the following:
    • $node->appendTo($node, $runValidation = true, $runTransaction = true)
    • These methods physically save the model, so they work as ActiveRecord::save() and it means that there is possibility to validation check before saving using $runValidation
    • They return true/false whether operation succeded or not
    • Often the inserting/moving operation has the consequences that some other nodes will need to be changed and saved and it is handy to run all this operation in one transaction using $runTransaction.
    • Also there is the setting MaterializedPathBehavior::$modelScenarioForAffectedModelsForSavingProcess which allows to set up some scenario for these additionally changed models before they are saved
  3. In these operations Root node could be used only in operations adding/moving nodes to it
  4. All operations like prependTo/insertBefore(After) need the rebuilding of the weight field of all new siblings. Solving this task this extension does not do the work of searching free intervals of weight or anнthing like this, it rebuilds "weights" of all new siblings and saves them all
  5. Seeing the situation when there are many trees in the table:
    • when creating new nodes tree condition for them will be set the same as node relevant to which we add new one
    • you are not allowed to move existed nodes to another tree

So the very operations

Add/Move node to new parent at the position of the last child

               $model->appendTo($node, $runValidation = true, $runTransaction = true)
  • $model will be added/moved to new parent $node at the position of the last child
  • When moving, all $model's descendants will be moved along with it naturally

There is a "mirror" method to the above one when to the left node we can add child:

               $model->add($child, $runValidation = true, $runTransaction = true)

Examples: 1, 2, 3

Add/Move node to new parent at the position of the first child

               $model->prependTo($node, $runValidation = true, $runTransaction = true)
  • $model will be added/moved to new parent $node at the position of the first child

Example

Add/Move node to the position before another node

               $model->insertBefore($node, $runValidation = true, $runTransaction = true)
  • $model will be added/moved to the position before $node. Their parent will be the same

Example.

Add/Move node to the position after another node

               $model->insertAfter($node, $runValidation = true, $runTransaction = true)
  • $modelwill be added/moved to the position after$node`

Example.

Add/Move node to new parent at the position specified as a number

               $model->insertAsChildAtPosition($node, $position, $runValidation = true, $runTransaction = true)
  • $model will be added/moved to new parent $node and among it's children it will occupy $position position (counting from zero)
  • technically this is a wrapper around the methods above

Example

Deleting of a node

Deleting of a node is happening by means of existed in Yii method ActiveRecord::delete()

               $model->delete()

When deleting of a node the next MaterializedPathBehavior::$moveChildrenWnenDeletingParent behavior setting gives two options for node's descendants:

  1. true - descendants will be moved to parent of the node being deleted (by means of appendTo). This is default setting.
  2. false - descendants will be deleted along with it

Because delete() method is the framework's inner one if you want to run it in transaction you need to set up this for yourself following documentation. Have a look at the example of this setting at catalog model

In the catalog model example deleting operation was wrapped in transaction but if descendants are to be deleted too we are not interested to run all these additional deletions in nested transactions so by using the setting MaterializedPathBehavior::$modelScenarioForChildrenNodesWhenTheyDeletedAfterParent we can set some scenario for these descendant nodes before they are deleted ( different from scenario set up in transactions()).

Example

Service to manage trees

Common

This service gives additional functionality to manage trees when we need to manipulate many nodes.

We can get this service the next way:

use mgrechanik\yiimaterializedpath\ServiceInterface;
// trees managing service
$service = \Yii::createObject(ServiceInterface::class);

Or inject it using DI.
Technically singleton for this service is defined in the bootstrap of the extension: mgrechanik\yiimaterializedpath\tools\Bootstrap.

Building trees and their output

Common information

As we saw above $node->getDescendantsQuery() query (works for root node also if all tree is needed) gets needed amount of descendants of the node (through one database query). But we need to transform this query result array in a form handy to work with (and to be displayed too) as with a tree.

So in this service we transform this structure into two types of php trees:

  • buildTree, buildDescendantsTree builds hierarchical structure in the form of nodes of special type connected to one another. This structure is handy for recursive output of the tree in the form of nested <ul>-<li> lists
  • buildFlatTree based on the information above builds "flat" tree - simple one-dimensional array of nodes according their positions. It is handy for output of the tree at admin pages using one foreach - for <select> list or for Data Provider.

Hierarchical tree

               $service->buildTree($parent, $isArray = true, $exceptIds = [], $depth = null)
               $service->buildDescendantsTree($parent, $isArray = true, $exceptIds = [], $depth = null)

These methods will build the next tree:

  1. Common algorithm is the next
    • Choose the node from which we will start our descendants tree
    • Build the tree
    • Print it
  2. The result will be the array of objects of mgrechanik\yiimaterializedpath\tools\TreeNode type which are the node referring in children property to it's children
  3. buildTree builds the tree starting from $parent node when buildDescendantsTree starts the tree from children of the $parent node
  4. isArray - choose the format (array or AR object) in which we put our data into TreeNode::$node
  5. $exceptIds see above
  6. $depth see above
  7. This tree could be printed at the page in the form of nested <ul>-<li> list using simple widget like - mgrechanik\yiimaterializedpath\widgets\TreeToListWidget. This widget has very basic functionality and comes mostly as example but still it has the opportunity to create any label for tree item you may need

Example:

use mgrechanik\yiimaterializedpath\ServiceInterface;
use mgrechanik\yiimaterializedpath\tests\models\Animal;
use mgrechanik\yiimaterializedpath\widgets\TreeToListWidget;

$service = \Yii::createObject(ServiceInterface::class);

// 1) choose the model
$model1 = Animal::findOne(1);
// 2) build the tree
$tree = $service->buildTree($model1);

We will get a structure like this:

Array
(
    [0] => TreeNode Object
        (
            [node] => Array ([id] => 1, [name] => cat, ...))    // <---- The very node with id=1
            [parent] => 
            [children] => Array                                 // <---- This is it's children (ARRAY-Z)
                (
                    [0] => TreeNode Object
                        (
                            [node] => Array ([id] => 5, [name] => mouse, ...)
                            [parent] => ...
                            [children] => Array
                                (
                                    [0] => TreeNode Object
                                        (
                                            [node] => Array ([id] => 7, [name] => stag, ...)
                                            [parent] => ...
                                            [children] => Array ()
                                        )
                                )
                        )
                    [1] => TreeNode Object
                        (
                            [node] => Array ( [id] => 6, [name] => fox, ... )
                            [parent] => ...
                            [children] => Array ()
                        )
                )
        )
)

If tree had been built like this - $tree = $service->buildDescendantsTree($model1); the result would have been the ARRAY-Z above

// 3) print the tree:
print TreeToListWidget::widget(['tree' => $tree]);

We wiil get:

  • cat
    • mouse
      • stag
    • fox

Html of the code above is the next:

<ul>
    <li>cat
        <ul>
            <li>mouse
                <ul>
                    <li>stag</li>
                </ul>
            </li>
            <li>fox</li>
        </ul>
    </li>
</ul>

Example of output of ALL tree:

Код:

$root = $service->getRoot(Animal::class);
$tree = $service->buildDescendantsTree($root);
print TreeToListWidget::widget(['tree' => $tree]);

We will get:

  • cat
    • mouse
      • stag
    • fox
  • dog
  • snake
    • lion
    • hedgehog
  • bear

Example of the output of the two first levels of the tree (using $depth parameter):

Code:

$root = $service->getRoot(Animal::class);
$tree = $service->buildDescendantsTree($root, true, [], 2);
print TreeToListWidget::widget(['tree' => $tree]);

will print:

  • cat
    • mouse
    • fox
  • dog
  • snake
    • lion
    • hedgehog
  • bear

Flat tree

By term "flat" we would mean a tree in the form of simple array which could be outputted by one foreach in the form like this:

- root             
  - (1) cat        
    -- (5) mouse   
      --- (7) stag 
    -- (6) fox     
  - (2) dog        
  - (3) snake      
    -- (8) lion    
    -- (9) hedgehog
  - (4) bear 

This tree is created like this:

           $service->buildFlatTree($parent, $isArray = true, $includeItself = false, $indexBy = false, $exceptIds = [], $depth = null)

This method will build the next tree:

  1. Common algorithm is the next
    • Choose the node from which we will start our descendants tree
    • Build the tree
    • Print it
  2. The result will be the array of nodes represented like associative arrays ($isArray = true) or AR objects
  3. $includeItself sets up from what we start our tree - from $parent when $includeItself = true or from it's children
  4. $indexBy - whether to index result array by model's ids. Can be handy to be used together with Data Provider
  5. $exceptIds see above.
  6. $depth see above.

Example:

// 1) choose the node
$root = $service->getRoot(Animal::class);
// 2) build the tree
$tree = $service->buildFlatTree($root);

We will get the next array:

Array
(
    [0] => Array               // To make the keys equal to ids look at $indexBy above
        (
            [id] => 1
            [path] => 
            [level] => 1
            [weight] => 1
            [name] => cat
        )
    [1] => Array
        (
            [id] => 5
            [path] => 1/
            [level] => 2
            [weight] => 1
            [name] => mouse
        )
    [2] => Array
        (
            [id] => 7
            [path] => 1/5/
            [level] => 3
            [weight] => 1
            [name] => stag
        )
    // .....
    // .....
    [8] => Array
        (
            [id] => 4
            [path] => 
            [level] => 1
            [weight] => 4
            [name] => bear
        )
)

This array is ready (when used with $indexBy) to be given to Data Provider, for example have a look at the catalog view page - actionIndex - in catalog controller.

To transform this array into $items for Yii's built-in listBox we would need the next helper:

           $service->buildSelectItems($flatTreeArray, callable $createLabel, $indexKey = 'id', $isArray = true);
  1. $flatTreeArray - array we built by buildFlatTree
  2. $createLabel - anonymous function to create item label. This function receives node processed and returns string - the item's label we see in <select> list
  3. $indexKey - what field to use to index select item
  4. Result will be the array of options [id1 => label1, id2 => label2, ...]
// 3) Build select list
$items = $service->buildSelectItems($tree,
	function($node) {
		return ($node['id'] < 0) ? '- root' : str_repeat('  ', $node['level']) . str_repeat('-', $node['level']) . 
				' (' . $node['id'] . ') ' . $node['name'];
	}
); 

which will make the next list:

  - (1) cat        
    -- (5) mouse   
      --- (7) stag 
    -- (6) fox     
  - (2) dog        
  - (3) snake      
    -- (8) lion    
    -- (9) hedgehog
  - (4) bear 

Example of output of all tree including root:

$root = $service->getRoot(Animal::class);
$tree = $service->buildFlatTree($root, true, true);
$items = $service->buildSelectItems($tree,
	function($node) {
		return ($node['id'] < 0) ? '- root' : str_repeat('  ', $node['level']) . str_repeat('-', $node['level']) . 
				' (' . $node['id'] . ') ' . $node['name'];
	} 
); 

We will have:

- root             
  - (1) cat        
    -- (5) mouse   
      --- (7) stag 
    -- (6) fox     
  - (2) dog        
  - (3) snake      
    -- (8) lion    
    -- (9) hedgehog
  - (4) bear 

You can see example of this code working in the editing form of catalog element.

Cloning

           $service->cloneSubtree($sourceNode, $destNode, $withSourceNode = true, $scenario = null)
  1. Common:
    • Operation will be performed in transaction
    • Cloning is allowed only among models of the same type
    • You can clone into another tree
    • $sourceNode, $destNode - Active Record models or root nodes
  2. $sourceNode - the root of the subtree we are cloning
  3. $destNode - the node to which we are cloning
  4. $withSourceNode - whether we start cloning from $sourceNode itself or from it's children.
    For example we need to set it to false if $sourceNode is the root. All tree will be cloned
  5. $scenario - optionally we might set scenario to cloned nodes before they are saved

Cloning examples

Other opportunities

Get the root of the tree

           $service->getRoot($className, $treeCondition = [])
  1. $className - ActiveRecord model name
  2. $treeCondition - tree condition in the case when table holds many trees. It is an array like ['treeid' => 1]

Get any node by it's id

           $service->getModelById($className, $id, $treeCondition = [])
  1. It is a wrapper over $className::findOne($id) which is able to find root node by it's negative $id. It is used when we have a html form and root node could be choosen there along with other nodes
  2. $className - ActiveRecord model name
  3. $id - unique identifier of the model or negative number as root's id
  4. $treeCondition - tree condition in the case when table holds many trees. You need to give it only if tree condition is made of more than one field. For conditions with one field like - ['treeid' => 2] omit this parameter because it will be figured out from $id.

Get ids of the nodes of some subtree

           $service->buildSubtreeIdRange($parent, $includeItself = false, $exceptIds = [], $depth = null)
  1. Allows to get array of node ids for certain descendants of $parent node
  2. $includeItself - whether to include id of the $parent
  3. $exceptIds see above.
  4. $depth see above.
  5. This functionality is interesting to work together with yii\validators\RangeValidator

Tree condition of the node

           $service->getTreeCondition($model)
  1. $model - node we are checking
  2. It will return array like ['treeid' => 1] - tree condition by which $model node belongs to it's tree

Get the parent's id from path

           $service->getParentidFromPath($path)
  1. $path - path
  2. It will return last id from the path or null if path is empty

Appendix A: Building a catalog example

Example about how to create/edit tree nodes at admin pages and display trees is shown in the Creating a catalog at Yii2 article where you can see all this architecture in work.

Appendix B: Examples of working with API

Common

All examples are going to work with Animal table at following start state:

- root             
  - (1) cat        
    -- (5) mouse   
      --- (7) stag 
    -- (6) fox     
  - (2) dog        
  - (3) snake      
    -- (8) lion    
    -- (9) hedgehog
  - (4) bear 

Also implicitly there is the next beginning of all code examples:

use mgrechanik\yiimaterializedpath\ServiceInterface;
use mgrechanik\yiimaterializedpath\tests\models\Animal;
// tree managing service
$service = \Yii::createObject(ServiceInterface::class);

Work with Root node

Add new node to Root node using add() or appendTo()

Whether this way:

$root = $service->getRoot(Animal::class);
$root->add(new Animal(['name' => 'new']));

or another:

$root = $service->getRoot(Animal::class);
$newModel = new Animal(['name' => 'new']);
$newModel->appendTo($root);

The next change will happen:

- root                        - root
  - (1) cat                     - (1) cat
    -- (5) mouse                  -- (5) mouse
      --- (7) stag                  --- (7) stag
    -- (6) fox                    -- (6) fox
  - (2) dog            ==>      - (2) dog
  - (3) snake                   - (3) snake
    -- (8) lion                   -- (8) lion
    -- (9) hedgehog               -- (9) hedgehog
  - (4) bear                    - (4) bear
                                - (10) new      

Move existed node to the root using appendTo()

$model7 = Animal::findOne(7);
$root = $model7->getRoot();
$model7->appendTo($root);

The next change will happen:

- root                        - root
  - (1) cat                     - (1) cat
    -- (5) mouse                  -- (5) mouse
      --- (7) stag                -- (6) fox
    -- (6) fox                  - (2) dog
  - (2) dog            ==>      - (3) snake
  - (3) snake                     -- (8) lion
    -- (8) lion                   -- (9) hedgehog
    -- (9) hedgehog             - (4) bear
  - (4) bear                    - (7) stag

appendTo()

Moving the subtree into new position:

$model1 = Animal::findOne(1);
$model3 = Animal::findOne(3);
$model1->appendTo($model3);

The next change will happen:

- root                        - root
  - (1) cat                     - (2) dog
    -- (5) mouse                - (3) snake
      --- (7) stag                -- (8) lion
    -- (6) fox                    -- (9) hedgehog
  - (2) dog            ==>        -- (1) cat
  - (3) snake                       --- (5) mouse
    -- (8) lion                       ---- (7) stag
    -- (9) hedgehog                 --- (6) fox
  - (4) bear                    - (4) bear

prependTo()

Add new node as first child of another node:

$model1 = Animal::findOne(1);
$newModel = new Animal(['name' => 'new']);
$newModel->prependTo($model1);

The next change will happen:

- root                        - root
  - (1) cat                     - (1) cat
    -- (5) mouse                  -- (12) new
      --- (7) stag                -- (5) mouse
    -- (6) fox                      --- (7) stag
  - (2) dog            ==>        -- (6) fox
  - (3) snake                   - (2) dog
    -- (8) lion                 - (3) snake
    -- (9) hedgehog               -- (8) lion
  - (4) bear                      -- (9) hedgehog
                                - (4) bear

insertBefore()

Add new node before another node:

$model3 = Animal::findOne(3);
$newModel = new Animal(['name' => 'new']);
$newModel->insertBefore($model3);

The next change will happen:

- root                        - root
  - (1) cat                     - (1) cat
    -- (5) mouse                  -- (5) mouse
      --- (7) stag                  --- (7) stag
    -- (6) fox                    -- (6) fox
  - (2) dog            ==>      - (2) dog
  - (3) snake                   - (13) new
    -- (8) lion                 - (3) snake
    -- (9) hedgehog               -- (8) lion
  - (4) bear                      -- (9) hedgehog
                                - (4) bear

insertAfter()

Move existed node right after another node:

$model7 = Animal::findOne(7);
$model8 = Animal::findOne(8);
$model7->insertAfter($model8);

The next change will happen:

- root                        - root
  - (1) cat                     - (1) cat
    -- (5) mouse                  -- (5) mouse
      --- (7) stag                -- (6) fox
    -- (6) fox                  - (2) dog
  - (2) dog            ==>      - (3) snake
  - (3) snake                     -- (8) lion
    -- (8) lion                   -- (7) stag
    -- (9) hedgehog               -- (9) hedgehog
  - (4) bear                    - (4) bear

insertAsChildAtPosition()

Insert new model as third child of the root (position 2):

$root = $service->getRoot(Animal::class);
$newModel = new Animal(['name' => 'new']);
$newModel->insertAsChildAtPosition($root, 2);

The next change will happen:

- root                        - root                        
  - (1) cat                     - (1) cat                   
    -- (5) mouse                  -- (5) mouse
      --- (7) stag                  --- (7) stag
    -- (6) fox                    -- (6) fox
  - (2) dog            ==>      - (2) dog                   
  - (3) snake                   - (14) new                  
    -- (8) lion                 - (3) snake
    -- (9) hedgehog               -- (8) lion
  - (4) bear                      -- (9) hedgehog
                                - (4) bear

delete()

Delete existing node with it's descendants moving to it's parent:

$model3 = Animal::findOne(3);
$model3->delete()

The next change will happen:

- root                        - root
  - (1) cat                     - (1) cat
    -- (5) mouse                  -- (5) mouse
      --- (7) stag                  --- (7) stag
    -- (6) fox                    -- (6) fox
  - (2) dog            ==>      - (2) dog
  - (3) snake                   - (4) bear
    -- (8) lion                 - (8) lion
    -- (9) hedgehog             - (9) hedgehog
  - (4) bear                  

Cloning

Cloning one node

$model7 = Animal::findOne(7);
$model8 = Animal::findOne(8);
$service->cloneSubtree($model7, $model8);

The next change will happen:

- root                        - root
  - (1) cat                     - (1) cat
    -- (5) mouse                  -- (5) mouse
      --- (7) stag                  --- (7) stag
    -- (6) fox                    -- (6) fox
  - (2) dog            ==>      - (2) dog
  - (3) snake                   - (3) snake
    -- (8) lion                   -- (8) lion
    -- (9) hedgehog                 --- (197) stag
  - (4) bear                      -- (9) hedgehog
                                - (4) bear                

Cloning all subtree

Cloning all subtree (default mode):

$model1 = Animal::findOne(1);
$model8 = Animal::findOne(8);
$service->cloneSubtree($model1, $model8);

The next change will happen:

- root                        - root
  - (1) cat                     - (1) cat
    -- (5) mouse                  -- (5) mouse
      --- (7) stag                  --- (7) stag
    -- (6) fox                    -- (6) fox
  - (2) dog            ==>      - (2) dog
  - (3) snake                   - (3) snake
    -- (8) lion                   -- (8) lion
    -- (9) hedgehog                 --- (198) cat
  - (4) bear                          ---- (199) mouse
                                        ----- (200) stag
                                      ---- (201) fox
                                  -- (9) hedgehog
                                - (4) bear               

Cloning subtree without it's root

Cloning subtree starting with the children of the source node:

$model1 = Animal::findOne(1);
$model8 = Animal::findOne(8);
$service->cloneSubtree($model1, $model8, false);

The next change will happen:

- root                        - root
  - (1) cat                     - (1) cat
    -- (5) mouse                  -- (5) mouse
      --- (7) stag                  --- (7) stag
    -- (6) fox                    -- (6) fox
  - (2) dog            ==>      - (2) dog
  - (3) snake                   - (3) snake
    -- (8) lion                   -- (8) lion
    -- (9) hedgehog                 --- (202) mouse
  - (4) bear                          ---- (203) stag
                                    --- (204) fox
                                  -- (9) hedgehog
                                - (4) bear              

Dublicate descendants of the node

$model1 = Animal::findOne(1);
$service->cloneSubtree($model1, $model1, false);

The next change will happen:

- root                        - root
  - (1) cat                     - (1) cat
    -- (5) mouse                  -- (5) mouse
      --- (7) stag                  --- (7) stag
    -- (6) fox                    -- (6) fox
  - (2) dog            ==>        -- (205) mouse
  - (3) snake                       --- (206) stag
    -- (8) lion                   -- (207) fox
    -- (9) hedgehog             - (2) dog
  - (4) bear                    - (3) snake
                                  -- (8) lion
                                  -- (9) hedgehog
                                - (4) bear             

Cloning all tree into a new tree

Original state of Menuitem table is shown here.
Cloning all tree nodes into a new tree:

// root of not empty tree
$root1 = $service->getRoot(Menuitem::class, ['treeid' => 1]);
// root of new and empty tree
$root5 = $service->getRoot(Menuitem::class, ['treeid' => 5]);
// cloning by starting from root1's children
$service->cloneSubtree($root1, $root5, false);

The next change will happen:

- root1                     
  - (1) red                   
    -- (4) black              
    -- (5) yellow    ==>      ...   
  - (2) green                
    -- (6) blue              
  - (3) brown                
  
- root5                     - root5
                              - (32) red
                                -- (33) black
                     ==>        -- (34) yellow
                              - (35) green
                                -- (36) blue
                              - (37) brown