adrema/app/Member/Member.php

397 lines
12 KiB
PHP

<?php
namespace App\Member;
use App\Activity;
use App\Bill\BillKind;
use App\Confession;
use App\Country;
use App\Course\Models\CourseMember;
use App\Group;
use App\Nationality;
use App\Payment\Payment;
use App\Payment\Subscription;
use App\Region;
use App\Setting\NamiSettings;
use App\Subactivity;
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 Sabre\VObject\Component\VCard;
use Zoomyboy\LaravelNami\Api;
use Zoomyboy\LaravelNami\Data\MembershipEntry;
/**
* @property string $subscription_name
* @property int $pending_payment
* @property bool $is_confirmed
* @property string $age_group_icon
* @property bool $is_leader
* @property \Carbon\Carbon $try_created_at
*/
class Member extends Model
{
use Notifiable;
use HasFactory;
use Sluggable;
public $guarded = [];
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'];
public $dates = ['try_created_at', 'joined_at', 'birthday'];
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',
'is_confirmed' => 'boolean',
'has_svk' => 'boolean',
'has_vk' => 'boolean',
'multiply_pv' => 'boolean',
'multiply_more_pv' => 'boolean',
'is_leader' => 'boolean',
];
public function sluggable(): array
{
return [
'slug' => ['source' => ['firstname', 'lastname']],
];
}
public function scopeSearch(Builder $q, ?string $text): Builder
{
if (is_null($text)) {
return $q;
}
return $q->where('firstname', 'LIKE', '%'.$text.'%')
->orWhere('lastname', 'LIKE', '%'.$text.'%')
->orWhere('address', 'LIKE', '%'.$text.'%')
->orWhere('zip', 'LIKE', '%'.$text.'%')
->orWhere('location', 'LIKE', '%'.$text.'%');
}
// ---------------------------------- Actions ----------------------------------
public function syncVersion(): void
{
$version = app(NamiSettings::class)->login()->member($this->group->nami_id, $this->nami_id)['version'];
$this->update(['version' => $version]);
}
public function createPayment(array $attributes): void
{
$this->payments()->create(array_merge($attributes, [
'last_remembered_at' => now(),
]));
}
// ----------------------------------- 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 getEtagAttribute(): string
{
return $this->updated_at->timestamp.'_'.$this->version;
}
public function getFullAddressAttribute(): string
{
return $this->address.', '.$this->zip.' '.$this->location;
}
public function getEfzLink(): ?string
{
return $this->isLeader()
? route('efz', ['member' => $this])
: null;
}
public function getHasNamiAttribute(): bool
{
return null !== $this->nami_id;
}
/**
* @return Collection<int, MembershipEntry>
*/
public function getNamiMemberships(Api $api): array
{
return $api->group($this->group->nami_id)->member($this->nami_id)->memberships();
}
public function getNamiFeeId(): ?int
{
if (!$this->subscription) {
return null;
}
return $this->subscription->fee->nami_id;
}
public function isLeader(): bool
{
return $this->memberships()->whereHas('activity', fn (Builder $query) => $query->where('has_efz', true))->exists();
}
public function getAge(): int
{
return $this->birthday->diffInYears(now());
}
// ---------------------------------- Relations ----------------------------------
public function country(): BelongsTo
{
return $this->belongsTo(Country::class);
}
public function gender(): BelongsTo
{
return $this->belongsTo(\App\Gender::class);
}
public function region(): BelongsTo
{
return $this->belongsTo(Region::class);
}
public function confession(): BelongsTo
{
return $this->belongsTo(Confession::class);
}
public function payments(): HasMany
{
return $this->hasMany(Payment::class)->orderBy('nr');
}
public function nationality(): BelongsTo
{
return $this->belongsTo(Nationality::class);
}
public function memberships(): HasMany
{
return $this->hasMany(Membership::class);
}
public function subscription(): BelongsTo
{
return $this->belongsTo(Subscription::class);
}
public function billKind(): BelongsTo
{
return $this->belongsTo(BillKind::class);
}
public function group(): BelongsTo
{
return $this->belongsTo(Group::class);
}
public function firstActivity(): BelongsTo
{
return $this->belongsTo(Activity::class, 'first_activity_id');
}
public function firstSubActivity(): BelongsTo
{
return $this->belongsTo(Subactivity::class, 'first_subactivity_id');
}
public function courses(): HasMany
{
return $this->hasMany(CourseMember::class);
}
public static function booted()
{
static::deleting(function (self $model): void {
$model->payments->each->delete();
});
}
// ---------------------------------- Scopes -----------------------------------
public function scopeOrdered(Builder $q): Builder
{
return $q->orderByRaw('lastname, firstname');
}
public function scopeSlangOrdered(Builder $q): Builder
{
return $q->orderByRaw('firstname, lastname');
}
public function scopeWithIsConfirmed(Builder $q): Builder
{
return $q->selectSub('DATEDIFF(NOW(), IFNULL(confirmed_at, DATE_SUB(NOW(), INTERVAL 3 YEAR))) < 712', 'is_confirmed');
}
public function scopeWithSubscriptionName(Builder $q): Builder
{
return $q->addSelect([
'subscription_name' => Subscription::select('name')->whereColumn('subscriptions.id', 'members.subscription_id')->limit(1),
]);
}
public function scopeWithPendingPayment(Builder $q): Builder
{
return $q->addSelect([
'pending_payment' => Payment::selectRaw('SUM(subscriptions.amount)')
->whereColumn('payments.member_id', 'members.id')
->whereNeedsPayment()
->join('subscriptions', 'subscriptions.id', 'payments.subscription_id'),
]);
}
public function scopeWithAgeGroup(Builder $q): Builder
{
return $q->addSelect([
'age_group_icon' => Subactivity::select('slug')
->join('memberships', 'memberships.subactivity_id', 'subactivities.id')
->where('subactivities.is_age_group', true)
->whereColumn('memberships.member_id', 'members.id')
->limit(1),
]);
}
public function scopeWithIsLeader(Builder $q): Builder
{
return $q->selectSub('EXISTS(SELECT memberships.id FROM memberships INNER JOIN activities ON activities.id=memberships.activity_id INNER JOIN subactivities ON subactivities.id=memberships.subactivity_id WHERE members.id=memberships.member_id AND subactivities.is_age_group=1 AND activities.has_efz=1)', 'is_leader');
}
public function scopeWhereHasPendingPayment(Builder $q): Builder
{
return $q->whereHas('payments', function (Builder $q): void {
$q->whereNeedsPayment();
});
}
public function scopeWhereAusstand(Builder $q): Builder
{
return $q->whereHas('payments', function ($q) {
return $q->whereHas('status', fn ($q) => $q->where('is_remember', true));
});
}
public function scopePayable(Builder $q): Builder
{
return $q->where('bill_kind_id', '!=', null)->where('subscription_id', '!=', null);
}
public function scopeWhereNoPayment(Builder $q, int $year): Builder
{
return $q->whereDoesntHave('payments', function (Builder $q) use ($year) {
$q->where('nr', '=', $year);
});
}
public function scopeForDashboard(Builder $q): Builder
{
return $q->selectRaw('SUM(id)');
}
public function scopeFilter(Builder $q, array $filter): Builder
{
if (true === data_get($filter, 'ausstand', false)) {
$q->whereAusstand();
}
if (data_get($filter, 'bill_kind', false)) {
$q->where('bill_kind_id', $filter['bill_kind']);
}
if (data_get($filter, 'subactivity_id', false) || data_get($filter, 'activity_id', false)) {
$q->whereHas('memberships', function ($q) use ($filter) {
if (data_get($filter, 'subactivity_id', false)) {
$q->where('subactivity_id', $filter['subactivity_id']);
}
if (data_get($filter, 'activity_id', false)) {
$q->where('activity_id', $filter['activity_id']);
}
});
}
return $q;
}
public function scopeEndingTries(Builder $q): Builder
{
return $q->whereHas('memberships', fn ($q) => $q
->where('created_at', '<=', now()->subWeeks(7))
->whereHas('activity', fn ($q) => $q->where('is_try', true)))
->addSelect([
'try_created_at' => Membership::select('created_at')
->whereColumn('memberships.member_id', 'members.id')
->join('activities', 'activities.id', 'memberships.activity_id')
->where('activities.is_try', true),
]);
}
public function toVcard(): VCard
{
$card = new VCard([
'FN' => $this->fullname,
'TEL' => $this->mobile_phone,
'N' => [$this->lastname, $this->firstname, '', '', ''],
'BDAY' => $this->birthday->format('Ymd'),
]);
if ($this->preferred_phone) {
$card->add('TEL', $this->preferred_phone, ['type' => 'pref,voice']);
}
if ($this->mobile_phone) {
$card->add('TEL', $this->mobile_phone, ['type' => 'cell']);
}
if ($this->work_phone) {
$card->add('TEL', $this->work_phone, ['type' => 'work']);
}
if ($this->email) {
$card->add('EMAIL', $this->email, ['type' => 'pref']);
}
if ($this->email_parents) {
$card->add('EMAIL', $this->email_parents, ['type' => 'internet']);
}
$card->add('ADDR', [
$this->fullname,
$this->fullAddress,
$this->address ?: '',
$this->location ?: '',
$this->region?->name ?: '',
$this->zip ?: '',
$this->country?->name ?: '',
], ['type' => 'postal']);
return $card;
}
}