Para registrar los movimientos requerimos especificar que tipo de movimiento es como puede ser entradas por compra, salidas por venta, entrada por devolución de cliente, saluda por devolución a proveedor

Para ello crearemos el siguiente catalogo de Tipos de movimiento con los siguientes campos

  • Empresa
  • Descripción
  • tipo
  • Es traspaso

Primero creamos el archivo de migración App/Database/2023-08-17222335_Tipos_movimientos_inventario.php con el siguiente código

<?php

namespace App\Database\Migrations;

use CodeIgniter\Database\Migration;

class Tipos_movimientos_inventario extends Migration {

    public function up() {
        // Tipos_movimientos_inventario
        $this->forge->addField([
            'id' => ['type' => 'int', 'constraint' => 11, 'unsigned' => true, 'auto_increment' => true],
            'idEmpresa' => ['type' => 'bigint', 'constraint' => 20, 'null' => true],
            'descripcion' => ['type' => 'varchar', 'constraint' => 256, 'null' => true],
            'tipo' => ['type' => 'varchar', 'constraint' => 3, 'null' => true],
            'esTraspaso' => ['type' => 'varchar', 'constraint' => 3, 'null' => true],
            'created_at' => ['type' => 'datetime', 'null' => true],
            'updated_at' => ['type' => 'datetime', 'null' => true],
            'deleted_at' => ['type' => 'datetime', 'null' => true],
        ]);
        $this->forge->addKey('id', true);
        $this->forge->createTable('tipos_movimientos_inventario', true);
    }

    public function down() {
        $this->forge->dropTable('tipos_movimientos_inventario', true);
    }

}

Creamos el archivo modelo para el acceso a la base de datos App/Models/Tipos_movimientos_inventarioModel.php con el siguiente código

<?php

namespace App\Models;

use CodeIgniter\Model;

class Tipos_movimientos_inventarioModel extends Model {

    protected $table = 'tipos_movimientos_inventario';
    protected $primaryKey = 'id';
    protected $useAutoIncrement = true;
    protected $returnType = 'array';
    protected $useSoftDeletes = true;
    protected $allowedFields = ['id', 'idEmpresa', 'descripcion', 'tipo', 'esTraspaso', 'created_at', 'updated_at', 'deleted_at'];
    protected $useTimestamps = true;
    protected $createdField = 'created_at';
    protected $deletedField = 'deleted_at';
    protected $validationRules = [
    ];
    protected $validationMessages = [];
    protected $skipValidation = false;

    public function mdlGetTipos_movimientos_inventario($idEmpresas) {

        $result = $this->db->table('tipos_movimientos_inventario a, empresas b')
                ->select('a.id,a.idEmpresa,a.descripcion,a.tipo,a.esTraspaso,a.created_at,a.updated_at,a.deleted_at ,b.nombre as nombreEmpresa')
                ->where('a.idEmpresa', 'b.id', FALSE)
                ->whereIn('a.idEmpresa', $idEmpresas);

        return $result;
    }

}

Para el controlador creamos el archivo App/Controllers/Tipos_movimientos_inventarioController.php con el siguiente código

<?php

namespace App\Controllers;

use App\Controllers\BaseController;
use \App\Models\{
    Tipos_movimientos_inventarioModel
};
use App\Models\LogModel;
use CodeIgniter\API\ResponseTrait;
use App\Models\EmpresasModel;

class Tipos_movimientos_inventarioController extends BaseController {

    use ResponseTrait;

    protected $log;
    protected $tipos_movimientos_inventario;

    public function __construct() {
        $this->tipos_movimientos_inventario = new Tipos_movimientos_inventarioModel();
        $this->log = new LogModel();
        $this->empresa = new EmpresasModel();
        helper('menu');
        helper('utilerias');
    }

