Taskx mittels Laraknife: Unterschied zwischen den Versionen

Aus Vokabulabor
Zur Navigation springen Zur Suche springen
 
(73 dazwischenliegende Versionen von 2 Benutzern werden nicht angezeigt)
Zeile 2: Zeile 2:
* [[Laraknife]]
* [[Laraknife]]
* [[Laravel]]
* [[Laravel]]
* [[LaraKnife Kochbuch]]
* [[Deployment Laraknife Projekt]]


= Zielsetzung =
= Zielsetzung =
Zeile 12: Zeile 14:
* Benutzerverwaltung: nur angemeldete Benutzer können Applikation nutzen
* Benutzerverwaltung: nur angemeldete Benutzer können Applikation nutzen
* Rechteverwaltung mittels Rollen
* Rechteverwaltung mittels Rollen
* Verwaltung von Notizen: Titel, Text, Kategorie, Text, Status: offen, erledigt
* Verwaltung von Notizen: Titel, Text, Kategorie, Status: offen, erledigt


= Installation =
= Installation =
* php muss installiert sein! [[PHP einrichten]] php-sqlite3
* mariadb muss installiert sein: [[mysql]]
* als normaler Benutzer (nicht root):
* als normaler Benutzer (nicht root):
<syntaxhighlight lang="bash">
<syntaxhighlight lang="bash">
sudo apt install php-laravel-framework npm php-intl
PROJ=taskx


PROJ=taskx
SCRIPT=/var/tmp/CreateLaraProj.sh
wget "https://public.hamatoma.de/public/CreateLaraProj.sh" -O $SCRIPT
chmod +x $SCRIPT
PASSW=topsecret
PASSW=topsecret
BASE=/home/ws/php/$PROJ
BASE=/home/ws/php
cd $(dirname $BASE)
sudo apt update
composer create-project laravel/laravel $PROJ
sudo apt install composer -y
cd $BASE
 
composer require laravel/ui
$SCRIPT $PROJ "$PASSW" "$BASE"
composer require spatie/laravel-permission
</syntaxhighlight>
php artisan ui bootstrap --auth
* Am Ende wird dazu aufgefordert, das Script /tmp/IncludeLara.sh zu starten:
dbtool create-db-and-user lrv$PROJ $PROJ "$PASSW"
<syntaxhighlight lang="bash">
sed -i -e "s/DB_DATABASE=.*/DB_DATABASE=lrv$PROJ/" \
# wget https://public.hamatoma.de/public/IncludeLara.sh -O /tmp/IncludeLara.sh
  -e "s/DB_USERNAME=.*/DB_USERNAME=$PROJ/" \
# chmod +x /tmp/IncludeLara.sh
  -e "s/DB_PASSWORD=.*/DB_PASSWORD=$PASSW/" .env
/tmp/IncludeLara.sh $BASE/$PROJ local
M_HOST=mail.gmx.net
# oder
M_PORT=587
/tmp/IncludeLara.sh $BASE/$PROJ git
M_USER=example@gmx.de
M_PW=Top.Secret42
sed -i \
  -e "s/APP_NAME=.*/APP_NAME=$PROJ"/ \
  -e "s/MAIL_MAILER=.*/MAIL_MAILER=smtp/" \
  -e "s/MAIL_HOST=.*/MAIL_HOST=$M_HOST/" \
  -e "s/MAIL_PORT=.*/MAIL_PORT=$M_PORT/" \
  -e "s/MAIL_USERNAME=.*/MAIL_USERNAME=$M_USER/" \
  -e "s/MAIL_PASSWORD=.*/MAIL_PASSWORD=$M_PW/" \
  -e "s/MAIL_ENCRYPTION=.*/MAIL_ENCRYPTION=STARTTLS/" \
  -e "s/MAIL_FROM_ADDRESS=.*/MAIL_FROM_ADDRESS=\"$M_USER\"/" .env
sudo sed -i -e "3 i 127.0.0.1 $PROJ.test" /etc/hosts
grep "$PROJ.test" /etc/hosts
php artisan migrate
npm install
npm run dev
# Abbruch mit Strg-C
</syntaxhighlight>
</syntaxhighlight>


== Einrichten LaraKnife ==
== Restliche Installation ==
<syntaxhighlight lang="bash">
<syntaxhighlight lang="bash">
composer config repositories.laraknife vcs https://github.com/hamatoma/laraknife
cd $BASE/$PROJ
# Branch main:
larascripts/laraknife-tool.sh rest
composer require hamatoma/laraknife:dev-main


