This commit is contained in:
Adrian Hopek 2022-01-14 15:09:08 +01:00
parent d4f43421cd
commit 7dca5d1e6c
27 changed files with 871 additions and 150 deletions

View File

@ -48,3 +48,5 @@ DOCKER_INSTALL_XDEBUG=false
GOOGLE_CLIENT_ID=
GOOGLE_CLIENT_SECRET=
GOOGLE_REDIRECT=
USER_EMAIL=

View File

@ -0,0 +1,24 @@
<?php
declare(strict_types=1);
namespace Toby\Console\Commands;
use Illuminate\Console\Command;
use Toby\Models\User;
class CreateUserCommand extends Command
{
protected $signature = "user:create
{email : an email for the user}";
protected $description = "Creates user";
public function handle(): void
{
$email = $this->argument("email");
User::factory(["email" => $email])->create();
$this->info("User has been created");
}
}

View File

@ -0,0 +1,18 @@
<?php
declare(strict_types=1);
namespace Toby\Enums;
enum FormOfEmployment: string
{
case EMPLOYMENT_CONTRACT = "employment_contract";
case COMMISSION_CONTRACT = "commission_contract";
case B2B_CONTRACT = "b2b_contract";
case BOARD_MEMBER_CONTRACT = "board_member_contract";
public function label(): string
{
return __($this->value);
}
}

View File

@ -4,16 +4,23 @@ declare(strict_types=1);
namespace Toby\Http\Controllers;
use Illuminate\Http\Request;
use Inertia\Response;
use Toby\Http\Resources\UserResource;
use Toby\Models\User;
class UserController extends Controller
{
public function index(): Response
public function index(Request $request): Response
{
$users = User::query()
->search($request->query("search"))
->paginate()
->withQueryString();
return inertia("Users/Index", [
"users" => UserResource::collection(User::all()),
"users" => UserResource::collection($users),
"filters" => $request->only("search"),
]);
}
}

View File

@ -15,7 +15,9 @@ class UserResource extends JsonResource
"name" => $this->name,
"email" => $this->email,
"role" => "Human Resources Manager",
"avatar" => "https://images.unsplash.com/photo-1550525811-e5869dd03032?ixlib=rb-1.2.1&ixid=eyJhcHBfaWQiOjEyMDd9&auto=format&fit=facearea&facepad=2&w=256&h=256&q=80",
"avatar" => asset($this->avatar),
"employmentForm" => $this->employment_form->label(),
"employmentStartDate" => $this->employment_start_date->translatedFormat("j F Y"),
];
}
}

View File

@ -4,26 +4,53 @@ declare(strict_types=1);
namespace Toby\Models;
use Illuminate\Database\Eloquent\Builder;
use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Foundation\Auth\User as Authenticatable;
use Illuminate\Notifications\Notifiable;
use Illuminate\Support\Carbon;
use Toby\Enums\FormOfEmployment;
/**
* @property int $id
* @property string $name
* @property string $email
* @property string $avatar
* @property FormOfEmployment $employment_form
* @property Carbon $empoyment_start_date
*/
class User extends Authenticatable
{
use HasFactory;
use Notifiable;
protected $perPage = 10;
protected $fillable = [
"name",
"email",
"avatar",
"employment_form",
"employment_start_date",
];
protected $casts = [
"employment_form" => FormOfEmployment::class,
"employment_start_date" => "datetime",
];
protected $hidden = [
"remember_token",
];
public function scopeSearch(Builder $query, ?string $text): Builder
{
if ($text == null) {
return $query;
}
return $query
->where("name", "LIKE", "%{$text}%")
->orWhere("email", "LIKE", "%{$text}%");
}
}

View File

@ -0,0 +1,52 @@
<?php
declare(strict_types=1);
namespace Toby\Observers;
use Illuminate\Support\Arr;
use Illuminate\Support\Facades\Storage;
use Toby\Models\User;
use LasseRafn\InitialAvatarGenerator\InitialAvatar;
class UserObserver
{
public function __construct(protected InitialAvatar $generator)
{
}
public function created(User $user): void
{
$user->avatar = $this->generateAvatar($user);
$user->save();
}
public function updating(User $user): void
{
if ($user->isDirty("name")) {
$user->avatar = $this->generateAvatar($user);
}
}
public function deleted(User $user): void
{
Storage::delete($user->avatar);
}
protected function generateAvatar(User $user): string
{
$path = "avatars/{$user->id}.svg";
Storage::put($path, $this->generator->rounded()->background($this->getRandomColor())->generateSvg($user->name));
return $path;
}
protected function getRandomColor(): string
{
$colors = config("colors");
return Arr::random($colors);
}
}

View File

@ -5,7 +5,13 @@ declare(strict_types=1);
namespace Toby\Providers;
use Illuminate\Support\ServiceProvider;
use Toby\Models\User;
use Toby\Observers\UserObserver;
class AppServiceProvider extends ServiceProvider
{
public function boot(): void
{
User::observe(UserObserver::class);
}
}

