Compare commits
53 Commits
Author | SHA1 | Date |
---|---|---|
philipp lang | a456cc1889 | |
philipp lang | 784b49c3dd | |
philipp lang | c3a0381a0a | |
philipp lang | 101ac7402a | |
philipp lang | 2c38488962 | |
philipp lang | 8a26276e28 | |
philipp lang | 4651a4cb96 | |
philipp lang | 688e10e230 | |
philipp lang | 267da38c6b | |
philipp lang | 499c61afff | |
philipp lang | 02531938b3 | |
philipp lang | 1c53e81f1a | |
philipp lang | 98712093b2 | |
philipp lang | a32ef33113 | |
philipp lang | 088ea4f0a2 | |
philipp lang | 4ac89da124 | |
philipp lang | 286ceea0d4 | |
philipp lang | 52ac5937a4 | |
philipp lang | 4690e86bbf | |
philipp lang | 1a4a7c3652 | |
philipp lang | b05c5025c0 | |
philipp lang | acf7b94094 | |
philipp lang | 57b1efb065 | |
philipp lang | 3e563f8390 | |
philipp lang | 0a5590d533 | |
philipp lang | 10deaf15a1 | |
philipp lang | ae447a7a60 | |
philipp lang | baea21807a | |
philipp lang | 58426c5537 | |
philipp lang | 84cd13c085 | |
philipp lang | b8d2b37057 | |
philipp lang | 44eb3719b9 | |
philipp lang | c65a208e34 | |
philipp lang | 95a8d4d689 | |
philipp lang | e1e8669beb | |
philipp lang | a3ecfa756d | |
philipp lang | 70e085a49e | |
philipp lang | 1178b011e0 | |
philipp lang | 6a91e857d2 | |
philipp lang | 82baf67a73 | |
philipp lang | 36466420f6 | |
philipp lang | 54c37fccd1 | |
philipp lang | d03f036a2b | |
philipp lang | 75d11f9860 | |
philipp lang | da4d553ea6 | |
philipp lang | 5bad88c5f4 | |
philipp lang | 75dc0d9769 | |
philipp lang | 9fc8548504 | |
philipp lang | c990ce3171 | |
philipp lang | e7f63d19e5 | |
philipp lang | f549b31862 | |
philipp lang | 1f4efb22d3 | |
philipp lang | 3e725d9842 |
|
@ -2,10 +2,11 @@ FROM php:8.3.11-fpm as php
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
RUN ls /app
|
RUN ls /app
|
||||||
RUN apt-get update
|
RUN apt-get update
|
||||||
RUN apt-get install -y rsync libcurl3-dev apt-utils zlib1g-dev libpng-dev libicu-dev libonig-dev unzip poppler-utils libpng-dev libjpeg-dev default-mysql-client libzip-dev
|
RUN apt-get install -y rsync libcurl3-dev apt-utils zlib1g-dev libpng-dev libicu-dev libonig-dev unzip poppler-utils libpng-dev libjpeg-dev default-mysql-client libzip-dev imagemagick libmagickwand-dev
|
||||||
RUN apt-get install -y --no-install-recommends texlive-base texlive-latex-base texlive-pictures texlive-latex-extra texlive-lang-german texlive-plain-generic texlive-fonts-recommended texlive-fonts-extra texlive-extra-utils
|
RUN apt-get install -y --no-install-recommends texlive-base texlive-latex-base texlive-pictures texlive-latex-extra texlive-lang-german texlive-plain-generic texlive-fonts-recommended texlive-fonts-extra texlive-extra-utils
|
||||||
RUN docker-php-ext-install pdo_mysql curl exif intl mbstring pcntl zip
|
RUN docker-php-ext-install pdo_mysql curl exif intl mbstring pcntl zip
|
||||||
RUN pecl install redis && docker-php-ext-enable redis
|
RUN pecl install redis && docker-php-ext-enable redis
|
||||||
|
RUN pecl install imagick && docker-php-ext-enable imagick
|
||||||
RUN docker-php-ext-configure gd --with-jpeg
|
RUN docker-php-ext-configure gd --with-jpeg
|
||||||
RUN docker-php-ext-install gd
|
RUN docker-php-ext-install gd
|
||||||
RUN usermod -s /bin/bash www-data
|
RUN usermod -s /bin/bash www-data
|
||||||
|
|
|
@ -40,3 +40,4 @@ Homestead.json
|
||||||
/public/sprite.svg
|
/public/sprite.svg
|
||||||
/.php-cs-fixer.cache
|
/.php-cs-fixer.cache
|
||||||
/groups.sql
|
/groups.sql
|
||||||
|
/.phpunit.cache
|
||||||
|
|
17
CHANGELOG.md
17
CHANGELOG.md
|
@ -1,5 +1,22 @@
|
||||||
# Letzte Änderungen
|
# Letzte Änderungen
|
||||||
|
|
||||||
|
### 1.12.2
|
||||||
|
|
||||||
|
- Zuschussliste Gallier
|
||||||
|
|
||||||
|
### 1.12.1
|
||||||
|
|
||||||
|
- In Teilnehmer-Liste von Veranstaltungen kann nun sortiert und gefiltert werden.
|
||||||
|
- Formulare: Feld Registrierung von / bis
|
||||||
|
|
||||||
|
### 1.11.5
|
||||||
|
|
||||||
|
- Fix: Synchronisation von NaMi-Mitgliedern
|
||||||
|
|
||||||
|
### 1.11.4
|
||||||
|
|
||||||
|
- Fix: Nicht an Präventions-Unterlagen für vergangene Veranstaltungen erinnern
|
||||||
|
|
||||||
### 1.11.1
|
### 1.11.1
|
||||||
|
|
||||||
- Es kann nun auch das Feld "Datenweiterverwendung" über Adrema gepflegt werden.
|
- Es kann nun auch das Feld "Datenweiterverwendung" über Adrema gepflegt werden.
|
||||||
|
|
|
@ -8,6 +8,7 @@ use App\Contribution\Documents\RdpNrwDocument;
|
||||||
use App\Contribution\Documents\CityRemscheidDocument;
|
use App\Contribution\Documents\CityRemscheidDocument;
|
||||||
use App\Contribution\Documents\CitySolingenDocument;
|
use App\Contribution\Documents\CitySolingenDocument;
|
||||||
use App\Contribution\Documents\CityFrankfurtMainDocument;
|
use App\Contribution\Documents\CityFrankfurtMainDocument;
|
||||||
|
use App\Contribution\Documents\WuppertalDocument;
|
||||||
use Illuminate\Support\Collection;
|
use Illuminate\Support\Collection;
|
||||||
use Illuminate\Validation\Rule;
|
use Illuminate\Validation\Rule;
|
||||||
|
|
||||||
|
@ -22,6 +23,7 @@ class ContributionFactory
|
||||||
CityRemscheidDocument::class,
|
CityRemscheidDocument::class,
|
||||||
CityFrankfurtMainDocument::class,
|
CityFrankfurtMainDocument::class,
|
||||||
BdkjHesse::class,
|
BdkjHesse::class,
|
||||||
|
WuppertalDocument::class,
|
||||||
];
|
];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -30,7 +32,7 @@ class ContributionFactory
|
||||||
public function compilerSelect(): Collection
|
public function compilerSelect(): Collection
|
||||||
{
|
{
|
||||||
return collect($this->documents)->map(fn ($document) => [
|
return collect($this->documents)->map(fn ($document) => [
|
||||||
'title' => $document::getName(),
|
'title' => $document::buttonName(),
|
||||||
'class' => $document,
|
'class' => $document,
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
|
@ -72,13 +72,23 @@ class MemberData extends Data
|
||||||
return $this->zip . ' ' . $this->location;
|
return $this->zip . ' ' . $this->location;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function age(): string
|
public function age(): int
|
||||||
{
|
{
|
||||||
return (string) $this->birthday->diffInYears(now()) ?: '';
|
return intval($this->birthday->diffInYears(now()));
|
||||||
}
|
}
|
||||||
|
|
||||||
public function birthYear(): string
|
public function birthYear(): string
|
||||||
{
|
{
|
||||||
return (string) $this->birthday->year;
|
return (string) $this->birthday->year;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function birthdayHuman(): string
|
||||||
|
{
|
||||||
|
return $this->birthday->format('d.m.Y');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function genderLetter(): string
|
||||||
|
{
|
||||||
|
return $this->gender->short;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,15 +3,16 @@
|
||||||
namespace App\Contribution\Documents;
|
namespace App\Contribution\Documents;
|
||||||
|
|
||||||
use App\Contribution\Data\MemberData;
|
use App\Contribution\Data\MemberData;
|
||||||
|
use App\Contribution\Traits\HasPdfBackground;
|
||||||
use App\Country;
|
use App\Country;
|
||||||
use Carbon\Carbon;
|
use Carbon\Carbon;
|
||||||
use Illuminate\Support\Collection;
|
use Illuminate\Support\Collection;
|
||||||
use Illuminate\Support\Str;
|
|
||||||
use Zoomyboy\Tex\Engine;
|
|
||||||
use Zoomyboy\Tex\Template;
|
|
||||||
|
|
||||||
class BdkjHesse extends ContributionDocument
|
class BdkjHesse extends ContributionDocument
|
||||||
{
|
{
|
||||||
|
|
||||||
|
use HasPdfBackground;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param Collection<int, Collection<int, MemberData>> $members
|
* @param Collection<int, Collection<int, MemberData>> $members
|
||||||
*/
|
*/
|
||||||
|
@ -25,6 +26,7 @@ class BdkjHesse extends ContributionDocument
|
||||||
public ?string $filename = '',
|
public ?string $filename = '',
|
||||||
public string $type = 'F',
|
public string $type = 'F',
|
||||||
) {
|
) {
|
||||||
|
$this->setEventName($eventName);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function dateFrom(): string
|
public function dateFrom(): string
|
||||||
|
@ -114,36 +116,9 @@ class BdkjHesse extends ContributionDocument
|
||||||
return $member->birthYear();
|
return $member->birthYear();
|
||||||
}
|
}
|
||||||
|
|
||||||
public function basename(): string
|
|
||||||
{
|
|
||||||
return 'zuschuesse-bdkj-hessen' . Str::slug($this->eventName);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function view(): string
|
|
||||||
{
|
|
||||||
return 'tex.contribution.bdkj-hesse';
|
|
||||||
}
|
|
||||||
|
|
||||||
public function template(): Template
|
|
||||||
{
|
|
||||||
return Template::make('tex.templates.contribution');
|
|
||||||
}
|
|
||||||
|
|
||||||
public function setFilename(string $filename): static
|
|
||||||
{
|
|
||||||
$this->filename = $filename;
|
|
||||||
|
|
||||||
return $this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getEngine(): Engine
|
|
||||||
{
|
|
||||||
return Engine::PDFLATEX;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static function getName(): string
|
public static function getName(): string
|
||||||
{
|
{
|
||||||
return 'Für BDKJ Hessen erstellen';
|
return 'BDKJ Hessen';
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -156,7 +131,6 @@ class BdkjHesse extends ContributionDocument
|
||||||
'dateUntil' => 'required|string|date_format:Y-m-d',
|
'dateUntil' => 'required|string|date_format:Y-m-d',
|
||||||
'country' => 'required|integer|exists:countries,id',
|
'country' => 'required|integer|exists:countries,id',
|
||||||
'zipLocation' => 'required|string',
|
'zipLocation' => 'required|string',
|
||||||
'eventName' => 'required|string',
|
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,16 +3,17 @@
|
||||||
namespace App\Contribution\Documents;
|
namespace App\Contribution\Documents;
|
||||||
|
|
||||||
use App\Contribution\Data\MemberData;
|
use App\Contribution\Data\MemberData;
|
||||||
|
use App\Contribution\Traits\FormatsDates;
|
||||||
|
use App\Contribution\Traits\HasPdfBackground;
|
||||||
use App\Country;
|
use App\Country;
|
||||||
use App\Invoice\InvoiceSettings;
|
use App\Invoice\InvoiceSettings;
|
||||||
use Carbon\Carbon;
|
|
||||||
use Illuminate\Support\Collection;
|
use Illuminate\Support\Collection;
|
||||||
use Illuminate\Support\Str;
|
|
||||||
use Zoomyboy\Tex\Engine;
|
|
||||||
use Zoomyboy\Tex\Template;
|
|
||||||
|
|
||||||
class CityFrankfurtMainDocument extends ContributionDocument
|
class CityFrankfurtMainDocument extends ContributionDocument
|
||||||
{
|
{
|
||||||
|
use HasPdfBackground;
|
||||||
|
use FormatsDates;
|
||||||
|
|
||||||
public string $fromName;
|
public string $fromName;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -28,6 +29,7 @@ class CityFrankfurtMainDocument extends ContributionDocument
|
||||||
public ?string $filename = '',
|
public ?string $filename = '',
|
||||||
public string $type = 'F',
|
public string $type = 'F',
|
||||||
) {
|
) {
|
||||||
|
$this->setEventName($eventName);
|
||||||
$this->fromName = app(InvoiceSettings::class)->from_long;
|
$this->fromName = app(InvoiceSettings::class)->from_long;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -61,17 +63,6 @@ class CityFrankfurtMainDocument extends ContributionDocument
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function dateFromHuman(): string
|
|
||||||
{
|
|
||||||
return Carbon::parse($this->dateFrom)->format('d.m.Y');
|
|
||||||
}
|
|
||||||
|
|
||||||
public function dateUntilHuman(): string
|
|
||||||
{
|
|
||||||
return Carbon::parse($this->dateUntil)->format('d.m.Y');
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
public function countryName(): string
|
public function countryName(): string
|
||||||
{
|
{
|
||||||
return $this->country->name;
|
return $this->country->name;
|
||||||
|
@ -82,56 +73,9 @@ class CityFrankfurtMainDocument extends ContributionDocument
|
||||||
return count($this->members);
|
return count($this->members);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function memberShort(MemberData $member): string
|
|
||||||
{
|
|
||||||
return $member->isLeader ? 'L' : '';
|
|
||||||
}
|
|
||||||
|
|
||||||
public function memberName(MemberData $member): string
|
|
||||||
{
|
|
||||||
return $member->separatedName();
|
|
||||||
}
|
|
||||||
|
|
||||||
public function memberAddress(MemberData $member): string
|
|
||||||
{
|
|
||||||
return $member->fullAddress();
|
|
||||||
}
|
|
||||||
|
|
||||||
public function memberAge(MemberData $member): string
|
|
||||||
{
|
|
||||||
return $member->age();
|
|
||||||
}
|
|
||||||
|
|
||||||
public function basename(): string
|
|
||||||
{
|
|
||||||
return 'zuschuesse-frankfurt-' . Str::slug($this->eventName);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function view(): string
|
|
||||||
{
|
|
||||||
return 'tex.contribution.city-frankfurt-main';
|
|
||||||
}
|
|
||||||
|
|
||||||
public function template(): Template
|
|
||||||
{
|
|
||||||
return Template::make('tex.templates.contribution');
|
|
||||||
}
|
|
||||||
|
|
||||||
public function setFilename(string $filename): static
|
|
||||||
{
|
|
||||||
$this->filename = $filename;
|
|
||||||
|
|
||||||
return $this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getEngine(): Engine
|
|
||||||
{
|
|
||||||
return Engine::PDFLATEX;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static function getName(): string
|
public static function getName(): string
|
||||||
{
|
{
|
||||||
return 'Für Frankfurt erstellen';
|
return 'Frankfurt';
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -144,7 +88,6 @@ class CityFrankfurtMainDocument extends ContributionDocument
|
||||||
'dateUntil' => 'required|string|date_format:Y-m-d',
|
'dateUntil' => 'required|string|date_format:Y-m-d',
|
||||||
'country' => 'required|integer|exists:countries,id',
|
'country' => 'required|integer|exists:countries,id',
|
||||||
'zipLocation' => 'required|string',
|
'zipLocation' => 'required|string',
|
||||||
'eventName' => 'required|string',
|
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,15 +3,17 @@
|
||||||
namespace App\Contribution\Documents;
|
namespace App\Contribution\Documents;
|
||||||
|
|
||||||
use App\Contribution\Data\MemberData;
|
use App\Contribution\Data\MemberData;
|
||||||
|
use App\Contribution\Traits\FormatsDates;
|
||||||
|
use App\Contribution\Traits\HasPdfBackground;
|
||||||
use App\Country;
|
use App\Country;
|
||||||
use App\Member\Member;
|
use App\Member\Member;
|
||||||
use Carbon\Carbon;
|
|
||||||
use Illuminate\Support\Collection;
|
use Illuminate\Support\Collection;
|
||||||
use Zoomyboy\Tex\Engine;
|
|
||||||
use Zoomyboy\Tex\Template;
|
|
||||||
|
|
||||||
class CityRemscheidDocument extends ContributionDocument
|
class CityRemscheidDocument extends ContributionDocument
|
||||||
{
|
{
|
||||||
|
use HasPdfBackground;
|
||||||
|
use FormatsDates;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param Collection<int, Collection<int, Member>> $leaders
|
* @param Collection<int, Collection<int, Member>> $leaders
|
||||||
* @param Collection<int, Collection<int, Member>> $children
|
* @param Collection<int, Collection<int, Member>> $children
|
||||||
|
@ -25,17 +27,9 @@ class CityRemscheidDocument extends ContributionDocument
|
||||||
public Collection $children,
|
public Collection $children,
|
||||||
public ?string $filename = '',
|
public ?string $filename = '',
|
||||||
public string $type = 'F',
|
public string $type = 'F',
|
||||||
|
public string $eventName = '',
|
||||||
) {
|
) {
|
||||||
}
|
$this->setEventName($eventName);
|
||||||
|
|
||||||
public function niceDateFrom(): string
|
|
||||||
{
|
|
||||||
return Carbon::parse($this->dateFrom)->format('d.m.Y');
|
|
||||||
}
|
|
||||||
|
|
||||||
public function niceDateUntil(): string
|
|
||||||
{
|
|
||||||
return Carbon::parse($this->dateUntil)->format('d.m.Y');
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -52,6 +46,7 @@ class CityRemscheidDocument extends ContributionDocument
|
||||||
country: Country::where('id', $request['country'])->firstOrFail(),
|
country: Country::where('id', $request['country'])->firstOrFail(),
|
||||||
leaders: $leaders->values()->toBase()->chunk(6),
|
leaders: $leaders->values()->toBase()->chunk(6),
|
||||||
children: $children->values()->toBase()->chunk(20),
|
children: $children->values()->toBase()->chunk(20),
|
||||||
|
eventName: $request['eventName'],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -70,39 +65,13 @@ class CityRemscheidDocument extends ContributionDocument
|
||||||
country: Country::where('id', $request['country'])->firstOrFail(),
|
country: Country::where('id', $request['country'])->firstOrFail(),
|
||||||
leaders: $leaders->values()->toBase()->chunk(6),
|
leaders: $leaders->values()->toBase()->chunk(6),
|
||||||
children: $children->values()->toBase()->chunk(20),
|
children: $children->values()->toBase()->chunk(20),
|
||||||
|
eventName: $request['eventName'],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function basename(): string
|
|
||||||
{
|
|
||||||
return 'zuschuesse-remscheid';
|
|
||||||
}
|
|
||||||
|
|
||||||
public function view(): string
|
|
||||||
{
|
|
||||||
return 'tex.contribution.city-remscheid';
|
|
||||||
}
|
|
||||||
|
|
||||||
public function template(): Template
|
|
||||||
{
|
|
||||||
return Template::make('tex.templates.contribution');
|
|
||||||
}
|
|
||||||
|
|
||||||
public function setFilename(string $filename): static
|
|
||||||
{
|
|
||||||
$this->filename = $filename;
|
|
||||||
|
|
||||||
return $this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getEngine(): Engine
|
|
||||||
{
|
|
||||||
return Engine::PDFLATEX;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static function getName(): string
|
public static function getName(): string
|
||||||
{
|
{
|
||||||
return 'Für Remscheid erstellen';
|
return 'Remscheid';
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -6,9 +6,7 @@ use App\Contribution\Data\MemberData;
|
||||||
use App\Invoice\InvoiceSettings;
|
use App\Invoice\InvoiceSettings;
|
||||||
use Carbon\Carbon;
|
use Carbon\Carbon;
|
||||||
use Illuminate\Support\Collection;
|
use Illuminate\Support\Collection;
|
||||||
use Illuminate\Support\Str;
|
|
||||||
use Zoomyboy\Tex\Engine;
|
use Zoomyboy\Tex\Engine;
|
||||||
use Zoomyboy\Tex\Template;
|
|
||||||
|
|
||||||
class CitySolingenDocument extends ContributionDocument
|
class CitySolingenDocument extends ContributionDocument
|
||||||
{
|
{
|
||||||
|
@ -25,6 +23,7 @@ class CitySolingenDocument extends ContributionDocument
|
||||||
public string $eventName,
|
public string $eventName,
|
||||||
public string $type = 'F',
|
public string $type = 'F',
|
||||||
) {
|
) {
|
||||||
|
$this->setEventName($eventName);
|
||||||
$this->fromName = app(InvoiceSettings::class)->from_long;
|
$this->fromName = app(InvoiceSettings::class)->from_long;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -74,11 +73,6 @@ class CitySolingenDocument extends ContributionDocument
|
||||||
return Carbon::parse($this->dateUntil)->format('d.m.Y');
|
return Carbon::parse($this->dateUntil)->format('d.m.Y');
|
||||||
}
|
}
|
||||||
|
|
||||||
public function template(): Template
|
|
||||||
{
|
|
||||||
return Template::make('tex.templates.contribution');
|
|
||||||
}
|
|
||||||
|
|
||||||
public function checkboxes(): string
|
public function checkboxes(): string
|
||||||
{
|
{
|
||||||
$output = '';
|
$output = '';
|
||||||
|
@ -94,16 +88,6 @@ class CitySolingenDocument extends ContributionDocument
|
||||||
return $firstRow . "\n" . $secondRow;
|
return $firstRow . "\n" . $secondRow;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function basename(): string
|
|
||||||
{
|
|
||||||
return 'zuschuesse-solingen-' . Str::slug($this->eventName);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function view(): string
|
|
||||||
{
|
|
||||||
return 'tex.contribution.city-solingen';
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getEngine(): Engine
|
public function getEngine(): Engine
|
||||||
{
|
{
|
||||||
return Engine::PDFLATEX;
|
return Engine::PDFLATEX;
|
||||||
|
@ -111,7 +95,7 @@ class CitySolingenDocument extends ContributionDocument
|
||||||
|
|
||||||
public static function getName(): string
|
public static function getName(): string
|
||||||
{
|
{
|
||||||
return 'Für Stadt Solingen erstellen';
|
return 'Stadt Solingen';
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -123,7 +107,6 @@ class CitySolingenDocument extends ContributionDocument
|
||||||
'dateFrom' => 'required|string|date_format:Y-m-d',
|
'dateFrom' => 'required|string|date_format:Y-m-d',
|
||||||
'dateUntil' => 'required|string|date_format:Y-m-d',
|
'dateUntil' => 'required|string|date_format:Y-m-d',
|
||||||
'zipLocation' => 'required|string',
|
'zipLocation' => 'required|string',
|
||||||
'eventName' => 'required|string',
|
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,9 +3,12 @@
|
||||||
namespace App\Contribution\Documents;
|
namespace App\Contribution\Documents;
|
||||||
|
|
||||||
use Zoomyboy\Tex\Document;
|
use Zoomyboy\Tex\Document;
|
||||||
|
use Zoomyboy\Tex\Template;
|
||||||
|
|
||||||
abstract class ContributionDocument extends Document
|
abstract class ContributionDocument extends Document
|
||||||
{
|
{
|
||||||
|
private string $eventName;
|
||||||
|
|
||||||
abstract public static function getName(): string;
|
abstract public static function getName(): string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -29,8 +32,34 @@ abstract class ContributionDocument extends Document
|
||||||
public static function globalRules(): array
|
public static function globalRules(): array
|
||||||
{
|
{
|
||||||
return [
|
return [
|
||||||
|
'eventName' => 'required|string',
|
||||||
'members' => 'present|array|min:1',
|
'members' => 'present|array|min:1',
|
||||||
'members.*' => 'integer|exists:members,id',
|
'members.*' => 'integer|exists:members,id',
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static function buttonName(): string
|
||||||
|
{
|
||||||
|
return 'Für ' . static::getName() . ' erstellen';;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setEventName(string $eventName): void
|
||||||
|
{
|
||||||
|
$this->eventName = $eventName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function basename(): string
|
||||||
|
{
|
||||||
|
return str('Zuschüsse ')->append($this->getName())->append(' ')->append($this->eventName)->slug();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function template(): Template
|
||||||
|
{
|
||||||
|
return Template::make('tex.templates.contribution');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function view(): string
|
||||||
|
{
|
||||||
|
return 'tex.contribution.' . str(class_basename(static::class))->replace('Document', '')->kebab()->toString();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,14 +3,16 @@
|
||||||
namespace App\Contribution\Documents;
|
namespace App\Contribution\Documents;
|
||||||
|
|
||||||
use App\Contribution\Data\MemberData;
|
use App\Contribution\Data\MemberData;
|
||||||
|
use App\Contribution\Traits\FormatsDates;
|
||||||
|
use App\Contribution\Traits\HasPdfBackground;
|
||||||
use App\Country;
|
use App\Country;
|
||||||
use Carbon\Carbon;
|
|
||||||
use Illuminate\Support\Collection;
|
use Illuminate\Support\Collection;
|
||||||
use Zoomyboy\Tex\Engine;
|
|
||||||
use Zoomyboy\Tex\Template;
|
|
||||||
|
|
||||||
class RdpNrwDocument extends ContributionDocument
|
class RdpNrwDocument extends ContributionDocument
|
||||||
{
|
{
|
||||||
|
use HasPdfBackground;
|
||||||
|
use FormatsDates;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param Collection<int, Collection<int, MemberData>> $members
|
* @param Collection<int, Collection<int, MemberData>> $members
|
||||||
*/
|
*/
|
||||||
|
@ -22,14 +24,9 @@ class RdpNrwDocument extends ContributionDocument
|
||||||
public Collection $members,
|
public Collection $members,
|
||||||
public ?string $filename = '',
|
public ?string $filename = '',
|
||||||
public string $type = 'F',
|
public string $type = 'F',
|
||||||
|
public string $eventName = '',
|
||||||
) {
|
) {
|
||||||
}
|
$this->setEventName($eventName);
|
||||||
|
|
||||||
public function dateRange(): string
|
|
||||||
{
|
|
||||||
return Carbon::parse($this->dateFrom)->format('d.m.Y')
|
|
||||||
. ' - '
|
|
||||||
. Carbon::parse($this->dateUntil)->format('d.m.Y');
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -43,6 +40,7 @@ class RdpNrwDocument extends ContributionDocument
|
||||||
zipLocation: $request['zipLocation'],
|
zipLocation: $request['zipLocation'],
|
||||||
country: Country::where('id', $request['country'])->firstOrFail(),
|
country: Country::where('id', $request['country'])->firstOrFail(),
|
||||||
members: MemberData::fromModels($request['members'])->chunk(17),
|
members: MemberData::fromModels($request['members'])->chunk(17),
|
||||||
|
eventName: $request['eventName'],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -57,6 +55,7 @@ class RdpNrwDocument extends ContributionDocument
|
||||||
zipLocation: $request['zipLocation'],
|
zipLocation: $request['zipLocation'],
|
||||||
country: Country::where('id', $request['country'])->firstOrFail(),
|
country: Country::where('id', $request['country'])->firstOrFail(),
|
||||||
members: MemberData::fromApi($request['member_data'])->chunk(17),
|
members: MemberData::fromApi($request['member_data'])->chunk(17),
|
||||||
|
eventName: $request['eventName'],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -65,65 +64,9 @@ class RdpNrwDocument extends ContributionDocument
|
||||||
return $this->country->name;
|
return $this->country->name;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function memberShort(MemberData $member): string
|
|
||||||
{
|
|
||||||
return $member->isLeader ? 'L' : '';
|
|
||||||
}
|
|
||||||
|
|
||||||
public function memberName(MemberData $member): string
|
|
||||||
{
|
|
||||||
return $member->separatedName();
|
|
||||||
}
|
|
||||||
|
|
||||||
public function memberAddress(MemberData $member): string
|
|
||||||
{
|
|
||||||
return $member->fullAddress();
|
|
||||||
}
|
|
||||||
|
|
||||||
public function memberGender(MemberData $member): string
|
|
||||||
{
|
|
||||||
if (!$member->gender) {
|
|
||||||
return '';
|
|
||||||
}
|
|
||||||
|
|
||||||
return strtolower(substr($member->gender->name, 0, 1));
|
|
||||||
}
|
|
||||||
|
|
||||||
public function memberAge(MemberData $member): string
|
|
||||||
{
|
|
||||||
return $member->age();
|
|
||||||
}
|
|
||||||
|
|
||||||
public function basename(): string
|
|
||||||
{
|
|
||||||
return 'zuschuesse-rdp-nrw';
|
|
||||||
}
|
|
||||||
|
|
||||||
public function view(): string
|
|
||||||
{
|
|
||||||
return 'tex.contribution.rdp-nrw';
|
|
||||||
}
|
|
||||||
|
|
||||||
public function template(): Template
|
|
||||||
{
|
|
||||||
return Template::make('tex.templates.contribution');
|
|
||||||
}
|
|
||||||
|
|
||||||
public function setFilename(string $filename): static
|
|
||||||
{
|
|
||||||
$this->filename = $filename;
|
|
||||||
|
|
||||||
return $this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getEngine(): Engine
|
|
||||||
{
|
|
||||||
return Engine::PDFLATEX;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static function getName(): string
|
public static function getName(): string
|
||||||
{
|
{
|
||||||
return 'Für RdP NRW erstellen';
|
return 'RdP NRW';
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -136,7 +79,6 @@ class RdpNrwDocument extends ContributionDocument
|
||||||
'dateUntil' => 'required|string|date_format:Y-m-d',
|
'dateUntil' => 'required|string|date_format:Y-m-d',
|
||||||
'country' => 'required|integer|exists:countries,id',
|
'country' => 'required|integer|exists:countries,id',
|
||||||
'zipLocation' => 'required|string',
|
'zipLocation' => 'required|string',
|
||||||
'eventName' => 'required|string',
|
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,79 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Contribution\Documents;
|
||||||
|
|
||||||
|
use App\Contribution\Data\MemberData;
|
||||||
|
use App\Contribution\Traits\FormatsDates;
|
||||||
|
use App\Contribution\Traits\HasPdfBackground;
|
||||||
|
use App\Country;
|
||||||
|
use Illuminate\Support\Collection;
|
||||||
|
|
||||||
|
class WuppertalDocument extends ContributionDocument
|
||||||
|
{
|
||||||
|
|
||||||
|
use HasPdfBackground;
|
||||||
|
use FormatsDates;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param Collection<int, Collection<int, MemberData>> $members
|
||||||
|
*/
|
||||||
|
public function __construct(
|
||||||
|
public string $dateFrom,
|
||||||
|
public string $dateUntil,
|
||||||
|
public string $zipLocation,
|
||||||
|
public ?Country $country,
|
||||||
|
public Collection $members,
|
||||||
|
public ?string $filename = '',
|
||||||
|
public string $type = 'F',
|
||||||
|
public string $eventName = '',
|
||||||
|
) {
|
||||||
|
$this->setEventName($eventName);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritdoc}
|
||||||
|
*/
|
||||||
|
public static function fromRequest(array $request): self
|
||||||
|
{
|
||||||
|
return new self(
|
||||||
|
dateFrom: $request['dateFrom'],
|
||||||
|
dateUntil: $request['dateUntil'],
|
||||||
|
zipLocation: $request['zipLocation'],
|
||||||
|
country: Country::where('id', $request['country'])->firstOrFail(),
|
||||||
|
members: MemberData::fromModels($request['members'])->chunk(14),
|
||||||
|
eventName: $request['eventName'],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritdoc}
|
||||||
|
*/
|
||||||
|
public static function fromApiRequest(array $request): self
|
||||||
|
{
|
||||||
|
return new self(
|
||||||
|
dateFrom: $request['dateFrom'],
|
||||||
|
dateUntil: $request['dateUntil'],
|
||||||
|
zipLocation: $request['zipLocation'],
|
||||||
|
country: Country::where('id', $request['country'])->firstOrFail(),
|
||||||
|
members: MemberData::fromApi($request['member_data'])->chunk(14),
|
||||||
|
eventName: $request['eventName'],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function getName(): string
|
||||||
|
{
|
||||||
|
return 'Wuppertal';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<string, mixed>
|
||||||
|
*/
|
||||||
|
public static function rules(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'dateFrom' => 'required|string|date_format:Y-m-d',
|
||||||
|
'dateUntil' => 'required|string|date_format:Y-m-d',
|
||||||
|
'zipLocation' => 'required|string',
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,24 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Contribution\Traits;
|
||||||
|
|
||||||
|
use Carbon\Carbon;
|
||||||
|
|
||||||
|
trait FormatsDates
|
||||||
|
{
|
||||||
|
|
||||||
|
public function niceDateFrom(): string
|
||||||
|
{
|
||||||
|
return Carbon::parse($this->dateFrom)->format('d.m.Y');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function niceDateUntil(): string
|
||||||
|
{
|
||||||
|
return Carbon::parse($this->dateUntil)->format('d.m.Y');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function dateRange(): string
|
||||||
|
{
|
||||||
|
return implode(' - ', [$this->niceDateFrom(), $this->niceDateUntil()]);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,13 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Contribution\Traits;
|
||||||
|
|
||||||
|
use Zoomyboy\Tex\Engine;
|
||||||
|
|
||||||
|
trait HasPdfBackground
|
||||||
|
{
|
||||||
|
public function getEngine(): Engine
|
||||||
|
{
|
||||||
|
return Engine::PDFLATEX;
|
||||||
|
}
|
||||||
|
}
|
|
@ -22,8 +22,8 @@ class FormUpdateMetaAction
|
||||||
|
|
||||||
return [
|
return [
|
||||||
'sorting' => 'array',
|
'sorting' => 'array',
|
||||||
'sorting.0' => 'required|string',
|
'sorting.by' => 'required|string',
|
||||||
'sorting.1' => 'required|string|in:asc,desc',
|
'sorting.direction' => 'required|boolean',
|
||||||
'active_columns' => 'array',
|
'active_columns' => 'array',
|
||||||
'active_columns.*' => ['string', Rule::in([...$form->getFields()->pluck('key')->toArray(), 'created_at', 'prevention'])]
|
'active_columns.*' => ['string', Rule::in([...$form->getFields()->pluck('key')->toArray(), 'created_at', 'prevention'])]
|
||||||
];
|
];
|
||||||
|
|
|
@ -6,9 +6,8 @@ use App\Form\Models\Form;
|
||||||
use App\Form\Models\Participant;
|
use App\Form\Models\Participant;
|
||||||
use App\Form\Resources\ParticipantResource;
|
use App\Form\Resources\ParticipantResource;
|
||||||
use App\Form\Scopes\ParticipantFilterScope;
|
use App\Form\Scopes\ParticipantFilterScope;
|
||||||
use Illuminate\Database\Eloquent\Relations\HasMany;
|
|
||||||
use Illuminate\Http\Resources\Json\AnonymousResourceCollection;
|
use Illuminate\Http\Resources\Json\AnonymousResourceCollection;
|
||||||
use Illuminate\Pagination\LengthAwarePaginator;
|
use Laravel\Scout\Builder;
|
||||||
use Lorisleiva\Actions\Concerns\AsAction;
|
use Lorisleiva\Actions\Concerns\AsAction;
|
||||||
|
|
||||||
class ParticipantIndexAction
|
class ParticipantIndexAction
|
||||||
|
@ -16,29 +15,22 @@ class ParticipantIndexAction
|
||||||
use AsAction;
|
use AsAction;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return HasMany<Participant>
|
* @return Builder<Participant>
|
||||||
*/
|
*/
|
||||||
protected function getQuery(Form $form, ParticipantFilterScope $filter): HasMany
|
protected function getQuery(Form $form, ParticipantFilterScope $filter): Builder
|
||||||
{
|
{
|
||||||
return $form->participants()->withFilter($filter)->withCount('children')->with('form');
|
return $filter->setForm($form)->getQuery()
|
||||||
}
|
->query(fn ($q) => $q->withCount('children')->with('form'));
|
||||||
|
|
||||||
/**
|
|
||||||
* @return LengthAwarePaginator<Participant>
|
|
||||||
*/
|
|
||||||
public function handle(Form $form, ParticipantFilterScope $filter): LengthAwarePaginator
|
|
||||||
{
|
|
||||||
return $this->getQuery($form, $filter)->paginate(15);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function asController(Form $form, ?int $parent = null): AnonymousResourceCollection
|
public function asController(Form $form, ?int $parent = null): AnonymousResourceCollection
|
||||||
{
|
{
|
||||||
$filter = ParticipantFilterScope::fromRequest(request()->input('filter'));
|
$filter = ParticipantFilterScope::fromRequest(request()->input('filter', ''))->parent($parent);
|
||||||
|
|
||||||
$data = match ($parent) {
|
$data = match ($parent) {
|
||||||
null => $this->handle($form, $filter),
|
null => $this->getQuery($form, $filter)->paginate(15), // initial all elements - paginate
|
||||||
-1 => $this->getQuery($form, $filter)->where('parent_id', null)->paginate(15),
|
-1 => $this->getQuery($form, $filter)->paginate(15), // initial root elements - parinate
|
||||||
default => $this->getQuery($form, $filter)->where('parent_id', $parent)->get(),
|
default => $this->getQuery($form, $filter)->get(), // specific parent element - show all
|
||||||
};
|
};
|
||||||
|
|
||||||
return ParticipantResource::collection($data)->additional(['meta' => ParticipantResource::meta($form)]);
|
return ParticipantResource::collection($data)->additional(['meta' => ParticipantResource::meta($form)]);
|
||||||
|
|
|
@ -17,7 +17,12 @@ class PreventionRememberAction
|
||||||
|
|
||||||
public function handle(): void
|
public function handle(): void
|
||||||
{
|
{
|
||||||
$query = Participant::whereHas('form', fn ($form) => $form->where('needs_prevention', true))
|
$query = Participant::whereHas(
|
||||||
|
'form',
|
||||||
|
fn ($form) => $form
|
||||||
|
->where('needs_prevention', true)
|
||||||
|
->where('from', '>=', now())
|
||||||
|
)
|
||||||
->where(
|
->where(
|
||||||
fn ($q) => $q
|
fn ($q) => $q
|
||||||
->where('last_remembered_at', '<=', now()->subWeeks(2))
|
->where('last_remembered_at', '<=', now()->subWeeks(2))
|
||||||
|
|
|
@ -7,6 +7,7 @@ use App\Form\Models\Form;
|
||||||
use App\Form\Models\Participant;
|
use App\Form\Models\Participant;
|
||||||
use App\Member\Member;
|
use App\Member\Member;
|
||||||
use Illuminate\Http\JsonResponse;
|
use Illuminate\Http\JsonResponse;
|
||||||
|
use Illuminate\Validation\ValidationException;
|
||||||
use Lorisleiva\Actions\ActionRequest;
|
use Lorisleiva\Actions\ActionRequest;
|
||||||
use Lorisleiva\Actions\Concerns\AsAction;
|
use Lorisleiva\Actions\Concerns\AsAction;
|
||||||
|
|
||||||
|
@ -19,6 +20,10 @@ class RegisterAction
|
||||||
*/
|
*/
|
||||||
public function handle(Form $form, array $input): Participant
|
public function handle(Form $form, array $input): Participant
|
||||||
{
|
{
|
||||||
|
if (!$form->canRegister()) {
|
||||||
|
throw ValidationException::withMessages(['event' => 'Anmeldung zzt nicht möglich.']);
|
||||||
|
}
|
||||||
|
|
||||||
$memberQuery = FieldCollection::fromRequest($form, $input)
|
$memberQuery = FieldCollection::fromRequest($form, $input)
|
||||||
->withNamiType()
|
->withNamiType()
|
||||||
->reduce(fn ($query, $field) => $field->namiType->performQuery($query, $field->value), (new Member())->newQuery());
|
->reduce(fn ($query, $field) => $field->namiType->performQuery($query, $field->value), (new Member())->newQuery());
|
||||||
|
|
|
@ -0,0 +1,31 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Form\Actions;
|
||||||
|
|
||||||
|
use App\Form\Models\Form;
|
||||||
|
use Lorisleiva\Actions\Concerns\AsAction;
|
||||||
|
|
||||||
|
class UpdateParticipantSearchIndexAction
|
||||||
|
{
|
||||||
|
use AsAction;
|
||||||
|
|
||||||
|
public function handle(Form $form): void
|
||||||
|
{
|
||||||
|
if (config('scout.driver') !== 'meilisearch') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$form->searchableUsing()->updateIndexSettings(
|
||||||
|
$form->participantsSearchableAs(),
|
||||||
|
[
|
||||||
|
'filterableAttributes' => [...$form->getFields()->filterables()->getKeys(), 'parent-id'],
|
||||||
|
'searchableAttributes' => $form->getFields()->searchables()->getKeys(),
|
||||||
|
'sortableAttributes' => [...$form->getFields()->sortables()->getKeys(), 'id', 'created_at'],
|
||||||
|
'displayedAttributes' => [...$form->getFields()->filterables()->getKeys(), ...$form->getFields()->searchables()->getKeys(), 'id'],
|
||||||
|
'pagination' => [
|
||||||
|
'maxTotalHits' => 1000000,
|
||||||
|
]
|
||||||
|
]
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,9 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Form\Contracts;
|
||||||
|
|
||||||
|
interface Filterable
|
||||||
|
{
|
||||||
|
/** @param mixed $value */
|
||||||
|
public function filter($value): string;
|
||||||
|
}
|
|
@ -2,6 +2,7 @@
|
||||||
|
|
||||||
namespace App\Form\Data;
|
namespace App\Form\Data;
|
||||||
|
|
||||||
|
use App\Form\Contracts\Filterable;
|
||||||
use App\Form\Enums\SpecialType;
|
use App\Form\Enums\SpecialType;
|
||||||
use App\Form\Fields\Field;
|
use App\Form\Fields\Field;
|
||||||
use App\Form\Fields\NamiField;
|
use App\Form\Fields\NamiField;
|
||||||
|
@ -117,4 +118,27 @@ class FieldCollection extends Collection
|
||||||
{
|
{
|
||||||
return $this->first(fn ($field) => $field->specialType === $specialType);
|
return $this->first(fn ($field) => $field->specialType === $specialType);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function searchables(): self
|
||||||
|
{
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function sortables(): self
|
||||||
|
{
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function filterables(): self
|
||||||
|
{
|
||||||
|
return $this->filter(fn ($field) => $field instanceof Filterable);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<int, string>
|
||||||
|
*/
|
||||||
|
public function getKeys(): array
|
||||||
|
{
|
||||||
|
return $this->map(fn ($field) => $field->key)->toArray();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
|
|
||||||
namespace App\Form\Fields;
|
namespace App\Form\Fields;
|
||||||
|
|
||||||
|
use App\Form\Contracts\Filterable;
|
||||||
use App\Form\Matchers\BooleanMatcher;
|
use App\Form\Matchers\BooleanMatcher;
|
||||||
use App\Form\Matchers\Matcher;
|
use App\Form\Matchers\Matcher;
|
||||||
use App\Form\Models\Form;
|
use App\Form\Models\Form;
|
||||||
|
@ -9,9 +10,8 @@ use App\Form\Models\Participant;
|
||||||
use App\Form\Presenters\BooleanPresenter;
|
use App\Form\Presenters\BooleanPresenter;
|
||||||
use App\Form\Presenters\Presenter;
|
use App\Form\Presenters\Presenter;
|
||||||
use Faker\Generator;
|
use Faker\Generator;
|
||||||
use Illuminate\Validation\Rule;
|
|
||||||
|
|
||||||
class CheckboxField extends Field
|
class CheckboxField extends Field implements Filterable
|
||||||
{
|
{
|
||||||
public bool $required;
|
public bool $required;
|
||||||
public string $description;
|
public string $description;
|
||||||
|
@ -86,4 +86,11 @@ class CheckboxField extends Field
|
||||||
{
|
{
|
||||||
return app(BooleanMatcher::class);
|
return app(BooleanMatcher::class);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function filter($value): string
|
||||||
|
{
|
||||||
|
$asString = $value ? 'true' : 'false';
|
||||||
|
|
||||||
|
return "{$this->key} = $asString";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
|
|
||||||
namespace App\Form\Fields;
|
namespace App\Form\Fields;
|
||||||
|
|
||||||
|
use App\Form\Contracts\Filterable;
|
||||||
use App\Form\Matchers\Matcher;
|
use App\Form\Matchers\Matcher;
|
||||||
use App\Form\Matchers\SingleValueMatcher;
|
use App\Form\Matchers\SingleValueMatcher;
|
||||||
use App\Form\Models\Form;
|
use App\Form\Models\Form;
|
||||||
|
@ -9,7 +10,7 @@ use App\Form\Models\Participant;
|
||||||
use Faker\Generator;
|
use Faker\Generator;
|
||||||
use Illuminate\Validation\Rule;
|
use Illuminate\Validation\Rule;
|
||||||
|
|
||||||
class DropdownField extends Field
|
class DropdownField extends Field implements Filterable
|
||||||
{
|
{
|
||||||
public bool $required;
|
public bool $required;
|
||||||
/** @var array<int, string> */
|
/** @var array<int, string> */
|
||||||
|
@ -87,4 +88,14 @@ class DropdownField extends Field
|
||||||
{
|
{
|
||||||
return app(SingleValueMatcher::class);
|
return app(SingleValueMatcher::class);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** @inheritdoc */
|
||||||
|
public function filter($value): string
|
||||||
|
{
|
||||||
|
if (is_null($value)) {
|
||||||
|
return "{$this->key} IS NULL";
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->key . ' = \'' . $value . '\'';
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -180,4 +180,10 @@ abstract class Field extends Data
|
||||||
{
|
{
|
||||||
return app(SingleValueMatcher::class);
|
return app(SingleValueMatcher::class);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** @param mixed $value */
|
||||||
|
public function filter($value): string
|
||||||
|
{
|
||||||
|
return '';
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
|
|
||||||
namespace App\Form\Fields;
|
namespace App\Form\Fields;
|
||||||
|
|
||||||
|
use App\Form\Contracts\Filterable;
|
||||||
use App\Form\Matchers\Matcher;
|
use App\Form\Matchers\Matcher;
|
||||||
use App\Form\Matchers\SingleValueMatcher;
|
use App\Form\Matchers\SingleValueMatcher;
|
||||||
use App\Form\Models\Form;
|
use App\Form\Models\Form;
|
||||||
|
@ -9,7 +10,7 @@ use App\Form\Models\Participant;
|
||||||
use Faker\Generator;
|
use Faker\Generator;
|
||||||
use Illuminate\Validation\Rule;
|
use Illuminate\Validation\Rule;
|
||||||
|
|
||||||
class RadioField extends Field
|
class RadioField extends Field implements Filterable
|
||||||
{
|
{
|
||||||
public bool $required;
|
public bool $required;
|
||||||
/** @var array<int, string> */
|
/** @var array<int, string> */
|
||||||
|
@ -87,4 +88,13 @@ class RadioField extends Field
|
||||||
{
|
{
|
||||||
return app(SingleValueMatcher::class);
|
return app(SingleValueMatcher::class);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function filter($value): string
|
||||||
|
{
|
||||||
|
if (is_null($value)) {
|
||||||
|
return "{$this->key} IS NULL";
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->key . ' = \'' . $value . '\'';
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,11 +2,13 @@
|
||||||
|
|
||||||
namespace App\Form\Models;
|
namespace App\Form\Models;
|
||||||
|
|
||||||
|
use App\Form\Actions\UpdateParticipantSearchIndexAction;
|
||||||
use App\Form\Data\ExportData;
|
use App\Form\Data\ExportData;
|
||||||
use App\Form\Data\FieldCollection;
|
use App\Form\Data\FieldCollection;
|
||||||
use App\Form\Data\FormConfigData;
|
use App\Form\Data\FormConfigData;
|
||||||
use App\Lib\Editor\Condition;
|
use App\Lib\Editor\Condition;
|
||||||
use App\Lib\Editor\EditorData;
|
use App\Lib\Editor\EditorData;
|
||||||
|
use App\Lib\Sorting;
|
||||||
use Cviebrock\EloquentSluggable\Sluggable;
|
use Cviebrock\EloquentSluggable\Sluggable;
|
||||||
use Database\Factories\Form\Models\FormFactory;
|
use Database\Factories\Form\Models\FormFactory;
|
||||||
use Illuminate\Database\Eloquent\Factories\HasFactory;
|
use Illuminate\Database\Eloquent\Factories\HasFactory;
|
||||||
|
@ -14,7 +16,6 @@ use Illuminate\Database\Eloquent\Model;
|
||||||
use Illuminate\Database\Eloquent\Relations\HasMany;
|
use Illuminate\Database\Eloquent\Relations\HasMany;
|
||||||
use Laravel\Scout\Searchable;
|
use Laravel\Scout\Searchable;
|
||||||
use Spatie\Image\Enums\Fit;
|
use Spatie\Image\Enums\Fit;
|
||||||
use Spatie\Image\Manipulations;
|
|
||||||
use Spatie\MediaLibrary\HasMedia;
|
use Spatie\MediaLibrary\HasMedia;
|
||||||
use Spatie\MediaLibrary\InteractsWithMedia;
|
use Spatie\MediaLibrary\InteractsWithMedia;
|
||||||
use Spatie\MediaLibrary\MediaCollections\Models\Media;
|
use Spatie\MediaLibrary\MediaCollections\Models\Media;
|
||||||
|
@ -159,7 +160,7 @@ class Form extends Model implements HasMedia
|
||||||
if (is_null(data_get($model->meta, 'active_columns'))) {
|
if (is_null(data_get($model->meta, 'active_columns'))) {
|
||||||
$model->setAttribute('meta', [
|
$model->setAttribute('meta', [
|
||||||
'active_columns' => $model->getFields()->count() ? $model->getFields()->take(4)->pluck('key')->toArray() : null,
|
'active_columns' => $model->getFields()->count() ? $model->getFields()->take(4)->pluck('key')->toArray() : null,
|
||||||
'sorting' => $model->getFields()->count() ? [$model->getFields()->first()->key, 'asc'] : null,
|
'sorting' => Sorting::by('id'),
|
||||||
]);
|
]);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -172,5 +173,32 @@ class Form extends Model implements HasMedia
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
static::saved(function ($model) {
|
||||||
|
UpdateParticipantSearchIndexAction::dispatch($model);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public function participantsSearchableAs(): string
|
||||||
|
{
|
||||||
|
return config('scout.prefix') . 'forms_' . $this->id . '_participants';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function defaultSorting(): Sorting
|
||||||
|
{
|
||||||
|
return Sorting::from($this->meta['sorting']);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function canRegister(): bool
|
||||||
|
{
|
||||||
|
if ($this->registration_from && $this->registration_from->gt(now())) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->registration_until && $this->registration_until->lt(now())) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,6 +15,7 @@ use Illuminate\Database\Eloquent\Model;
|
||||||
use Illuminate\Database\Eloquent\Relations\BelongsTo;
|
use Illuminate\Database\Eloquent\Relations\BelongsTo;
|
||||||
use Illuminate\Database\Eloquent\Relations\HasMany;
|
use Illuminate\Database\Eloquent\Relations\HasMany;
|
||||||
use Illuminate\Support\Facades\Mail;
|
use Illuminate\Support\Facades\Mail;
|
||||||
|
use Laravel\Scout\Searchable;
|
||||||
use stdClass;
|
use stdClass;
|
||||||
|
|
||||||
class Participant extends Model implements Preventable
|
class Participant extends Model implements Preventable
|
||||||
|
@ -22,6 +23,7 @@ class Participant extends Model implements Preventable
|
||||||
|
|
||||||
/** @use HasFactory<ParticipantFactory> */
|
/** @use HasFactory<ParticipantFactory> */
|
||||||
use HasFactory;
|
use HasFactory;
|
||||||
|
use Searchable;
|
||||||
|
|
||||||
public $guarded = [];
|
public $guarded = [];
|
||||||
|
|
||||||
|
@ -46,15 +48,6 @@ class Participant extends Model implements Preventable
|
||||||
return $this->hasMany(self::class, 'parent_id');
|
return $this->hasMany(self::class, 'parent_id');
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* @param Builder<self> $query
|
|
||||||
* @return Builder<self>
|
|
||||||
*/
|
|
||||||
public function scopeWithFilter(Builder $query, ParticipantFilterScope $filter): Builder
|
|
||||||
{
|
|
||||||
return $filter->apply($query);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return BelongsTo<Member, self>
|
* @return BelongsTo<Member, self>
|
||||||
*/
|
*/
|
||||||
|
@ -110,4 +103,15 @@ class Participant extends Model implements Preventable
|
||||||
{
|
{
|
||||||
return 'Nachweise erforderlich für deine Anmeldung zu ' . $this->form->name;
|
return 'Nachweise erforderlich für deine Anmeldung zu ' . $this->form->name;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function searchableAs(): string
|
||||||
|
{
|
||||||
|
return $this->form->participantsSearchableAs();
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @return array<string, mixed> */
|
||||||
|
public function toSearchableArray(): array
|
||||||
|
{
|
||||||
|
return [...$this->data, 'parent-id' => $this->parent_id, 'created_at' => $this->created_at->timestamp];
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -35,6 +35,7 @@ class FormApiResource extends JsonResource
|
||||||
'image' => $this->getMedia('headerImage')->first()->getFullUrl('square'),
|
'image' => $this->getMedia('headerImage')->first()->getFullUrl('square'),
|
||||||
'is_active' => $this->is_active,
|
'is_active' => $this->is_active,
|
||||||
'is_private' => $this->is_private,
|
'is_private' => $this->is_private,
|
||||||
|
'can_register' => $this->getModel()->canRegister(),
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -4,32 +4,76 @@ namespace App\Form\Scopes;
|
||||||
|
|
||||||
use App\Form\Models\Form;
|
use App\Form\Models\Form;
|
||||||
use App\Form\Models\Participant;
|
use App\Form\Models\Participant;
|
||||||
use App\Lib\Filter;
|
use App\Lib\ScoutFilter;
|
||||||
use Illuminate\Database\Eloquent\Builder;
|
use App\Lib\Sorting;
|
||||||
use Illuminate\Support\Arr;
|
use Illuminate\Support\Arr;
|
||||||
|
use Laravel\Scout\Builder;
|
||||||
use Spatie\LaravelData\Attributes\MapInputName;
|
use Spatie\LaravelData\Attributes\MapInputName;
|
||||||
use Spatie\LaravelData\Attributes\MapOutputName;
|
use Spatie\LaravelData\Attributes\MapOutputName;
|
||||||
use Spatie\LaravelData\Mappers\SnakeCaseMapper;
|
use Spatie\LaravelData\Mappers\SnakeCaseMapper;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @extends Filter<Participant>
|
* @extends ScoutFilter<Participant>
|
||||||
*/
|
*/
|
||||||
#[MapInputName(SnakeCaseMapper::class)]
|
#[MapInputName(SnakeCaseMapper::class)]
|
||||||
#[MapOutputName(SnakeCaseMapper::class)]
|
#[MapOutputName(SnakeCaseMapper::class)]
|
||||||
class ParticipantFilterScope extends Filter
|
class ParticipantFilterScope extends ScoutFilter
|
||||||
{
|
{
|
||||||
|
|
||||||
|
public static string $nan = 'deeb3ef4-d185-44b1-a4bc-0a4e7addebc3d8900c6f-a344-4afb-b54e-065ed483a7ba';
|
||||||
|
private Form $form;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param array<string, mixed> $data
|
* @param array<string, mixed> $data
|
||||||
|
* @param array<string, mixed> $options
|
||||||
*/
|
*/
|
||||||
public function __construct(
|
public function __construct(
|
||||||
public array $data = [],
|
public array $data = [],
|
||||||
|
public string $search = '',
|
||||||
|
public array $options = [],
|
||||||
|
public ?int $parent = null,
|
||||||
|
public ?Sorting $sort = null
|
||||||
) {
|
) {
|
||||||
}
|
}
|
||||||
|
|
||||||
public static string $nan = 'deeb3ef4-d185-44b1-a4bc-0a4e7addebc3d8900c6f-a344-4afb-b54e-065ed483a7ba';
|
public function getQuery(): Builder
|
||||||
|
{
|
||||||
|
$this->search = $this->search ?: '';
|
||||||
|
|
||||||
|
return Participant::search($this->search, function ($engine, string $query, array $options) {
|
||||||
|
$filter = collect([]);
|
||||||
|
|
||||||
|
foreach ($this->form->getFields()->filterables() as $field) {
|
||||||
|
if ($this->data[$field->key] === static::$nan) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$filter->push($field->filter($this->data[$field->key]));
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->parent === -1) {
|
||||||
|
$filter->push('parent-id IS NULL');
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->parent !== null && $this->parent !== -1) {
|
||||||
|
$filter->push('parent-id = ' . $this->parent);
|
||||||
|
}
|
||||||
|
|
||||||
|
$options['filter'] = $filter->map(fn ($expression) => "($expression)")->implode(' AND ');
|
||||||
|
|
||||||
|
$options['sort'] = $this->sort->toMeilisearch();
|
||||||
|
|
||||||
|
return $engine->search($query, [...$this->options, ...$options]);
|
||||||
|
})->within($this->form->participantsSearchableAs());
|
||||||
|
}
|
||||||
|
|
||||||
public function setForm(Form $form): self
|
public function setForm(Form $form): self
|
||||||
{
|
{
|
||||||
|
$this->form = $form;
|
||||||
|
|
||||||
|
if (is_null($this->sort)) {
|
||||||
|
$this->sort = $this->form->defaultSorting();
|
||||||
|
}
|
||||||
|
|
||||||
foreach ($form->getFields() as $field) {
|
foreach ($form->getFields() as $field) {
|
||||||
if (!Arr::has($this->data, $field->key)) {
|
if (!Arr::has($this->data, $field->key)) {
|
||||||
data_set($this->data, $field->key, static::$nan);
|
data_set($this->data, $field->key, static::$nan);
|
||||||
|
@ -39,18 +83,10 @@ class ParticipantFilterScope extends Filter
|
||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
public function parent(?int $parent): self
|
||||||
* @inheritdoc
|
|
||||||
*/
|
|
||||||
public function apply(Builder $query): Builder
|
|
||||||
{
|
{
|
||||||
foreach ($this->data as $key => $value) {
|
$this->parent = $parent;
|
||||||
if ($value === static::$nan) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
$query = $query->where('data->' . $key, $value);
|
|
||||||
}
|
|
||||||
|
|
||||||
return $query;
|
return $this;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -25,6 +25,15 @@ class Gender extends Model
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getShortAttribute(): string
|
||||||
|
{
|
||||||
|
return match ($this->name) {
|
||||||
|
'Männlich' => 'm',
|
||||||
|
'Weiblich' => 'w',
|
||||||
|
default => ''
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
public static function fromString(string $title): self
|
public static function fromString(string $title): self
|
||||||
{
|
{
|
||||||
return self::firstWhere('name', $title);
|
return self::firstWhere('name', $title);
|
||||||
|
|
|
@ -6,9 +6,9 @@ use Lorisleiva\Actions\Concerns\AsAction;
|
||||||
use App\Invoice\Models\Invoice;
|
use App\Invoice\Models\Invoice;
|
||||||
use App\Invoice\Resources\InvoiceResource;
|
use App\Invoice\Resources\InvoiceResource;
|
||||||
use App\Invoice\Scopes\InvoiceFilterScope;
|
use App\Invoice\Scopes\InvoiceFilterScope;
|
||||||
use Illuminate\Pagination\LengthAwarePaginator;
|
|
||||||
use Inertia\Inertia;
|
use Inertia\Inertia;
|
||||||
use Inertia\Response;
|
use Inertia\Response;
|
||||||
|
use Laravel\Scout\Builder;
|
||||||
use Lorisleiva\Actions\ActionRequest;
|
use Lorisleiva\Actions\ActionRequest;
|
||||||
|
|
||||||
class InvoiceIndexAction
|
class InvoiceIndexAction
|
||||||
|
@ -17,11 +17,11 @@ class InvoiceIndexAction
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return LengthAwarePaginator<Invoice>
|
* @return Builder<Invoice>
|
||||||
*/
|
*/
|
||||||
public function handle(InvoiceFilterScope $filter): LengthAwarePaginator
|
public function handle(InvoiceFilterScope $filter): Builder
|
||||||
{
|
{
|
||||||
return Invoice::withFilter($filter)->with('positions')->paginate(15);
|
return $filter->getQuery()->query(fn ($q) => $q->with('positions'));
|
||||||
}
|
}
|
||||||
|
|
||||||
public function asController(ActionRequest $request): Response
|
public function asController(ActionRequest $request): Response
|
||||||
|
@ -32,7 +32,7 @@ class InvoiceIndexAction
|
||||||
$filter = InvoiceFilterScope::fromRequest($request->input('filter', ''));
|
$filter = InvoiceFilterScope::fromRequest($request->input('filter', ''));
|
||||||
|
|
||||||
return Inertia::render('invoice/Index', [
|
return Inertia::render('invoice/Index', [
|
||||||
'data' => InvoiceResource::collection($this->handle($filter)),
|
'data' => InvoiceResource::collection($this->handle($filter)->paginate(15)),
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,27 +8,17 @@ use Lorisleiva\Actions\ActionRequest;
|
||||||
|
|
||||||
class InvoiceSettings extends LocalSettings implements Storeable
|
class InvoiceSettings extends LocalSettings implements Storeable
|
||||||
{
|
{
|
||||||
public string $from_long;
|
public ?string $from_long;
|
||||||
|
public ?string $from;
|
||||||
public string $from;
|
public ?string $mobile;
|
||||||
|
public ?string $email;
|
||||||
public string $mobile;
|
public ?string $website;
|
||||||
|
public ?string $address;
|
||||||
public string $email;
|
public ?string $place;
|
||||||
|
public ?string $zip;
|
||||||
public string $website;
|
public ?string $iban;
|
||||||
|
public ?string $bic;
|
||||||
public string $address;
|
public ?int $rememberWeeks;
|
||||||
|
|
||||||
public string $place;
|
|
||||||
|
|
||||||
public string $zip;
|
|
||||||
|
|
||||||
public string $iban;
|
|
||||||
|
|
||||||
public string $bic;
|
|
||||||
|
|
||||||
public int $rememberWeeks;
|
|
||||||
|
|
||||||
public static function group(): string
|
public static function group(): string
|
||||||
{
|
{
|
||||||
|
|
|
@ -17,12 +17,14 @@ use Illuminate\Database\Eloquent\Collection;
|
||||||
use Illuminate\Database\Eloquent\Factories\HasFactory;
|
use Illuminate\Database\Eloquent\Factories\HasFactory;
|
||||||
use Illuminate\Database\Eloquent\Model;
|
use Illuminate\Database\Eloquent\Model;
|
||||||
use Illuminate\Database\Eloquent\Relations\HasMany;
|
use Illuminate\Database\Eloquent\Relations\HasMany;
|
||||||
|
use Laravel\Scout\Searchable;
|
||||||
use stdClass;
|
use stdClass;
|
||||||
|
|
||||||
class Invoice extends Model
|
class Invoice extends Model
|
||||||
{
|
{
|
||||||
/** @use HasFactory<InvoiceFactory> */
|
/** @use HasFactory<InvoiceFactory> */
|
||||||
use HasFactory;
|
use HasFactory;
|
||||||
|
use Searchable;
|
||||||
|
|
||||||
public $guarded = [];
|
public $guarded = [];
|
||||||
|
|
||||||
|
@ -120,15 +122,6 @@ class Invoice extends Model
|
||||||
->where('last_remembered_at', '<=', now()->subWeeks($weeks));
|
->where('last_remembered_at', '<=', now()->subWeeks($weeks));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* @param Builder<self> $query
|
|
||||||
* @return Builder<self>
|
|
||||||
*/
|
|
||||||
public function scopeWithFilter(Builder $query, InvoiceFilterScope $filter): Builder
|
|
||||||
{
|
|
||||||
return $filter->apply($query);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getMailRecipient(): stdClass
|
public function getMailRecipient(): stdClass
|
||||||
{
|
{
|
||||||
return (object) [
|
return (object) [
|
||||||
|
@ -154,4 +147,20 @@ class Invoice extends Model
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the indexable data array for the model.
|
||||||
|
*
|
||||||
|
* @return array<string, mixed>
|
||||||
|
*/
|
||||||
|
public function toSearchableArray(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'to' => implode(', ', $this->to),
|
||||||
|
'usage' => $this->usage,
|
||||||
|
'greeting' => $this->greeting,
|
||||||
|
'mail_email' => $this->mail_email,
|
||||||
|
'status' => $this->status->value,
|
||||||
|
];
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,42 +5,43 @@ namespace App\Invoice\Scopes;
|
||||||
use App\Invoice\Enums\InvoiceStatus;
|
use App\Invoice\Enums\InvoiceStatus;
|
||||||
use App\Invoice\Models\Invoice;
|
use App\Invoice\Models\Invoice;
|
||||||
use App\Lib\Filter;
|
use App\Lib\Filter;
|
||||||
use Illuminate\Database\Eloquent\Builder;
|
use App\Lib\ScoutFilter;
|
||||||
|
use Laravel\Scout\Builder;
|
||||||
use Spatie\LaravelData\Attributes\MapInputName;
|
use Spatie\LaravelData\Attributes\MapInputName;
|
||||||
use Spatie\LaravelData\Attributes\MapOutputName;
|
use Spatie\LaravelData\Attributes\MapOutputName;
|
||||||
use Spatie\LaravelData\Mappers\SnakeCaseMapper;
|
use Spatie\LaravelData\Mappers\SnakeCaseMapper;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @extends Filter<Invoice>
|
* @extends ScoutFilter<Invoice>
|
||||||
*/
|
*/
|
||||||
#[MapInputName(SnakeCaseMapper::class)]
|
#[MapInputName(SnakeCaseMapper::class)]
|
||||||
#[MapOutputName(SnakeCaseMapper::class)]
|
#[MapOutputName(SnakeCaseMapper::class)]
|
||||||
class InvoiceFilterScope extends Filter
|
class InvoiceFilterScope extends ScoutFilter
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* @param array<int, string> $statuses
|
* @param array<int, string> $statuses
|
||||||
*/
|
*/
|
||||||
public function __construct(
|
public function __construct(
|
||||||
public ?array $statuses = null,
|
public ?array $statuses = null,
|
||||||
|
public ?string $search = null
|
||||||
) {
|
) {
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @inheritdoc
|
|
||||||
*/
|
|
||||||
public function apply(Builder $query): Builder
|
|
||||||
{
|
|
||||||
$query = $query->whereIn('status', $this->statuses);
|
|
||||||
|
|
||||||
return $query;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @inheritdoc
|
|
||||||
*/
|
|
||||||
public function toDefault(): self
|
|
||||||
{
|
|
||||||
$this->statuses = $this->statuses === null ? InvoiceStatus::defaultVisibleValues()->toArray() : $this->statuses;
|
$this->statuses = $this->statuses === null ? InvoiceStatus::defaultVisibleValues()->toArray() : $this->statuses;
|
||||||
return $this;
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @inheritdoc
|
||||||
|
*/
|
||||||
|
public function getQuery(): Builder
|
||||||
|
{
|
||||||
|
$this->search = $this->search ?: '';
|
||||||
|
|
||||||
|
return Invoice::search($this->search, function ($engine, string $query, array $options) {
|
||||||
|
if (empty($this->statuses)) {
|
||||||
|
$filter = 'status = "asa6aeruuni4BahC7Wei6ahm1"';
|
||||||
|
} else {
|
||||||
|
$filter = collect($this->statuses)->map(fn (string $status) => "status = \"$status\"")->join(' OR ');
|
||||||
|
}
|
||||||
|
return $engine->search($query, [...$options, 'filter' => $filter]);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,25 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Lib;
|
||||||
|
|
||||||
|
use Spatie\LaravelData\Data;
|
||||||
|
|
||||||
|
class Sorting extends Data
|
||||||
|
{
|
||||||
|
public static function by(string $by): self
|
||||||
|
{
|
||||||
|
return static::factory()->withoutMagicalCreation()->from(['by' => $by]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function __construct(public string $by, public bool $direction = false)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<int, string>
|
||||||
|
*/
|
||||||
|
public function toMeilisearch(): array
|
||||||
|
{
|
||||||
|
return [$this->by . ':' . ($this->direction ? 'desc' : 'asc')];
|
||||||
|
}
|
||||||
|
}
|
|
@ -6,3 +6,5 @@ echo "create database scoutrobot;" | sudo mysql
|
||||||
ssh -l stammsilva zoomyboy.de "cd /usr/share/webapps/nami_silva && docker compose exec db mysqldump -udb -p$SCOUTROBOT_DB_PASSWORD db" > db.tmp
|
ssh -l stammsilva zoomyboy.de "cd /usr/share/webapps/nami_silva && docker compose exec db mysqldump -udb -p$SCOUTROBOT_DB_PASSWORD db" > db.tmp
|
||||||
sudo mysql scoutrobot < db.tmp
|
sudo mysql scoutrobot < db.tmp
|
||||||
rm db.tmp
|
rm db.tmp
|
||||||
|
|
||||||
|
echo 'app(\App\Form\FormSettings::class)->fill(["registerUrl" => "http://stammsilva.test/anmeldung/{slug}/register", "clearCacheUrl" => "http://stammsilva.test/adrema/clear-cache"])->save();' | php artisan tinker
|
||||||
|
|
20
bin/run
20
bin/run
|
@ -1,5 +1,17 @@
|
||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
|
|
||||||
|
set -e
|
||||||
|
|
||||||
|
FILESHARE=false
|
||||||
|
WEB=false
|
||||||
|
|
||||||
|
while getopts f opt; do
|
||||||
|
case $opt in
|
||||||
|
f) FILESHARE=true ;;
|
||||||
|
w) WEB=true ;;
|
||||||
|
esac
|
||||||
|
done
|
||||||
|
|
||||||
tmux new-session -d -s test
|
tmux new-session -d -s test
|
||||||
tmux send-keys -t test "a serve" Enter
|
tmux send-keys -t test "a serve" Enter
|
||||||
|
|
||||||
|
@ -12,11 +24,11 @@ tmux send-keys -t test "SS start docker && duu socketi" Enter
|
||||||
tmux new-window -t test
|
tmux new-window -t test
|
||||||
tmux send-keys -t test "nrh" Enter
|
tmux send-keys -t test "nrh" Enter
|
||||||
|
|
||||||
tmux new-window -t test
|
$WEB && tmux new-window -t test
|
||||||
tmux send-keys -t test "ggwdk && cd plugins/silva/adrema/assets/vendor/adrema-form && nrd" Enter
|
$WEB && tmux send-keys -t test "ggwdk && cd plugins/silva/adrema/assets/vendor/adrema-form && nrd" Enter
|
||||||
|
|
||||||
tmux new-window -t test
|
$FILESHARE && tmux new-window -t test
|
||||||
tmux send-keys -t test "cd tests/Fileshare && docker compose up" Enter
|
$FILESHARE && tmux send-keys -t test "cd tests/Fileshare && docker compose up" Enter
|
||||||
|
|
||||||
tmux attach-session -t test
|
tmux attach-session -t test
|
||||||
|
|
||||||
|
|
|
@ -69,6 +69,23 @@ return [
|
||||||
|
|
||||||
],
|
],
|
||||||
|
|
||||||
|
/*
|
||||||
|
|--------------------------------------------------------------------------
|
||||||
|
| Job Batching
|
||||||
|
|--------------------------------------------------------------------------
|
||||||
|
|
|
||||||
|
| The following options configure the database and table that store job
|
||||||
|
| batching information. These options can be updated to any database
|
||||||
|
| connection and table which has been defined by your application.
|
||||||
|
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
'batching' => [
|
||||||
|
'database' => env('DB_CONNECTION', 'mysql'),
|
||||||
|
'table' => 'job_batches',
|
||||||
|
],
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
| Failed Queue Jobs
|
| Failed Queue Jobs
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
<?php
|
<?php
|
||||||
|
|
||||||
use App\Form\Models\Form;
|
use App\Form\Models\Form;
|
||||||
|
use App\Invoice\Models\Invoice;
|
||||||
use App\Member\Member;
|
use App\Member\Member;
|
||||||
|
|
||||||
return [
|
return [
|
||||||
|
@ -153,7 +154,16 @@ return [
|
||||||
'pagination' => [
|
'pagination' => [
|
||||||
'maxTotalHits' => 1000000,
|
'maxTotalHits' => 1000000,
|
||||||
]
|
]
|
||||||
]
|
],
|
||||||
|
Invoice::class => [
|
||||||
|
'filterableAttributes' => ['to', 'usage', 'greeting', 'mail_email', 'status', 'id'],
|
||||||
|
'searchableAttributes' => ['to', 'usage', 'greeting', 'mail_email', 'status', 'id'],
|
||||||
|
'sortableAttributes' => [],
|
||||||
|
'displayedAttributes' => ['to', 'usage', 'greeting', 'mail_email', 'status', 'id'],
|
||||||
|
'pagination' => [
|
||||||
|
'maxTotalHits' => 1000000,
|
||||||
|
]
|
||||||
|
],
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
|
|
||||||
|
|
|
@ -7,6 +7,7 @@ use App\Form\Models\Form;
|
||||||
use App\Lib\Editor\Condition;
|
use App\Lib\Editor\Condition;
|
||||||
use Database\Factories\Traits\FakesMedia;
|
use Database\Factories\Traits\FakesMedia;
|
||||||
use Illuminate\Database\Eloquent\Factories\Factory;
|
use Illuminate\Database\Eloquent\Factories\Factory;
|
||||||
|
use Illuminate\Support\Carbon;
|
||||||
use Tests\Feature\Form\FormtemplateFieldRequest;
|
use Tests\Feature\Form\FormtemplateFieldRequest;
|
||||||
use Tests\Feature\Form\FormtemplateSectionRequest;
|
use Tests\Feature\Form\FormtemplateSectionRequest;
|
||||||
use Tests\RequestFactories\EditorRequestFactory;
|
use Tests\RequestFactories\EditorRequestFactory;
|
||||||
|
@ -48,8 +49,8 @@ class FormFactory extends Factory
|
||||||
'config' => ['sections' => []],
|
'config' => ['sections' => []],
|
||||||
'from' => $this->faker->dateTimeBetween('+1 week', '+4 weeks')->format('Y-m-d H:i:s'),
|
'from' => $this->faker->dateTimeBetween('+1 week', '+4 weeks')->format('Y-m-d H:i:s'),
|
||||||
'to' => $this->faker->dateTimeBetween('+1 week', '+4 weeks')->format('Y-m-d H:i:s'),
|
'to' => $this->faker->dateTimeBetween('+1 week', '+4 weeks')->format('Y-m-d H:i:s'),
|
||||||
'registration_from' => $this->faker->dateTimeBetween('-2 weeks', 'now')->format('Y-m-d H:i:s'),
|
'registration_from' => $this->faker->dateTimeBetween(Carbon::parse('-2 weeks'), now())->format('Y-m-d H:i:s'),
|
||||||
'registration_until' => $this->faker->dateTimeBetween('now', '+2 weeks')->format('Y-m-d H:i:s'),
|
'registration_until' => $this->faker->dateTimeBetween(now(), Carbon::parse('+2 weeks'))->format('Y-m-d H:i:s'),
|
||||||
'mail_top' => EditorRequestFactory::new()->toData(),
|
'mail_top' => EditorRequestFactory::new()->toData(),
|
||||||
'mail_bottom' => EditorRequestFactory::new()->toData(),
|
'mail_bottom' => EditorRequestFactory::new()->toData(),
|
||||||
'is_active' => true,
|
'is_active' => true,
|
||||||
|
|
|
@ -0,0 +1,31 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
use App\Form\Actions\UpdateParticipantSearchIndexAction;
|
||||||
|
use App\Form\Models\Form;
|
||||||
|
use App\Lib\Sorting;
|
||||||
|
use Illuminate\Database\Migrations\Migration;
|
||||||
|
|
||||||
|
return new class extends Migration
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Run the migrations.
|
||||||
|
*/
|
||||||
|
public function up(): void
|
||||||
|
{
|
||||||
|
foreach (Form::get() as $form) {
|
||||||
|
UpdateParticipantSearchIndexAction::run($form);
|
||||||
|
foreach ($form->participants as $participant) {
|
||||||
|
$participant->searchable();
|
||||||
|
}
|
||||||
|
$form->updateQuietly(['meta' => [...$form->meta, 'sorting' => Sorting::by('id')]]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reverse the migrations.
|
||||||
|
*/
|
||||||
|
public function down(): void
|
||||||
|
{
|
||||||
|
//
|
||||||
|
}
|
||||||
|
};
|
|
@ -0,0 +1,27 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
use App\Invoice\Models\Invoice;
|
||||||
|
use Illuminate\Database\Migrations\Migration;
|
||||||
|
use Laravel\Scout\Console\SyncIndexSettingsCommand;
|
||||||
|
|
||||||
|
return new class extends Migration
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Run the migrations.
|
||||||
|
*/
|
||||||
|
public function up(): void
|
||||||
|
{
|
||||||
|
Artisan::call(SyncIndexSettingsCommand::class);
|
||||||
|
foreach (Invoice::get() as $invoice) {
|
||||||
|
$invoice->searchable();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reverse the migrations.
|
||||||
|
*/
|
||||||
|
public function down(): void
|
||||||
|
{
|
||||||
|
//
|
||||||
|
}
|
||||||
|
};
|
|
@ -0,0 +1,30 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
use App\Form\Actions\UpdateParticipantSearchIndexAction;
|
||||||
|
use App\Form\Models\Form;
|
||||||
|
use App\Lib\Sorting;
|
||||||
|
use Illuminate\Database\Migrations\Migration;
|
||||||
|
|
||||||
|
return new class extends Migration
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Run the migrations.
|
||||||
|
*/
|
||||||
|
public function up(): void
|
||||||
|
{
|
||||||
|
foreach (Form::get() as $form) {
|
||||||
|
UpdateParticipantSearchIndexAction::run($form);
|
||||||
|
foreach ($form->participants as $participant) {
|
||||||
|
$participant->searchable();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reverse the migrations.
|
||||||
|
*/
|
||||||
|
public function down(): void
|
||||||
|
{
|
||||||
|
//
|
||||||
|
}
|
||||||
|
};
|
|
@ -1 +1 @@
|
||||||
Subproject commit 77cfe9bbc7a11ee6494b205458ac5c75ff5c166d
|
Subproject commit 424ca30932610dcef496640e4097644ed404f13b
|
|
@ -1 +1 @@
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" xml:space="preserve" width="687.866" height="704.496" style="shape-rendering:geometricPrecision;text-rendering:geometricPrecision;image-rendering:optimizeQuality;fill-rule:evenodd;clip-rule:evenodd" viewBox="0 0 182.004 186.404"><defs><style>.fil0{fill:#000}</style></defs><g id="Ebene_x0020_1"><path class="fil0" d="m72.601.2-.6.2-.8.4-1 .6-1 1-.6 1-.6 1.6 1.2 18-39.4.4L10.6 43.2l61.001 4.601 2.6 28-2.4-1.6a12.425 12.425 0 0 1-2.2-1.4c-.916-.523-2.05-1.05-2.8-1.8l-1.8-1.2-1.6-1-1.6-1-2-1.2-1.4-.8-1.6-1-4.2-2.4-1.8-1-3-1.4-2.8-1.4-1.6-.8-2.2-1-2.4-1-2.8-1-3.2-1-4.4-1-2.8-.4h-5.4l-1 .2-2 .4-1.6.4-1 .4-1.401.6-.6.4-1 .6-1.4 1-2 2c-.262.524-.672.745-1 1.2l-.6.8c-.1.138-.3.662-.4.8l-1.2 2-.8 1.6-.6 1.4-.8 2.6-.6 1.6-.4 1.8-.4 1.8-.4 2-.4 2.4-.4 3.4-.2 3.4v10.401l.2 3.4.4 3.8.4 2.6.2 1.6.4 2.2.6 3 .4 1.6.6 2.4.4 1.4.4 1.4.2.6.4 1.2c.292.585.353 1.203.6 1.8l.4 1.2.4 1 .2.601.4 1 .6 1.6.4.8.6 1.4.4.8.4.8.6 1.2.4.8.6 1.2.6 1.2 1 1.8.4.6.6 1 .4.6 1 1.6.8 1.2.8 1.2 1 1.4 1 1.4.8 1 1 1.2.6.8 1 1.2 3 3 1 .8 1 .6 1.2.6.8.4 1 .4 2.8.6H38l3.2-.6 1.2-.4 3-1.4 1.4-.8 1-.6 2.001-1.2 1.6-1.2 1.4-1 1.6-1.2 2.6-2.2 1-.8 2.4-2 2-1.8 2.2-2 2.2-1.8c.704-.703 1.89-1.955 2.2-2.2.1-.053.874-.82 1.4-1.2l1.2-1 .2-.2.2-.4v-.6l-.2-.2-5.8-.2-2.2-.4-1.8-.6-1-.6-1-.8-1-1-.8-1-1.2-2.2-1.2-3.4-.8-2.6-.6-3.2-.6-3.4-.6-2.6-.6-3.4-.6-3.2-.8-4v-2l.8-.8h2.6l2 .4 2.4.6c1.357.542 2.688.943 4 1.6l2.2 1 1.4.6.8.4c.807.58 2.086.885 2.8 1.6l2 1.2 2.2 1.6 2.4 2.4.8 1.6 1.4 13.6h23.401l1-10.4.6-2.6 1.2-2.2 2.4-2.4.8-.6c.69-.346 1.377-.933 2-1.4.81-.54 1.721-1.25 2.6-1.6 1.04-.52 1.991-1.096 3-1.6l1.8-.8c.757-.33 1.629-.718 2.4-1l1.8-.6 1-.4 2.4-.6 2-.4h2.601l.8.8v2c-.015.296-1.42 6.984-1.6 8l-.4 2.6-.4 2-.6 3.4-.4 1.8-.4 2-.6 2.2-.6 2-.8 1.8-.6 1.2-.6 1-.8 1-1 1-2 1.4-1.8.6-2.2.4-5.8.2-.2.2v.6l.2.4c.16.222 1.13.954 1.4 1.2l1 1 2 1.8 1 1 1.4 1.2 2.6 2.2 1.6 1.6 4.8 4 1.6 1.2 8.4 5.6s2.962 1.606 3.6 1.8l2 .6 2.4.4h5.6l2.8-.6c.245-.074 2.257-.918 2.4-1l2.2-1.6 3.6-3.6c1.182-1.18 2.016-2.615 3.2-3.8l1.4-2.2 3.4-5 2.4-4.2 1.4-2.6 1-2 1.2-2.8 1-2.4 1.4-3.8.801-2.4 1-3.4.8-3.6 1.2-5.6.2-1.6.4-2.6.2-2.2.4-5V85.201l-.2-3.4-.4-3.4-.4-2.4-.2-1.4-.4-1.8-.4-1.6-.2-.8-.4-1.2-.4-1.4c-.165-.484-.78-1.904-1-2.4l-1-2.2-1-1.8-1.2-1.8-2.2-2.4c-.22-.2-3.7-2.745-3.8-2.8l-2-.8-2-.6-3-.6h-5.4l-2.8.4-3.4.8-3 .8-3 1-2.397.978-2.404 1.021-3 1.4-4 2-4.8 2.6-1.4.8-7 4.2-1.8 1.2-3.2 2-4 2.6-3.2 2.2 2.6-28.8 61.6-4.6-19.2-19.6-40.019-.2 1.418-17.8-.6-1.8-.8-1.2-1.2-1.2-1.4-.8-.6-.2-1.4-.2h-33.6l-1.2.2z"/><path class="fil0" d="m51.201 175.604-.6 1v.6l.2.4.4.4.4.2.8.4.8.4 1.6.6 1 .4 2.4.6 1.8.2h1.6l1-.2.6-.4.8-.6 1.8-1.6c3.438-3.437 7.332-6.59 11-9.8l-.8 14v2.4l.2 1 .2.4.2.2.8.2h27.201l.8-.2.2-.2.2-.4.2-1v-2.4l-.8-14 12 10.6.8.8.8.6.6.4 1 .2h1.6l1.8-.2 1.601-.4 1.8-.6 1-.4.6-.2.8-.4.8-.4.4-.2.4-.4.2-.4v-.6c0 .3-29.401-43.8-29.4-43.8H80.001l-28.8 42.8z"/></g></svg>
|
<svg xmlns="http://www.w3.org/2000/svg" xml:space="preserve" width="687.866" height="704.496" style="shape-rendering:geometricPrecision;text-rendering:geometricPrecision;image-rendering:optimizeQuality;fill-rule:evenodd;clip-rule:evenodd" viewBox="0 0 182.004 186.404"><defs></defs><g id="Ebene_x0020_1"><path d="m72.601.2-.6.2-.8.4-1 .6-1 1-.6 1-.6 1.6 1.2 18-39.4.4L10.6 43.2l61.001 4.601 2.6 28-2.4-1.6a12.425 12.425 0 0 1-2.2-1.4c-.916-.523-2.05-1.05-2.8-1.8l-1.8-1.2-1.6-1-1.6-1-2-1.2-1.4-.8-1.6-1-4.2-2.4-1.8-1-3-1.4-2.8-1.4-1.6-.8-2.2-1-2.4-1-2.8-1-3.2-1-4.4-1-2.8-.4h-5.4l-1 .2-2 .4-1.6.4-1 .4-1.401.6-.6.4-1 .6-1.4 1-2 2c-.262.524-.672.745-1 1.2l-.6.8c-.1.138-.3.662-.4.8l-1.2 2-.8 1.6-.6 1.4-.8 2.6-.6 1.6-.4 1.8-.4 1.8-.4 2-.4 2.4-.4 3.4-.2 3.4v10.401l.2 3.4.4 3.8.4 2.6.2 1.6.4 2.2.6 3 .4 1.6.6 2.4.4 1.4.4 1.4.2.6.4 1.2c.292.585.353 1.203.6 1.8l.4 1.2.4 1 .2.601.4 1 .6 1.6.4.8.6 1.4.4.8.4.8.6 1.2.4.8.6 1.2.6 1.2 1 1.8.4.6.6 1 .4.6 1 1.6.8 1.2.8 1.2 1 1.4 1 1.4.8 1 1 1.2.6.8 1 1.2 3 3 1 .8 1 .6 1.2.6.8.4 1 .4 2.8.6H38l3.2-.6 1.2-.4 3-1.4 1.4-.8 1-.6 2.001-1.2 1.6-1.2 1.4-1 1.6-1.2 2.6-2.2 1-.8 2.4-2 2-1.8 2.2-2 2.2-1.8c.704-.703 1.89-1.955 2.2-2.2.1-.053.874-.82 1.4-1.2l1.2-1 .2-.2.2-.4v-.6l-.2-.2-5.8-.2-2.2-.4-1.8-.6-1-.6-1-.8-1-1-.8-1-1.2-2.2-1.2-3.4-.8-2.6-.6-3.2-.6-3.4-.6-2.6-.6-3.4-.6-3.2-.8-4v-2l.8-.8h2.6l2 .4 2.4.6c1.357.542 2.688.943 4 1.6l2.2 1 1.4.6.8.4c.807.58 2.086.885 2.8 1.6l2 1.2 2.2 1.6 2.4 2.4.8 1.6 1.4 13.6h23.401l1-10.4.6-2.6 1.2-2.2 2.4-2.4.8-.6c.69-.346 1.377-.933 2-1.4.81-.54 1.721-1.25 2.6-1.6 1.04-.52 1.991-1.096 3-1.6l1.8-.8c.757-.33 1.629-.718 2.4-1l1.8-.6 1-.4 2.4-.6 2-.4h2.601l.8.8v2c-.015.296-1.42 6.984-1.6 8l-.4 2.6-.4 2-.6 3.4-.4 1.8-.4 2-.6 2.2-.6 2-.8 1.8-.6 1.2-.6 1-.8 1-1 1-2 1.4-1.8.6-2.2.4-5.8.2-.2.2v.6l.2.4c.16.222 1.13.954 1.4 1.2l1 1 2 1.8 1 1 1.4 1.2 2.6 2.2 1.6 1.6 4.8 4 1.6 1.2 8.4 5.6s2.962 1.606 3.6 1.8l2 .6 2.4.4h5.6l2.8-.6c.245-.074 2.257-.918 2.4-1l2.2-1.6 3.6-3.6c1.182-1.18 2.016-2.615 3.2-3.8l1.4-2.2 3.4-5 2.4-4.2 1.4-2.6 1-2 1.2-2.8 1-2.4 1.4-3.8.801-2.4 1-3.4.8-3.6 1.2-5.6.2-1.6.4-2.6.2-2.2.4-5V85.201l-.2-3.4-.4-3.4-.4-2.4-.2-1.4-.4-1.8-.4-1.6-.2-.8-.4-1.2-.4-1.4c-.165-.484-.78-1.904-1-2.4l-1-2.2-1-1.8-1.2-1.8-2.2-2.4c-.22-.2-3.7-2.745-3.8-2.8l-2-.8-2-.6-3-.6h-5.4l-2.8.4-3.4.8-3 .8-3 1-2.397.978-2.404 1.021-3 1.4-4 2-4.8 2.6-1.4.8-7 4.2-1.8 1.2-3.2 2-4 2.6-3.2 2.2 2.6-28.8 61.6-4.6-19.2-19.6-40.019-.2 1.418-17.8-.6-1.8-.8-1.2-1.2-1.2-1.4-.8-.6-.2-1.4-.2h-33.6l-1.2.2z"/><path d="m51.201 175.604-.6 1v.6l.2.4.4.4.4.2.8.4.8.4 1.6.6 1 .4 2.4.6 1.8.2h1.6l1-.2.6-.4.8-.6 1.8-1.6c3.438-3.437 7.332-6.59 11-9.8l-.8 14v2.4l.2 1 .2.4.2.2.8.2h27.201l.8-.2.2-.2.2-.4.2-1v-2.4l-.8-14 12 10.6.8.8.8.6.6.4 1 .2h1.6l1.8-.2 1.601-.4 1.8-.6 1-.4.6-.2.8-.4.8-.4.4-.2.4-.4.2-.4v-.6c0 .3-29.401-43.8-29.4-43.8H80.001l-28.8 42.8z"/></g></svg>
|
||||||
|
|
Before Width: | Height: | Size: 2.8 KiB After Width: | Height: | Size: 2.8 KiB |
|
@ -0,0 +1,34 @@
|
||||||
|
<template>
|
||||||
|
<th @click="$emit('update:model-value')">
|
||||||
|
<div class="flex justify-between items-center">
|
||||||
|
<span v-text="label"></span>
|
||||||
|
<ui-sprite v-if="value.by === column && value.direction === false" src="chevron" class="w-3 h-3 text-primaryfg ml-2">ASC</ui-sprite>
|
||||||
|
<ui-sprite v-if="value.by === column && value.direction === true" src="chevron" class="rotate-180 w-3 h-3 text-primaryfg ml-2">DESC</ui-sprite>
|
||||||
|
</div>
|
||||||
|
</th>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup>
|
||||||
|
defineEmits(['update:model-value']);
|
||||||
|
|
||||||
|
defineProps({
|
||||||
|
column: {
|
||||||
|
type: String,
|
||||||
|
default: () => '',
|
||||||
|
},
|
||||||
|
label: {
|
||||||
|
type: String,
|
||||||
|
default: () => '',
|
||||||
|
},
|
||||||
|
value: {
|
||||||
|
type: Object,
|
||||||
|
default: () => {
|
||||||
|
return {by: '', direction: false};
|
||||||
|
},
|
||||||
|
},
|
||||||
|
sortable: {
|
||||||
|
type: Boolean,
|
||||||
|
default: false,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
|
@ -28,9 +28,9 @@ export function useApiIndex(firstUrl, siteName = null) {
|
||||||
inner.meta.value = response.meta;
|
inner.meta.value = response.meta;
|
||||||
}
|
}
|
||||||
|
|
||||||
async function reloadPage(page) {
|
async function reloadPage(page, p = {}) {
|
||||||
inner.meta.value.current_page = page;
|
inner.meta.value.current_page = page;
|
||||||
await reload(false);
|
await reload(false, p);
|
||||||
}
|
}
|
||||||
|
|
||||||
function create() {
|
function create() {
|
||||||
|
|
|
@ -56,6 +56,8 @@
|
||||||
></f-singlefile>
|
></f-singlefile>
|
||||||
<f-text id="from" v-model="single.from" type="date" label="Von" required></f-text>
|
<f-text id="from" v-model="single.from" type="date" label="Von" required></f-text>
|
||||||
<f-text id="to" v-model="single.to" type="date" label="Bis" required></f-text>
|
<f-text id="to" v-model="single.to" type="date" label="Bis" required></f-text>
|
||||||
|
<f-text id="registration_from" v-model="single.registration_from" type="datetime-local" label="Registrierung von" required></f-text>
|
||||||
|
<f-text id="registration_until" v-model="single.registration_until" type="datetime-local" label="Registrierung bis" required></f-text>
|
||||||
<f-textarea
|
<f-textarea
|
||||||
id="excerpt"
|
id="excerpt"
|
||||||
v-model="single.excerpt"
|
v-model="single.excerpt"
|
||||||
|
@ -64,13 +66,15 @@
|
||||||
:rows="5"
|
:rows="5"
|
||||||
required
|
required
|
||||||
></f-textarea>
|
></f-textarea>
|
||||||
<f-editor id="description" v-model="single.description" name="description" label="Beschreibung" :rows="10" required></f-editor>
|
|
||||||
</div>
|
</div>
|
||||||
<div v-if="active === 1">
|
<div v-if="active === 1">
|
||||||
|
<f-editor id="description" v-model="single.description" name="description" label="Beschreibung" :rows="10" required></f-editor>
|
||||||
|
</div>
|
||||||
|
<div v-if="active === 2">
|
||||||
<ui-note class="mt-2"> Sobald sich der erste Teilnehmer für die Veranstaltung angemeldet hat, kann dieses Formular nicht mehr geändert werden. </ui-note>
|
<ui-note class="mt-2"> Sobald sich der erste Teilnehmer für die Veranstaltung angemeldet hat, kann dieses Formular nicht mehr geändert werden. </ui-note>
|
||||||
<form-builder v-model="single.config" :meta="meta"></form-builder>
|
<form-builder v-model="single.config" :meta="meta"></form-builder>
|
||||||
</div>
|
</div>
|
||||||
<div v-show="active === 2" class="grid grid-cols-[1fr_300px] gap-3">
|
<div v-show="active === 3" class="grid grid-cols-[1fr_300px] gap-3">
|
||||||
<ui-note class="mt-2 col-span-full">
|
<ui-note class="mt-2 col-span-full">
|
||||||
Hier kannst du die E-Mail anpassen, die nach der Anmeldung an den Teilnehmer verschickt wird.<br />
|
Hier kannst du die E-Mail anpassen, die nach der Anmeldung an den Teilnehmer verschickt wird.<br />
|
||||||
Es gibt dafür einen ersten E-Mail-Teil und einen zweiten E-Mail-Teil. Dazwischen werden die Daten des Teilnehmers aufgelistet.<br />
|
Es gibt dafür einen ersten E-Mail-Teil und einen zweiten E-Mail-Teil. Dazwischen werden die Daten des Teilnehmers aufgelistet.<br />
|
||||||
|
@ -108,14 +112,14 @@
|
||||||
</template>
|
</template>
|
||||||
</f-multiplefiles>
|
</f-multiplefiles>
|
||||||
</div>
|
</div>
|
||||||
<div v-if="active === 3">
|
<div v-if="active === 4">
|
||||||
<div class="grid gap-3">
|
<div class="grid gap-3">
|
||||||
<ui-remote-resource id="export" v-model="single.export.root" label="Haupt-Ordner"></ui-remote-resource>
|
<ui-remote-resource id="export" v-model="single.export.root" label="Haupt-Ordner"></ui-remote-resource>
|
||||||
<f-select id="group_by" v-model="single.export.group_by" :options="allFields" label="Gruppieren nach" name="group_by"></f-select>
|
<f-select id="group_by" v-model="single.export.group_by" :options="allFields" label="Gruppieren nach" name="group_by"></f-select>
|
||||||
<f-select id="to_group_field" v-model="single.export.to_group_field" :options="allFields" label="Nach Gruppe schreiben" name="to_group_field"></f-select>
|
<f-select id="to_group_field" v-model="single.export.to_group_field" :options="allFields" label="Nach Gruppe schreiben" name="to_group_field"></f-select>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div v-show="active === 4" class="grid grid-cols-2 gap-3">
|
<div v-show="active === 5" class="grid grid-cols-2 gap-3">
|
||||||
<f-switch id="needs_prevention" v-model="single.needs_prevention" name="needs_prevention" label="Prävention"></f-switch>
|
<f-switch id="needs_prevention" v-model="single.needs_prevention" name="needs_prevention" label="Prävention"></f-switch>
|
||||||
<f-editor
|
<f-editor
|
||||||
id="prevention_text"
|
id="prevention_text"
|
||||||
|
@ -207,7 +211,7 @@ const fileSettingPopup = ref(null);
|
||||||
|
|
||||||
const active = ref(0);
|
const active = ref(0);
|
||||||
const activeMailTab = ref(0);
|
const activeMailTab = ref(0);
|
||||||
const tabs = [{ title: 'Allgemeines' }, { title: 'Formular' }, { title: 'Bestätigungs-E-Mail' }, { title: 'Export' }, { title: 'Prävention' }];
|
const tabs = [{ title: 'Allgemeines' }, { title: 'Beschreibung' }, { title: 'Formular' }, { title: 'Bestätigungs-E-Mail' }, { title: 'Export' }, { title: 'Prävention' }];
|
||||||
const mailTabs = [{ title: 'vor Daten' }, { title: 'nach Daten' }];
|
const mailTabs = [{ title: 'vor Daten' }, { title: 'nach Daten' }];
|
||||||
|
|
||||||
const allFields = computed(() => {
|
const allFields = computed(() => {
|
||||||
|
|
|
@ -23,6 +23,7 @@
|
||||||
</ui-popup>
|
</ui-popup>
|
||||||
<page-filter breakpoint="lg">
|
<page-filter breakpoint="lg">
|
||||||
<template #buttons>
|
<template #buttons>
|
||||||
|
<f-text id="search" v-model="innerFilter.search" name="search" label="Suchen" size="sm"></f-text>
|
||||||
<ui-icon-button icon="plus" @click="editing = {participant: null, preview: JSON.stringify(meta.form_config)}">Hinzufügen</ui-icon-button>
|
<ui-icon-button icon="plus" @click="editing = {participant: null, preview: JSON.stringify(meta.form_config)}">Hinzufügen</ui-icon-button>
|
||||||
<f-switch v-if="meta.has_nami_field" id="group_participants" v-model="groupParticipants" label="Gruppieren" size="sm" name="group_participants"></f-switch>
|
<f-switch v-if="meta.has_nami_field" id="group_participants" v-model="groupParticipants" label="Gruppieren" size="sm" name="group_participants"></f-switch>
|
||||||
<f-multipleselect id="active_columns" v-model="activeColumnsConfig" :options="meta.columns" label="Aktive Spalten" size="sm"></f-multipleselect>
|
<f-multipleselect id="active_columns" v-model="activeColumnsConfig" :options="meta.columns" label="Aktive Spalten" size="sm"></f-multipleselect>
|
||||||
|
@ -68,7 +69,15 @@
|
||||||
</page-filter>
|
</page-filter>
|
||||||
<table cellspacing="0" cellpadding="0" border="0" class="custom-table custom-table-sm">
|
<table cellspacing="0" cellpadding="0" border="0" class="custom-table custom-table-sm">
|
||||||
<thead>
|
<thead>
|
||||||
<th v-for="column in activeColumns" :key="column.id" v-text="column.name"></th>
|
<ui-th
|
||||||
|
v-for="column in activeColumns"
|
||||||
|
:key="column.id"
|
||||||
|
:column="column.id"
|
||||||
|
:label="column.name"
|
||||||
|
:value="getSort"
|
||||||
|
sortable
|
||||||
|
@update:model-value="setSort(column.id, {filter: toFilterString(innerFilter)})"
|
||||||
|
></ui-th>
|
||||||
<th></th>
|
<th></th>
|
||||||
</thead>
|
</thead>
|
||||||
|
|
||||||
|
@ -117,7 +126,7 @@
|
||||||
</template>
|
</template>
|
||||||
</table>
|
</table>
|
||||||
<div class="px-6">
|
<div class="px-6">
|
||||||
<ui-pagination class="mt-4" :value="meta" @reload="reloadPage"></ui-pagination>
|
<ui-pagination class="mt-4" :value="meta" @reload="reloadPage($event, {filter: toFilterString(innerFilter)})"></ui-pagination>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
@ -173,6 +182,13 @@ var { meta, data, reload, reloadPage, axios, remove, toFilterString, url, update
|
||||||
|
|
||||||
const activeColumns = computed(() => meta.value.columns.filter((c) => meta.value.form_meta.active_columns.includes(c.id)));
|
const activeColumns = computed(() => meta.value.columns.filter((c) => meta.value.form_meta.active_columns.includes(c.id)));
|
||||||
|
|
||||||
|
const getSort = computed(() => innerFilter.value.sort);
|
||||||
|
|
||||||
|
async function setSort(column) {
|
||||||
|
innerFilter.value.sort = getSort.value.by === column ? {by: column, direction: !getSort.value.direction} : {by: column, direction: false};
|
||||||
|
sortingConfig.value = innerFilter.value.sort;
|
||||||
|
}
|
||||||
|
|
||||||
const activeColumnsConfig = computed({
|
const activeColumnsConfig = computed({
|
||||||
get: () => meta.value.form_meta.active_columns,
|
get: () => meta.value.form_meta.active_columns,
|
||||||
set: async (v) => {
|
set: async (v) => {
|
||||||
|
@ -185,6 +201,18 @@ const activeColumnsConfig = computed({
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const sortingConfig = computed({
|
||||||
|
get: () => meta.value.form_meta.sorting,
|
||||||
|
set: async (v) => {
|
||||||
|
const response = await axios.patch(meta.value.links.update_form_meta, {
|
||||||
|
...meta.value.form_meta,
|
||||||
|
sorting: v,
|
||||||
|
});
|
||||||
|
|
||||||
|
meta.value.form_meta = response.data;
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
async function handleDelete() {
|
async function handleDelete() {
|
||||||
await remove(deleting.value);
|
await remove(deleting.value);
|
||||||
deleting.value = null;
|
deleting.value = null;
|
||||||
|
|
|
@ -74,6 +74,7 @@
|
||||||
</ui-popup>
|
</ui-popup>
|
||||||
<page-filter breakpoint="xl" :filterable="false">
|
<page-filter breakpoint="xl" :filterable="false">
|
||||||
<template #buttons>
|
<template #buttons>
|
||||||
|
<f-text id="search" :model-value="getFilter('search')" label="Suchen …" size="sm" @update:model-value="setFilter('search', $event)"></f-text>
|
||||||
<f-multipleselect
|
<f-multipleselect
|
||||||
id="statuses"
|
id="statuses"
|
||||||
:options="meta.statuses"
|
:options="meta.statuses"
|
||||||
|
@ -114,6 +115,7 @@
|
||||||
<div class="flex space-x-2">
|
<div class="flex space-x-2">
|
||||||
<ui-action-button tooltip="Anschauen" :href="invoice.links.pdf" class="btn-info" icon="eye" blank></ui-action-button>
|
<ui-action-button tooltip="Anschauen" :href="invoice.links.pdf" class="btn-info" icon="eye" blank></ui-action-button>
|
||||||
<ui-action-button tooltip="Erinnerung anschauen" :href="invoice.links.rememberpdf" class="btn-info" icon="document" blank></ui-action-button>
|
<ui-action-button tooltip="Erinnerung anschauen" :href="invoice.links.rememberpdf" class="btn-info" icon="document" blank></ui-action-button>
|
||||||
|
<ui-action-button tooltip="Als Bezahlt markieren" class="btn-warning" icon="money" blank @click.prevent="markAsPaid(invoice)"></ui-action-button>
|
||||||
<ui-action-button :data-cy="`edit-button-${invoice.id}`" tooltip="Bearbeiten" class="btn-warning" icon="pencil" @click.prevent="edit(invoice)"></ui-action-button>
|
<ui-action-button :data-cy="`edit-button-${invoice.id}`" tooltip="Bearbeiten" class="btn-warning" icon="pencil" @click.prevent="edit(invoice)"></ui-action-button>
|
||||||
<ui-action-button tooltip="Löschen" class="btn-danger" icon="trash" @click.prevent="deleting = invoice"></ui-action-button>
|
<ui-action-button tooltip="Löschen" class="btn-danger" icon="trash" @click.prevent="deleting = invoice"></ui-action-button>
|
||||||
</div>
|
</div>
|
||||||
|
@ -143,4 +145,9 @@ async function sendMassstore() {
|
||||||
await axios.post(meta.value.links['mass-store'], massstore.value);
|
await axios.post(meta.value.links['mass-store'], massstore.value);
|
||||||
massstore.value = null;
|
massstore.value = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function markAsPaid(invoice) {
|
||||||
|
await axios.patch(invoice.links.update, {...invoice, status: 'Rechnung beglichen'});
|
||||||
|
await reloadPage();
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
|
@ -18,15 +18,15 @@
|
||||||
\node[anchor=base west, text width=205.3mm] at (52.9mm,31.0mm) {\bfseries{\large{<<<!!$fromName!!>>>}}}; %Feld: Jugendverband/-Gruppe
|
\node[anchor=base west, text width=205.3mm] at (52.9mm,31.0mm) {\bfseries{\large{<<<!!$fromName!!>>>}}}; %Feld: Jugendverband/-Gruppe
|
||||||
\node[anchor=base west, text width=215.9mm] at (41.3mm,38.8mm) {\bfseries{\large{<<<!!$eventName!!>>>}}}; %Feld: Art der Maßnahme
|
\node[anchor=base west, text width=215.9mm] at (41.3mm,38.8mm) {\bfseries{\large{<<<!!$eventName!!>>>}}}; %Feld: Art der Maßnahme
|
||||||
\node[anchor=base west, text width=104.8mm] at (17.5mm,47.0mm) {\bfseries{\large{<<<!!$zipLocation!!>>>, <<<!!$countryName!!>>>}}};
|
\node[anchor=base west, text width=104.8mm] at (17.5mm,47.0mm) {\bfseries{\large{<<<!!$zipLocation!!>>>, <<<!!$countryName!!>>>}}};
|
||||||
\node[anchor=base west, text width=41.3mm, align=center] at (170.7mm,47.0mm) {\bfseries{\large{<<<!!$dateFromHuman()!!>>>}}};
|
\node[anchor=base west, text width=41.3mm, align=center] at (170.7mm,47.0mm) {\bfseries{\large{<<<!!$niceDateFrom()!!>>>}}};
|
||||||
\node[anchor=base west, text width=38.4mm, align=center] at (219.9mm,47.0mm) {\bfseries{\large{<<<!!$dateUntilHuman()!!>>>}}};
|
\node[anchor=base west, text width=38.4mm, align=center] at (219.9mm,47.0mm) {\bfseries{\large{<<<!!$niceDateUntil()!!>>>}}};
|
||||||
|
|
||||||
@foreach($chunk as $j => $member)
|
@foreach($chunk as $j => $member)
|
||||||
\node[anchor=base, text width=4mm, align=center] at ($(8.0mm, 69.0mm + 8.05mm * <<<$j%15>>>)$) {<<<$memberShort($member)>>>};
|
\node[anchor=base, text width=4mm, align=center] at ($(8.0mm, 69.0mm + 8.05mm * <<<$j%15>>>)$) {<<<$member->isLeader ? 'L' : ''>>>};
|
||||||
\node[anchor=base, text width=6mm, align=center] at ($(13.0mm, 69.0mm + 8.05mm * <<<$j%15>>>)$) {<<<$j+1>>>};
|
\node[anchor=base, text width=6mm, align=center] at ($(13.0mm, 69.0mm + 8.05mm * <<<$j%15>>>)$) {<<<$j+1>>>};
|
||||||
\node[anchor=base, text width=67.5mm, align=center] at ($(50.65mm, 69.0mm + 8.05mm * <<<$j%15>>>)$) {<<<$memberName($member)>>>};
|
\node[anchor=base, text width=67.5mm, align=center] at ($(50.65mm, 69.0mm + 8.05mm * <<<$j%15>>>)$) {<<<$member->separatedName()>>>};
|
||||||
\node[anchor=base, text width=14.6mm, align=center] at ($(92.2mm, 69.0mm + 8.05mm * <<<$j%15>>>)$) {<<<$memberAge($member)>>>};
|
\node[anchor=base, text width=14.6mm, align=center] at ($(92.2mm, 69.0mm + 8.05mm * <<<$j%15>>>)$) {<<<$member->age()>>>};
|
||||||
\node[anchor=base, text width=79.4mm, align=center] at ($(139.7mm, 69.0mm + 8.05mm * <<<$j%15>>>)$) {<<<$memberAddress($member)>>>};
|
\node[anchor=base, text width=79.4mm, align=center] at ($(139.7mm, 69.0mm + 8.05mm * <<<$j%15>>>)$) {<<<$member->fullAddress()>>>};
|
||||||
@endforeach
|
@endforeach
|
||||||
|
|
||||||
\node[anchor=base, text width=23.0mm, align=center] at (278.2mm,196.1mm) {Seite <<<!!$i + 1!!>>> von <<<!!$pages!!>>>};
|
\node[anchor=base, text width=23.0mm, align=center] at (278.2mm,196.1mm) {Seite <<<!!$i + 1!!>>> von <<<!!$pages!!>>>};
|
||||||
|
|
|
@ -23,11 +23,11 @@
|
||||||
|
|
||||||
@foreach($chunk as $i => $member)
|
@foreach($chunk as $i => $member)
|
||||||
\node[anchor=base, text width=7.75mm, align=center] at ($(16.35mm, 76.6mm + 7mm * <<<$i % 17>>>)$) {<<<$i+1>>>};
|
\node[anchor=base, text width=7.75mm, align=center] at ($(16.35mm, 76.6mm + 7mm * <<<$i % 17>>>)$) {<<<$i+1>>>};
|
||||||
\node[anchor=base, text width=18mm, align=center] at ($(32.55mm, 76.6mm + 7mm * <<<$i%17>>>)$) {<<<$memberShort($member)>>>};
|
\node[anchor=base, text width=18mm, align=center] at ($(32.55mm, 76.6mm + 7mm * <<<$i%17>>>)$) {<<<$member->isLeader ? 'L' : ''>>>};
|
||||||
\node[anchor=base, text width=70mm, align=center] at ($(80.25mm, 76.6mm + 7mm * <<<$i%17>>>)$) {<<<$memberName($member)>>>};
|
\node[anchor=base, text width=70mm, align=center] at ($(80.25mm, 76.6mm + 7mm * <<<$i%17>>>)$) {<<<$member->separatedName()>>>};
|
||||||
\node[anchor=base, text width=118mm, align=center] at ($(178.25mm, 76.6mm + 7mm * <<<$i%17>>>)$) {<<<$memberAddress($member)>>>};
|
\node[anchor=base, text width=118mm, align=center] at ($(178.25mm, 76.6mm + 7mm * <<<$i%17>>>)$) {<<<$member->fullAddress()>>>};
|
||||||
\node[anchor=base, text width=16mm, align=center] at ($(249.50mm, 76.6mm + 7mm * <<<$i%17>>>)$) {<<<$memberGender($member)>>>};
|
\node[anchor=base, text width=16mm, align=center] at ($(249.50mm, 76.6mm + 7mm * <<<$i%17>>>)$) {<<<$member->genderLetter()>>>};
|
||||||
\node[anchor=base, text width=16mm, align=center] at ($(269.50mm, 76.6mm + 7mm * <<<$i%17>>>)$) {<<<$memberAge($member)>>>};
|
\node[anchor=base, text width=16mm, align=center] at ($(269.50mm, 76.6mm + 7mm * <<<$i%17>>>)$) {<<<$member->age()>>>};
|
||||||
@endforeach
|
@endforeach
|
||||||
|
|
||||||
\end{tikzpicture}
|
\end{tikzpicture}
|
||||||
|
|
|
@ -0,0 +1,39 @@
|
||||||
|
\documentclass[a4paper,landscape]{article}
|
||||||
|
|
||||||
|
\usepackage[landscape,top=0cm,left=0cm,bottom=0cm,right=0cm]{geometry}
|
||||||
|
\usepackage{tikz}
|
||||||
|
\usepackage{background}
|
||||||
|
\usepackage{blindtext}
|
||||||
|
\usetikzlibrary{matrix, shapes.misc, calc}
|
||||||
|
|
||||||
|
\pagestyle{empty}
|
||||||
|
\setlength{\parindent}{0cm}
|
||||||
|
\backgroundsetup{scale = 1, angle = 0, opacity = 1, color=black, contents = {\includegraphics[width = \paperwidth, height = \paperheight] {wuppertal.pdf}}}
|
||||||
|
|
||||||
|
\begin{document}
|
||||||
|
\noindent \sffamily
|
||||||
|
|
||||||
|
@foreach($members as $chunk)
|
||||||
|
\begin{tikzpicture}[remember picture,overlay,yscale=-1]
|
||||||
|
\node[anchor=base west] at (17mm,29.62mm) {\bfseries{\small{<<<$niceDateFrom>>>}}};
|
||||||
|
\node[anchor=base west] at (34mm,29.62mm) {\bfseries{\small{<<<$niceDateUntil>>>}}};
|
||||||
|
\node[] at (203mm,15.55mm) {\bfseries{\small{<<<$zipLocation>>>}}};
|
||||||
|
|
||||||
|
@foreach($chunk as $i => $member)
|
||||||
|
\node[anchor=center, text width=9.75mm, align=center] at ($(19.35mm, 59.7mm + 9.2mm * <<<$i%14>>>)$) {<<<$i+1>>>};
|
||||||
|
\node[anchor=center, text width=41.75mm, align=center] at ($(48.35mm, 59.7mm + 9.2mm * <<<$i%14>>>)$) {<<<$member->lastname>>>};
|
||||||
|
\node[anchor=center, text width=37.75mm, align=center] at ($(91.35mm, 59.7mm + 9.2mm * <<<$i%14>>>)$) {<<<$member->firstname>>>};
|
||||||
|
\node[anchor=center, text width=45.75mm, align=center] at ($(135.35mm, 59.7mm + 9.2mm * <<<$i%14>>>)$) {<<<$member->address>>>};
|
||||||
|
\node[anchor=center, text width=26.75mm, align=center] at ($(174.35mm, 59.7mm + 9.2mm * <<<$i%14>>>)$) {<<<$member->city()>>>};
|
||||||
|
\node[anchor=center, text width=19.75mm, align=center] at ($(199.35mm, 59.7mm + 9.2mm * <<<$i%14>>>)$) {<<<$member->birthdayHuman()>>>};
|
||||||
|
\node[anchor=center, text width=7.75mm, align=center] at ($(216.35mm, 59.7mm + 9.2mm * <<<$i%14>>>)$) {<<<$member->genderLetter()>>>};
|
||||||
|
\node[anchor=center, text width=7.75mm, align=center] at ($(276.35mm, 59.7mm + 9.2mm * <<<$i%14>>>)$) {<<<$member->isLeader ? 'GL' : 'T'>>>};
|
||||||
|
@endforeach
|
||||||
|
|
||||||
|
\end{tikzpicture}
|
||||||
|
|
||||||
|
\pagebreak
|
||||||
|
|
||||||
|
@endforeach
|
||||||
|
\end{document}
|
||||||
|
|
Binary file not shown.
|
@ -1,22 +0,0 @@
|
||||||
<?php
|
|
||||||
|
|
||||||
namespace Tests;
|
|
||||||
|
|
||||||
use Illuminate\Contracts\Console\Kernel;
|
|
||||||
|
|
||||||
trait CreatesApplication
|
|
||||||
{
|
|
||||||
/**
|
|
||||||
* Creates the application.
|
|
||||||
*
|
|
||||||
* @return \Illuminate\Foundation\Application
|
|
||||||
*/
|
|
||||||
public function createApplication()
|
|
||||||
{
|
|
||||||
$app = require __DIR__.'/../bootstrap/app.php';
|
|
||||||
|
|
||||||
$app->make(Kernel::class)->bootstrap();
|
|
||||||
|
|
||||||
return $app;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -3,14 +3,12 @@
|
||||||
namespace Tests\EndToEnd\Form;
|
namespace Tests\EndToEnd\Form;
|
||||||
|
|
||||||
use App\Form\Models\Form;
|
use App\Form\Models\Form;
|
||||||
use App\Membership\TestersBlock;
|
|
||||||
use App\Subactivity;
|
use App\Subactivity;
|
||||||
use Carbon\Carbon;
|
|
||||||
use Illuminate\Foundation\Testing\DatabaseTransactions;
|
use Illuminate\Foundation\Testing\DatabaseTransactions;
|
||||||
|
use Illuminate\Support\Carbon;
|
||||||
use Illuminate\Support\Facades\Storage;
|
use Illuminate\Support\Facades\Storage;
|
||||||
use Tests\Feature\Form\FormtemplateSectionRequest;
|
use Tests\Feature\Form\FormtemplateSectionRequest;
|
||||||
use Tests\RequestFactories\EditorRequestFactory;
|
use Tests\RequestFactories\EditorRequestFactory;
|
||||||
use Tests\TestCase;
|
|
||||||
|
|
||||||
uses(FormTestCase::class);
|
uses(FormTestCase::class);
|
||||||
uses(DatabaseTransactions::class);
|
uses(DatabaseTransactions::class);
|
||||||
|
@ -42,11 +40,24 @@ it('testItDisplaysForms', function () {
|
||||||
->assertJsonPath('data.0.dates', '05.05.2023 - 07.06.2023')
|
->assertJsonPath('data.0.dates', '05.05.2023 - 07.06.2023')
|
||||||
->assertJsonPath('data.0.from_human', '05.05.2023')
|
->assertJsonPath('data.0.from_human', '05.05.2023')
|
||||||
->assertJsonPath('data.0.to_human', '07.06.2023')
|
->assertJsonPath('data.0.to_human', '07.06.2023')
|
||||||
|
->assertJsonPath('data.0.can_register', true)
|
||||||
->assertJsonPath('meta.per_page', 15)
|
->assertJsonPath('meta.per_page', 15)
|
||||||
->assertJsonPath('meta.base_url', url(''))
|
->assertJsonPath('meta.base_url', url(''))
|
||||||
->assertJsonPath('meta.total', 1);
|
->assertJsonPath('meta.total', 1);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('displays registration not possible', function () {
|
||||||
|
Storage::fake('temp');
|
||||||
|
$this->loginNami()->withoutExceptionHandling();
|
||||||
|
Form::factory()
|
||||||
|
->registrationFrom(now()->addDay())
|
||||||
|
->withImage('headerImage', 'lala-2.jpg')
|
||||||
|
->create();
|
||||||
|
|
||||||
|
sleep(1);
|
||||||
|
$this->get('/api/form?perPage=15')->assertJsonPath('data.0.can_register', false);
|
||||||
|
});
|
||||||
|
|
||||||
it('testItDisplaysDefaultValueOfField', function () {
|
it('testItDisplaysDefaultValueOfField', function () {
|
||||||
Storage::fake('temp');
|
Storage::fake('temp');
|
||||||
$this->loginNami()->withoutExceptionHandling();
|
$this->loginNami()->withoutExceptionHandling();
|
||||||
|
|
|
@ -0,0 +1,309 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Tests\Feature\Form;
|
||||||
|
|
||||||
|
use App\Form\Fields\TextField;
|
||||||
|
use App\Form\Models\Form;
|
||||||
|
use App\Form\Models\Participant;
|
||||||
|
use App\Form\Scopes\ParticipantFilterScope;
|
||||||
|
use App\Group;
|
||||||
|
use App\Member\Member;
|
||||||
|
use Carbon\Carbon;
|
||||||
|
use Tests\EndToEndTestCase;
|
||||||
|
use Tests\Lib\CreatesFormFields;
|
||||||
|
|
||||||
|
uses(EndToEndTestCase::class);
|
||||||
|
uses(CreatesFormFields::class);
|
||||||
|
|
||||||
|
it('testItShowsParticipantsAndColumns', function () {
|
||||||
|
$this->login()->loginNami()->withoutExceptionHandling();
|
||||||
|
$group = Group::factory()->innerName('Stamm')->create();
|
||||||
|
$form = Form::factory()
|
||||||
|
->has(Participant::factory()->state(['member_id' => 55])->data(['vorname' => 'Max', 'select' => ['A', 'B'], 'stufe' => 'Pfadfinder', 'test1' => '', 'test2' => '', 'test3' => '', 'birthday' => '1991-04-20', 'bezirk' => $group->id]))
|
||||||
|
->fields([
|
||||||
|
$this->textField('vorname')->name('Vorname'),
|
||||||
|
$this->checkboxesField('select')->options(['A', 'B', 'C']),
|
||||||
|
$this->dropdownField('stufe')->options(['Wölfling', 'Jungpfadfinder', 'Pfadfinder']),
|
||||||
|
$this->textField('test1')->name('Test 1'),
|
||||||
|
$this->textField('test2')->name('Test 2'),
|
||||||
|
$this->textField('test3')->name('Test 3'),
|
||||||
|
$this->dateField('birthday')->name('Geburtsdatum'),
|
||||||
|
$this->groupField('bezirk')->name('bezirk'),
|
||||||
|
])
|
||||||
|
->create();
|
||||||
|
|
||||||
|
sleep(2);
|
||||||
|
$this->callFilter('form.participant.index', [], ['form' => $form])
|
||||||
|
->assertOk()
|
||||||
|
->assertJsonPath('data.0.id', $form->participants->first()->id)
|
||||||
|
->assertJsonPath('data.0.vorname', 'Max')
|
||||||
|
->assertJsonPath('data.0.vorname_display', 'Max')
|
||||||
|
->assertJsonPath('data.0.stufe', 'Pfadfinder')
|
||||||
|
->assertJsonPath('data.0.bezirk', $group->id)
|
||||||
|
->assertJsonPath('data.0.member_id', 55)
|
||||||
|
->assertJsonPath('data.0.bezirk_display', 'Stamm')
|
||||||
|
->assertJsonPath('data.0.birthday_display', '20.04.1991')
|
||||||
|
->assertJsonPath('data.0.birthday', '1991-04-20')
|
||||||
|
->assertJsonPath('data.0.select', ['A', 'B'])
|
||||||
|
->assertJsonPath('data.0.select_display', 'A, B')
|
||||||
|
->assertJsonPath('data.0.links.destroy', route('participant.destroy', ['participant' => $form->participants->first()]))
|
||||||
|
->assertJsonPath('data.0.links.assign', route('participant.assign', ['participant' => $form->participants->first()]))
|
||||||
|
->assertJsonPath('data.0.links.fields', route('participant.fields', ['participant' => $form->participants->first()]))
|
||||||
|
->assertJsonPath('data.0.links.update', route('participant.update', ['participant' => $form->participants->first()]))
|
||||||
|
->assertJsonPath('meta.columns.0.name', 'Vorname')
|
||||||
|
->assertJsonPath('meta.columns.0.base_type', class_basename(TextField::class))
|
||||||
|
->assertJsonPath('meta.columns.0.id', 'vorname')
|
||||||
|
->assertJsonPath('meta.columns.6.display_attribute', 'birthday_display')
|
||||||
|
->assertJsonPath('meta.columns.0.display_attribute', 'vorname_display')
|
||||||
|
->assertJsonPath('meta.form_meta.active_columns', ['vorname', 'select', 'stufe', 'test1'])
|
||||||
|
->assertJsonPath('meta.has_nami_field', false)
|
||||||
|
->assertJsonPath('meta.links.update_form_meta', route('form.update-meta', ['form' => $form]))
|
||||||
|
->assertJsonPath('meta.links.store_participant', route('form.participant.store', ['form' => $form]))
|
||||||
|
->assertJsonPath('meta.form_meta.sorting', ['by' => 'id', 'direction' => false])
|
||||||
|
->assertJsonPath('meta.form_config.sections.0.fields.0.key', 'vorname');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('testItShowsEmptyFilters', function () {
|
||||||
|
$this->login()->loginNami()->withoutExceptionHandling();
|
||||||
|
$form = Form::factory()->fields([$this->checkboxField('check')->name('Checked')])->create();
|
||||||
|
|
||||||
|
sleep(2);
|
||||||
|
$this->callFilter('form.participant.index', [], ['form' => $form])
|
||||||
|
->assertOk()
|
||||||
|
->assertJsonPath('meta.filters.0.name', 'Checked')
|
||||||
|
->assertJsonPath('meta.filters.0.key', 'check')
|
||||||
|
->assertJsonPath('meta.filters.0.base_type', 'CheckboxField')
|
||||||
|
->assertJsonPath('meta.default_filter_value', ParticipantFilterScope::$nan);
|
||||||
|
|
||||||
|
$this->callFilter('form.participant.index', ['data' => ['check' => null]], ['form' => $form])->assertHasJsonPath('meta.filter.data.check')->assertJsonPath('meta.filter.data.check', null);
|
||||||
|
$this->callFilter('form.participant.index', ['data' => ['check' => 'A']], ['form' => $form])->assertJsonPath('meta.filter.data.check', 'A');
|
||||||
|
$this->callFilter('form.participant.index', ['data' => []], ['form' => $form])->assertJsonPath('meta.filter.data.check', ParticipantFilterScope::$nan);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('sorts by active colums sorting by default', function (array $sorting, string $by, bool $direction) {
|
||||||
|
$this->login()->loginNami()->withoutExceptionHandling();
|
||||||
|
$form = Form::factory()->fields([
|
||||||
|
$this->checkboxField('check'),
|
||||||
|
$this->checkboxField('vorname'),
|
||||||
|
])->create();
|
||||||
|
$form->update(['meta' => ['active_columns' => [], 'sorting' => $sorting]]);
|
||||||
|
|
||||||
|
sleep(2);
|
||||||
|
$this->callFilter('form.participant.index', [], ['form' => $form])
|
||||||
|
->assertOk()
|
||||||
|
->assertJsonPath('meta.filter.sort.by', $by)
|
||||||
|
->assertJsonPath('meta.filter.sort.direction', $direction);
|
||||||
|
})->with([
|
||||||
|
[['by' => 'vorname', 'direction' => true], 'vorname', true],
|
||||||
|
[['by' => 'created_at', 'direction' => true], 'created_at', true],
|
||||||
|
]);
|
||||||
|
|
||||||
|
|
||||||
|
it('testItDisplaysHasNamiField', function () {
|
||||||
|
$this->login()->loginNami()->withoutExceptionHandling();
|
||||||
|
$form = Form::factory()->fields([$this->namiField('mitglieder')])->create();
|
||||||
|
sleep(2);
|
||||||
|
$this->callFilter('form.participant.index', [], ['form' => $form])->assertJsonPath('meta.has_nami_field', true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('testItFiltersParticipantsByCheckboxValue', function () {
|
||||||
|
$this->login()->loginNami()->withoutExceptionHandling();
|
||||||
|
$form = Form::factory()->fields([$this->checkboxField('check')])
|
||||||
|
->has(Participant::factory()->data(['check' => true])->count(1))
|
||||||
|
->has(Participant::factory()->data(['check' => false])->count(2))
|
||||||
|
->create();
|
||||||
|
|
||||||
|
sleep(2);
|
||||||
|
$this->callFilter('form.participant.index', ['data' => []], ['form' => $form])
|
||||||
|
->assertJsonCount(3, 'data');
|
||||||
|
$this->callFilter('form.participant.index', ['data' => ['check' => ParticipantFilterScope::$nan]], ['form' => $form])
|
||||||
|
->assertJsonCount(3, 'data');
|
||||||
|
$this->callFilter('form.participant.index', ['data' => ['check' => true]], ['form' => $form])
|
||||||
|
->assertJsonCount(1, 'data');
|
||||||
|
$this->callFilter('form.participant.index', ['data' => ['check' => false]], ['form' => $form])
|
||||||
|
->assertJsonCount(2, 'data');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('test it handles full text search', function (array $memberAttributes, string $search, bool $includes) {
|
||||||
|
$this->login()->loginNami()->withoutExceptionHandling();
|
||||||
|
$form = Form::factory()
|
||||||
|
->has(Participant::factory()->data(['vorname' => 'Max', 'select' => 'Pfadfinder', ...$memberAttributes]))
|
||||||
|
->fields([
|
||||||
|
$this->textField('vorname')->name('Vorname'),
|
||||||
|
$this->checkboxesField('select')->options(['Wölflinge', 'Pfadfinder']),
|
||||||
|
])
|
||||||
|
->create();
|
||||||
|
|
||||||
|
sleep(2);
|
||||||
|
$this->callFilter('form.participant.index', ['search' => $search], ['form' => $form])
|
||||||
|
->assertJsonCount($includes ? 1 : 0, 'data');
|
||||||
|
})->with([
|
||||||
|
[['vorname' => 'Max'], 'Max', true],
|
||||||
|
[['vorname' => 'Jane'], 'Max', false],
|
||||||
|
[['select' => 'Pfadfinder'], 'Pfadfinder', true],
|
||||||
|
[['select' => 'Pfadfinder'], 'Rov', false],
|
||||||
|
[['select' => 'Wölflinge'], 'Wölflinge', true],
|
||||||
|
[['select' => 'Wölflinge'], 'Wölf', true],
|
||||||
|
[['vorname' => 'Max', 'nachname' => 'Muster'], 'Max Muster', true],
|
||||||
|
[['vorname' => 'Max', 'nachname' => 'Muster'], 'Jane Doe', false],
|
||||||
|
]);
|
||||||
|
|
||||||
|
it('testItFiltersParticipantsByDropdownValue', function () {
|
||||||
|
$this->login()->loginNami()->withoutExceptionHandling();
|
||||||
|
$form = Form::factory()->fields([$this->dropdownField('drop')->options(['A', 'B'])])
|
||||||
|
->has(Participant::factory()->data(['drop' => null])->count(1))
|
||||||
|
->has(Participant::factory()->data(['drop' => 'A'])->count(2))
|
||||||
|
->has(Participant::factory()->data(['drop' => 'B'])->count(4))
|
||||||
|
->create();
|
||||||
|
|
||||||
|
sleep(2);
|
||||||
|
$this->callFilter('form.participant.index', ['data' => ['drop' => ParticipantFilterScope::$nan]], ['form' => $form])
|
||||||
|
->assertJsonCount(7, 'data');
|
||||||
|
$this->callFilter('form.participant.index', ['data' => ['drop' => null]], ['form' => $form])
|
||||||
|
->assertJsonCount(1, 'data');
|
||||||
|
$this->callFilter('form.participant.index', ['data' => ['drop' => 'A']], ['form' => $form])
|
||||||
|
->assertJsonCount(2, 'data');
|
||||||
|
$this->callFilter('form.participant.index', ['data' => ['drop' => 'B']], ['form' => $form])
|
||||||
|
->assertJsonCount(4, 'data');
|
||||||
|
$this->callFilter('form.participant.index', ['data' => ['drop' => 'Z*Z']], ['form' => $form])
|
||||||
|
->assertJsonCount(0, 'data');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('testItFiltersParticipantsByRadioValue', function () {
|
||||||
|
$this->login()->loginNami()->withoutExceptionHandling();
|
||||||
|
$form = Form::factory()->fields([$this->radioField('drop')->options(['A', 'B'])])
|
||||||
|
->has(Participant::factory()->data(['drop' => null])->count(1))
|
||||||
|
->has(Participant::factory()->data(['drop' => 'A'])->count(2))
|
||||||
|
->has(Participant::factory()->data(['drop' => 'B'])->count(4))
|
||||||
|
->create();
|
||||||
|
|
||||||
|
sleep(2);
|
||||||
|
$this->callFilter('form.participant.index', ['data' => ['drop' => ParticipantFilterScope::$nan]], ['form' => $form])
|
||||||
|
->assertJsonCount(7, 'data');
|
||||||
|
$this->callFilter('form.participant.index', ['data' => ['drop' => 'A']], ['form' => $form])
|
||||||
|
->assertJsonCount(2, 'data');
|
||||||
|
$this->callFilter('form.participant.index', ['data' => ['drop' => 'B']], ['form' => $form])
|
||||||
|
->assertJsonCount(4, 'data');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('testItPresentsNamiField', function () {
|
||||||
|
$this->login()->loginNami()->withoutExceptionHandling();
|
||||||
|
$form = Form::factory()
|
||||||
|
->has(Participant::factory()->data(['mitglieder' => [['id' => 393], ['id' => 394]]]))
|
||||||
|
->has(Participant::factory()->nr(393)->data(['mitglieder' => []]))
|
||||||
|
->has(Participant::factory()->nr(394)->data(['mitglieder' => []]))
|
||||||
|
->fields([
|
||||||
|
$this->namiField('mitglieder'),
|
||||||
|
])
|
||||||
|
->create();
|
||||||
|
|
||||||
|
sleep(2);
|
||||||
|
$this->callFilter('form.participant.index', [], ['form' => $form])
|
||||||
|
->assertJsonPath('data.0.mitglieder_display', '393, 394');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('testItShowsRegisteredAtColumnAndAttribute', function () {
|
||||||
|
Carbon::setTestNow(Carbon::parse('2023-03-05 06:00:00'));
|
||||||
|
$this->login()->loginNami()->withoutExceptionHandling();
|
||||||
|
$form = Form::factory()
|
||||||
|
->has(Participant::factory()->data(['vorname' => 'Max']))
|
||||||
|
->fields([
|
||||||
|
$this->textField('vorname')->name('Vorname'),
|
||||||
|
])
|
||||||
|
->create();
|
||||||
|
|
||||||
|
sleep(2);
|
||||||
|
$this->callFilter('form.participant.index', [], ['form' => $form])
|
||||||
|
->assertJsonPath('data.0.vorname', 'Max')
|
||||||
|
->assertJsonPath('data.0.vorname_display', 'Max')
|
||||||
|
->assertJsonPath('data.0.created_at', '2023-03-05 06:00:00')
|
||||||
|
->assertJsonPath('data.0.created_at_display', '05.03.2023')
|
||||||
|
->assertJsonPath('meta.columns.1.name', 'Registriert am')
|
||||||
|
->assertJsonPath('meta.columns.1.id', 'created_at')
|
||||||
|
->assertJsonPath('meta.columns.1.display_attribute', 'created_at_display');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('testItShowsOnlyParentParticipantsWhenFilterEnabled', function () {
|
||||||
|
$this->login()->loginNami()->withoutExceptionHandling();
|
||||||
|
$form = Form::factory()->create();
|
||||||
|
$participant = Participant::factory()
|
||||||
|
->has(Participant::factory()->for($form)->count(2), 'children')
|
||||||
|
->for($form)
|
||||||
|
->create();
|
||||||
|
|
||||||
|
sleep(2);
|
||||||
|
$this->callFilter('form.participant.index', [], ['form' => $form])->assertJsonCount(3, 'data');
|
||||||
|
$this->callFilter('form.participant.index', [], ['form' => $form, 'parent' => -1])->assertJsonCount(1, 'data');
|
||||||
|
$this->callFilter('form.participant.index', [], ['form' => $form, 'parent' => $participant->id])->assertJsonCount(2, 'data');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('testItShowsChildrenCount', function () {
|
||||||
|
$this->login()->loginNami()->withoutExceptionHandling();
|
||||||
|
$form = Form::factory()->create();
|
||||||
|
$participant = Participant::factory()
|
||||||
|
->has(Participant::factory()->for($form)->count(2), 'children')
|
||||||
|
->for($form)
|
||||||
|
->create();
|
||||||
|
Participant::factory()->for($form)->create();
|
||||||
|
|
||||||
|
sleep(2);
|
||||||
|
$this->callFilter('form.participant.index', [], ['form' => $form, 'parent' => -1])
|
||||||
|
->assertJsonPath('data.0.children_count', 2)
|
||||||
|
->assertJsonPath('data.1.children_count', 0)
|
||||||
|
->assertJsonPath('data.0.links.children', route('form.participant.index', ['form' => $form, 'parent' => $participant->id]))
|
||||||
|
->assertJsonPath('meta.current_page', 1);
|
||||||
|
$this->callFilter('form.participant.index', [], ['form' => $form, 'parent' => $participant->id])->assertJsonPath('data.0.children_count', 0);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('testItShowsPreventionState', function () {
|
||||||
|
$this->login()->loginNami()->withoutExceptionHandling();
|
||||||
|
$participant = Participant::factory()->data(['vorname' => 'Max'])
|
||||||
|
->for(Member::factory()->defaults()->state(['efz' => null]))
|
||||||
|
->for(Form::factory())
|
||||||
|
->create();
|
||||||
|
|
||||||
|
sleep(2);
|
||||||
|
$this->callFilter('form.participant.index', [], ['form' => $participant->form])
|
||||||
|
->assertJsonPath('data.0.prevention_items.0.letter', 'F')
|
||||||
|
->assertJsonPath('data.0.prevention_items.0.value', false)
|
||||||
|
->assertJsonPath('data.0.prevention_items.0.tooltip', 'erweitertes Führungszeugnis nicht vorhanden');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('test it orders participants by value', function (array $values, array $sorting, array $expected) {
|
||||||
|
list($key, $direction) = $sorting;
|
||||||
|
$this->login()->loginNami()->withoutExceptionHandling();
|
||||||
|
$form = Form::factory()
|
||||||
|
->fields([
|
||||||
|
$this->textField('vorname')->name('Vorname'),
|
||||||
|
$this->checkboxesField('select')->options(['Wölflinge', 'Pfadfinder']),
|
||||||
|
]);
|
||||||
|
foreach ($values as $value) {
|
||||||
|
$form = $form->has(Participant::factory()->data(['vorname' => 'Max', 'select' => 'Pfadfinder', $key => $value]));
|
||||||
|
}
|
||||||
|
$form = $form->create();
|
||||||
|
|
||||||
|
sleep(2);
|
||||||
|
$response = $this->callFilter('form.participant.index', ['sort' => ['by' => $key, 'direction' => $direction]], ['form' => $form]);
|
||||||
|
$response->assertJsonPath('meta.filter.sort.by', $key);
|
||||||
|
$response->assertJsonPath('meta.filter.sort.direction', $direction);
|
||||||
|
|
||||||
|
foreach ($expected as $index => $value) {
|
||||||
|
$response->assertJsonPath("data.{$index}.{$key}", $value);
|
||||||
|
}
|
||||||
|
})->with([
|
||||||
|
[
|
||||||
|
['Anna', 'Sarah', 'Ben'],
|
||||||
|
['vorname', false],
|
||||||
|
['Anna', 'Ben', 'Sarah'],
|
||||||
|
],
|
||||||
|
[
|
||||||
|
['Anna', 'Sarah', 'Ben'],
|
||||||
|
['vorname', true],
|
||||||
|
['Sarah', 'Ben', 'Anna'],
|
||||||
|
],
|
||||||
|
[
|
||||||
|
['Wölflinge', 'Pfadfinder'],
|
||||||
|
['select', false],
|
||||||
|
['Pfadfinder', 'Wölflinge'],
|
||||||
|
]
|
||||||
|
]);
|
|
@ -1,6 +1,6 @@
|
||||||
<?php
|
<?php
|
||||||
|
|
||||||
namespace Tests\Feature\Invoice;
|
namespace Tests\Feature\EndToEnd;
|
||||||
|
|
||||||
use App\Invoice\BillKind;
|
use App\Invoice\BillKind;
|
||||||
use App\Invoice\Enums\InvoiceStatus;
|
use App\Invoice\Enums\InvoiceStatus;
|
||||||
|
@ -9,8 +9,11 @@ use App\Invoice\Models\InvoicePosition;
|
||||||
use App\Member\Member;
|
use App\Member\Member;
|
||||||
use App\Payment\Subscription;
|
use App\Payment\Subscription;
|
||||||
use Illuminate\Foundation\Testing\DatabaseTransactions;
|
use Illuminate\Foundation\Testing\DatabaseTransactions;
|
||||||
|
use Tests\EndToEndTestCase;
|
||||||
|
use Tests\Feature\Invoice\ReceiverRequestFactory;
|
||||||
|
|
||||||
uses(DatabaseTransactions::class);
|
uses(DatabaseTransactions::class);
|
||||||
|
uses(EndToEndTestCase::class);
|
||||||
|
|
||||||
it('testItDisplaysInvoices', function () {
|
it('testItDisplaysInvoices', function () {
|
||||||
$this->login()->loginNami()->withoutExceptionHandling();
|
$this->login()->loginNami()->withoutExceptionHandling();
|
||||||
|
@ -25,6 +28,7 @@ it('testItDisplaysInvoices', function () {
|
||||||
->status(InvoiceStatus::SENT)
|
->status(InvoiceStatus::SENT)
|
||||||
->create(['usage' => 'Usa', 'mail_email' => 'a@b.de']);
|
->create(['usage' => 'Usa', 'mail_email' => 'a@b.de']);
|
||||||
|
|
||||||
|
sleep(2);
|
||||||
test()->get(route('invoice.index'))
|
test()->get(route('invoice.index'))
|
||||||
->assertInertiaPath('data.data.0.to.name', 'Familie Blabla')
|
->assertInertiaPath('data.data.0.to.name', 'Familie Blabla')
|
||||||
->assertInertiaPath('data.data.0.id', $invoice->id)
|
->assertInertiaPath('data.data.0.id', $invoice->id)
|
||||||
|
@ -78,19 +82,35 @@ it('testValuesCanBeNull', function () {
|
||||||
test()->login()->loginNami()->withoutExceptionHandling();
|
test()->login()->loginNami()->withoutExceptionHandling();
|
||||||
Invoice::factory()->create();
|
Invoice::factory()->create();
|
||||||
|
|
||||||
|
sleep(2);
|
||||||
test()->get(route('invoice.index'))
|
test()->get(route('invoice.index'))
|
||||||
->assertInertiaPath('data.data.0.sent_at_human', '');
|
->assertInertiaPath('data.data.0.sent_at_human', '');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('testItFiltersForInvoiceStatus', function () {
|
it('filters for invoice status', function (array $filter, int $count) {
|
||||||
test()->login()->loginNami()->withoutExceptionHandling();
|
test()->login()->loginNami()->withoutExceptionHandling();
|
||||||
Invoice::factory()->status(InvoiceStatus::NEW)->create();
|
Invoice::factory()->status(InvoiceStatus::NEW)->create();
|
||||||
Invoice::factory()->status(InvoiceStatus::SENT)->count(2)->create();
|
Invoice::factory()->status(InvoiceStatus::SENT)->count(2)->create();
|
||||||
Invoice::factory()->status(InvoiceStatus::PAID)->count(3)->create();
|
Invoice::factory()->status(InvoiceStatus::PAID)->count(3)->create();
|
||||||
|
|
||||||
test()->callFilter('invoice.index', [])->assertInertiaCount('data.data', 3);
|
sleep(2);
|
||||||
test()->callFilter('invoice.index', ['statuses' => []])->assertInertiaCount('data.data', 0);
|
test()->callFilter('invoice.index', $filter)->assertInertiaCount('data.data', $count);
|
||||||
test()->callFilter('invoice.index', ['statuses' => ['Neu']])->assertInertiaCount('data.data', 1);
|
})->with([
|
||||||
test()->callFilter('invoice.index', ['statuses' => ['Neu', 'Rechnung beglichen']])->assertInertiaCount('data.data', 4);
|
[[], 3],
|
||||||
test()->callFilter('invoice.index', ['statuses' => ['Neu', 'Rechnung beglichen', 'Rechnung gestellt']])->assertInertiaCount('data.data', 6);
|
[['statuses' => []], 0],
|
||||||
});
|
[['statuses' => ['Neu']], 1],
|
||||||
|
[['statuses' => ['Neu', 'Rechnung beglichen']], 4],
|
||||||
|
[['statuses' => ['Neu', 'Rechnung beglichen', 'Rechnung gestellt']], 6],
|
||||||
|
]);
|
||||||
|
|
||||||
|
it('searches invoice usage', function (array $filter, int $count) {
|
||||||
|
test()->login()->loginNami()->withoutExceptionHandling();
|
||||||
|
Invoice::factory()->status(InvoiceStatus::NEW)->create(['usage' => 'Kein Zweck']);
|
||||||
|
Invoice::factory()->status(InvoiceStatus::NEW)->create(['usage' => 'Mitgliedsbeitrag']);
|
||||||
|
|
||||||
|
sleep(2);
|
||||||
|
test()->callFilter('invoice.index', $filter)->assertInertiaCount('data.data', $count);
|
||||||
|
})->with([
|
||||||
|
[['search' => 'Mitgliedsbeitrag'], 1],
|
||||||
|
[['search' => 'Kein'], 1],
|
||||||
|
]);
|
|
@ -3,6 +3,7 @@
|
||||||
namespace Tests;
|
namespace Tests;
|
||||||
|
|
||||||
use App\Form\Models\Form;
|
use App\Form\Models\Form;
|
||||||
|
use App\Invoice\Models\Invoice;
|
||||||
use App\Member\Member;
|
use App\Member\Member;
|
||||||
use Laravel\Scout\Console\FlushCommand;
|
use Laravel\Scout\Console\FlushCommand;
|
||||||
use Laravel\Scout\Console\SyncIndexSettingsCommand;
|
use Laravel\Scout\Console\SyncIndexSettingsCommand;
|
||||||
|
@ -26,6 +27,7 @@ abstract class EndToEndTestCase extends TestCase
|
||||||
config()->set('scout.driver', 'meilisearch');
|
config()->set('scout.driver', 'meilisearch');
|
||||||
Artisan::call(FlushCommand::class, ['model' => Member::class]);
|
Artisan::call(FlushCommand::class, ['model' => Member::class]);
|
||||||
Artisan::call(FlushCommand::class, ['model' => Form::class]);
|
Artisan::call(FlushCommand::class, ['model' => Form::class]);
|
||||||
|
Artisan::call(FlushCommand::class, ['model' => Invoice::class]);
|
||||||
Artisan::call(SyncIndexSettingsCommand::class);
|
Artisan::call(SyncIndexSettingsCommand::class);
|
||||||
|
|
||||||
return $this;
|
return $this;
|
||||||
|
|
|
@ -2,211 +2,202 @@
|
||||||
|
|
||||||
namespace Tests\Feature\Contribution;
|
namespace Tests\Feature\Contribution;
|
||||||
|
|
||||||
use App\Contribution\Documents\ContributionDocument;
|
|
||||||
use App\Contribution\Documents\RdpNrwDocument;
|
use App\Contribution\Documents\RdpNrwDocument;
|
||||||
use App\Contribution\Documents\CitySolingenDocument;
|
use App\Contribution\Documents\CitySolingenDocument;
|
||||||
|
use App\Contribution\Documents\WuppertalDocument;
|
||||||
use App\Country;
|
use App\Country;
|
||||||
use App\Gender;
|
use App\Gender;
|
||||||
use App\Invoice\InvoiceSettings;
|
use App\Invoice\InvoiceSettings;
|
||||||
use App\Member\Member;
|
use App\Member\Member;
|
||||||
use Generator;
|
|
||||||
use Illuminate\Foundation\Testing\DatabaseTransactions;
|
use Illuminate\Foundation\Testing\DatabaseTransactions;
|
||||||
use Laravel\Passport\Client;
|
use Laravel\Passport\Client;
|
||||||
use Laravel\Passport\Passport;
|
use Laravel\Passport\Passport;
|
||||||
use PHPUnit\Framework\Attributes\DataProvider;
|
|
||||||
use Tests\RequestFactories\ContributionMemberApiRequestFactory;
|
use Tests\RequestFactories\ContributionMemberApiRequestFactory;
|
||||||
use Tests\RequestFactories\ContributionRequestFactory;
|
use Tests\RequestFactories\ContributionRequestFactory;
|
||||||
use Tests\TestCase;
|
|
||||||
use Zoomyboy\Tex\Tex;
|
use Zoomyboy\Tex\Tex;
|
||||||
|
|
||||||
class StoreTest extends TestCase
|
uses(DatabaseTransactions::class);
|
||||||
{
|
|
||||||
use DatabaseTransactions;
|
|
||||||
|
|
||||||
/**
|
dataset('validation', function () {
|
||||||
* @testWith ["App\\Contribution\\Documents\\CitySolingenDocument", ["Super tolles Lager", "Max Muster", "Jane Muster", "15.06.1991"]]
|
return [
|
||||||
* ["App\\Contribution\\Documents\\RdpNrwDocument", ["Muster, Max", "Muster, Jane", "15.06.1991", "42777 SG"]]
|
[
|
||||||
* ["App\\Contribution\\Documents\\CityRemscheidDocument", ["Max", "Muster", "Jane"]]
|
|
||||||
* ["App\\Contribution\\Documents\\CityFrankfurtMainDocument", ["Max", "Muster", "Jane"]]
|
|
||||||
* ["App\\Contribution\\Documents\\BdkjHesse", ["Max", "Muster", "Jane"]]
|
|
||||||
*
|
|
||||||
* @param array<int, string> $bodyChecks
|
|
||||||
*/
|
|
||||||
public function testItCompilesContributionDocumentsViaRequest(string $type, array $bodyChecks): void
|
|
||||||
{
|
|
||||||
$this->withoutExceptionHandling();
|
|
||||||
Tex::spy();
|
|
||||||
$this->login()->loginNami();
|
|
||||||
$member1 = Member::factory()->defaults()->create(['address' => 'Maxstr 44', 'zip' => '42719', 'firstname' => 'Max', 'lastname' => 'Muster']);
|
|
||||||
$member2 = Member::factory()->defaults()->create(['address' => 'Maxstr 44', 'zip' => '42719', 'firstname' => 'Jane', 'lastname' => 'Muster']);
|
|
||||||
|
|
||||||
$response = $this->call('GET', '/contribution-generate', [
|
|
||||||
'payload' => ContributionRequestFactory::new()->type($type)->state([
|
|
||||||
'dateFrom' => '1991-06-15',
|
|
||||||
'dateUntil' => '1991-06-16',
|
|
||||||
'eventName' => 'Super tolles Lager',
|
|
||||||
'members' => [$member1->id, $member2->id],
|
|
||||||
'type' => $type,
|
|
||||||
'zipLocation' => '42777 SG',
|
|
||||||
])->toBase64(),
|
|
||||||
]);
|
|
||||||
|
|
||||||
$response->assertSessionDoesntHaveErrors();
|
|
||||||
$response->assertOk();
|
|
||||||
Tex::assertCompiled($type, fn ($document) => $document->hasAllContent($bodyChecks));
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testItCompilesGroupNameInSolingenDocument(): void
|
|
||||||
{
|
|
||||||
$this->withoutExceptionHandling()->login()->loginNami();
|
|
||||||
Tex::spy();
|
|
||||||
InvoiceSettings::fake(['from_long' => 'Stamm BiPi']);
|
|
||||||
|
|
||||||
$this->call('GET', '/contribution-generate', [
|
|
||||||
'payload' => ContributionRequestFactory::new()->type(CitySolingenDocument::class)->toBase64(),
|
|
||||||
]);
|
|
||||||
|
|
||||||
Tex::assertCompiled(CitySolingenDocument::class, fn ($document) => $document->hasAllContent(['Stamm BiPi']));
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testItCompilesContributionDocumentsViaApi(): void
|
|
||||||
{
|
|
||||||
$this->withoutExceptionHandling();
|
|
||||||
Tex::spy();
|
|
||||||
Gender::factory()->female()->create();
|
|
||||||
Gender::factory()->male()->create();
|
|
||||||
Passport::actingAsClient(Client::factory()->create(), ['contribution-generate']);
|
|
||||||
$country = Country::factory()->create();
|
|
||||||
Member::factory()->defaults()->create(['address' => 'Maxstr 44', 'zip' => '42719', 'firstname' => 'Max', 'lastname' => 'Muster']);
|
|
||||||
Member::factory()->defaults()->create(['address' => 'Maxstr 44', 'zip' => '42719', 'firstname' => 'Jane', 'lastname' => 'Muster']);
|
|
||||||
|
|
||||||
$response = $this->postJson('/api/contribution-generate', [
|
|
||||||
'country' => $country->id,
|
|
||||||
'dateFrom' => '1991-06-15',
|
|
||||||
'dateUntil' => '1991-06-16',
|
|
||||||
'eventName' => 'Super tolles Lager',
|
|
||||||
'type' => CitySolingenDocument::class,
|
|
||||||
'zipLocation' => '42777 SG',
|
|
||||||
'member_data' => [
|
|
||||||
ContributionMemberApiRequestFactory::new()->create(),
|
|
||||||
ContributionMemberApiRequestFactory::new()->create(),
|
|
||||||
],
|
|
||||||
]);
|
|
||||||
|
|
||||||
$response->assertSessionDoesntHaveErrors();
|
|
||||||
$response->assertOk();
|
|
||||||
Tex::assertCompiled(CitySolingenDocument::class, fn ($document) => $document->hasAllContent(['Super']));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @testWith [""]
|
|
||||||
* ["aaaa"]
|
|
||||||
* ["YWFhCg=="]
|
|
||||||
*/
|
|
||||||
public function testInputShouldBeBase64EncodedJson(string $payload): void
|
|
||||||
{
|
|
||||||
$this->login()->loginNami();
|
|
||||||
|
|
||||||
$this->call('GET', '/contribution-generate', ['payload' => $payload])->assertSessionHasErrors('payload');
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param array<string, string> $input
|
|
||||||
* @param class-string<ContributionDocument> $documentClass
|
|
||||||
*/
|
|
||||||
#[DataProvider('validationDataProvider')]
|
|
||||||
public function testItValidatesInput(array $input, string $documentClass, string $errorField): void
|
|
||||||
{
|
|
||||||
$this->login()->loginNami();
|
|
||||||
Country::factory()->create();
|
|
||||||
Member::factory()->defaults()->create();
|
|
||||||
|
|
||||||
$this->postJson('/contribution-validate', ContributionRequestFactory::new()->type($documentClass)->state($input)->create())
|
|
||||||
->assertJsonValidationErrors($errorField);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param array<string, string> $input
|
|
||||||
* @param class-string<ContributionDocument> $documentClass
|
|
||||||
*/
|
|
||||||
#[DataProvider('validationDataProvider')]
|
|
||||||
public function testItValidatesInputBeforeGeneration(array $input, string $documentClass, string $errorField): void
|
|
||||||
{
|
|
||||||
$this->login()->loginNami();
|
|
||||||
Country::factory()->create();
|
|
||||||
Member::factory()->defaults()->create();
|
|
||||||
|
|
||||||
$this->call('GET', '/contribution-generate', [
|
|
||||||
'payload' => ContributionRequestFactory::new()->type($documentClass)->state($input)->toBase64(),
|
|
||||||
])->assertSessionHasErrors($errorField);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static function validationDataProvider(): Generator
|
|
||||||
{
|
|
||||||
yield [
|
|
||||||
['type' => 'aaa'],
|
['type' => 'aaa'],
|
||||||
CitySolingenDocument::class,
|
CitySolingenDocument::class,
|
||||||
'type',
|
'type',
|
||||||
];
|
],
|
||||||
yield [
|
[
|
||||||
['type' => ''],
|
['type' => ''],
|
||||||
CitySolingenDocument::class,
|
CitySolingenDocument::class,
|
||||||
'type',
|
'type',
|
||||||
];
|
],
|
||||||
yield [
|
[
|
||||||
['dateFrom' => ''],
|
['dateFrom' => ''],
|
||||||
CitySolingenDocument::class,
|
CitySolingenDocument::class,
|
||||||
'dateFrom',
|
'dateFrom',
|
||||||
];
|
],
|
||||||
yield [
|
[
|
||||||
['dateFrom' => '2022-01'],
|
['dateFrom' => '2022-01'],
|
||||||
CitySolingenDocument::class,
|
CitySolingenDocument::class,
|
||||||
'dateFrom',
|
'dateFrom',
|
||||||
];
|
],
|
||||||
yield [
|
[
|
||||||
['dateUntil' => ''],
|
['dateUntil' => ''],
|
||||||
CitySolingenDocument::class,
|
CitySolingenDocument::class,
|
||||||
'dateUntil',
|
'dateUntil',
|
||||||
];
|
],
|
||||||
yield [
|
[
|
||||||
['dateUntil' => '2022-01'],
|
['dateUntil' => '2022-01'],
|
||||||
CitySolingenDocument::class,
|
CitySolingenDocument::class,
|
||||||
'dateUntil',
|
'dateUntil',
|
||||||
];
|
],
|
||||||
yield [
|
[
|
||||||
['country' => -1],
|
['country' => -1],
|
||||||
RdpNrwDocument::class,
|
RdpNrwDocument::class,
|
||||||
'country',
|
'country',
|
||||||
];
|
],
|
||||||
yield [
|
[
|
||||||
['country' => 'AAAA'],
|
['country' => 'AAAA'],
|
||||||
RdpNrwDocument::class,
|
RdpNrwDocument::class,
|
||||||
'country',
|
'country',
|
||||||
];
|
],
|
||||||
yield [
|
[
|
||||||
['members' => 'A'],
|
['members' => 'A'],
|
||||||
RdpNrwDocument::class,
|
RdpNrwDocument::class,
|
||||||
'members',
|
'members',
|
||||||
];
|
],
|
||||||
yield [
|
[
|
||||||
['members' => [99999]],
|
['members' => [99999]],
|
||||||
RdpNrwDocument::class,
|
RdpNrwDocument::class,
|
||||||
'members.0',
|
'members.0',
|
||||||
];
|
],
|
||||||
yield [
|
[
|
||||||
['members' => ['lalala']],
|
['members' => ['lalala']],
|
||||||
RdpNrwDocument::class,
|
RdpNrwDocument::class,
|
||||||
'members.0',
|
'members.0',
|
||||||
];
|
],
|
||||||
yield [
|
[
|
||||||
['eventName' => ''],
|
['eventName' => ''],
|
||||||
CitySolingenDocument::class,
|
CitySolingenDocument::class,
|
||||||
'eventName',
|
'eventName',
|
||||||
];
|
],
|
||||||
yield [
|
[
|
||||||
['zipLocation' => ''],
|
['zipLocation' => ''],
|
||||||
CitySolingenDocument::class,
|
CitySolingenDocument::class,
|
||||||
'zipLocation',
|
'zipLocation',
|
||||||
];
|
],
|
||||||
}
|
[
|
||||||
}
|
['zipLocation' => ''],
|
||||||
|
WuppertalDocument::class,
|
||||||
|
'zipLocation',
|
||||||
|
],
|
||||||
|
[
|
||||||
|
['dateFrom' => ''],
|
||||||
|
WuppertalDocument::class,
|
||||||
|
'dateFrom',
|
||||||
|
],
|
||||||
|
[
|
||||||
|
['dateUntil' => ''],
|
||||||
|
WuppertalDocument::class,
|
||||||
|
'dateUntil',
|
||||||
|
],
|
||||||
|
];
|
||||||
|
});
|
||||||
|
|
||||||
|
it('compiles documents via api', function (string $type, array $bodyChecks) {
|
||||||
|
$this->withoutExceptionHandling();
|
||||||
|
Tex::spy();
|
||||||
|
$this->login()->loginNami();
|
||||||
|
$member1 = Member::factory()->defaults()->male()->create(['address' => 'Maxstr 44', 'zip' => '42719', 'firstname' => 'Max', 'lastname' => 'Muster']);
|
||||||
|
$member2 = Member::factory()->defaults()->female()->create(['address' => 'Maxstr 44', 'zip' => '42719', 'firstname' => 'Jane', 'lastname' => 'Muster']);
|
||||||
|
|
||||||
|
$response = $this->call('GET', '/contribution-generate', [
|
||||||
|
'payload' => ContributionRequestFactory::new()->type($type)->state([
|
||||||
|
'dateFrom' => '1991-06-15',
|
||||||
|
'dateUntil' => '1991-06-16',
|
||||||
|
'eventName' => 'Super tolles Lager',
|
||||||
|
'members' => [$member1->id, $member2->id],
|
||||||
|
'type' => $type,
|
||||||
|
'zipLocation' => '42777 SG',
|
||||||
|
])->toBase64(),
|
||||||
|
]);
|
||||||
|
|
||||||
|
$response->assertSessionDoesntHaveErrors();
|
||||||
|
$response->assertOk();
|
||||||
|
Tex::assertCompiled($type, fn ($document) => $document->hasAllContent($bodyChecks));
|
||||||
|
})->with([
|
||||||
|
["App\\Contribution\\Documents\\CitySolingenDocument", ["Super tolles Lager", "Max Muster", "Jane Muster", "15.06.1991"]],
|
||||||
|
["App\\Contribution\\Documents\\RdpNrwDocument", ["Muster, Max", "Muster, Jane", "15.06.1991", "42777 SG"]],
|
||||||
|
["App\\Contribution\\Documents\\CityRemscheidDocument", ["Max", "Muster", "Jane"]],
|
||||||
|
["App\\Contribution\\Documents\\CityFrankfurtMainDocument", ["Max", "Muster", "Jane"]],
|
||||||
|
["App\\Contribution\\Documents\\BdkjHesse", ["Max", "Muster", "Jane"]],
|
||||||
|
["App\\Contribution\\Documents\\WuppertalDocument", ["Max", "Muster", "Jane", "42777 SG", "15.06.1991", "16.06.1991"]],
|
||||||
|
]);
|
||||||
|
|
||||||
|
it('testItCompilesGroupNameInSolingenDocument', function () {
|
||||||
|
$this->withoutExceptionHandling()->login()->loginNami();
|
||||||
|
Tex::spy();
|
||||||
|
InvoiceSettings::fake(['from_long' => 'Stamm BiPi']);
|
||||||
|
|
||||||
|
$this->call('GET', '/contribution-generate', [
|
||||||
|
'payload' => ContributionRequestFactory::new()->type(CitySolingenDocument::class)->toBase64(),
|
||||||
|
]);
|
||||||
|
|
||||||
|
Tex::assertCompiled(CitySolingenDocument::class, fn ($document) => $document->hasAllContent(['Stamm BiPi']));
|
||||||
|
});
|
||||||
|
|
||||||
|
it('testItCompilesContributionDocumentsViaApi', function () {
|
||||||
|
$this->withoutExceptionHandling();
|
||||||
|
Tex::spy();
|
||||||
|
Gender::factory()->female()->create();
|
||||||
|
Gender::factory()->male()->create();
|
||||||
|
Passport::actingAsClient(Client::factory()->create(), ['contribution-generate']);
|
||||||
|
$country = Country::factory()->create();
|
||||||
|
Member::factory()->defaults()->create(['address' => 'Maxstr 44', 'zip' => '42719', 'firstname' => 'Max', 'lastname' => 'Muster']);
|
||||||
|
Member::factory()->defaults()->create(['address' => 'Maxstr 44', 'zip' => '42719', 'firstname' => 'Jane', 'lastname' => 'Muster']);
|
||||||
|
|
||||||
|
$response = $this->postJson('/api/contribution-generate', [
|
||||||
|
'country' => $country->id,
|
||||||
|
'dateFrom' => '1991-06-15',
|
||||||
|
'dateUntil' => '1991-06-16',
|
||||||
|
'eventName' => 'Super tolles Lager',
|
||||||
|
'type' => CitySolingenDocument::class,
|
||||||
|
'zipLocation' => '42777 SG',
|
||||||
|
'member_data' => [
|
||||||
|
ContributionMemberApiRequestFactory::new()->create(),
|
||||||
|
ContributionMemberApiRequestFactory::new()->create(),
|
||||||
|
],
|
||||||
|
]);
|
||||||
|
|
||||||
|
$response->assertSessionDoesntHaveErrors();
|
||||||
|
$response->assertOk();
|
||||||
|
Tex::assertCompiled(CitySolingenDocument::class, fn ($document) => $document->hasAllContent(['Super']));
|
||||||
|
});
|
||||||
|
|
||||||
|
it('testInputShouldBeBase64EncodedJson', function (string $payload) {
|
||||||
|
$this->login()->loginNami();
|
||||||
|
|
||||||
|
$this->call('GET', '/contribution-generate', ['payload' => $payload])->assertSessionHasErrors('payload');
|
||||||
|
})->with([
|
||||||
|
[""],
|
||||||
|
["aaaa"],
|
||||||
|
["YWFhCg=="],
|
||||||
|
]);
|
||||||
|
|
||||||
|
it('testItValidatesInput', function (array $input, string $documentClass, string $errorField) {
|
||||||
|
$this->login()->loginNami();
|
||||||
|
Country::factory()->create();
|
||||||
|
Member::factory()->defaults()->create();
|
||||||
|
|
||||||
|
$this->postJson('/contribution-validate', ContributionRequestFactory::new()->type($documentClass)->state($input)->create())
|
||||||
|
->assertJsonValidationErrors($errorField);
|
||||||
|
})->with('validation');
|
||||||
|
|
||||||
|
it('testItValidatesInputBeforeGeneration', function (array $input, string $documentClass, string $errorField) {
|
||||||
|
$this->login()->loginNami();
|
||||||
|
Country::factory()->create();
|
||||||
|
Member::factory()->defaults()->create();
|
||||||
|
|
||||||
|
$this->call('GET', '/contribution-generate', [
|
||||||
|
'payload' => ContributionRequestFactory::new()->type($documentClass)->state($input)->toBase64(),
|
||||||
|
])->assertSessionHasErrors($errorField);
|
||||||
|
})->with('validation');
|
||||||
|
|
|
@ -55,6 +55,22 @@ class FormRegisterActionTest extends FormTestCase
|
||||||
$this->assertEquals('Abraham', $participants->first()->data['spitzname']);
|
$this->assertEquals('Abraham', $participants->first()->data['spitzname']);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testItCannotRegisterWhenRegistrationFromReached(): void
|
||||||
|
{
|
||||||
|
$this->login()->loginNami();
|
||||||
|
$form = Form::factory()->registrationFrom(now()->addDay())->create();
|
||||||
|
|
||||||
|
$this->register($form, [])->assertJsonValidationErrors(['event' => 'Anmeldung zzt nicht möglich.']);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testItCannotRegisterWhenRegistrationUntilReached(): void
|
||||||
|
{
|
||||||
|
$this->login()->loginNami();
|
||||||
|
$form = Form::factory()->registrationUntil(now()->subDay())->create();
|
||||||
|
|
||||||
|
$this->register($form, [])->assertJsonValidationErrors(['event' => 'Anmeldung zzt nicht möglich.']);
|
||||||
|
}
|
||||||
|
|
||||||
public function testItSendsEmailToParticipant(): void
|
public function testItSendsEmailToParticipant(): void
|
||||||
{
|
{
|
||||||
$this->login()->loginNami()->withoutExceptionHandling();
|
$this->login()->loginNami()->withoutExceptionHandling();
|
||||||
|
|
|
@ -59,6 +59,19 @@ class FormStoreActionTest extends FormTestCase
|
||||||
$this->assertFrontendCacheCleared();
|
$this->assertFrontendCacheCleared();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testItStoresDefaultSorting(): void
|
||||||
|
{
|
||||||
|
Event::fake([Succeeded::class]);
|
||||||
|
$this->login()->loginNami()->withoutExceptionHandling();
|
||||||
|
FormRequest::new()->fields([$this->textField()])->fake();
|
||||||
|
|
||||||
|
$this->postJson(route('form.store'))->assertOk();
|
||||||
|
|
||||||
|
$form = Form::latest()->first();
|
||||||
|
$this->assertEquals('id', $form->meta['sorting']['by']);
|
||||||
|
$this->assertFalse(false, $form->meta['sorting']['direction']);
|
||||||
|
}
|
||||||
|
|
||||||
public function testRegistrationDatesCanBeNull(): void
|
public function testRegistrationDatesCanBeNull(): void
|
||||||
{
|
{
|
||||||
$this->login()->loginNami()->withoutExceptionHandling();
|
$this->login()->loginNami()->withoutExceptionHandling();
|
||||||
|
|
|
@ -5,8 +5,6 @@ namespace Tests\Feature\Form;
|
||||||
use App\Fileshare\Data\FileshareResourceData;
|
use App\Fileshare\Data\FileshareResourceData;
|
||||||
use App\Form\Data\ExportData;
|
use App\Form\Data\ExportData;
|
||||||
use App\Form\Models\Form;
|
use App\Form\Models\Form;
|
||||||
use App\Lib\Editor\Condition;
|
|
||||||
use App\Lib\Editor\EditorData;
|
|
||||||
use Illuminate\Foundation\Testing\DatabaseTransactions;
|
use Illuminate\Foundation\Testing\DatabaseTransactions;
|
||||||
use Tests\RequestFactories\EditorRequestFactory;
|
use Tests\RequestFactories\EditorRequestFactory;
|
||||||
|
|
||||||
|
@ -31,6 +29,20 @@ class FormUpdateActionTest extends FormTestCase
|
||||||
$this->assertTrue($form->config->sections->get(0)->fields->get(0)->maxToday);
|
$this->assertTrue($form->config->sections->get(0)->fields->get(0)->maxToday);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testItSetsRegistrationDates(): void
|
||||||
|
{
|
||||||
|
$this->login()->loginNami()->withoutExceptionHandling();
|
||||||
|
$form = Form::factory()->create();
|
||||||
|
$payload = FormRequest::new()->registrationFrom('2023-05-04 01:00:00')->registrationUntil('2023-07-07 01:00:00')->create();
|
||||||
|
|
||||||
|
$this->patchJson(route('form.update', ['form' => $form]), $payload)->assertOk();
|
||||||
|
|
||||||
|
$form = $form->fresh();
|
||||||
|
|
||||||
|
$this->assertEquals('2023-05-04 01:00', $form->registration_from->format('Y-m-d H:i'));
|
||||||
|
$this->assertEquals('2023-07-07 01:00', $form->registration_until->format('Y-m-d H:i'));
|
||||||
|
}
|
||||||
|
|
||||||
public function testItSetsTexts(): void
|
public function testItSetsTexts(): void
|
||||||
{
|
{
|
||||||
$this->login()->loginNami()->withoutExceptionHandling();
|
$this->login()->loginNami()->withoutExceptionHandling();
|
||||||
|
|
|
@ -21,13 +21,13 @@ class FormUpdateMetaActionTest extends FormTestCase
|
||||||
|
|
||||||
$this->patchJson(route('form.update-meta', ['form' => $form]), [
|
$this->patchJson(route('form.update-meta', ['form' => $form]), [
|
||||||
'active_columns' => ['textone'],
|
'active_columns' => ['textone'],
|
||||||
'sorting' => ['textone', 'desc'],
|
'sorting' => ['by' => 'textone', 'direction' => false],
|
||||||
])->assertOk()
|
])->assertOk()
|
||||||
->assertJsonPath('active_columns.0', 'textone')
|
->assertJsonPath('active_columns.0', 'textone')
|
||||||
->assertJsonPath('sorting.1', 'desc');
|
->assertJsonPath('sorting.by', 'textone');
|
||||||
|
|
||||||
$form = Form::latest()->first();
|
$form = Form::latest()->first();
|
||||||
$this->assertEquals(['textone', 'desc'], $form->meta['sorting']);
|
$this->assertEquals(['by' => 'textone', 'direction' => false], $form->meta['sorting']);
|
||||||
$this->assertEquals(['textone'], $form->meta['active_columns']);
|
$this->assertEquals(['textone'], $form->meta['active_columns']);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -38,11 +38,11 @@ class FormUpdateMetaActionTest extends FormTestCase
|
||||||
|
|
||||||
$this->patchJson(route('form.update-meta', ['form' => $form]), [
|
$this->patchJson(route('form.update-meta', ['form' => $form]), [
|
||||||
'active_columns' => ['created_at'],
|
'active_columns' => ['created_at'],
|
||||||
'sorting' => ['created_at', 'desc'],
|
'sorting' => ['by' => 'textone', 'direction' => false],
|
||||||
])->assertOk();
|
])->assertOk();
|
||||||
|
|
||||||
$form = Form::latest()->first();
|
$form = Form::latest()->first();
|
||||||
$this->assertEquals(['created_at', 'desc'], $form->fresh()->meta['sorting']);
|
$this->assertEquals(['by' => 'textone', 'direction' => false], $form->fresh()->meta['sorting']);
|
||||||
$this->assertEquals(['created_at'], $form->fresh()->meta['active_columns']);
|
$this->assertEquals(['created_at'], $form->fresh()->meta['active_columns']);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,225 +0,0 @@
|
||||||
<?php
|
|
||||||
|
|
||||||
namespace Tests\Feature\Form;
|
|
||||||
|
|
||||||
use App\Form\Fields\TextField;
|
|
||||||
use App\Form\Models\Form;
|
|
||||||
use App\Form\Models\Participant;
|
|
||||||
use App\Form\Scopes\ParticipantFilterScope;
|
|
||||||
use App\Group;
|
|
||||||
use App\Member\Member;
|
|
||||||
use Carbon\Carbon;
|
|
||||||
use Illuminate\Foundation\Testing\DatabaseTransactions;
|
|
||||||
|
|
||||||
class ParticipantIndexActionTest extends FormTestCase
|
|
||||||
{
|
|
||||||
|
|
||||||
use DatabaseTransactions;
|
|
||||||
|
|
||||||
public function testItShowsParticipantsAndColumns(): void
|
|
||||||
{
|
|
||||||
$this->login()->loginNami()->withoutExceptionHandling();
|
|
||||||
$group = Group::factory()->innerName('Stamm')->create();
|
|
||||||
$form = Form::factory()
|
|
||||||
->has(Participant::factory()->state(['member_id' => 55])->data(['vorname' => 'Max', 'select' => ['A', 'B'], 'stufe' => 'Pfadfinder', 'test1' => '', 'test2' => '', 'test3' => '', 'birthday' => '1991-04-20', 'bezirk' => $group->id]))
|
|
||||||
->fields([
|
|
||||||
$this->textField('vorname')->name('Vorname'),
|
|
||||||
$this->checkboxesField('select')->options(['A', 'B', 'C']),
|
|
||||||
$this->dropdownField('stufe')->options(['Wölfling', 'Jungpfadfinder', 'Pfadfinder']),
|
|
||||||
$this->textField('test1')->name('Test 1'),
|
|
||||||
$this->textField('test2')->name('Test 2'),
|
|
||||||
$this->textField('test3')->name('Test 3'),
|
|
||||||
$this->dateField('birthday')->name('Geburtsdatum'),
|
|
||||||
$this->groupField('bezirk')->name('bezirk'),
|
|
||||||
])
|
|
||||||
->create();
|
|
||||||
|
|
||||||
$this->callFilter('form.participant.index', [], ['form' => $form])
|
|
||||||
->assertOk()
|
|
||||||
->assertJsonPath('data.0.id', $form->participants->first()->id)
|
|
||||||
->assertJsonPath('data.0.vorname', 'Max')
|
|
||||||
->assertJsonPath('data.0.vorname_display', 'Max')
|
|
||||||
->assertJsonPath('data.0.stufe', 'Pfadfinder')
|
|
||||||
->assertJsonPath('data.0.bezirk', $group->id)
|
|
||||||
->assertJsonPath('data.0.member_id', 55)
|
|
||||||
->assertJsonPath('data.0.bezirk_display', 'Stamm')
|
|
||||||
->assertJsonPath('data.0.birthday_display', '20.04.1991')
|
|
||||||
->assertJsonPath('data.0.birthday', '1991-04-20')
|
|
||||||
->assertJsonPath('data.0.select', ['A', 'B'])
|
|
||||||
->assertJsonPath('data.0.select_display', 'A, B')
|
|
||||||
->assertJsonPath('data.0.links.destroy', route('participant.destroy', ['participant' => $form->participants->first()]))
|
|
||||||
->assertJsonPath('data.0.links.assign', route('participant.assign', ['participant' => $form->participants->first()]))
|
|
||||||
->assertJsonPath('data.0.links.fields', route('participant.fields', ['participant' => $form->participants->first()]))
|
|
||||||
->assertJsonPath('data.0.links.update', route('participant.update', ['participant' => $form->participants->first()]))
|
|
||||||
->assertJsonPath('meta.columns.0.name', 'Vorname')
|
|
||||||
->assertJsonPath('meta.columns.0.base_type', class_basename(TextField::class))
|
|
||||||
->assertJsonPath('meta.columns.0.id', 'vorname')
|
|
||||||
->assertJsonPath('meta.columns.6.display_attribute', 'birthday_display')
|
|
||||||
->assertJsonPath('meta.columns.0.display_attribute', 'vorname_display')
|
|
||||||
->assertJsonPath('meta.form_meta.active_columns', ['vorname', 'select', 'stufe', 'test1'])
|
|
||||||
->assertJsonPath('meta.has_nami_field', false)
|
|
||||||
->assertJsonPath('meta.links.update_form_meta', route('form.update-meta', ['form' => $form]))
|
|
||||||
->assertJsonPath('meta.links.store_participant', route('form.participant.store', ['form' => $form]))
|
|
||||||
->assertJsonPath('meta.form_meta.sorting', ['vorname', 'asc'])
|
|
||||||
->assertJsonPath('meta.form_config.sections.0.fields.0.key', 'vorname');
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testItShowsEmptyFilters(): void
|
|
||||||
{
|
|
||||||
$this->login()->loginNami()->withoutExceptionHandling();
|
|
||||||
$form = Form::factory()->fields([$this->checkboxField('check')->name('Checked')])->create();
|
|
||||||
|
|
||||||
$this->callFilter('form.participant.index', [], ['form' => $form])
|
|
||||||
->assertOk()
|
|
||||||
->assertJsonPath('meta.filters.0.name', 'Checked')
|
|
||||||
->assertJsonPath('meta.filters.0.key', 'check')
|
|
||||||
->assertJsonPath('meta.filters.0.base_type', 'CheckboxField')
|
|
||||||
->assertJsonPath('meta.default_filter_value', ParticipantFilterScope::$nan);
|
|
||||||
|
|
||||||
$this->callFilter('form.participant.index', ['data' => ['check' => null]], ['form' => $form])->assertHasJsonPath('meta.filter.data.check')->assertJsonPath('meta.filter.data.check', null);
|
|
||||||
$this->callFilter('form.participant.index', ['data' => ['check' => 'A']], ['form' => $form])->assertJsonPath('meta.filter.data.check', 'A');
|
|
||||||
$this->callFilter('form.participant.index', ['data' => []], ['form' => $form])->assertJsonPath('meta.filter.data.check', ParticipantFilterScope::$nan);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testItDisplaysHasNamiField(): void
|
|
||||||
{
|
|
||||||
$this->login()->loginNami()->withoutExceptionHandling();
|
|
||||||
$form = Form::factory()->fields([$this->namiField('mitglieder')])->create();
|
|
||||||
$this->callFilter('form.participant.index', [], ['form' => $form])->assertJsonPath('meta.has_nami_field', true);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testItFiltersParticipantsByCheckboxValue(): void
|
|
||||||
{
|
|
||||||
$this->login()->loginNami()->withoutExceptionHandling();
|
|
||||||
$form = Form::factory()->fields([$this->checkboxField('check')])
|
|
||||||
->has(Participant::factory()->data(['check' => true])->count(1))
|
|
||||||
->has(Participant::factory()->data(['check' => false])->count(2))
|
|
||||||
->create();
|
|
||||||
|
|
||||||
$this->callFilter('form.participant.index', ['data' => ['check' => ParticipantFilterScope::$nan]], ['form' => $form])
|
|
||||||
->assertJsonCount(3, 'data');
|
|
||||||
$this->callFilter('form.participant.index', ['data' => ['check' => true]], ['form' => $form])
|
|
||||||
->assertJsonCount(1, 'data');
|
|
||||||
$this->callFilter('form.participant.index', ['data' => ['check' => false]], ['form' => $form])
|
|
||||||
->assertJsonCount(2, 'data');
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testItFiltersParticipantsByDropdownValue(): void
|
|
||||||
{
|
|
||||||
$this->login()->loginNami()->withoutExceptionHandling();
|
|
||||||
$form = Form::factory()->fields([$this->dropdownField('drop')->options(['A', 'B'])])
|
|
||||||
->has(Participant::factory()->data(['drop' => null])->count(1))
|
|
||||||
->has(Participant::factory()->data(['drop' => 'A'])->count(2))
|
|
||||||
->has(Participant::factory()->data(['drop' => 'B'])->count(4))
|
|
||||||
->create();
|
|
||||||
|
|
||||||
$this->callFilter('form.participant.index', ['data' => ['drop' => ParticipantFilterScope::$nan]], ['form' => $form])
|
|
||||||
->assertJsonCount(7, 'data');
|
|
||||||
$this->callFilter('form.participant.index', ['data' => ['drop' => null]], ['form' => $form])
|
|
||||||
->assertJsonCount(1, 'data');
|
|
||||||
$this->callFilter('form.participant.index', ['data' => ['drop' => 'A']], ['form' => $form])
|
|
||||||
->assertJsonCount(2, 'data');
|
|
||||||
$this->callFilter('form.participant.index', ['data' => ['drop' => 'B']], ['form' => $form])
|
|
||||||
->assertJsonCount(4, 'data');
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testItFiltersParticipantsByRadioValue(): void
|
|
||||||
{
|
|
||||||
$this->login()->loginNami()->withoutExceptionHandling();
|
|
||||||
$form = Form::factory()->fields([$this->radioField('drop')->options(['A', 'B'])])
|
|
||||||
->has(Participant::factory()->data(['drop' => null])->count(1))
|
|
||||||
->has(Participant::factory()->data(['drop' => 'A'])->count(2))
|
|
||||||
->has(Participant::factory()->data(['drop' => 'B'])->count(4))
|
|
||||||
->create();
|
|
||||||
|
|
||||||
$this->callFilter('form.participant.index', ['data' => ['drop' => ParticipantFilterScope::$nan]], ['form' => $form])
|
|
||||||
->assertJsonCount(7, 'data');
|
|
||||||
$this->callFilter('form.participant.index', ['data' => ['drop' => 'A']], ['form' => $form])
|
|
||||||
->assertJsonCount(2, 'data');
|
|
||||||
$this->callFilter('form.participant.index', ['data' => ['drop' => 'B']], ['form' => $form])
|
|
||||||
->assertJsonCount(4, 'data');
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testItPresentsNamiField(): void
|
|
||||||
{
|
|
||||||
$this->login()->loginNami()->withoutExceptionHandling();
|
|
||||||
$form = Form::factory()
|
|
||||||
->has(Participant::factory()->data(['mitglieder' => [['id' => 393], ['id' => 394]]]))
|
|
||||||
->has(Participant::factory()->nr(393)->data(['mitglieder' => []]))
|
|
||||||
->has(Participant::factory()->nr(394)->data(['mitglieder' => []]))
|
|
||||||
->fields([
|
|
||||||
$this->namiField('mitglieder'),
|
|
||||||
])
|
|
||||||
->create();
|
|
||||||
|
|
||||||
$this->callFilter('form.participant.index', [], ['form' => $form])
|
|
||||||
->assertJsonPath('data.0.mitglieder_display', '393, 394');
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testItShowsRegisteredAtColumnAndAttribute(): void
|
|
||||||
{
|
|
||||||
Carbon::setTestNow(Carbon::parse('2023-03-05 06:00:00'));
|
|
||||||
$this->login()->loginNami()->withoutExceptionHandling();
|
|
||||||
$form = Form::factory()
|
|
||||||
->has(Participant::factory()->data(['vorname' => 'Max']))
|
|
||||||
->fields([
|
|
||||||
$this->textField('vorname')->name('Vorname'),
|
|
||||||
])
|
|
||||||
->create();
|
|
||||||
|
|
||||||
$this->callFilter('form.participant.index', [], ['form' => $form])
|
|
||||||
->assertJsonPath('data.0.vorname', 'Max')
|
|
||||||
->assertJsonPath('data.0.vorname_display', 'Max')
|
|
||||||
->assertJsonPath('data.0.created_at', '2023-03-05 06:00:00')
|
|
||||||
->assertJsonPath('data.0.created_at_display', '05.03.2023')
|
|
||||||
->assertJsonPath('meta.columns.1.name', 'Registriert am')
|
|
||||||
->assertJsonPath('meta.columns.1.id', 'created_at')
|
|
||||||
->assertJsonPath('meta.columns.1.display_attribute', 'created_at_display');
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testItShowsOnlyParentParticipantsWhenFilterEnabled(): void
|
|
||||||
{
|
|
||||||
$this->login()->loginNami()->withoutExceptionHandling();
|
|
||||||
$form = Form::factory()->create();
|
|
||||||
$participant = Participant::factory()
|
|
||||||
->has(Participant::factory()->for($form)->count(2), 'children')
|
|
||||||
->for($form)
|
|
||||||
->create();
|
|
||||||
|
|
||||||
$this->callFilter('form.participant.index', [], ['form' => $form])->assertJsonCount(3, 'data');
|
|
||||||
$this->callFilter('form.participant.index', [], ['form' => $form, 'parent' => -1])->assertJsonCount(1, 'data');
|
|
||||||
$this->callFilter('form.participant.index', [], ['form' => $form, 'parent' => $participant->id])->assertJsonCount(2, 'data');
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testItShowsChildrenCount(): void
|
|
||||||
{
|
|
||||||
$this->login()->loginNami()->withoutExceptionHandling();
|
|
||||||
$form = Form::factory()->create();
|
|
||||||
$participant = Participant::factory()
|
|
||||||
->has(Participant::factory()->for($form)->count(2), 'children')
|
|
||||||
->for($form)
|
|
||||||
->create();
|
|
||||||
Participant::factory()->for($form)->create();
|
|
||||||
|
|
||||||
$this->callFilter('form.participant.index', [], ['form' => $form, 'parent' => -1])
|
|
||||||
->assertJsonPath('data.0.children_count', 2)
|
|
||||||
->assertJsonPath('data.1.children_count', 0)
|
|
||||||
->assertJsonPath('data.0.links.children', route('form.participant.index', ['form' => $form, 'parent' => $participant->id]))
|
|
||||||
->assertJsonPath('meta.current_page', 1);
|
|
||||||
$this->callFilter('form.participant.index', [], ['form' => $form, 'parent' => $participant->id])->assertJsonPath('data.0.children_count', 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testItShowsPreventionState(): void
|
|
||||||
{
|
|
||||||
$this->login()->loginNami()->withoutExceptionHandling();
|
|
||||||
$participant = Participant::factory()->data(['vorname' => 'Max'])
|
|
||||||
->for(Member::factory()->defaults()->state(['efz' => null]))
|
|
||||||
->for(Form::factory())
|
|
||||||
->create();
|
|
||||||
|
|
||||||
$this->callFilter('form.participant.index', [], ['form' => $participant->form])
|
|
||||||
->assertJsonPath('data.0.prevention_items.0.letter', 'F')
|
|
||||||
->assertJsonPath('data.0.prevention_items.0.value', false)
|
|
||||||
->assertJsonPath('data.0.prevention_items.0.tooltip', 'erweitertes Führungszeugnis nicht vorhanden');
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -39,6 +39,18 @@ class PreventionTest extends TestCase
|
||||||
$this->assertEquals(now()->format('Y-m-d'), $participant->fresh()->last_remembered_at->format('Y-m-d'));
|
$this->assertEquals(now()->format('Y-m-d'), $participant->fresh()->last_remembered_at->format('Y-m-d'));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testItDoesntRememberPastEvents(): void
|
||||||
|
{
|
||||||
|
Mail::fake();
|
||||||
|
$form = $this->createForm();
|
||||||
|
$participant = $this->createParticipant($form);
|
||||||
|
$form->update(['from' => now()->subDay()]);
|
||||||
|
|
||||||
|
PreventionRememberAction::run();
|
||||||
|
|
||||||
|
$this->assertNull($participant->fresh()->last_remembered_at);
|
||||||
|
}
|
||||||
|
|
||||||
public function testItDoesntRememberWhenConditionDoesntMatch(): void
|
public function testItDoesntRememberWhenConditionDoesntMatch(): void
|
||||||
{
|
{
|
||||||
Mail::fake();
|
Mail::fake();
|
||||||
|
|
|
@ -21,7 +21,6 @@ use Zoomyboy\TableDocument\TestsExcelDocuments;
|
||||||
|
|
||||||
class TestCase extends BaseTestCase
|
class TestCase extends BaseTestCase
|
||||||
{
|
{
|
||||||
use CreatesApplication;
|
|
||||||
use TestsInertia;
|
use TestsInertia;
|
||||||
use MakesHttpCalls;
|
use MakesHttpCalls;
|
||||||
use TestsExcelDocuments;
|
use TestsExcelDocuments;
|
||||||
|
|
Loading…
Reference in New Issue