composer update
./Own.sh all
vendor/hamatoma/laraknife/scripts/laraknife-tool.sh build-links
</syntaxhighlight>
</syntaxhighlight>


== Mehrsprachigkeit (I18N) ==
Das folgende sind die einzelnen Schritte, '''normalerweise nicht''' erforderlich:
* config/app.php
 
<pre>
'locale' => 'en',
# ersetzen durch:
'locale' => 'de_DE',
'available_locales' => [
  'English' => 'en',
  'German' => 'de_DE',
],
</pre>
* Übersetzungen aus LaraKnife aktivieren:
<syntaxhighlight lang="bash">
<syntaxhighlight lang="bash">
# App-spezifische Übersetzung anlegen:
# Links erstellen/auffrischen:
cat <<EOS >resources/lang/sources/$PROJ.de.json
larascripts/laraknife-tool.sh build-links [--force]
{
# Mehrsprachigkeit
"!comment": "Bitte alphabetisch sortiert eintragen",
larascripts/laraknife-tool.sh init-i18n
"ZZZZZ_last": ""
# Die neuen Übersetzungen müssen in resources/lang/sources/taskx.de.json eingetragen werden.
}
# Roles und SProperties füllen
EOS
larascripts/laraknife-tool.sh fill-db
./Join
# Laraknife-Module anpassen
larascripts/laraknife-tool.sh adapt-modules
# Layout erstellen
larascripts/laraknife-tool.sh create-layout
# Homepage einrichten
larascripts/laraknife-tool.sh create-home
</syntaxhighlight>
</syntaxhighlight>
* Die neuen Übersetzungen müssen in resources/lang/sources/taskx.de.json eingetragen werden.


= Starten Webserver =
= Entwicklungsumgebung starten =
<syntaxhighlight lang="bash">
<syntaxhighlight lang="bash">
cd /home/ws/php/taskx
./Build
# In einem ANDEREN Terminal:
php artisan serve
php artisan serve
</syntaxhighlight>
</syntaxhighlight>
* Im Browser: http://localhost:8000


== Roles und SProperties füllen ==
= Modul Device erstellen =
<syntaxhighlight lang="bash">
php artisan migrate
sudo mysql lrv$PROJ <<'EOS'
insert into roles (name, priority, created_at, updated_at) values
('Administrator', 10, '2023.12.28', '2023-12-28'),
('Manager', 20, '2023.12.28', '2023-12-28'),
('User', 30, '2023.12.28', '2023-12-28'),
('Guest', 90, '2023.12.28', '2023-12-28');
insert into sproperties (id, scope, name, `order`, shortname, created_at) values
(1001, 'status', 'active', 10, 'A', '2023-12-28'),
(1002, 'status', 'inactive', 20, 'I', '2023-12-28'),
(2001, 'category', 'undefined', 10, '-', '2023-12-28'),
(2002, 'category', 'private', 20, 'P', '2023-12-28'),
(2003, 'category', 'work', 30, 'W', '2023-12-28'),
(2501, 'group', 'undefined', 10, '-', '2023-12-28'),
(2502, 'group', 'private', 20, 'P', '2023-12-28'),
(2503, 'group', 'work', 20, 'W', '2023-12-28'),
(2601, 'notestatus', 'open', 10, 'O', '2023-12-28'),
(2602, 'notestatus', 'closed', 20, 'C', '2023-12-28');
EOS
</syntaxhighlight>
 
== User anpassen ==
* Datei app/Models/User.php ergänzen:
<pre>
    protected $fillable = [
        'name',
        'email',
        'password',
        'role_id', // <====
    ];
</pre>
 
= Modul Notes erstellen =
== Tabellenbeschreibung erstellen ==
== Tabellenbeschreibung erstellen ==
* Konvention: Tabelle wird kleingeschrieben, im Plural
* Konvention: Tabelle wird kleingeschrieben, im Plural
<syntaxhighlight lang="bash">
<syntaxhighlight lang="bash">
TABLE=notes
TABLE=devices
php artisan make:migration create_${TABLE}_table
php artisan make:migration create_${TABLE}_table
</syntaxhighlight>
</syntaxhighlight>
* Es wird die Datei database/migrations/2023_12_29_180821_create_notes_table.php erzeugt
* Es wird die Datei database/migrations/2023_12_29_180821_create_devices_table.php erzeugt, wobei Datum/Uhrzeit bei jeder Generierung anders ist.
* Diese Datei anpassen:
* Diese Datei anpassen:
<pre>
<pre>
         Schema::create('notes', function (Blueprint $table) {
         Schema::create('devices', function (Blueprint $table) {
             $table->id();
             $table->id();
             $table->timestamps();
             $table->timestamps();
             $table->string('title');
             $table->string('name');
             $table->text('body');
             $table->text('description');
            // foreign key of sproperties.
             $table->integer('devicegroup_scope');
             $table->integer('category_scope');
             $table->foreignId('owner_id')->nullable()->references('id')->on('users');
            // foreign key of sproperties.
             $table->integer('status_scope');
            // foreign key of sproperties.
            $table->integer('group_scope');
            $table->foreignId('user_id')->references('id')->on('users')->nullable();
         });
         });
    }