View File

@ -13,7 +13,8 @@
"laravel/framework": "^8.75",
"laravel/socialite": "^5.2",
"laravel/telescope": "^4.6",
"laravel/tinker": "^2.5"
"laravel/tinker": "^2.5",
"lasserafn/php-initial-avatar-generator": "^4.2"
},
"require-dev": {
"blumilksoftware/codestyle": "^0.9.0",

372
composer.lock generated
View File

@ -4,7 +4,7 @@
"Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies",
"This file is @generated automatically"
],
"content-hash": "6c0c7586f9003a71d9299165b3d5030d",
"content-hash": "ad1daca5c42430dda8fb886074c9fc2a",
"packages": [
{
"name": "asm89/stack-cors",
@ -1028,6 +1028,90 @@
],
"time": "2022-01-12T16:18:26+00:00"
},
{
"name": "intervention/image",
"version": "2.7.1",
"source": {
"type": "git",
"url": "https://github.com/Intervention/image.git",
"reference": "744ebba495319501b873a4e48787759c72e3fb8c"
},
"dist": {
"type": "zip",
"url": "https://api.github.com/repos/Intervention/image/zipball/744ebba495319501b873a4e48787759c72e3fb8c",
"reference": "744ebba495319501b873a4e48787759c72e3fb8c",
"shasum": ""
},
"require": {
"ext-fileinfo": "*",
"guzzlehttp/psr7": "~1.1 || ^2.0",
"php": ">=5.4.0"
},
"require-dev": {
"mockery/mockery": "~0.9.2",
"phpunit/phpunit": "^4.8 || ^5.7 || ^7.5.15"
},
"suggest": {
"ext-gd": "to use GD library based image processing.",
"ext-imagick": "to use Imagick based image processing.",
"intervention/imagecache": "Caching extension for the Intervention Image library"
},
"type": "library",
"extra": {
"branch-alias": {
"dev-master": "2.4-dev"
},
"laravel": {
"providers": [
"Intervention\\Image\\ImageServiceProvider"
],
"aliases": {
"Image": "Intervention\\Image\\Facades\\Image"
}
}
},
"autoload": {
"psr-4": {
"Intervention\\Image\\": "src/Intervention/Image"
}
},
"notification-url": "https://packagist.org/downloads/",
"license": [
"MIT"
],
"authors": [
{
"name": "Oliver Vogel",
"email": "oliver@olivervogel.com",
"homepage": "http://olivervogel.com/"
}
],
"description": "Image handling and manipulation library with support for Laravel integration",
"homepage": "http://image.intervention.io/",
"keywords": [
"gd",
"image",
"imagick",
"laravel",
"thumbnail",
"watermark"
],
"support": {
"issues": "https://github.com/Intervention/image/issues",
"source": "https://github.com/Intervention/image/tree/2.7.1"
},
"funding": [
{
"url": "https://www.paypal.me/interventionphp",
"type": "custom"
},
{
"url": "https://github.com/Intervention",
"type": "github"
}
],
"time": "2021-12-16T16:49:26+00:00"
},
{
"name": "laravel/framework",
"version": "v8.79.0",
@ -1464,6 +1548,169 @@
},
"time": "2022-01-10T08:52:49+00:00"
},
{
"name": "lasserafn/php-initial-avatar-generator",
"version": "4.2.1",
"source": {
"type": "git",
"url": "https://github.com/LasseRafn/php-initial-avatar-generator.git",
"reference": "49d0b10cc8819af831e0f6fb1056a7d5ed9512d0"
},
"dist": {
"type": "zip",
"url": "https://api.github.com/repos/LasseRafn/php-initial-avatar-generator/zipball/49d0b10cc8819af831e0f6fb1056a7d5ed9512d0",
"reference": "49d0b10cc8819af831e0f6fb1056a7d5ed9512d0",
"shasum": ""
},
"require": {
"ext-json": "*",
"intervention/image": "^2.3",
"lasserafn/php-initials": "^3.0",
"lasserafn/php-string-script-language": "^0.3.0",
"meyfa/php-svg": "^0.9.0",
"overtrue/pinyin": "^4.0",
"php": "^7.0|^7.1|^7.2|^7.3|^7.4|^8.0"
},
"require-dev": {
"doctrine/instantiator": "1.0.*",
"phpunit/phpunit": "^6.5",
"satooshi/php-coveralls": "^1.0"
},
"type": "library",
"autoload": {
"psr-4": {
"LasseRafn\\InitialAvatarGenerator\\": "src/"
}
},
"notification-url": "https://packagist.org/downloads/",
"license": [
"MIT"
],
"authors": [
{
"name": "Lasse Rafn",
"email": "lasserafn@gmail.com"
}
],
"description": "A package to generate avatars with initials for PHP",
"keywords": [
"Initials",
"avatar",
"image",
"svg"
],
"support": {
"issues": "https://github.com/LasseRafn/php-initial-avatar-generator/issues",
"source": "https://github.com/LasseRafn/php-initial-avatar-generator/tree/4.2.1"
},
"funding": [
{
"url": "https://opencollective.com/ui-avatars",
"type": "open_collective"
}
],
"time": "2020-12-24T13:12:12+00:00"
},
{
"name": "lasserafn/php-initials",
"version": "3.1",
"source": {
"type": "git",
"url": "https://github.com/LasseRafn/php-initials.git",
"reference": "d287e1542687390eb68de779949bc0adc49e2d52"
},
"dist": {
"type": "zip",
"url": "https://api.github.com/repos/LasseRafn/php-initials/zipball/d287e1542687390eb68de779949bc0adc49e2d52",
"reference": "d287e1542687390eb68de779949bc0adc49e2d52",
"shasum": ""
},
"require": {
"php": "^5.6|^7.0|^7.1|^8.0"
},
"require-dev": {
"phpunit/phpunit": "^5.7",
"satooshi/php-coveralls": "^1.0"
},
"type": "library",
"autoload": {
"psr-4": {
"LasseRafn\\Initials\\": "src/"
}
},
"notification-url": "https://packagist.org/downloads/",
"license": [
"MIT"
],
"authors": [
{
"name": "Lasse Rafn",
"email": "lasserafn@gmail.com"
}
],
"description": "A package to generate initials in PHP",
"keywords": [
"Initials",
"php"
],
"support": {
"issues": "https://github.com/LasseRafn/php-initials/issues",
"source": "https://github.com/LasseRafn/php-initials/tree/3.1"
},
"time": "2020-12-24T12:25:51+00:00"
},
{
"name": "lasserafn/php-string-script-language",
"version": "0.3",
"source": {
"type": "git",
"url": "https://github.com/LasseRafn/php-string-script-language.git",
"reference": "49a09d4a5e38c1e59a2656ac05b601d615c7cddb"
},
"dist": {
"type": "zip",
"url": "https://api.github.com/repos/LasseRafn/php-string-script-language/zipball/49a09d4a5e38c1e59a2656ac05b601d615c7cddb",
"reference": "49a09d4a5e38c1e59a2656ac05b601d615c7cddb",
"shasum": ""
},
"require": {
"php": "^5.6|^7.0|^7.1|^8.0"
},
"require-dev": {
"doctrine/instantiator": "1.0.5",
"phpunit/phpunit": "^5.6",
"phpunit/phpunit-mock-objects": "3.2.4",
"satooshi/php-coveralls": "^1.0",
"sebastian/exporter": "^1.2"
},
"type": "library",
"autoload": {
"psr-4": {
"LasseRafn\\": "src/"
}
},
"notification-url": "https://packagist.org/downloads/",
"license": [
"MIT"
],
"authors": [
{
"name": "Lasse Rafn",
"email": "lasserafn@gmail.com"
}
],
"description": "Detect language/encoding of a string in PHP",
"keywords": [
"language",
"php",
"string"
],
"support": {
"issues": "https://github.com/LasseRafn/php-string-script-language/issues",
"source": "https://github.com/LasseRafn/php-string-script-language/tree/0.3"
},
"time": "2020-12-24T12:43:59+00:00"
},
{
"name": "league/commonmark",
"version": "2.1.1",
@ -1875,6 +2122,56 @@
},
"time": "2021-08-15T23:05:49+00:00"
},
{
"name": "meyfa/php-svg",
"version": "v0.9.1",
"source": {
"type": "git",
"url": "https://github.com/meyfa/php-svg.git",
"reference": "34401edef1f724898f468f71b85505fbcc8351bb"
},
"dist": {
"type": "zip",
"url": "https://api.github.com/repos/meyfa/php-svg/zipball/34401edef1f724898f468f71b85505fbcc8351bb",
"reference": "34401edef1f724898f468f71b85505fbcc8351bb",
"shasum": ""
},
"require": {
"ext-gd": "*",
"ext-simplexml": "*",
"php": ">=5.3.3"
},
"require-dev": {
"meyfa/phpunit-assert-gd": "^1.1",
"phpunit/phpunit": "^4.8"
},
"type": "library",
"autoload": {
"psr-4": {
"SVG\\": "src/"
}
},
"notification-url": "https://packagist.org/downloads/",
"license": [
"MIT"
],
"authors": [
{
"name": "Fabian Meyer",
"homepage": "http://meyfa.net"
}
],
"description": "Read, edit, write, and render SVG files with PHP",
"homepage": "https://github.com/meyfa/php-svg",
"keywords": [
"svg"
],
"support": {
"issues": "https://github.com/meyfa/php-svg/issues",
"source": "https://github.com/meyfa/php-svg/tree/v0.9.1"
},
"time": "2019-07-30T18:41:25+00:00"
},
{
"name": "monolog/monolog",
"version": "2.3.5",
@ -2338,6 +2635,79 @@
},
"time": "2021-04-09T13:42:10+00:00"
},
{
"name": "overtrue/pinyin",
"version": "4.0.8",
"source": {
"type": "git",
"url": "https://github.com/overtrue/pinyin.git",
"reference": "04bdb4d33d50e8fb1aa5a824064c5151c4b15dc2"
},
"dist": {
"type": "zip",
"url": "https://api.github.com/repos/overtrue/pinyin/zipball/04bdb4d33d50e8fb1aa5a824064c5151c4b15dc2",
"reference": "04bdb4d33d50e8fb1aa5a824064c5151c4b15dc2",
"shasum": ""
},
"require": {
"php": ">=7.1"
},
"require-dev": {
"brainmaestro/composer-git-hooks": "^2.7",
"friendsofphp/php-cs-fixer": "^2.16",
"phpunit/phpunit": "~8.0"
},
"type": "library",
"extra": {
"hooks": {
"pre-commit": [
"composer test",
"composer fix-style"
],
"pre-push": [
"composer test",
"composer check-style"
]
}
},
"autoload": {
"psr-4": {
"Overtrue\\Pinyin\\": "src/"
},
"files": [
"src/const.php"
]
},
"notification-url": "https://packagist.org/downloads/",
"license": [
"MIT"
],
"authors": [
{
"name": "overtrue",
"email": "anzhengchao@gmail.com",
"homepage": "http://github.com/overtrue"
}
],
"description": "Chinese to pinyin translator.",
"homepage": "https://github.com/overtrue/pinyin",
"keywords": [
"Chinese",
"Pinyin",
"cn2pinyin"
],
"support": {
"issues": "https://github.com/overtrue/pinyin/issues",
"source": "https://github.com/overtrue/pinyin/tree/4.0.8"
},
"funding": [
{
"url": "https://www.patreon.com/overtrue",
"type": "patreon"
}
],
"time": "2021-07-19T03:43:32+00:00"
},
{
"name": "phpoption/phpoption",
"version": "1.8.1",

7
config/colors.php Normal file
View File

@ -0,0 +1,7 @@
<?php
declare(strict_types=1);
return [
"#F0E9E9",
];

View File

@ -18,5 +18,6 @@ return [
],
"links" => [
public_path("storage") => storage_path("app/public"),
public_path("avatars") => storage_path("app/avatars"),
],
];