    public function index() {



        helper('auth');

        $idUser = user()->id;
        $titulos["empresas"] = $this->empresa->mdlEmpresasPorUsuario($idUser);

        if (count($titulos["empresas"]) == "0") {

            $empresasID[0] = "0";
        } else {

            $empresasID = array_column($titulos["empresas"], "id");
        }




        if ($this->request->isAJAX()) {
            $datos = $this->tipos_movimientos_inventario->mdlGetTipos_movimientos_inventario($empresasID);

            return \Hermawan\DataTables\DataTable::of($datos)->toJson(true);
        }
        $titulos["title"] = lang('tipos_movimientos_inventario.title');
        $titulos["subtitle"] = lang('tipos_movimientos_inventario.subtitle');
        return view('tipos_movimientos_inventario', $titulos);
    }

    /**
     * Read Tipos_movimientos_inventario
     */
    public function getTipos_movimientos_inventario() {

        helper('auth');

        $idUser = user()->id;
        $titulos["empresas"] = $this->empresa->mdlEmpresasPorUsuario($idUser);

        if (count($titulos["empresas"]) == "0") {

            $empresasID[0] = "0";
        } else {

            $empresasID = array_column($titulos["empresas"], "id");
        }


        $idTipos_movimientos_inventario = $this->request->getPost("idTipos_movimientos_inventario");
        $datosTipos_movimientos_inventario = $this->tipos_movimientos_inventario->whereIn('idEmpresa', $empresasID)
                        ->where("id", $idTipos_movimientos_inventario)->first();
        echo json_encode($datosTipos_movimientos_inventario);
    }

    /**
     * Save or update Tipos_movimientos_inventario
     */
    public function save() {
        helper('auth');
        $userName = user()->username;
        $idUser = user()->id;
        $datos = $this->request->getPost();
        if ($datos["idTipos_movimientos_inventario"] == 0) {
            try {
                if ($this->tipos_movimientos_inventario->save($datos) === false) {
                    $errores = $this->tipos_movimientos_inventario->errors();
                    foreach ($errores as $field => $error) {
                        echo $error . " ";
                    }
                    return;
                }
                $dateLog["description"] = lang("vehicles.logDescription") . json_encode($datos);
                $dateLog["user"] = $userName;
                $this->log->save($dateLog);
                echo "Guardado Correctamente";
            } catch (\PHPUnit\Framework\Exception $ex) {
                echo "Error al guardar " . $ex->getMessage();
            }
        } else {
            if ($this->tipos_movimientos_inventario->update($datos["idTipos_movimientos_inventario"], $datos) == false) {
                $errores = $this->tipos_movimientos_inventario->errors();
                foreach ($errores as $field => $error) {
                    echo $error . " ";
                }
                return;
            } else {
                $dateLog["description"] = lang("tipos_movimientos_inventario.logUpdated") . json_encode($datos);
                $dateLog["user"] = $userName;
                $this->log->save($dateLog);
                echo "Actualizado Correctamente";
                return;
            }
        }
        return;
    }

    /**
     * Delete Tipos_movimientos_inventario
     * @param type $id
     * @return type
     */
    public function delete($id) {
        $infoTipos_movimientos_inventario = $this->tipos_movimientos_inventario->find($id);
        helper('auth');
        $userName = user()->username;
        if (!$found = $this->tipos_movimientos_inventario->delete($id)) {
            return $this->failNotFound(lang('tipos_movimientos_inventario.msg.msg_get_fail'));
        }
        $this->tipos_movimientos_inventario->purgeDeleted();
        $logData["description"] = lang("tipos_movimientos_inventario.logDeleted") . json_encode($infoTipos_movimientos_inventario);
        $logData["user"] = $userName;
        $this->log->save($logData);
        return $this->respondDeleted($found, lang('tipos_movimientos_inventario.msg_delete'));
    }

