I have an array in PHP, which looks like this:
array (
    [0] => array (
        [id] => 1
        [title] => "Title 1"
        [parent_id] => NULL
        [depth] => 0
    )
    [1] => array (
        [id] => 2
        [title] => "Title 2"
        [parent_id] => NULL
        [depth] => 0
    )
    [2] => array (
        [id] => 3
        [title] => "Title 3"
        [parent_id] => 2
        [depth] => 1
    )
    [3] => array (
        [id] => 4
        [title] => "Title 4"
        [parent_id] => 2
        [depth] => 1
    )
    [4] => array (
        [id] => 5
        [title] => "Title 5"
        [parent_id] => NULL
        [depth] => 0
    )
    [5] => array (
        [id] => 6
        [title] => "Title 6"
        [parent_id] => 4
        [depth] => 2
    )
)
What i want to do is iterate over this array and create a nested <ol> list from it. So the result should look like this:
<ol>
    <li>Title 1</li> // id = 1
    <li>Title 2</li> // id = 2
    <ol>
        <li>Title 3</li> // id = 3 -> parent_id = 2
        <li>Title 4</li> // id = 4 -> parent_id = 2
        <ol>
            <li>Title 6</li> // id = 6 -> parent_id = 4
        </ol>
    </ol>
    <li>Title 5</li> // id = 5
</ol>
I've been trying to think of a way how i could get this done. But so far every attempt failed...
Anyone any idea how i can create such a nested <ol> list from an array like that?
Please note that i do not have any control on the given data. I simply make a call to an API and it returns json data, which i convert to an array. And the array looks exactly like the one i described.
You should use recursion:
First the array in 'php' syntax:
<?php
$a=array (
    '0' => array (
        'id' => 1,
        'title' => "Title 1",
        'parent_id' => 'NULL',
        'depth' => 0
    ),
    '1' => array (
        'id' => 2,
        'title' => "Title 2",
        'parent_id' => 'NULL',
        'depth' => 0
    ),
    '2' => array (
        'id' => 3,
        'title' => "Title 3",
        'parent_id' => 2,
        'depth' => 1
    ),
    '3' => array (
        'id' => 4,
        'title' => "Title 4",
        'parent_id' => 2,
        'depth' => 1
    ),
    '4' => array (
        'id' => 5,
        'title' => "Title 5",
        'parent_id' => 'NULL',
        'depth' => 0
    ),
    '5' => array (
        'id' => 6,
        'title' => "Title 6",
        'parent_id' => 4,
        'depth' => 0
    )
);
Here the code:
$level = 'NULL';
function r( $a, $level) {
   $r = "<ol>";
   foreach ( $a as $i ) {
       if ($i['parent_id'] == $level ) {
          $r = $r . "<li>" . $i['title'] . r( $a, $i['id'] ) . "</li>";
       }
   }
   $r = $r . "</ol>";
   return $r;
}
print r( $a, $level );
?>
The results:
<ol><li>Title 1<ol></ol></li><li>Title 2<ol><li>Title 3<ol>
</ol></li><li>Title 4<ol><li>Title 6<ol></ol></li></ol></li></ol></li><li>Title 5
<ol></ol></li></ol>
EDITED AFTER CHECK AS SOLUTION
To avoid empty leafs:
function r( $a, $level) {
   $r = '' ;
   foreach ( $a as $i ) {
       if ($i['parent_id'] == $level ) {
          $r = $r . "<li>" . $i['title'] . r( $a, $i['id'] ) . "</li>";
       }
   }
   return ($r==''?'':"<ol>". $r . "</ol>");
}
                        You can try the following
$array = array (
    "0" => array (
        "id" => 1,
        "title" => "Title 1",
        "parent_id" => NULL,
        "depth" => 0
    ),
    "1" => array (
        "id" => 2,
        "title" => "Title 2",
        "parent_id" => NULL,
        "depth" => 0
    ),
    "2" => array (
        "id" => 3,
        "title" => "Title 3",
        "parent_id" => 2,
        "depth" => 1
    ),
    "3" => array (
        "id" => 4,
        "title" => "Title 4",
        "parent_id" => 2,
        "depth" => 1
    ),
    "4" => array (
        "id" => 5,
        "title" => "Title 5",
        "parent_id" => NULL,
        "depth" => 0
    ),
    "5" => array (
        "id" => 6,
        "title" => "Title 6",
        "parent_id" => 4,
        "depth" => 0
    )
);
echo(make($array));
Output
<ol>
    <li>Title 1</li>
    <li>Title 2</li>
    <ol>
        <li>Title 3</li>
        <li>Title 4</li>
        <ol>
            <li>Title 6</li>
        </ol>
    </ol>
    <li>Title 5</li>
</ol>
Function Used
function make(array $array, $no = 0) {
    $child = hasChildren($array, $no);
    if (empty($child))
        return "";
    $content = "<ol>\n";
    foreach ( $child as $value ) {
        $content .= sprintf("\t<li>%s</li>\n", $value['title']);
        $content .= make($array, $value['id']);
    }
    $content .= "</ol>\n";
    return $content;
}
function hasChildren($array, $id) {
    return array_filter($array, function ($var) use($id) {
        return $var['parent_id'] == $id;
    });
}
See Live Demo
The following array:
Array
(
    [0] => Content
    [1] => Array
        (
            [0] => International
            [1] => Array
                (
                    [0] => Mexico
                    [1] => Array
                        (
                            [0] => Tamaulipas
                        )
                    [2] => USA
                )
        )
)
With this function:
function r($element) {
    foreach ($element as $value) {
        if (!is_array($value)) {
            echo "<li>";
            echo $value;
        } else {
            echo "<ul>";
            r($value);
            echo "</li>";
            echo "</ul>";
        }
    }
}
PHP code:
echo "<ul>";
r($array);
echo "</ul>";
Returns:
<ul>
    <li>Public</li>
    <li>User</li>
    <li>Content
        <ul>
            <li>International
                <ul>
                    <li>Mexico
                        <ul>
                            <li>Tamaulipas</li>
                        </ul>  
                    </li>
                    <li>USA</li>
                </ul>
            </li>
        </ul>
    </li>
</ul>
                        If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With