View File

@ -6,14 +6,17 @@ namespace Database\Factories;
use Illuminate\Database\Eloquent\Factories\Factory;
use Illuminate\Support\Str;
use Toby\Enums\FormOfEmployment;
class UserFactory extends Factory
{
public function definition(): array
{
return [
"name" => $this->faker->name(),
"name" => "{$this->faker->firstName} {$this->faker->lastName}",
"email" => $this->faker->unique()->safeEmail(),
"employment_form" => $this->faker->randomElement(FormOfEmployment::cases()),
"employment_start_date" => $this->faker->dateTimeBetween("2020-10-27"),
"remember_token" => Str::random(10),
];
}

View File

@ -13,6 +13,9 @@ return new class() extends Migration {
$table->id();
$table->string("name");
$table->string("email")->unique();
$table->string("avatar")->nullable();
$table->string("employment_form");
$table->dateTime("employment_start_date");
$table->rememberToken();
$table->timestamps();
});

View File

@ -11,6 +11,7 @@ class DatabaseSeeder extends Seeder
{
public function run(): void
{
User::factory(10)->create();
User::factory(35)->create();
User::factory(["email" => env("USER_EMAIL")])->create();
}
}

1
package-lock.json generated
View File

@ -16,6 +16,7 @@
"@vue/compiler-sfc": "^3.2.26",
"autoprefixer": "^10.4.2",
"laravel-mix": "^6.0.6",
"lodash": "^4.17.21",
"postcss": "^8.4.5",
"tailwindcss": "^3.0.13",
"vue": "^3.2.26",