    public function getTiposMovimientoAjax() {

        helper('auth');

        $idUser = user()->id;
        $titulos["empresas"] = $this->empresa->mdlEmpresasPorUsuario($idUser);

        if (count($titulos["empresas"]) == "0") {

            $empresasID[0] = "0";
        } else {

            $empresasID = array_column($titulos["empresas"], "id");
        }

        $request = service('request');
        $postData = $request->getPost();

        $response = array();

        // Read new token and assign in $response['token']
        $response['token'] = csrf_hash();

        if (!isset($postData['searchTerm'])) {
            // Fetch record

            $listTiposMovimiento = $this->tipos_movimientos_inventario->select('id,descripcion')->where("deleted_at", null)
                    ->where("idEmpresa", $postData['idEmpresa'])
                    ->orderBy('id')
                    ->orderBy('descripcion')
                    ->findAll(10);
        } else {
            $searchTerm = $postData['searchTerm'];

            // Fetch record

            $listTiposMovimiento = $this->tipos_movimientos_inventario->select('id,descripcion')
                    ->where("deleted_at", null)
                    ->where("idEmpresa", $postData['idEmpresa'])
                    ->like('descripcion', $searchTerm)
                    ->orLike('id', $searchTerm)
                    ->findAll(10);
        }

        $data = array();
        foreach ($listTiposMovimiento as $tipoMovimiento) {
            $data[] = array(
                "id" => $tipoMovimiento['id'],
                "text" => $tipoMovimiento['descripcion'],
            );
        }

        $response['data'] = $data;

        return $this->response->setJSON($response);
    }

}

Para la vista principal creamos el siguiente fichero en App/Views/tipos_movimientos_inventario.php

<?= $this->include('julio101290\boilerplate\Views\load\select2') ?>
<?= $this->include('julio101290\boilerplate\Views\load\datatables') ?>
<?= $this->include('julio101290\boilerplate\Views\load\nestable') ?>
<!-- Extend from layout index -->
<?= $this->extend('julio101290\boilerplate\Views\layout\index') ?>

<!-- Section content -->
<?= $this->section('content') ?>

<?= $this->include('modulesTipos_movimientos_inventario/modalCaptureTipos_movimientos_inventario') ?>

<!-- SELECT2 EXAMPLE -->
<div class="card card-default">
    <div class="card-header">
        <div class="float-right">
            <div class="btn-group">

                <button class="btn btn-primary btnAddTipos_movimientos_inventario" data-toggle="modal" data-target="#modalAddTipos_movimientos_inventario"><i class="fa fa-plus"></i>

                    <?= lang('tipos_movimientos_inventario.add') ?>

                </button>

            </div>
        </div>
    </div>
    <div class="card-body">
        <div class="row">
            <div class="col-md-12">
                <div class="table-responsive">
                    <table id="tableTipos_movimientos_inventario" class="table table-striped table-hover va-middle tableTipos_movimientos_inventario">
                        <thead>
                            <tr>

                                <th>#</th>
                                <th><?= lang('tipos_movimientos_inventario.fields.idEmpresa') ?></th>
                                <th><?= lang('tipos_movimientos_inventario.fields.descripcion') ?></th>
                                <th><?= lang('tipos_movimientos_inventario.fields.tipo') ?></th>
                                <th><?= lang('tipos_movimientos_inventario.fields.esTraspaso') ?></th>
                                <th><?= lang('tipos_movimientos_inventario.fields.created_at') ?></th>
                                <th><?= lang('tipos_movimientos_inventario.fields.updated_at') ?></th>
                                <th><?= lang('tipos_movimientos_inventario.fields.deleted_at') ?></th>

                                <th><?= lang('tipos_movimientos_inventario.fields.actions') ?> </th>

                            </tr>
                        </thead>
                        <tbody>
                        </tbody>
                    </table>
                </div>
            </div>
        </div>
    </div>
</div>
<!-- /.card -->

<?= $this->endSection() ?>


