558 lines
16 KiB
PHP
558 lines
16 KiB
PHP
<?php
|
|
|
|
namespace App\Member;
|
|
|
|
use App\Confession;
|
|
use App\Country;
|
|
use App\Course\Models\CourseMember;
|
|
use App\Gender;
|
|
use App\Group;
|
|
use App\Invoice\BillKind;
|
|
use App\Invoice\Models\InvoicePosition;
|
|
use App\Nami\HasNamiField;
|
|
use App\Nationality;
|
|
use App\Payment\Subscription;
|
|
use App\Pdf\Sender;
|
|
use App\Region;
|
|
use App\Setting\NamiSettings;
|
|
use Carbon\Carbon;
|
|
use Cviebrock\EloquentSluggable\Sluggable;
|
|
use Illuminate\Database\Eloquent\Builder;
|
|
use Illuminate\Database\Eloquent\Factories\HasFactory;
|
|
use Illuminate\Database\Eloquent\Model;
|
|
use Illuminate\Database\Eloquent\Relations\BelongsTo;
|
|
use Illuminate\Database\Eloquent\Relations\HasMany;
|
|
use Illuminate\Notifications\Notifiable;
|
|
use Laravel\Scout\Searchable;
|
|
use Sabre\VObject\Component\VCard;
|
|
use Sabre\VObject\Reader;
|
|
use Spatie\LaravelData\Lazy;
|
|
use Zoomyboy\Osm\Address;
|
|
use Zoomyboy\Osm\Coordinate;
|
|
use Zoomyboy\Osm\Geolocatable;
|
|
use Zoomyboy\Osm\HasGeolocation;
|
|
use Zoomyboy\Phone\HasPhoneNumbers;
|
|
use App\Prevention\Enums\Prevention;
|
|
use Database\Factories\Member\MemberFactory;
|
|
|
|
/**
|
|
* @property string $subscription_name
|
|
* @property int $pending_payment
|
|
*/
|
|
class Member extends Model implements Geolocatable
|
|
{
|
|
use Notifiable;
|
|
use HasNamiField;
|
|
/** @use HasFactory<MemberFactory> */
|
|
use HasFactory;
|
|
use Sluggable;
|
|
use Searchable;
|
|
use HasPhoneNumbers;
|
|
use HasGeolocation;
|
|
|
|
/**
|
|
* @var array<string, string>
|
|
*/
|
|
public $guarded = [];
|
|
|
|
/**
|
|
* @var array<int, string>
|
|
*/
|
|
public static array $namiFields = ['firstname', 'lastname', 'joined_at', 'birthday', 'send_newspaper', 'address', 'zip', 'location', 'nickname', 'other_country', 'further_address', 'main_phone', 'mobile_phone', 'work_phone', 'fax', 'email', 'email_parents', 'gender_id', 'confession_id', 'region_id', 'country_id', 'fee_id', 'nationality_id', 'slug', 'subscription_id', 'keepdata'];
|
|
|
|
/**
|
|
* @var array<string, string>
|
|
*/
|
|
public $casts = [
|
|
'pending_payment' => 'integer',
|
|
'send_newspaper' => 'boolean',
|
|
'gender_id' => 'integer',
|
|
'way_id' => 'integer',
|
|
'country_id' => 'integer',
|
|
'region_id' => 'integer',
|
|
'confession_id' => 'integer',
|
|
'nami_id' => 'integer',
|
|
'has_svk' => 'boolean',
|
|
'has_vk' => 'boolean',
|
|
'multiply_pv' => 'boolean',
|
|
'multiply_more_pv' => 'boolean',
|
|
'is_leader' => 'boolean',
|
|
'keepdata' => 'boolean',
|
|
'bill_kind' => BillKind::class,
|
|
'mitgliedsnr' => 'integer',
|
|
|
|
'try_created_at' => 'datetime',
|
|
'recertified_at' => 'datetime',
|
|
'joined_at' => 'datetime',
|
|
'birthday' => 'datetime',
|
|
'efz' => 'datetime',
|
|
'ps_at' => 'datetime',
|
|
'more_ps_at' => 'datetime',
|
|
'without_education_at' => 'datetime',
|
|
'without_efz_at' => 'datetime',
|
|
];
|
|
|
|
/**
|
|
* @return array<int, string>
|
|
*/
|
|
public function phoneNumbers(): array
|
|
{
|
|
return ['main_phone', 'mobile_phone', 'work_phone', 'children_phone', 'fax'];
|
|
}
|
|
|
|
/**
|
|
* @return SluggableConfig
|
|
*/
|
|
public function sluggable(): array
|
|
{
|
|
return [
|
|
'slug' => ['source' => ['firstname', 'lastname']],
|
|
];
|
|
}
|
|
|
|
// ---------------------------------- Actions ----------------------------------
|
|
public function syncVersion(): void
|
|
{
|
|
$version = app(NamiSettings::class)->login()->member($this->group->nami_id, $this->nami_id)->version;
|
|
|
|
$this->update(['version' => $version]);
|
|
}
|
|
|
|
// ----------------------------------- Getters -----------------------------------
|
|
public function getFullnameAttribute(): string
|
|
{
|
|
return $this->firstname . ' ' . $this->lastname;
|
|
}
|
|
|
|
public function getPreferredPhoneAttribute(): ?string
|
|
{
|
|
if ($this->mobile_phone) {
|
|
return $this->mobile_phone;
|
|
}
|
|
|
|
if ($this->main_phone) {
|
|
return $this->main_phone;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
public function getPreferredEmailAttribute(): ?string
|
|
{
|
|
if ($this->email) {
|
|
return $this->email;
|
|
}
|
|
|
|
if ($this->email_parents) {
|
|
return $this->email_parents;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
public function getEtagAttribute(): string
|
|
{
|
|
return $this->updated_at->timestamp . '_' . $this->version;
|
|
}
|
|
|
|
public function getFullAddressAttribute(): string
|
|
{
|
|
return $this->address && $this->zip && $this->location
|
|
? $this->address . ', ' . $this->zip . ' ' . $this->location
|
|
: '';
|
|
}
|
|
|
|
public function getEfzLink(): ?string
|
|
{
|
|
return $this->address && $this->zip && $this->location && $this->birthday
|
|
? route('efz', ['member' => $this])
|
|
: null;
|
|
}
|
|
|
|
public function getNamiFeeId(): ?int
|
|
{
|
|
if (!$this->subscription) {
|
|
return null;
|
|
}
|
|
|
|
return $this->subscription->fee->nami_id;
|
|
}
|
|
|
|
public function isLeader(): bool
|
|
{
|
|
return $this->leaderMemberships->count() > 0;
|
|
}
|
|
|
|
public function getAge(): ?int
|
|
{
|
|
return $this->birthday ? intval($this->birthday->diffInYears(now())) : null;
|
|
}
|
|
|
|
protected function getAusstand(): int
|
|
{
|
|
return (int) $this->invoicePositions()->whereHas('invoice', fn ($query) => $query->whereNeedsPayment())->sum('price');
|
|
}
|
|
|
|
// ---------------------------------- Relations ----------------------------------
|
|
/**
|
|
* @return BelongsTo<Country, self>
|
|
*/
|
|
public function country(): BelongsTo
|
|
{
|
|
return $this->belongsTo(Country::class);
|
|
}
|
|
|
|
/**
|
|
* @return BelongsTo<Gender, self>
|
|
*/
|
|
public function gender(): BelongsTo
|
|
{
|
|
return $this->belongsTo(Gender::class);
|
|
}
|
|
|
|
/**
|
|
* @return BelongsTo<Region, self>
|
|
*/
|
|
public function region(): BelongsTo
|
|
{
|
|
return $this->belongsTo(Region::class)->withDefault([
|
|
'name' => '-- kein --',
|
|
'nami_id' => null,
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* @return HasMany<InvoicePosition>
|
|
*/
|
|
public function invoicePositions(): HasMany
|
|
{
|
|
return $this->hasMany(InvoicePosition::class);
|
|
}
|
|
|
|
/**
|
|
* @return BelongsTo<Confession, self>
|
|
*/
|
|
public function confession(): BelongsTo
|
|
{
|
|
return $this->belongsTo(Confession::class);
|
|
}
|
|
|
|
/**
|
|
* @return BelongsTo<Nationality, self>
|
|
*/
|
|
public function nationality(): BelongsTo
|
|
{
|
|
return $this->belongsTo(Nationality::class);
|
|
}
|
|
|
|
/**
|
|
* @return BelongsTo<Subscription, self>
|
|
*/
|
|
public function subscription(): BelongsTo
|
|
{
|
|
return $this->belongsTo(Subscription::class);
|
|
}
|
|
|
|
/**
|
|
* @return BelongsTo<Group, self>
|
|
*/
|
|
public function group(): BelongsTo
|
|
{
|
|
return $this->belongsTo(Group::class);
|
|
}
|
|
|
|
/**
|
|
* @return HasMany<CourseMember>
|
|
*/
|
|
public function courses(): HasMany
|
|
{
|
|
return $this->hasMany(CourseMember::class);
|
|
}
|
|
|
|
/**
|
|
* @return HasMany<Membership>
|
|
*/
|
|
public function memberships(): HasMany
|
|
{
|
|
return $this->hasMany(Membership::class);
|
|
}
|
|
|
|
/**
|
|
* @return HasMany<Membership>
|
|
*/
|
|
public function leaderMemberships(): HasMany
|
|
{
|
|
return $this->ageGroupMemberships()->isLeader()->active();
|
|
}
|
|
|
|
/**
|
|
* @return HasMany<Membership>
|
|
*/
|
|
public function ageGroupMemberships(): HasMany
|
|
{
|
|
return $this->memberships()->isAgeGroup()->active();
|
|
}
|
|
|
|
public static function booted()
|
|
{
|
|
static::deleting(function (self $model): void {
|
|
$model->memberships->each->delete();
|
|
$model->courses->each->delete();
|
|
$model->invoicePositions->each(function ($position) {
|
|
$position->delete();
|
|
});
|
|
});
|
|
}
|
|
|
|
// ---------------------------------- Scopes -----------------------------------
|
|
/**
|
|
* @param Builder<self> $query
|
|
*
|
|
* @return Builder<self>
|
|
*/
|
|
public function scopeOrdered(Builder $query): Builder
|
|
{
|
|
return $query->orderByRaw('lastname, firstname');
|
|
}
|
|
|
|
/**
|
|
* @param Builder<self> $query
|
|
*
|
|
* @return Builder<self>
|
|
*/
|
|
public function scopeWithPendingPayment(Builder $query): Builder
|
|
{
|
|
return $query->addSelect([
|
|
'pending_payment' => InvoicePosition::selectRaw('SUM(price)')
|
|
->whereColumn('invoice_positions.member_id', 'members.id')
|
|
->whereHas('invoice', fn ($query) => $query->whereNeedsPayment()),
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* @param Builder<self> $query
|
|
*
|
|
* @return Builder<self>
|
|
*/
|
|
public function scopeWhereHasPendingPayment(Builder $query): Builder
|
|
{
|
|
return $query->whereHas('invoicePositions', fn ($q) => $q->whereHas('invoice', fn ($q) => $q->whereNeedsPayment()));
|
|
}
|
|
|
|
/**
|
|
* @param Builder<self> $query
|
|
*
|
|
* @return Builder<self>
|
|
*/
|
|
public function scopePayable(Builder $query): Builder
|
|
{
|
|
return $query->where('bill_kind', '!=', null)->where('subscription_id', '!=', null);
|
|
}
|
|
|
|
/**
|
|
* @return array<int, Prevention>
|
|
*/
|
|
public function preventions(?Carbon $date = null): array
|
|
{
|
|
$date = $date ?: now();
|
|
|
|
/** @var array<int, Prevention> */
|
|
$preventions = [];
|
|
|
|
if ($this->efz === null || $this->efz->diffInYears($date) >= 5) {
|
|
$preventions[] = Prevention::EFZ;
|
|
}
|
|
|
|
if (!$this->has_vk) {
|
|
$preventions[] = Prevention::VK;
|
|
}
|
|
|
|
if ($this->more_ps_at === null) {
|
|
if ($this->ps_at === null) {
|
|
$preventions[] = Prevention::PS;
|
|
} else if ($this->ps_at->diffInYears($date) >= 5) {
|
|
$preventions[] = Prevention::MOREPS;
|
|
}
|
|
} else {
|
|
if ($this->more_ps_at === null || $this->more_ps_at->diffInYears($date) >= 5) {
|
|
$preventions[] = Prevention::MOREPS;
|
|
}
|
|
}
|
|
|
|
return $preventions;
|
|
}
|
|
|
|
|
|
/**
|
|
* @param Builder<self> $query
|
|
*
|
|
* @return Builder<self>
|
|
*/
|
|
public function scopeForDashboard(Builder $query): Builder
|
|
{
|
|
return $query->selectRaw('SUM(id)');
|
|
}
|
|
|
|
/**
|
|
* @param Builder<self> $query
|
|
*
|
|
* @return Builder<self>
|
|
*/
|
|
public function scopeWhereCurrentGroup(Builder $query): Builder
|
|
{
|
|
$group = app(NamiSettings::class)->localGroup();
|
|
|
|
if (!$group) {
|
|
return $query;
|
|
}
|
|
|
|
return $query->where('group_id', $group->id);
|
|
}
|
|
|
|
public static function fromVcard(string $url, string $data): self
|
|
{
|
|
$settings = app(NamiSettings::class);
|
|
$card = Reader::read($data);
|
|
[$lastname, $firstname] = $card->N->getParts();
|
|
[$deprecated1, $deprecated2, $address, $location, $region, $zip, $country] = $card->ADR->getParts();
|
|
|
|
return new self([
|
|
'joined_at' => now(),
|
|
'send_newspaper' => false,
|
|
'firstname' => $firstname,
|
|
'lastname' => $lastname,
|
|
'birthday' => Carbon::createFromFormat('Ymd', $card->BDAY->getValue()),
|
|
'slug' => pathinfo($url, PATHINFO_FILENAME),
|
|
'address' => $address,
|
|
'zip' => $zip,
|
|
'location' => $location,
|
|
'group_id' => $settings->default_group_id,
|
|
'nationality_id' => Nationality::firstWhere('name', 'deutsch')->id,
|
|
'subscription_id' => Subscription::firstWhere('name', 'Voll')->id,
|
|
]);
|
|
}
|
|
|
|
public function toVcard(): Vcard
|
|
{
|
|
$card = new VCard([
|
|
'VERSION' => '3.0',
|
|
'FN' => $this->fullname,
|
|
'N' => [$this->lastname, $this->firstname, '', '', ''],
|
|
'CATEGORIES' => 'Scoutrobot',
|
|
'UID' => $this->slug,
|
|
]);
|
|
|
|
if ($this->birthday) {
|
|
$card->add('BDAY', $this->birthday->format('Ymd'));
|
|
}
|
|
|
|
if ($this->main_phone) {
|
|
$card->add('TEL', $this->main_phone, ['type' => 'voice']);
|
|
}
|
|
|
|
if ($this->mobile_phone) {
|
|
$card->add('TEL', $this->mobile_phone, ['type' => 'work']);
|
|
}
|
|
|
|
if ($this->children_phone) {
|
|
$card->add('TEL', $this->children_phone, ['type' => 'cell']);
|
|
}
|
|
|
|
if ($this->email) {
|
|
$card->add('EMAIL', $this->email, ['type' => 'internet']);
|
|
}
|
|
|
|
if ($this->email_parents) {
|
|
$card->add('EMAIL', $this->email_parents, ['type' => 'aol']);
|
|
}
|
|
|
|
$card->add('ADR', [
|
|
'',
|
|
'',
|
|
$this->address ?: '',
|
|
$this->location ?: '',
|
|
$this->region?->name ?: '',
|
|
$this->zip ?: '',
|
|
$this->country?->name ?: '',
|
|
]);
|
|
|
|
return $card;
|
|
}
|
|
|
|
public function toSender(): Sender
|
|
{
|
|
return Sender::from([
|
|
'name' => $this->fullname,
|
|
'address' => $this->address,
|
|
'zipLocation' => $this->zip . ' ' . $this->location,
|
|
'mglnr' => Lazy::create(fn () => 'Mglnr.: ' . $this->nami_id),
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* @return array<int, array{id: int, name: string}>
|
|
*/
|
|
public static function forSelect(): array
|
|
{
|
|
return static::select(['id', 'firstname', 'lastname'])->get()->map(fn ($member) => ['id' => $member->id, 'name' => $member->fullname])->toArray();
|
|
}
|
|
|
|
// -------------------------------- Geolocation --------------------------------
|
|
// *****************************************************************************
|
|
public function fillCoordinate(Coordinate $coordinate): void
|
|
{
|
|
$this->updateQuietly(['lat' => $coordinate->lat, 'lon' => $coordinate->lon]);
|
|
}
|
|
|
|
public function getAddressForGeolocation(): ?Address
|
|
{
|
|
return new Address($this->address, $this->zip, $this->location);
|
|
}
|
|
|
|
public function destroyCoordinate(): void
|
|
{
|
|
$this->updateQuietly([
|
|
'lat' => null,
|
|
'lon' => null,
|
|
]);
|
|
}
|
|
|
|
public function needsGeolocationUpdate(): bool
|
|
{
|
|
return $this->getOriginal('address') !== $this->address
|
|
|| $this->getOriginal('zip') !== $this->zip
|
|
|| $this->getOriginal('location') !== $this->location;
|
|
}
|
|
|
|
// --------------------------------- Searching ---------------------------------
|
|
// *****************************************************************************
|
|
|
|
/**
|
|
* Get the indexable data array for the model.
|
|
*
|
|
* @return array<string, mixed>
|
|
*/
|
|
public function toSearchableArray()
|
|
{
|
|
return [
|
|
'address' => $this->fullAddress,
|
|
'fullname' => $this->fullname,
|
|
'firstname' => $this->firstname,
|
|
'lastname' => $this->lastname,
|
|
'birthday' => $this->birthday?->format('Y-m-d'),
|
|
'ausstand' => $this->getAusstand(),
|
|
'bill_kind' => $this->bill_kind?->value,
|
|
'group_id' => $this->group->id,
|
|
'group_name' => $this->group->inner_name ?: $this->group->name,
|
|
'links' => [
|
|
'show' => route('member.show', ['member' => $this], false),
|
|
'edit' => route('member.edit', ['member' => $this], false),
|
|
],
|
|
'age_group_icon' => $this->ageGroupMemberships->first()?->subactivity->slug,
|
|
'is_leader' => $this->leaderMemberships()->count() > 0,
|
|
'memberships' => $this->memberships()->active()->get()
|
|
->map(fn ($membership) => [...$membership->only('activity_id', 'subactivity_id'), 'both' => $membership->activity_id . '|' . $membership->subactivity_id, 'with_group' => $membership->group_id . '|' . $membership->activity_id . '|' . $membership->subactivity_id]),
|
|
];
|
|
}
|
|
}
|