View File

@ -23,6 +23,7 @@
"@vue/compiler-sfc": "^3.2.26",
"autoprefixer": "^10.4.2",
"laravel-mix": "^6.0.6",
"lodash": "^4.17.21",
"postcss": "^8.4.5",
"tailwindcss": "^3.0.13",
"vue": "^3.2.26",

1
public/avatars Symbolic link
View File

@ -0,0 +1 @@
/application/storage/app/avatars

View File

@ -1,5 +1,6 @@
<template>
<div class="bg-white sm:rounded-lg shado-md">
<InertiaHead title="Użytkownicy" />
<div class="bg-white sm:rounded-lg shadow-md">
<div class="p-4 sm:px-6">
<h2 class="text-lg leading-6 font-medium text-gray-900">
Użytkownicy w organizacji
@ -9,146 +10,164 @@
</p>
</div>
<div class="border-t border-gray-200">
<div class="px-4 py-3">
<div class="relative max-w-md">
<div class="absolute inset-y-0 left-0 pl-3 flex items-center pointer-events-none">
<SearchIcon class="h-5 w-5 text-gray-400" />
</div>
<input
v-model.trim="search"
type="search"
class="block w-full bg-white border border-gray-300 rounded-md py-2 pl-10 pr-3 text-sm placeholder-gray-500 focus:outline-none focus:text-gray-900 focus:placeholder-gray-400 focus:ring-1 focus:ring-blumilk-500 focus:border-blumilk-500 sm:text-sm mt-1"
placeholder="Szukaj"
>
</div>
</div>
<div class="overflow-x-auto xl:overflow-x-visible overflow-y-auto xl:overflow-y-visible">
<div class="align-middle inline-block min-w-full">
<div class="shadow sm:rounded-b-lg">
<table
class="min-w-full divide-y divide-gray-200"
<table class="min-w-full divide-y divide-gray-200">
<thead class="bg-gray-50">
<tr>
<th
scope="col"
class="px-6 py-3 text-left text-xs font-semibold text-gray-500 uppercase tracking-wider"
>
Imię i nazwisko
</th>
<th
scope="col"
class="px-6 py-3 text-left text-xs font-semibold text-gray-500 uppercase tracking-wider"
>
Rola
</th>
<th
scope="col"
class="px-6 py-3 text-left text-xs font-semibold text-gray-500 uppercase tracking-wider"
>
Forma zatrudnienia
</th>
<th
scope="col"
class="px-6 py-3 text-left text-xs font-semibold text-gray-500 uppercase tracking-wider"
>
Data rozpoczęcia
</th>
<th
scope="col"
class="px-6 py-3 text-left text-xs font-semibold text-gray-500 uppercase tracking-wider"
/>
</tr>
</thead>
<tbody class="bg-white divide-y divide-gray-100">
<tr
v-for="user in users.data"
:key="user.id"
class="hover:bg-white"
>
<thead class="bg-gray-50">
<tr>
<th
scope="col"
class="px-6 py-3 text-left text-xs font-semibold text-gray-500 uppercase tracking-wider"
<td class="px-4 py-4 whitespace-nowrap text-sm text-gray-500">
<div class="flex">
<span
class="inline-flex items-center justify-center h-10 w-10 rounded-full bg-blumilk-500"
>
Imię i nazwisko
</th>
<th
scope="col"
class="px-6 py-3 text-left text-xs font-semibold text-gray-500 uppercase tracking-wider"
>
Status
</th>
<th
scope="col"
class="px-6 py-3 text-left text-xs font-semibold text-gray-500 uppercase tracking-wider"
>
Rola
</th>
<th
scope="col"
class="px-6 py-3 text-left text-xs font-semibold text-gray-500 uppercase tracking-wider"
>
Data dodania
</th>
<th
scope="col"
class="px-6 py-3 text-left text-xs font-semibold text-gray-500 uppercase tracking-wider"
/>
</tr>
</thead>
<tbody class="bg-white divide-y divide-gray-100">
<tr
v-for="user in users.data"
:key="user.id"
class="hover:bg-white"
>
<td class="px-4 py-4 whitespace-nowrap text-sm text-gray-500">
<div class="flex">
<span class="inline-flex items-center justify-center h-10 w-10 rounded-full bg-indigo-500">
<img
class="h-10 w-10 rounded-full"
:src="user.avatar"
alt=""
>
</span>
<div class="ml-3">
<p class="text-sm font-medium break-all text-gray-900">
{{ user.name }}
</p>
<p class="text-sm break-all text-gray-500">
{{ user.email }}
</p>
</div>
</div>
</td>
<td class="px-4 py-4 whitespace-nowrap text-sm text-gray-500">
<span
class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-yellow-100 text-yellow-800"
data-cy="invited-status"
<img
class="h-10 w-10 rounded-full"
:src="user.avatar"
alt=""
>
Zaproszony
</span>
</td>
<td class="px-4 py-4 whitespace-nowrap text-sm text-gray-500">
{{ user.role }}
</td>
<td class="px-4 py-4 whitespace-nowrap text-sm text-gray-500">
5 listopada 2021
</td>
<td class="px-4 py-4 whitespace-nowrap text-sm text-gray-500 text-right">
<div
x-data="dropdown"
class="relative inline-block text-left"
@keydown.escape="close()"
@click.outside="close()"
</span>
<div class="ml-3">
<p class="text-sm font-medium break-all text-gray-900">
{{ user.name }}
</p>
<p class="text-sm break-all text-gray-500">
{{ user.email }}
</p>
</div>
</div>
</td>
<td class="px-4 py-4 whitespace-nowrap text-sm text-gray-500">
{{ user.role }}
</td>
<td class="px-4 py-4 whitespace-nowrap text-sm text-gray-500">
{{ user.employmentForm }}
</td>
<td class="px-4 py-4 whitespace-nowrap text-sm text-gray-500">
{{ user.employmentStartDate }}
</td>
<td class="px-4 py-4 whitespace-nowrap text-sm text-gray-500 text-right">
<div>
<button
class="rounded-full flex items-center text-gray-400 hover:text-gray-600 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-offset-gray-100 focus:ring-blumilk-500"
>
<svg
class="h-5 w-5"
xmlns="http://www.w3.org/2000/svg"
viewBox="0 0 20 20"
fill="currentColor"
>
<div>
<button
class="rounded-full flex items-center text-gray-400 hover:text-gray-600 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-offset-gray-100 focus:ring-indigo-500"
data-cy="options-button"
@click="toggle()"
>
<svg
class="h-5 w-5"
xmlns="http://www.w3.org/2000/svg"
viewBox="0 0 20 20"
fill="currentColor"
>
<path d="M10 6a2 2 0 110-4 2 2 0 010 4zM10 12a2 2 0 110-4 2 2 0 010 4zM10 18a2 2 0 110-4 2 2 0 010 4z" />
</svg>
</button>
</div>
<div
x-show="show"
x-transition:enter="transition ease-out duration-100"
x-transition:enter-start="transform opacity-0 scale-95"
x-transition:enter-end="transform opacity-100 scale-100"
x-transition:leave="transition ease-in duration-75"
x-transition:leave-start="transform opacity-100 scale-100"
x-transition:leave-end="transform opacity-0 scale-95"
class="origin-top-right absolute right-6 mt-2 w-56 z-10 rounded-md shadow-lg bg-white ring-1 ring-black ring-opacity-5 divide-y divide-gray-200"
data-cy="options-list"
style="display: none;"
>
<a
href="#"
role="button"
class="group flex items-center px-4 py-2 text-sm font-semibold text-gray-500 hover:bg-gray-100"
wire:click.prevent="inviteAgain"
data-cy="invite-again-button"
@click="close()"
>
Wyślij zaproszenie ponownie
</a>
<a
href="#"
role="button"
class="group flex items-center px-4 py-2 text-sm font-semibold text-gray-500 hover:bg-gray-100"
wire:click.prevent="cancelInvitation"
data-cy="cancel-invitation-button"
@click="close()"
>
<span class="text-red-500">
Anuluj zaproszenie
</span>
</a>
</div>
</div>
</td>
</tr>
</tbody>
</table>
<path
d="M10 6a2 2 0 110-4 2 2 0 010 4zM10 12a2 2 0 110-4 2 2 0 010 4zM10 18a2 2 0 110-4 2 2 0 010 4z"
/>
</svg>
</button>
</div>
</td>
</tr>
<tr
v-if="! users.data.length"
>
<td
colspan="100%"
class="text-center py-4 text-xl leading-5 text-gray-700"
>
Brak danych
</td>
</tr>
</tbody>
</table>
<div
v-if="users.data.length"
class="bg-white px-4 py-3 flex items-center justify-between border-t border-gray-200 sm:px-6 rounded-b-lg"
>
<div class="flex-1 flex justify-between sm:hidden">
<InertiaLink
:is="users.links.prev ? 'InertiaLink': 'span'"
:href="users.links.prev"
class="relative inline-flex items-center px-4 py-2 border border-gray-300 text-sm font-medium rounded-md text-gray-700 bg-white hover:bg-gray-50"
>
Poprzednia
</InertiaLink>
<Component
:is="users.links.next ? 'InertiaLink': 'span'"
:href="users.links.next"
class="ml-3 relative inline-flex items-center px-4 py-2 border border-gray-300 text-sm font-medium rounded-md text-gray-700 bg-white hover:bg-gray-50"
>
Następna
</Component>
</div>
<div class="hidden sm:flex-1 sm:flex sm:items-center sm:justify-between">
<div class="text-sm text-gray-700">
Wyświetlanie
<span class="font-medium">{{ users.meta.from }}</span>
od
<span class="font-medium">{{ users.meta.to }}</span>
do
<span class="font-medium">{{ users.meta.total }}</span>
wyników
</div>
<nav class="relative z-0 inline-flex space-x-1">
<template
v-for="(link, index) in users.meta.links"
:key="index"
>
<Component
:is="link.url ? 'InertiaLink' : 'span'"
:href="link.url"
class="relative inline-flex items-center px-4 py-2 border rounded-md text-sm font-medium"
:class="{ 'z-10 bg-blumilk-25 border-blumilk-500 text-blumilk-600': link.active, 'bg-white border-gray-300 text-gray-500': !link.active, 'hover:bg-blumilk-25': link.url, 'border-none': !link.url}"
v-text="link.label"
/>
</template>
</nav>
</div>
</div>
</div>
@ -157,13 +176,39 @@
</template>
<script>
import { ref, watch } from 'vue';
import { Inertia } from '@inertiajs/inertia';
import { debounce } from 'lodash';
import { SearchIcon } from '@heroicons/vue/outline';
export default {
name: 'UserIndex',
components: {
SearchIcon,
},
props: {
users: {
type: Object,
default: () => {},
}
default: () => null,
},
filters: {
type: Object,
default: () => null,
},
},
setup(props) {
let search = ref(props.filters.search);
watch(search, debounce(value => {
Inertia.get('/users', value ? { search: value} : {}, {
preserveState: true,
replace: true,
});
}, 300));
return {
search,
};
}
};
</script>