<?= $this->section('js') ?>
<script>
    /**
     * Cargamos la tabla
     */

    var tableTipos_movimientos_inventario = $('#tableTipos_movimientos_inventario').DataTable({
        processing: true,
        serverSide: true,
        responsive: true,
        autoWidth: false,
        order: [
            [1, 'asc']
        ],

        ajax: {
            url: '<?= base_url('admin/tipos_movimientos_inventario') ?>',
            method: 'GET',
            dataType: "json"
        },
        columnDefs: [{
                orderable: false,
                targets: [8],
                searchable: false,
                targets: [8]

            }],
        columns: [{
                'data': 'id'
            },

            {
                'data': 'nombreEmpresa'
            },

            {
                'data': 'descripcion'
            },

            {
                'data': 'tipo'
            },

            {
                'data': 'esTraspaso'
            },

            {
                'data': 'created_at'
            },

            {
                'data': 'updated_at'
            },

            {
                'data': 'deleted_at'
            },

            {
                "data": function (data) {
                    return `<td class="text-right py-0 align-middle">
                         <div class="btn-group btn-group-sm">
                             <button class="btn btn-warning btnEditTipos_movimientos_inventario" data-toggle="modal" idTipos_movimientos_inventario="${data.id}" data-target="#modalAddTipos_movimientos_inventario">  <i class=" fa fa-edit"></i></button>
                             <button class="btn btn-danger btn-delete" data-id="${data.id}"><i class="fas fa-trash"></i></button>
                         </div>
                         </td>`
                }
            }
        ]
    });



    $(document).on('click', '#btnSaveTipos_movimientos_inventario', function (e) {


        var idTipos_movimientos_inventario = $("#idTipos_movimientos_inventario").val();
        var idEmpresa = $("#idEmpresa").val();
        var descripcion = $("#descripcion").val();
        var tipo = $("#tipo").val();
        var esTraspaso = $("#esTraspaso").val();


        if (idEmpresa == 0) {

            Toast.fire({
                icon: 'error',
                title: "Tiene que seleccionar la empresa"
            });
            return;
        }

        $("#btnSaveTipos_movimientos_inventario").attr("disabled", true);

        var datos = new FormData();
        datos.append("idTipos_movimientos_inventario", idTipos_movimientos_inventario);
        datos.append("idEmpresa", idEmpresa);
        datos.append("descripcion", descripcion);
        datos.append("tipo", tipo);
        datos.append("esTraspaso", esTraspaso);


        $.ajax({

            url: "<?= base_url('admin/tipos_movimientos_inventario/save') ?>",
            method: "POST",
            data: datos,
            cache: false,
            contentType: false,
            processData: false,
            success: function (respuesta) {
                if (respuesta.match(/Correctamente.*/)) {

                    Toast.fire({
                        icon: 'success',
                        title: "Guardado Correctamente"
                    });

                    tableTipos_movimientos_inventario.ajax.reload();
                    $("#btnSaveTipos_movimientos_inventario").removeAttr("disabled");


                    $('#modalAddTipos_movimientos_inventario').modal('hide');
                } else {

                    Toast.fire({
                        icon: 'error',
                        title: respuesta
                    });

                    $("#btnSaveTipos_movimientos_inventario").removeAttr("disabled");


                }

            }

        }

        )

    });



    /**
     * Carga datos actualizar
     */


    /*=============================================
     EDITAR Tipos_movimientos_inventario
     =============================================*/
    $(".tableTipos_movimientos_inventario").on("click", ".btnEditTipos_movimientos_inventario", function () {

        var idTipos_movimientos_inventario = $(this).attr("idTipos_movimientos_inventario");

        var datos = new FormData();
        datos.append("idTipos_movimientos_inventario", idTipos_movimientos_inventario);

        $.ajax({

            url: "<?= base_url('admin/tipos_movimientos_inventario/getTipos_movimientos_inventario') ?>",
            method: "POST",
            data: datos,
            cache: false,
            contentType: false,
            processData: false,
            dataType: "json",
            success: function (respuesta) {
                $("#idTipos_movimientos_inventario").val(respuesta["id"]);

                $("#idEmpresa").val(respuesta["idEmpresa"]);
                $("#descripcion").val(respuesta["descripcion"]);
                $("#tipo").val(respuesta["tipo"]);
                $("#esTraspaso").val(respuesta["esTraspaso"]);


            }

        })

    })


    /*=============================================
     ELIMINAR tipos_movimientos_inventario
     =============================================*/
    $(".tableTipos_movimientos_inventario").on("click", ".btn-delete", function () {

        var idTipos_movimientos_inventario = $(this).attr("data-id");

        Swal.fire({
            title: '<?= lang('boilerplate.global.sweet.title') ?>',
            text: "<?= lang('boilerplate.global.sweet.text') ?>",
            icon: 'warning',
            showCancelButton: true,
            confirmButtonColor: '#3085d6',
            cancelButtonColor: '#d33',
            confirmButtonText: '<?= lang('boilerplate.global.sweet.confirm_delete') ?>'
        })
                .then((result) => {
                    if (result.value) {
                        $.ajax({
                            url: `<?= base_url('admin/tipos_movimientos_inventario') ?>/` + idTipos_movimientos_inventario,
                            method: 'DELETE',
                        }).done((data, textStatus, jqXHR) => {
                            Toast.fire({
                                icon: 'success',
                                title: jqXHR.statusText,
                            });


                            tableTipos_movimientos_inventario.ajax.reload();
                        }).fail((error) => {
                            Toast.fire({
                                icon: 'error',
                                title: error.responseJSON.messages.error,
                            });
                        })
                    }
                })
    })

    $(function () {
        $("#modalAddTipos_movimientos_inventario").draggable();

    });
