<?php

namespace App\Models;

use App\Models\Concerns\Reportable;
use Illuminate\Database\Eloquent\Builder;
use Illuminate\Database\Eloquent\Collection;

/**
 * A data table object (referencing revisions)
 *
 * @property int $id
 * @property \Carbon\Carbon $created_at
 * @property \Carbon\Carbon $updated_at
 * @property int $owner_id
 * @property int $ancestor_id
 * @property int $revision_id
 * @property string $name
 * @property string $title
 * @property string $description
 * @property string $license
 * @property string $origin
 * @property int $visits
 * @property-read string $viewRoute
 * @property-read string $draftRoute
 * @property-read string $settingsRoute
 * @property-read string $deleteRoute
 * @property-read string $draftSessionKey
 * @property-read string $draftDiscardRoute
 * @property-read string $draftUpdateRoute
 * @property-read string $draftSubmitRoute
 * @property-read User $owner
 * @property-read Table $parentTable
 * @property-read Table[]|Collection $forks
 * @property-read Revision[]|Collection $revisions
 * @property-read Revision $revision
 * @property-read Proposal[]|Collection $proposal
 * @property-read TableComment[]|Collection $comments
 * @property-read User[]|Collection $favouritingUsers
 * @property-read User[]|Collection $discussionFollowers
 */
class Table extends BaseModel
{
    use Reportable;
    protected $guarded = [];

    protected $touches = ['owner'];

    public function countVisit()
    {
        // Temporarily disable timestamps to avoid touching updated_at
        $oldt = $this->touches;
        $this->touches = [];
        $this->timestamps = false;

        $this->increment('visits');

        $this->timestamps = true;
        $this->touches = $oldt;
    }

    /** Get owner from the instance cache (use when building table lists) */
    public function cachedOwner()
    {
        return User::getCached($this->owner_id);
    }

    /** Owning user */
    public function owner()
    {
        return $this->belongsTo(User::class, 'owner_id');
    }

    /** Fork precursor */
    public function parentTable()
    {
        return $this->belongsTo(Table::class, 'ancestor_id');
    }

    /** Fork descendants */
    public function forks()
    {
        return $this->hasMany(Table::class, 'ancestor_id');
    }

    /** All related revisions (this may include merged revisions) */
    public function revisions()
    {
        return $this->belongsToMany(Revision::class, 'table_revision_pivot');
    }

    /** Active revision */
    public function revision()
    {
        return $this->belongsTo(Revision::class, 'revision_id');
    }

    /** Proposals submitted to this table */
    public function proposals()
    {
        return $this->hasMany(Proposal::class);
    }

    /** User-submitted comments */
    public function comments()
    {
        return $this->hasMany(TableComment::class);
    }

    /** Users favouriting this table */
    public function favouritingUsers()
    {
        return $this->belongsToMany(User::class, 'table_favourites');
    }

    /**
     * Relation to the pivot table, for favourite counting
     */
    public function favourites()
    {
        return $this->hasMany(TableFavouritePivot::class);
    }

    /** Users to notify about comments */
    public function discussionFollowers()
    {
        return $this->belongsToMany(User::class, 'discussion_follows');
    }

    public function __get($name)
    {
        if (ends_with($name, 'Route')) {
            $arg = [
                'user' => $this->cachedOwner()->name,
                'table' => $this->name
            ];

            switch ($name) {
                case 'viewRoute':     return route('table.view', $arg);
                case 'settingsRoute': return route('table.conf', $arg);
                case 'draftRoute':    return route('table.draft', $arg);
                case 'deleteRoute':   return route('table.delete', $arg);
                case 'draftDiscardRoute':    return route('table.draft-discard', $arg);
                case 'draftUpdateRoute':    return route('table.draft-update', $arg);
                case 'draftSubmitRoute':    return route('table.draft-submit', $arg);
            }
        }

        if ($name == 'draftSessionKey') return "proposal.{$this->id}";

        return parent::__get($name);
    }

    public function getDraftRoute($tab=null)
    {
        return route('table.draft', [
            'user' => $this->cachedOwner()->name,
            'table' => $this->name,
            'tab' => $tab,
        ]);
    }

    public function scopeForList(Builder $query)
    {
        return $query->with('revision:id,row_count')->with('owner:id,name,title')
            ->withCount(['favourites', 'forks', 'revisions', 'proposals']);
    }
}