View File

@ -265,7 +265,7 @@ export default {
setup() {
const user = computed(() => usePage().props.value.auth.user);
const navigation = [
{name: 'Dashboard', href: '/', current: true},
{name: 'Strona główna', href: '/', current: true},
{name: 'Użytkownicy', href: '/users', current: false},
{name: 'Resources', href: '#', current: false},
{name: 'Company Directory', href: '#', current: false},

View File

@ -3,6 +3,6 @@
declare(strict_types=1);
return [
"previous" => "&laquo; Previous",
"next" => "Next &raquo;",
"previous" => "Previous",
"next" => "Next",
];

View File

@ -1,3 +1,7 @@
{
"User does not exist.": "Użytkownik nie istnieje."
"User does not exist.": "Użytkownik nie istnieje.",
"employment_contract": "Umowa o pracę",
"commission_contract": "Umowa zlecenie",
"b2b_contract": "Kontrakt B2B",
"board_member_contract": "Członek zarządu"
}

View File

@ -0,0 +1,9 @@
<?php
declare(strict_types=1);
return [
"failed" => "Błędny login lub hasło.",
"password" => "Podane hasło jest nieprawidłowe.",
"throttle" => "Za dużo nieudanych prób logowania. Proszę spróbować za :seconds sekund.",
];

View File

@ -0,0 +1,8 @@
<?php
declare(strict_types=1);
return [
"next" => "Następna",
"previous" => "Poprzednia",
];

View File

@ -0,0 +1,11 @@
<?php
declare(strict_types=1);
return [
"reset" => "Hasło zostało zresetowane!",
"sent" => "Przypomnienie hasła zostało wysłane!",
"throttled" => "Proszę zaczekać zanim spróbujesz ponownie.",
"token" => "Token resetowania hasła jest nieprawidłowy.",
"user" => "Nie znaleziono użytkownika z takim adresem e-mail.",
];

View File

@ -0,0 +1,117 @@
<?php
declare(strict_types=1);
return [
"accepted" => "Pole :attribute musi zostać zaakceptowane.",
"active_url" => "Pole :attribute jest nieprawidłowym adresem URL.",
"after" => "Pole :attribute musi być datą późniejszą od :date.",
"after_or_equal" => "Pole :attribute musi być datą nie wcześniejszą niż :date.",
"alpha" => "Pole :attribute może zawierać jedynie litery.",
"alpha_dash" => "Pole :attribute może zawierać jedynie litery, cyfry i myślniki.",
"alpha_num" => "Pole :attribute może zawierać jedynie litery i cyfry.",
"array" => "Pole :attribute musi być tablicą.",
"attached" => "Ten :attribute jest już dołączony.",
"before" => "Pole :attribute musi być datą wcześniejszą od :date.",
"before_or_equal" => "Pole :attribute musi być datą nie późniejszą niż :date.",
"between" => [
"array" => "Pole :attribute musi składać się z :min - :max elementów.",
"file" => "Pole :attribute musi zawierać się w granicach :min - :max kilobajtów.",
"numeric" => "Pole :attribute musi zawierać się w granicach :min - :max.",
"string" => "Pole :attribute musi zawierać się w granicach :min - :max znaków.",
],
"boolean" => "Pole :attribute musi mieć wartość logiczną prawda albo fałsz.",
"confirmed" => "Potwierdzenie pola :attribute nie zgadza się.",
"current_password" => "Hasło jest nieprawidłowe.",
"date" => "Pole :attribute nie jest prawidłową datą.",
"date_equals" => "Pole :attribute musi być datą równą :date.",
"date_format" => "Pole :attribute nie jest w formacie :format.",
"different" => "Pole :attribute oraz :other muszą się różnić.",
"digits" => "Pole :attribute musi składać się z :digits cyfr.",
"digits_between" => "Pole :attribute musi mieć od :min do :max cyfr.",
"dimensions" => "Pole :attribute ma niepoprawne wymiary.",
"distinct" => "Pole :attribute ma zduplikowane wartości.",
"email" => "Pole :attribute nie jest poprawnym adresem e-mail.",
"ends_with" => "Pole :attribute musi kończyć się jedną z następujących wartości: :values.",
"exists" => "Zaznaczone pole :attribute jest nieprawidłowe.",
"file" => "Pole :attribute musi być plikiem.",
"filled" => "Pole :attribute nie może być puste.",
"gt" => [
"array" => "Pole :attribute musi mieć więcej niż :value elementów.",
"file" => "Pole :attribute musi być większe niż :value kilobajtów.",
"numeric" => "Pole :attribute musi być większe niż :value.",
"string" => "Pole :attribute musi być dłuższe niż :value znaków.",
],
"gte" => [
"array" => "Pole :attribute musi mieć :value lub więcej elementów.",
"file" => "Pole :attribute musi być większe lub równe :value kilobajtów.",
"numeric" => "Pole :attribute musi być większe lub równe :value.",
"string" => "Pole :attribute musi być dłuższe lub równe :value znaków.",
],
"image" => "Pole :attribute musi być obrazkiem.",
"in" => "Zaznaczony element :attribute jest nieprawidłowy.",
"in_array" => "Pole :attribute nie znajduje się w :other.",
"integer" => "Pole :attribute musi być liczbą całkowitą.",
"ip" => "Pole :attribute musi być prawidłowym adresem IP.",
"ipv4" => "Pole :attribute musi być prawidłowym adresem IPv4.",
"ipv6" => "Pole :attribute musi być prawidłowym adresem IPv6.",
"json" => "Pole :attribute musi być poprawnym ciągiem znaków JSON.",
"lt" => [
"array" => "Pole :attribute musi mieć mniej niż :value elementów.",
"file" => "Pole :attribute musi być mniejsze niż :value kilobajtów.",
"numeric" => "Pole :attribute musi być mniejsze niż :value.",
"string" => "Pole :attribute musi być krótsze niż :value znaków.",
],
"lte" => [
"array" => "Pole :attribute musi mieć :value lub mniej elementów.",
"file" => "Pole :attribute musi być mniejsze lub równe :value kilobajtów.",
"numeric" => "Pole :attribute musi być mniejsze lub równe :value.",
"string" => "Pole :attribute musi być krótsze lub równe :value znaków.",
],
"max" => [
"array" => "Pole :attribute nie może mieć więcej niż :max elementów.",
"file" => "Pole :attribute nie może być większe niż :max kilobajtów.",
"numeric" => "Pole :attribute nie może być większe niż :max.",
"string" => "Pole :attribute nie może być dłuższe niż :max znaków.",
],
"mimes" => "Pole :attribute musi być plikiem typu :values.",
"mimetypes" => "Pole :attribute musi być plikiem typu :values.",
"min" => [
"array" => "Pole :attribute musi mieć przynajmniej :min elementów.",
"file" => "Pole :attribute musi mieć przynajmniej :min kilobajtów.",
"numeric" => "Pole :attribute musi być nie mniejsze od :min.",
"string" => "Pole :attribute musi mieć przynajmniej :min znaków.",
],
"multiple_of" => "Pole :attribute musi być wielokrotnością wartości :value",
"not_in" => "Zaznaczony :attribute jest nieprawidłowy.",
"not_regex" => "Format pola :attribute jest nieprawidłowy.",
"numeric" => "Pole :attribute musi być liczbą.",
"password" => "Hasło jest nieprawidłowe.",
"present" => "Pole :attribute musi być obecne.",
"prohibited" => "Pole :attribute jest zabronione.",
"prohibited_if" => "Pole :attribute jest zabronione, gdy :other to :value.",
"prohibited_unless" => "Pole :attribute jest zabronione, chyba że :other jest w :values.",
"regex" => "Format pola :attribute jest nieprawidłowy.",
"relatable" => "Ten :attribute może nie być powiązany z tym zasobem.",
"required" => "Pole :attribute jest wymagane.",
"required_if" => "Pole :attribute jest wymagane gdy :other ma wartość :value.",
"required_unless" => "Pole :attribute jest wymagane jeżeli :other nie znajduje się w :values.",
"required_with" => "Pole :attribute jest wymagane gdy :values jest obecny.",
"required_with_all" => "Pole :attribute jest wymagane gdy wszystkie :values są obecne.",
"required_without" => "Pole :attribute jest wymagane gdy :values nie jest obecny.",
"required_without_all" => "Pole :attribute jest wymagane gdy żadne z :values nie są obecne.",
"same" => "Pole :attribute i :other muszą być takie same.",
"size" => [
"array" => "Pole :attribute musi zawierać :size elementów.",
"file" => "Pole :attribute musi mieć :size kilobajtów.",
"numeric" => "Pole :attribute musi mieć :size.",
"string" => "Pole :attribute musi mieć :size znaków.",
],
"starts_with" => "Pole :attribute musi zaczynać się jedną z następujących wartości: :values.",
"string" => "Pole :attribute musi być ciągiem znaków.",
"timezone" => "Pole :attribute musi być prawidłową strefą czasową.",
"unique" => "Taki :attribute już występuje.",
"uploaded" => "Nie udało się wgrać pliku :attribute.",
"url" => "Format pola :attribute jest nieprawidłowy.",
"uuid" => "Pole :attribute musi być poprawnym identyfikatorem UUID.",
];