</script>
<?= $this->endSection() ?>

Creamos el archivo App/Views/modulesTipos_movimientos_inventario/modalCaptureTipos_movimientos_inventario que contiene el modal para registrar los tipos de movimiento

<!-- Modal Tipos_movimientos_inventario -->
<div class="modal fade" id="modalAddTipos_movimientos_inventario" tabindex="-1" role="dialog" aria-labelledby="modalAddTipos_movimientos_inventario" aria-hidden="true">
    <div class="modal-dialog modal-lg" role="document">
        <div class="modal-content">
            <div class="modal-header">
                <h5 class="modal-title"><?= lang('tipos_movimientos_inventario.createEdit') ?></h5>
                <button type="button" class="close" data-dismiss="modal" aria-label="Close">
                    <span aria-hidden="true">&times;</span>
                </button>
            </div>
            <div class="modal-body">
                <form id="form-tipos_movimientos_inventario" class="form-horizontal">
                    <input type="hidden" id="idTipos_movimientos_inventario" name="idTipos_movimientos_inventario" value="0">

                    <div class="form-group row">
                        <label for="idEmpresa" class="col-sm-2 col-form-label">Empresa</label>
                        <div class="col-sm-10">
                            <div class="input-group">
                                <div class="input-group-prepend">
                                    <span class="input-group-text"><i class="fas fa-pencil-alt"></i></span>
                                </div>

                                <select class="form-control" name="idEmpresa" id="idEmpresa" style = "width:80%;">
                                    <option value="0">Seleccione empresa</option>
                                    <?php
                                    foreach ($empresas as $key => $value) {

                                        echo "<option value='$value[id]' selected>$value[id] - $value[nombre] </option>  ";
                                    }
                                    ?>

                                </select>

                            </div>
                        </div>
                    </div>

                    <div class="form-group row">
                        <label for="descripcion" class="col-sm-2 col-form-label"><?= lang('tipos_movimientos_inventario.fields.descripcion') ?></label>
                        <div class="col-sm-10">
                            <div class="input-group">
                                <div class="input-group-prepend">
                                    <span class="input-group-text"><i class="fas fa-pencil-alt"></i></span>
                                </div>
                                <input type="text" name="descripcion" id="descripcion" class="form-control <?= session('error.descripcion') ? 'is-invalid' : '' ?>" value="<?= old('descripcion') ?>" placeholder="<?= lang('tipos_movimientos_inventario.fields.descripcion') ?>" autocomplete="off">
                            </div>
                        </div>
                    </div>


                    <div class="form-group row">
                        <label for="tipo" class="col-sm-2 col-form-label">Tipo</label>
                        <div class="col-sm-10">
                            <div class="input-group">
                                <div class="input-group-prepend">
                                    <span class="input-group-text"><i class="fas fa-pencil-alt"></i></span>
                                </div>

                                <select class="form-control" name="tipo" id="tipo" style = "width:80%;">
                                    <option value="ENT" selected>ENT Entrada </option>
                                    <option value="SAL">SAL Salida </option>

                                </select>

                            </div>
                        </div>
                    </div>

                    <div class="form-group row">
                        <label for="tipo" class="col-sm-2 col-form-label">Es traspaso</label>
                        <div class="col-sm-10">
                            <div class="input-group">
                                <div class="input-group-prepend">
                                    <span class="input-group-text"><i class="fas fa-pencil-alt"></i></span>
                                </div>

                                <select class="form-control esTraspaso" name="esTraspaso" id="esTraspaso" style = "width:80%;">
                                    <option value="No" selected>No </option>
                                    <option value="Si">Si</option>

                                </select>

                            </div>
                        </div>
                    </div>



                </form>
            </div>
            <div class="modal-footer">
                <button type="button" class="btn btn-secondary btn-sm" data-dismiss="modal"><?= lang('boilerplate.global.close') ?></button>
                <button type="button" class="btn btn-primary btn-sm" id="btnSaveTipos_movimientos_inventario"><?= lang('boilerplate.global.save') ?></button>
            </div>
        </div>
    </div>