</pre>
</pre>
* Syntax für Feldänderungen: [[Datenbank-Laravel#Feld%C3%A4nderungen]]
== Modul erzeugen ==
== Modul erzeugen ==
<syntaxhighlight lang="bash">
<syntaxhighlight lang="bash">
php artisan migrate
php artisan migrate
./Lara create:module database/migrations/*_create_notes_table.php
</syntaxhighlight>


== Layout erstellen ==
./Lara create:module database/migrations/*_create_${TABLE}_table.php
* Datei resources/views/layouts/taskx.blade.php erstellen:
# oder, wenn das Modul nicht automatisch gebildet werden kann:
<syntaxhighlight lang="html">
./Lara create:module database/migrations/*_create_${TABLE}_table.php --module=device
cat <<'EOS' >resources/views/layouts/taskx.blade.php
 
<!doctype html>
 
<html lang="{{ str_replace('_', '-', app()->getLocale()) }}">
php artisan migrate
<head>
    <meta charset="utf-8">
    <meta name="viewport" content="width=device-width, initial-scale=1">
    <!-- CSRF Token -->
    <meta name="csrf-token" content="{{ csrf_token() }}">
    <title>{{ config('app.name', 'Laravel') }}</title>
    <!-- Fonts -->
    <!-- link rel="dns-prefetch" href="//fonts.bunny.net">
    <link href="https://fonts.bunny.net/css?family=Nunito" rel="stylesheet" -->
    <link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.1/dist/css/bootstrap.min.css" rel="stylesheet">
    <!-- Scripts -->
    @vite(['resources/sass/app.scss', 'resources/js/app.js'])
    <link href="/css/laraknife.css" rel="stylesheet">
    <link href="/css/langutor.css" rel="stylesheet">
    <script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.1/dist/js/bootstrap.bundle.min.js"></script>
    <script src="/js/laraknife.js"></script>
</head>
<body>
    <header>
        <nav class="navbar navbar-expand-md bg-primary ">
            <a class="navbar-brand" href="/home">Booking</a>
            <button class="navbar-toggler" type="button" data-toggle="collapse" data-target="#navbarsExampleDefault" aria-expanded="false" aria-label="Toggle navigation">
              <span class="navbar-toggler-icon"></span>
            </button>
            <div class="collapse navbar-collapse" id="navbarTop">
                <ul class="navbar-nav mr-auto">
                    <li class="nav-item active">
                        <a class="nav-link" href="#">Start</a>
                    </li>
                    <li class="nav-item">
                        <a class="nav-link" href="/user-edit">{{ __('Settings') }}</a>
                    </li>
                    <li>
                        <a class="nav-link" href="/public/doc/Impressum.pdf" target="_blank">{{ __('Imprint') }}</a>
                    </li>
                    <li>
                        <a class="nav-link" href="/public/doc/Datenschutz.pdf" target="_blank">{{ __('Privacy') }}</a>
                    </li>
                </ul>
                <form class="form-inline my-2 my-lg-0" action="/logout" method="post">
                    <button class="btn btn-outline-success my-2 my-sm-0 logout" name="btnLogout"
                        type="submit"> {{ __('Logout') }}</button>
                </form>
                </div>               
        </nav>
    </header>
        @yield('content')
</body>
</html>
EOS
</syntaxhighlight>
</syntaxhighlight>


 
== Database Seeder ==
Um Tabellen mit sinnvollen Daten vorzubelegen, werden '''Database Seeder''' verwendet.
<syntaxhighlight lang="bash">
<syntaxhighlight lang="bash">
cd resources/views/layouts
php artisan make:seeder DeviceSeeder
ln -sv taskx.blade.php backend.blade.php
cd ../../..
</syntaxhighlight>
</syntaxhighlight>
 
Die entstehende Datei database/seeders/DeviceSeeder.php anpassen: Icons in https://icons.getbootstrap.com
== Homepage einrichten ==
<pre>
* Datei resources/views/home.blade.php
    public function run(): void
    {
        SProperty::insertIfNotExists(2001, 'devicegroup', 'Computer', 10, 'C');
        SProperty::insertIfNotExists(2002, 'devicegroup', 'Router', 20, '$');
        Menuitem::insertIfNotExists('articles', 'bi bi-journals');
        Module::insertIfNotExists('Article');
    }
</pre>
Aktivieren:
<syntaxhighlight lang="bash">
<syntaxhighlight lang="bash">
cat <<'EOS' >resources/views/home.blade.php
php artisan db:seed --class=DeviceSeeder
@extends('layouts.taskx')
 
@section('content')
<div class="container">
    <div class="row justify-content-center">
        <div class="col-md-8">
            <div class="card">
                <div class="card-header">{{ __('Dashboard') }}</div>
                <div class="card-body">
                    @if (session('status'))
                        <div class="alert alert-success" role="alert">
                            {{ session('status') }}
                        </div>
                    @endif
                    {{ __('You are logged in!') }}
                    <ul>
                        <li><a href="/sproperty-index">SProperties</a></li>
                        <li><a href="/user-index">Benutzer</a></li>
                        <li><a href="/role-index">Rollen</a></li>
                        <li><a href="/notes-index">Notizen</a></li>
                    </ul>
                </div>
            </div>
        </div>
    </div>
</div>
@endsection
EOS
</syntaxhighlight>
</syntaxhighlight>


Zeile 259: Zeile 133:
* In Datei routes/web.php einfügen:
* In Datei routes/web.php einfügen:
<syntaxhighlight lang="php">
<syntaxhighlight lang="php">
SPropertyController::routes();
...
RoleController::routes();
DeviceController::routes();
UserController::routes();
...
NoteController::routes();
</syntaxhighlight>
</syntaxhighlight>
 
* Die Klasse NodeController ist unterstrichen: mit Strg-Alt-I Klassen einbinden.


== Starten, Registrieren, Login ==
== Starten, Registrieren, Login ==
* Den Webserver starten:
* Den Webserver starten:
<syntaxhighlight lang="bash">
<syntaxhighlight lang="bash">
./Build
# in einem anderen Terminal:
php artisan serve
php artisan serve
</syntaxhighlight>
</syntaxhighlight>
Zeile 276: Zeile 151:
* Oben rechts: Login
* Oben rechts: Login


== In Homepage einfügen ==
* resources/views/home.blade.php:
<pre>
<li><a href="/device-index">Geräte</a></li>
</pre>


== Controller und Views anpassen ==
== Controller und Views anpassen ==
=== Übersicht (index) ===
=== Übersicht (index) ===
* Datei app/Http/Controllers/NoteController.php
* Datei app/Http/Controllers/DeviceController.php
<syntaxhighlight lang="php">
<syntaxhighlight lang="php">
     public function index()
     public function index()
     {
     {
        if (array_key_exists('btnSubmit', $_POST) && $_POST['btnSubmit'] === 'btnNew') {
            return redirect('/note-create');
        } else {
            $sql = 'SELECT id, title, body, user_id, cast(body AS VARCHAR(60)) as body2 FROM notes';
            $userId = auth()->id();
            $parameters = [':user_id' => $userId];
            $conditions = ['user_id = :user_id'];
            if (count($_POST) == 0) {
                $fields = [
                    'text' => '',
                    '_sortParams' => 'id:desc'
                ];
            } else {
                $fields = $_POST;
                ViewHelper::addConditionPattern($conditions, $parameters, 'title,body', 'text');
            }
            $sql = DbHelper::addConditions($sql, $conditions);
            $sql = DbHelper::addOrderBy($sql, $fields['_sortParams']);
            $records = DB::select($sql, $parameters);
            $pagination = new Pagination($sql, $parameters, $fields);
            return view('note.index', [
                'records' => $records,
                'fields' => $fields,
                'pagination' => $pagination
            ]);
        }
     }
     }
</syntaxhighlight>
</syntaxhighlight>
* Datei resources/views/note/index.blade.php
* Datei resources/views/device/index.blade.php
<syntaxhighlight lang="html">
<syntaxhighlight lang="html">
<form id="note-index" action="/note-index" method="POST">
<form id="device-index" action="/device-index" method="POST">
     @csrf
     @csrf
    <x-laraknife.index-panel title="{{ __('Notes') }}">
      <x-laraknife.filter-panel legend="{{ $pagination->legendText() }}">
        <x-laraknife.text position="alone" name="text" label="Text" value="{{$fields['text']}}" width2="4" />
      </x-laraknife.filter-panel>
      <x-laraknife.index-button-panel buttonType="new"/>
      <x-laraknife.sortable-table-panel :fields="$fields" :pagination="$pagination">
        <thead>
          <tr>
            <th></th>
            <th sortId="id">{{__('Id')}}</th>
            <th sortId="title">{{__('Title')}}</th>
            <th sortId="body">{{__('Body')}}</th>
            <th></th>
          </tr>
        </thead>
        <tbody>
      @foreach ($records as $note)
        <tr>
          <td><a href="/note-edit/{{$note->id}}">{{ __('Change')}}</a></td>
          <td>{{$note->id}}</td>
          <td>{{$note->title}}</td>
          <td>{{$note->body2}}</td>
          <td><a href="/note-delete/{{$note->id}}">{{ __('Delete')}}</a></td>
        </tr>
      @endforeach
      </tbody>
    </x-laraknife.sortable-table-panel>
  </x-laraknife.index-panel>
</syntaxhighlight>
</syntaxhighlight>


=== Anlegen (create) ===
=== Anlegen (create) ===
* Datei app/Http/Controllers/NoteController.php
* Datei app/Http/Controllers/DeviceController.php
* Die drei Methoden '''ersetzen''':
<syntaxhighlight lang="php">
<syntaxhighlight lang="php">
    public function create(Request $request)
    {
        if (array_key_exists('btnSubmit', $_POST) && $_POST['btnSubmit'] === 'btnCancel') {
            $rc = redirect('/note-index');
        } else {
            $rc = null;
            $error = null;
            if (count($_POST) > 0) {
                $fields = $_POST;
                try {
                    $incomingFields = $request->validate($this->rules(true));
                    $rc = $this->store($request);
                } catch (\Exception $e) {
                    $error = $e->getMessage();
                }
            } else {
                $fields = [
                    'title' => '',
                    'body' => '',
                    'category_scope' => '2001',
                    'status_scope' => '2601',
                    'group_scope' => '2501',
                ];
            }
            if ($rc == null) {
                $optionsCategory = SProperty::optionsByScope('category', $fields['category_scope'], '-');
                $optionsStatus = SProperty::optionsByScope('notestatus', $fields['status_scope'], '-');
                $optionsGroup = SProperty::optionsByScope('group', $fields['group_scope'], '-');
                $rc = view('note.create', ['fields' => $fields,
                    'optionsCategory' => $optionsCategory,
                    'optionsStatus' => $optionsStatus,
                    'optionsGroup' => $optionsGroup,
                    'error' => $error]);
            }
        }
        return $rc;
    }
    private function rules(bool $isCreation=false): array
    {
        $rc = [
            'title' => 'required',
            'body' => 'required',
            'category_scope' => 'required',
            'status_scope' => 'required',
            'group_scope' => 'required',
        ];
        return $rc;
    }
    public function store(Request $request)
    {
        if ($request->btnSubmit === 'btnStore') {
            $incomingFields = $request->validate($this->rules());
            $incomingFields['user_id'] = auth()->id();
            $incomingFields['title'] = strip_tags($incomingFields['title']);
            $incomingFields['body'] = strip_tags($incomingFields['body']);
            Note::create($incomingFields);
        }
        return redirect('/note-index');
    }
</syntaxhighlight>
</syntaxhighlight>


* Datei resources/views/note/create.blade.php
* Datei resources/views/device/create.blade.php
<syntaxhighlight lang="html">
<syntaxhighlight lang="html">
    <form id="note-create" action="/note-create" method="POST">
</syntaxhighlight>
        @csrf
* Übersetzungen in resources/lang/sources/taskx.de.json
        <x-laraknife.create-panel title="{{ __('Creation of a Note') }}" error="{{$error}}">
<pre>
            <x-laraknife.combobox position="first" name="status_scope" label="Status" value="{{$fields['status_scope']}}" :options="$optionsStatus" width2="4" />
"Devices": "Geräte",
            <x-laraknife.combobox position="last" name="group_scope" label="Group" value="{{$fields['group_scope']}}" :options="$optionsGroup" width2="4" />
</pre>
            <x-laraknife.combobox position="alone" name="category_scope" label="Category" value="{{$fields['category_scope']}}" :options="$optionsCategory" width2="4" />
* Aktivieren:
            <x-laraknife.text position="alone" name="title" label="Title" value="{{$fields['title']}}"  />
<syntaxhighlight lang="bash">
            <x-laraknife.bigtext position="alone" name="body" label="Body" value="{{$fields['body']}}" rows="10" />
./Join
        </x-laraknife.create-panel>
    </form>
</syntaxhighlight>
</syntaxhighlight>


=== Ändern (edit) ===
=== Ändern (edit) ===
* Datei app/Http/Controllers/NoteController.php
* Datei app/Http/Controllers/DeviceController.php
<syntaxhighlight lang="php">
<syntaxhighlight lang="php">
     public function edit(Note $note)
     public function edit(Device $device)
     {
     {
        if (array_key_exists('btnSubmit', $_POST) && $_POST['btnSubmit'] === 'btnCancel') {
    }
            $rc = redirect('/note-index');
    public function update(Device $device, Request $request)
        } else {
    {
            $optionsCategory = SProperty::optionsByScope('category', $note->category_scope, '-');
        $rc = null;
            $optionsStatus = SProperty::optionsByScope('notestatus', $note->status_scope, '-');
            $optionsGroup = SProperty::optionsByScope('group', $note->group_scope, '-');
            $rc = view('note.edit', [
                'note' => $note,
                'optionsCategory' => $optionsCategory,
                'optionsStatus' => $optionsStatus,
                'optionsGroup' => $optionsGroup,
            ]);
        }
         return $rc;
         return $rc;
     }
     }
</syntaxhighlight>
</syntaxhighlight>


* Datei resources/views/note/edit.blade.php
* Datei resources/views/device/edit.blade.php
<syntaxhighlight lang="html">
<syntaxhighlight lang="html">
    <form id="note-edit" action="/note-update/{{ $note->id }}" method="POST">
</syntaxhighlight>
        @csrf
 
        <x-laraknife.edit-panel title="{{ __('Change of a Note') }}">
* Übersetzungen in resources/lang/sources/taskx.de.json
            <x-laraknife.combobox position="first" name="status_scope" label="Status" value="{{$note->status_scope}}" :options="$optionsStatus" width2="4" />
<pre>
            <x-laraknife.combobox position="last" name="group_scope" label="Group" value="{{$note->group_scope}}" :options="$optionsGroup" width2="4" />
"Change of a Device": "Ändern eines Gerätes",
            <x-laraknife.combobox position="alone" name="category_scope" label="Category" value="{{$note->category_scope}}" :options="$optionsCategory" width2="4" />
</pre>
            <x-laraknife.text position="alone" name="title" label="Title" value="{{$note->title}}"  />
* Aktivieren:
            <x-laraknife.bigtext position="alone" name="body" label="Body" value="{{$note->body}}" rows="10" />
<syntaxhighlight lang="bash">
        </x-laraknife.edit-panel>
./Join
    </form>
</syntaxhighlight>
</syntaxhighlight>


=== Anzeigen (show/delete) ===
=== Anzeigen (show/delete) ===
* Datei app/Http/Controllers/NoteController.php
* Datei app/Http/Controllers/DeviceController.php
<syntaxhighlight lang="php">
<syntaxhighlight lang="php">
     public function show(Note $note)
     public function show(Device $device)
     {
     {
        if (array_key_exists('btnSubmit', $_POST) && $_POST['btnSubmit'] === 'btnCancel') {
            $rc = redirect('/note-index');
        } else {
            $optionsCategory = SProperty::optionsByScope('category', 'category', '');
            $optionsGroup = SProperty::optionsByScope('group', 'group', '');
            $optionsStatus = SProperty::optionsByScope('notestatus', 'status', '');
            $rc = view('note.show', [
                'note' => $note,
                'optionsCategory' => $optionsCategory,
                'optionsGroup' => $optionsGroup,
                'optionsStatus' => $optionsStatus,
                'mode' => 'delete'
            ]);
        }
        return $rc;
     }
     }
</syntaxhighlight>
</syntaxhighlight>


* Datei resources/views/note/show.blade.php
* Datei resources/views/device/show.blade.php
<syntaxhighlight lang="html">
<syntaxhighlight lang="html">
    <form id="note-show" action="/note-show/{{ $note->id }}/{{ $mode }}" method="POST">
</syntaxhighlight>
        @csrf
* Übersetzungen in resources/lang/sources/taskx.de.json
        @if ($mode === 'delete')
<pre>
            @method('DELETE')
"A Device" : "Ein Gerät",
        @endif
"Deletion of a Device": "Löschen eines Gerätes",
        <x-laraknife.show-panel title="{{ __($mode !== 'delete' ? 'A Note' : 'Deletion of a Note') }}"
</pre>
            mode="{{ $mode }}">
* Aktivieren:
            <x-laraknife.combobox position="first" name="status_scope" label="Status" value="{{ $note->status_scope }}"
<syntaxhighlight lang="bash">
                :options="$optionsStatus" width2="4" attribute="readonly" />
./Join
            <x-laraknife.combobox position="last" name="group_scope" label="Group" value="{{ $note->group_scope }}"
</syntaxhighlight>
                :options="$optionsGroup" width2="4" attribute="readonly" />
 
            <x-laraknife.combobox position="first" name="category_scope" label="Category" value="{{ $note->category_scope }}"
= Neue Instanz von taskx einrichten =
                :options="$optionsCategory" width2="4" attribute="readonly" />
* Lokal:
            <x-laraknife.text position="last" name="id" label="Id" value="{{ $note->id }}" width2="4"
<syntaxhighlight lang="bash">
                attribute="readonly" />
PROJ=taskx
            <x-laraknife.text position="alone" name="title" label="Title" value="{{ $note->title }}"
BASE=/home/ws/php
                attribute="readonly" />
cd $BASE
            <x-laraknife.bigtext position="alone" name="body" label="Body" value="{{ $note->body }}" rows="10"
git clone ssh://git@git.hamatoma.de:/home/git/repo/$PROJ.git
                attribute="readonly" />
cd $PROJ
        </x-laraknife.show-panel>
scripts/InitApp.sh
    </form>
</syntaxhighlight>
 
* auf einem Server:
<syntaxhighlight lang="bash">
PROJ=taskx
DOMAIN=$PROJ.hamatoma.de
BASE=/srv/www
cd $BASE
sudo -u www-data git config --global --add safe.directory /home/git/repo/$PROJ.git
sudo -u www-data git clone /home/git/repo/$PROJ.git
mv $PROJ $DOMAIN
cd $DOMAIN
sudo -u www-data git checkout main
../laraknife/scripts/InstallInitApp.sh
 
echo "DEV_USER=www-data" >.env.user
./Own.sh all
 
cp .env.example .env
vi .env
bash scripts/InitApp.sh
 
sudo -u www-data ./Build prod
sudo -u www-data ./artisan key:generate
</syntaxhighlight>
</syntaxhighlight>

Aktuelle Version vom 3. Januar 2025, 21:21 Uhr

Links

Zielsetzung

Es soll eine minimale Web-Applikation mit den Werkzeugen Laravel und Laraknife erstellt werden: Eine Verwaltung von Notizen.

Name

Eine Verballhornung des Wortes "Tasks": Die schwäbische Aussprache spricht ein x am Ende.

Eigenschaften

  • Benutzerverwaltung: nur angemeldete Benutzer können Applikation nutzen
  • Rechteverwaltung mittels Rollen
  • Verwaltung von Notizen: Titel, Text, Kategorie, Status: offen, erledigt

Installation

  • php muss installiert sein! PHP einrichten php-sqlite3
  • mariadb muss installiert sein: mysql
  • als normaler Benutzer (nicht root):
PROJ=taskx

SCRIPT=/var/tmp/CreateLaraProj.sh
wget "https://public.hamatoma.de/public/CreateLaraProj.sh" -O $SCRIPT
chmod +x $SCRIPT
PASSW=topsecret
BASE=/home/ws/php
sudo apt update
sudo apt install composer -y

$SCRIPT $PROJ "$PASSW" "$BASE"
  • Am Ende wird dazu aufgefordert, das Script /tmp/IncludeLara.sh zu starten:
# wget https://public.hamatoma.de/public/IncludeLara.sh -O /tmp/IncludeLara.sh
# chmod +x /tmp/IncludeLara.sh
/tmp/IncludeLara.sh $BASE/$PROJ local
# oder
/tmp/IncludeLara.sh $BASE/$PROJ git

Restliche Installation

cd $BASE/$PROJ
larascripts/laraknife-tool.sh rest

./Own.sh all

Das folgende sind die einzelnen Schritte, normalerweise nicht erforderlich:

# Links erstellen/auffrischen:
larascripts/laraknife-tool.sh build-links [--force]
# Mehrsprachigkeit
larascripts/laraknife-tool.sh init-i18n
# Die neuen Übersetzungen müssen in resources/lang/sources/taskx.de.json eingetragen werden.
# Roles und SProperties füllen
larascripts/laraknife-tool.sh fill-db
# Laraknife-Module anpassen
larascripts/laraknife-tool.sh adapt-modules
# Layout erstellen
larascripts/laraknife-tool.sh create-layout
# Homepage einrichten
larascripts/laraknife-tool.sh create-home

Entwicklungsumgebung starten

cd /home/ws/php/taskx
./Build
# In einem ANDEREN Terminal:
php artisan serve

Modul Device erstellen

Tabellenbeschreibung erstellen

  • Konvention: Tabelle wird kleingeschrieben, im Plural
TABLE=devices
php artisan make:migration create_${TABLE}_table
  • Es wird die Datei database/migrations/2023_12_29_180821_create_devices_table.php erzeugt, wobei Datum/Uhrzeit bei jeder Generierung anders ist.
  • Diese Datei anpassen:
        Schema::create('devices', function (Blueprint $table) {
            $table->id();
            $table->timestamps();
            $table->string('name');
            $table->text('description');
            $table->integer('devicegroup_scope');
            $table->foreignId('owner_id')->nullable()->references('id')->on('users');
        });
    }

Modul erzeugen

php artisan migrate

./Lara create:module database/migrations/*_create_${TABLE}_table.php
# oder, wenn das Modul nicht automatisch gebildet werden kann:
./Lara create:module database/migrations/*_create_${TABLE}_table.php --module=device


php artisan migrate

Database Seeder

Um Tabellen mit sinnvollen Daten vorzubelegen, werden Database Seeder verwendet.

php artisan make:seeder DeviceSeeder

Die entstehende Datei database/seeders/DeviceSeeder.php anpassen: Icons in https://icons.getbootstrap.com

    public function run(): void
    {
        SProperty::insertIfNotExists(2001, 'devicegroup', 'Computer', 10, 'C');
        SProperty::insertIfNotExists(2002, 'devicegroup', 'Router', 20, '$');
        Menuitem::insertIfNotExists('articles', 'bi bi-journals');
        Module::insertIfNotExists('Article');
    }

Aktivieren:

php artisan db:seed --class=DeviceSeeder

Routing einrichten

  • In Datei routes/web.php einfügen:
...
DeviceController::routes();
...
  • Die Klasse NodeController ist unterstrichen: mit Strg-Alt-I Klassen einbinden.

Starten, Registrieren, Login

  • Den Webserver starten:
./Build
# in einem anderen Terminal:
php artisan serve

In Homepage einfügen

  • resources/views/home.blade.php:
<li><a href="/device-index">Geräte</a></li>

Controller und Views anpassen

Übersicht (index)

  • Datei app/Http/Controllers/DeviceController.php
    public function index()
    {
    }
  • Datei resources/views/device/index.blade.php
<form id="device-index" action="/device-index" method="POST">
    @csrf

Anlegen (create)

  • Datei app/Http/Controllers/DeviceController.php
  • Die drei Methoden ersetzen:
  • Datei resources/views/device/create.blade.php
  • Übersetzungen in resources/lang/sources/taskx.de.json
"Devices": "Geräte",
  • Aktivieren:
./Join

Ändern (edit)

  • Datei app/Http/Controllers/DeviceController.php
    public function edit(Device $device)
    {
    }
    public function update(Device $device, Request $request)
    {
        $rc = null;
        return $rc;
    }
  • Datei resources/views/device/edit.blade.php
  • Übersetzungen in resources/lang/sources/taskx.de.json
"Change of a Device": "Ändern eines Gerätes",
  • Aktivieren:
./Join

Anzeigen (show/delete)

  • Datei app/Http/Controllers/DeviceController.php
    public function show(Device $device)
    {
    }
  • Datei resources/views/device/show.blade.php
  • Übersetzungen in resources/lang/sources/taskx.de.json
"A Device" : "Ein Gerät",
"Deletion of a Device": "Löschen eines Gerätes",
  • Aktivieren:
./Join

Neue Instanz von taskx einrichten

  • Lokal:
PROJ=taskx
BASE=/home/ws/php
cd $BASE
git clone ssh://git@git.hamatoma.de:/home/git/repo/$PROJ.git
cd $PROJ
scripts/InitApp.sh
  • auf einem Server:
PROJ=taskx
DOMAIN=$PROJ.hamatoma.de
BASE=/srv/www
cd $BASE
sudo -u www-data git config --global --add safe.directory /home/git/repo/$PROJ.git
sudo -u www-data git clone /home/git/repo/$PROJ.git
mv $PROJ $DOMAIN
cd $DOMAIN
sudo -u www-data git checkout main
../laraknife/scripts/InstallInitApp.sh

echo "DEV_USER=www-data" >.env.user
./Own.sh all

cp .env.example .env
vi .env
bash scripts/InitApp.sh

sudo -u www-data ./Build prod
sudo -u www-data ./artisan key:generate