*/ 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']; /** * @var array */ public $dates = ['try_created_at', 'joined_at', 'birthday']; /** * @var array */ 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', ]; /** * @return array}> */ 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 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; } public function getEfzLink(): ?string { return $this->isLeader() ? route('efz', ['member' => $this]) : null; } public function getHasNamiAttribute(): bool { return null !== $this->nami_id; } /** * @return Collection */ 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()->isLeader()->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)->withDefault([ 'name' => '-- kein --', 'nami_id' => null, ]); } 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 courses(): HasMany { return $this->hasMany(CourseMember::class); } /** * @return HasMany */ public function leaderMemberships(): HasMany { return $this->ageGroupMemberships()->isLeader(); } /** * @return HasMany */ public function ageGroupMemberships(): HasMany { return $this->memberships()->isAgeGroup(); } public static function booted() { static::deleting(function (self $model): void { $model->payments->each->delete(); $model->memberships->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 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 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 static function fromVcard(string $url, string $data): static { $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 static([ '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, '', '', ''], 'BDAY' => $this->birthday->format('Ymd'), 'CATEGORIES' => 'Scoutrobot', 'UID' => $this->slug, ]); 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), ]); } }