</div>

<?= $this->section('js') ?>


<script>
    $(document).on('click', '.btnAddTipos_movimientos_inventario', function (e) {


        $(".form-control").val("");

        $("#idTipos_movimientos_inventario").val("0");

        $("#btnSaveTipos_movimientos_inventario").removeAttr("disabled");

        $("#idEmpresa").val("0");

        $("#tipo").val("ENT");

        $("#esTraspaso").val("No");

        $("#idEmpresa").trigger("change");

    });

    /* 
     * AL hacer click al editar
     */

    $("#idEmpresa").select2();
    $("#tipo").select2();
    $("#esTraspaso").select2();


    $(document).on('click', '.btnEditTipos_movimientos_inventario', function (e) {


        var idTipos_movimientos_inventario = $(this).attr("idTipos_movimientos_inventario");

        //LIMPIAMOS CONTROLES
        $(".form-control").val("");

        $("#idTipos_movimientos_inventario").val(idTipos_movimientos_inventario);
        $("#btnGuardarTipos_movimientos_inventario").removeAttr("disabled");

    });
</script>


<?= $this->endSection() ?>

Creamo el archivo de lenguaje en ingles en App/Languaje/es/tipos_movimientos_inventario.php con el siguiente código

<?php

$tipos_movimientos_inventario["logDescription"] = "The tipos_movimientos_inventario was saved with the following data:";
$tipos_movimientos_inventario["logUpdate"] = "The tipos_movimientos_inventario was updated  with the following data:";
$tipos_movimientos_inventario["logDeleted"] = "The tipos_movimientos_inventario was deleted  with the following data:";
$tipos_movimientos_inventario["msg_delete"] = "The tipos_movimientos_inventario was deleted  correctly:";

$tipos_movimientos_inventario["add"] = "Add Tipos_movimientos_inventario";
$tipos_movimientos_inventario["edit"] = "Edit tipos_movimientos_inventario";
$tipos_movimientos_inventario["createEdit"] = "Create / Edit";
$tipos_movimientos_inventario["title"] = "tipos_movimientos_inventario management";
$tipos_movimientos_inventario["subtitle"] = "tipos_movimientos_inventario list";
$tipos_movimientos_inventario["fields"]["idEmpresa"] = "IdEmpresa";
$tipos_movimientos_inventario["fields"]["descripcion"] = "Descripcion";
$tipos_movimientos_inventario["fields"]["tipo"] = "Tipo";
$tipos_movimientos_inventario["fields"]["esTraspaso"] = "EsTraspaso";
$tipos_movimientos_inventario["fields"]["created_at"] = "Created_at";
$tipos_movimientos_inventario["fields"]["updated_at"] = "Updated_at";
$tipos_movimientos_inventario["fields"]["deleted_at"] = "Deleted_at";

$tipos_movimientos_inventario["fields"]["actions"] = "Actions";
$tipos_movimientos_inventario["msg"]["msg_insert"] = "The tipos_movimientos_inventario has been correctly added.";
$tipos_movimientos_inventario["msg"]["msg_update"] = "The tipos_movimientos_inventario has been correctly modified.";
$tipos_movimientos_inventario["msg"]["msg_delete"] = "The tipos_movimientos_inventario has been correctly deleted.";
$tipos_movimientos_inventario["msg"]["msg_get"] = "The Tipos_movimientos_inventario has been successfully get.";
$tipos_movimientos_inventario["msg"]["msg_get_fail"] = "The tipos_movimientos_inventario not found or already deleted.";

return $tipos_movimientos_inventario;

Creamos el archivo de lenguaje en español App/Languaje/es/tipos_movimientos_inventario.php con el siguiente código

 <?php
$tipos_movimientos_inventario["logDescription"] = "El registro en tipos_movimientos_inventario fue guardado con los siguientes datos:";
$tipos_movimientos_inventario["logUpdate"] = "El registro en tipos_movimientos_inventario fue actualizado con los siguientes datos:";
$tipos_movimientos_inventario["logDeleted"] = "El registro en tipos_movimientos_inventario fue eliminado con los siguientes datos:";
$tipos_movimientos_inventario["msg_delete"] = "El Registro en tipos_movimientos_inventario fue eliminado correctamente:";
$tipos_movimientos_inventario["add"] = "Agregar Tipo de Inventario";
$tipos_movimientos_inventario["edit"] = "Editar Tipo de inventario";
$tipos_movimientos_inventario["createEdit"] = "Crear / Editar";
$tipos_movimientos_inventario["title"] = "Admon. tipos de inventario";
$tipos_movimientos_inventario["subtitle"] = "Lista tipos de inventario";
$tipos_movimientos_inventario["fields"]["idEmpresa"] = "Empresa";
$tipos_movimientos_inventario["fields"]["descripcion"] = "Descripcion";
$tipos_movimientos_inventario["fields"]["tipo"] = "Tipo";
$tipos_movimientos_inventario["fields"]["esTraspaso"] = "Es Traspaso";
$tipos_movimientos_inventario["fields"]["created_at"] = "Fecha Creación";
$tipos_movimientos_inventario["fields"]["updated_at"] = "Actualizado";
$tipos_movimientos_inventario["fields"]["deleted_at"] = "Elimininado";

 $tipos_movimientos_inventario["fields"]["actions"] = "Acciones";       
$tipos_movimientos_inventario["msg"]["msg_insert"] = "Registro agregado correctamente.";
$tipos_movimientos_inventario["msg"]["msg_update"] = "Registro modificado correctamente.";
$tipos_movimientos_inventario["msg"]["msg_delete"] = "Registro eliminado correctamente.";
$tipos_movimientos_inventario["msg"]["msg_get"] = "Registro obtenido correctamente.";
$tipos_movimientos_inventario["msg"]["msg_get_fail"] = "Registro no encontrado o eliminado.";
return $tipos_movimientos_inventario;
        

Agregamos las rutas en el archivo App/Config/Routes.php en grupo admin

 $routes->post('tiposMovimientoInventario/getTiposMovimientoInventarioAjax', 'Tipos_movimientos_inventarioController::getTiposMovimientoAjax');

$routes->resource('tipos_movimientos_inventario', [
        'filter' => 'permission:tipos_movimientos_inventario-permission',
        'controller' => 'tipos_movimientos_inventarioController',
        'except' => 'show'
    ]);
    $routes->post('tipos_movimientos_inventario/save', 'Tipos_movimientos_inventarioController::save');
    $routes->post('tipos_movimientos_inventario/getTipos_movimientos_inventario', 'Tipos_movimientos_inventarioController::getTipos_movimientos_inventario');

Creamos el menú con los siguientes datos

Creamos los permisos tal cual se ven en la imagen, posteriormente lo agregamos a los roles y usuarios

Y listo ya tenemos nuestro catalogo de tipos de movimiento

Video Demostrativo