'boolean',
        'has_tasks' => 'boolean',
        'client_enable_auto_bill' => 'boolean',
        'has_expenses' => 'boolean',
    ];
    // used for custom invoice numbers
    /**
     * @var array
     */
    public static $patternFields = [
        'counter',
        'clientCounter',
        'clientIdNumber',
        'clientCustom1',
        'clientCustom2',
        'userId',
        'year',
        'date:',
    ];
    /**
     * @var array
     */
    public static $requestFields = [
        'invoice_number',
        'invoice_date',
        'due_date',
        'po_number',
        'discount',
        'partial',
    ];
    public static $statusClasses = [
        INVOICE_STATUS_SENT => 'info',
        INVOICE_STATUS_VIEWED => 'warning',
        INVOICE_STATUS_APPROVED => 'success',
        INVOICE_STATUS_PARTIAL => 'primary',
        INVOICE_STATUS_PAID => 'success',
    ];
    /**
     * @return array
     */
    public static function getImportColumns()
    {
        return [
            'name',
            'email',
            'invoice_number',
            'po_number',
            'invoice_date',
            'due_date',
            'paid',
            'terms',
            'public_notes',
            'private_notes',
            'item_product',
            'item_notes',
            'item_quantity',
            'item_cost',
            'item_tax1',
            'item_tax2',
        ];
    }
    /**
     * @return array
     */
    public static function getImportMap()
    {
        return [
            'number^po' => 'invoice_number',
            'client|organization' => 'name',
            'email' => 'email',
            'paid^date' => 'paid',
            'invoice date|create date' => 'invoice_date',
            'po number' => 'po_number',
            'due date' => 'due_date',
            'terms' => 'terms',
            'public notes' => 'public_notes',
            'private notes' => 'private_notes',
            'description' => 'item_notes',
            'quantity|qty' => 'item_quantity',
            'amount|cost' => 'item_cost',
            'product' => 'item_product',
            'tax' => 'item_tax1',
        ];
    }
    /**
     * @return string
     */
    public function getRoute()
    {
        if ($this->is_recurring) {
            $entityType = 'recurring_invoice';
        } else {
            $entityType = $this->getEntityType();
        }
        return "/{$entityType}s/{$this->public_id}/edit";
    }
    /**
     * @return mixed
     */
    public function getDisplayName()
    {
        return $this->is_recurring ? trans('texts.recurring') : $this->invoice_number;
    }
    /**
     * @return bool
     */
    public function affectsBalance()
    {
        return $this->isType(INVOICE_TYPE_STANDARD) && ! $this->is_recurring && $this->is_public;
    }
    /**
     * @return float|int
     */
    public function getAdjustment()
    {
        if (! $this->affectsBalance()) {
            return 0;
        }
        return $this->getRawAdjustment();
    }
    /**
     * @return float
     */
    private function getRawAdjustment()
    {
        // if we've just made the invoice public then apply the full amount
        if ($this->is_public && ! $this->getOriginal('is_public')) {
            return $this->amount;
        }
        return floatval($this->amount) - floatval($this->getOriginal('amount'));
    }
    public function isChanged()
    {
        if (Utils::isNinja()) {
            if ($this->getRawAdjustment() != 0) {
                return true;
            }
            foreach ([
                'invoice_number',
                'po_number',
                'invoice_date',
                'due_date',
                'terms',
                'public_notes',
                'invoice_footer',
                'partial',
                'partial_due_date',
            ] as $field) {
                if ($this->$field != $this->getOriginal($field)) {
                    return true;
                }
            }
            return false;
        } else {
            $dirty = $this->getDirty();
            unset($dirty['invoice_status_id']);
            unset($dirty['client_enable_auto_bill']);
            unset($dirty['quote_invoice_id']);
            return count($dirty) > 0;
        }
    }
    /**
     * @param bool $calculate
     *
     * @return int|mixed
     */
    public function getAmountPaid($calculate = false)
    {
        if ($this->isType(INVOICE_TYPE_QUOTE) || $this->is_recurring) {
            return 0;
        }
        if ($calculate) {
            $amount = 0;
            foreach ($this->payments as $payment) {
                if ($payment->payment_status_id == PAYMENT_STATUS_VOIDED || $payment->payment_status_id == PAYMENT_STATUS_FAILED) {
                    continue;
                }
                $amount += $payment->getCompletedAmount();
            }
            return $amount;
        } else {
            return $this->amount - $this->balance;
        }
    }
    /**
     * @return bool
     */
    public function trashed()
    {
        if ($this->client && $this->client->trashed()) {
            return true;
        }
        return self::parentTrashed();
    }
    /**
     * @return \Illuminate\Database\Eloquent\Relations\BelongsTo
     */
    public function account()
    {
        return $this->belongsTo('App\Models\Account');
    }
    /**
     * @return mixed
     */
    public function user()
    {
        return $this->belongsTo('App\Models\User')->withTrashed();
    }
    /**
     * @return mixed
     */
    public function client()
    {
        return $this->belongsTo('App\Models\Client')->withTrashed();
    }
    /**
     * @return mixed
     */
    public function invoice_items()
    {
        return $this->hasMany('App\Models\InvoiceItem')->orderBy('id');
    }
    /**
     * @return mixed
     */
    public function documents()
    {
        return $this->hasMany('App\Models\Document')->orderBy('id');
    }
    /**
     * @return mixed
     */
    public function allDocuments()
    {
        $documents = $this->documents;
        $documents = $documents->merge($this->account->defaultDocuments);
        foreach ($this->expenses as $expense) {
            if ($expense->invoice_documents) {
                $documents = $documents->merge($expense->documents);
            }
        }
        return $documents;
    }
    /**
     * @return \Illuminate\Database\Eloquent\Relations\BelongsTo
     */
    public function invoice_status()
    {
        return $this->belongsTo('App\Models\InvoiceStatus');
    }
    /**
     * @return \Illuminate\Database\Eloquent\Relations\BelongsTo
     */
    public function invoice_design()
    {
        return $this->belongsTo('App\Models\InvoiceDesign');
    }
    /**
     * @return \Illuminate\Database\Eloquent\Relations\HasMany
     */
    public function payments()
    {
        return $this->hasMany('App\Models\Payment', 'invoice_id', 'id');
    }
    /**
     * @return \Illuminate\Database\Eloquent\Relations\BelongsTo
     */
    public function recurring_invoice()
    {
        return $this->belongsTo('App\Models\Invoice');
    }
    /**
     * @return \Illuminate\Database\Eloquent\Relations\BelongsTo
     */
    public function quote()
    {
        return $this->belongsTo('App\Models\Invoice');
    }
    /**
     * @return \Illuminate\Database\Eloquent\Relations\HasMany
     */
    public function recurring_invoices()
    {
        return $this->hasMany('App\Models\Invoice', 'recurring_invoice_id');
    }
    /**
     * @return \Illuminate\Database\Eloquent\Relations\BelongsTo
     */
    public function frequency()
    {
        return $this->belongsTo('App\Models\Frequency');
    }
    /**
     * @return mixed
     */
    public function invitations()
    {
        return $this->hasMany('App\Models\Invitation')->orderBy('invitations.contact_id');
    }
    /**
     * @return mixed
     */
    public function expenses()
    {
        return $this->hasMany('App\Models\Expense', 'invoice_id', 'id')->withTrashed();
    }
    /**
     * @param $query
     *
     * @return mixed
     */
    public function scopeInvoices($query)
    {
        return $query->where('invoice_type_id', '=', INVOICE_TYPE_STANDARD)
                     ->where('is_recurring', '=', false);
    }
    /**
     * @param $query
     *
     * @return mixed
     */
    public function scopeRecurring($query)
    {
        return $query->where('invoice_type_id', '=', INVOICE_TYPE_STANDARD)
                     ->where('is_recurring', '=', true);
    }
    /**
     * @param $query
     *
     * @return mixed
     */
    public function scopeDateRange($query, $startDate, $endDate)
    {
        return $query->where(function ($query) use ($startDate, $endDate) {
            $query->whereBetween('invoice_date', [$startDate, $endDate]);
        })->orWhere(function ($query) use ($startDate, $endDate) {
            $query->whereBetween('due_date', [$startDate, $endDate]);
        });
    }
    /**
     * @param $query
     *
     * @return mixed
     */
    public function scopeQuotes($query)
    {
        return $query->where('invoice_type_id', '=', INVOICE_TYPE_QUOTE)
                     ->where('is_recurring', '=', false);
    }
    /**
     * @param $query
     * @param $typeId
     *
     * @return mixed
     */
    public function scopeInvoiceType($query, $typeId)
    {
        return $query->where('invoice_type_id', '=', $typeId);
    }
    /**
     * @param $query
     * @param $typeId
     *
     * @return mixed
     */
    public function scopeStatusIds($query, $statusIds)
    {
        if (! $statusIds || (is_array($statusIds) && ! count($statusIds))) {
            return $query;
        }
        return $query->where(function ($query) use ($statusIds) {
            foreach ($statusIds as $statusId) {
                $query->orWhere('invoice_status_id', '=', $statusId);
            }
            if (in_array(INVOICE_STATUS_UNPAID, $statusIds)) {
                $query->orWhere(function ($query) {
                    $query->where('balance', '>', 0)
                          ->where('is_public', '=', true);
                });
            }
            if (in_array(INVOICE_STATUS_OVERDUE, $statusIds)) {
                $query->orWhere(function ($query) {
                    $query->where('balance', '>', 0)
                          ->where('due_date', '<', date('Y-m-d'))
                          ->where('is_public', '=', true);
                });
            }
        });
    }
    /**
     * @param $typeId
     *
     * @return bool
     */
    public function isType($typeId)
    {
        return $this->invoice_type_id == $typeId;
    }
    /**
     * @return bool
     */
    public function isQuote()
    {
        return $this->isType(INVOICE_TYPE_QUOTE);
    }
    /**
     * @return bool
     */
    public function isStandard()
    {
        return $this->isType(INVOICE_TYPE_STANDARD) && ! $this->is_recurring;
    }
    public function markSentIfUnsent()
    {
        if (! $this->isSent()) {
            $this->markSent();
        }
    }
    public function markSent()
    {
        if ($this->is_deleted) {
            return;
        }
        if (! $this->isSent()) {
            $this->invoice_status_id = INVOICE_STATUS_SENT;
        }
        $this->is_public = true;
        $this->save();
        $this->markInvitationsSent();
    }
    /**
     * @param bool  $notify
     * @param mixed $reminder
     */
    public function markInvitationsSent($notify = false, $reminder = false)
    {
        if ($this->is_deleted) {
            return;
        }
        if (! $this->relationLoaded('invitations')) {
            $this->load('invitations');
        }
        foreach ($this->invitations as $invitation) {
            $this->markInvitationSent($invitation, false, $notify, $reminder);
        }
    }
    public function areInvitationsSent()
    {
        if (! $this->relationLoaded('invitations')) {
            $this->load('invitations');
        }
        foreach ($this->invitations as $invitation) {
            if (! $invitation->isSent()) {
                return false;
            }
        }
        return true;
    }
    /**
     * @param $invitation
     * @param bool  $messageId
     * @param bool  $notify
     * @param mixed $notes
     */
    public function markInvitationSent($invitation, $messageId = false, $notify = true, $notes = false)
    {
        if ($this->is_deleted) {
            return;
        }
        if (! $this->isSent()) {
            $this->is_public = true;
            $this->invoice_status_id = INVOICE_STATUS_SENT;
            $this->save();
        }
        $invitation->markSent($messageId);
        // if the user marks it as sent rather than acually sending it
        // then we won't track it in the activity log
        if (! $notify) {
            return;
        }
        if ($this->isType(INVOICE_TYPE_QUOTE)) {
            event(new QuoteInvitationWasEmailed($invitation, $notes));
        } else {
            event(new InvoiceInvitationWasEmailed($invitation, $notes));
        }
    }
    public function markViewed()
    {
        if (! $this->isViewed()) {
            $this->invoice_status_id = INVOICE_STATUS_VIEWED;
            $this->save();
        }
    }
    /**
     * @param bool $save
     */
    public function updatePaidStatus($paid = false, $save = true)
    {
        $statusId = false;
        if ($paid && $this->balance == 0) {
            $statusId = INVOICE_STATUS_PAID;
        } elseif ($paid && $this->balance > 0 && $this->balance < $this->amount) {
            $statusId = INVOICE_STATUS_PARTIAL;
        } elseif ($this->isPartial() && $this->balance > 0) {
            $statusId = ($this->balance == $this->amount ? INVOICE_STATUS_SENT : INVOICE_STATUS_PARTIAL);
        }
        if ($statusId && $statusId != $this->invoice_status_id) {
            $this->invoice_status_id = $statusId;
            if ($save) {
                $this->save();
            }
        }
    }
    public function markApproved()
    {
        if ($this->isType(INVOICE_TYPE_QUOTE)) {
            $this->invoice_status_id = INVOICE_STATUS_APPROVED;
            $this->save();
        }
    }
    /**
     * @param $balanceAdjustment
     * @param int $partial
     */
    public function updateBalances($balanceAdjustment, $partial = 0)
    {
        if ($this->is_deleted) {
            return;
        }
        $balanceAdjustment = floatval($balanceAdjustment);
        $partial = floatval($partial);
        if (! $balanceAdjustment && $this->partial == $partial) {
            return;
        }
        $this->balance = $this->balance + $balanceAdjustment;
        if ($this->partial > 0) {
            $this->partial = $partial;
            // clear the partial due date and set the due date
            // using payment terms if it's blank
            if (! $this->partial && $this->partial_due_date) {
                $this->partial_due_date = null;
                if (! $this->due_date) {
                    $this->due_date = $this->account->defaultDueDate($this->client);
                }
            }
        }
        $this->save();
        // mark fees as paid
        if ($balanceAdjustment != 0 && $this->account->gateway_fee_enabled) {
            if ($invoiceItem = $this->getGatewayFeeItem()) {
                $invoiceItem->markFeePaid();
            }
        }
    }
    public function activeUser()
    {
        if (! $this->user->trashed()) {
            return $this->user;
        }
        return $this->account->users->first();
    }
    /**
     * @return mixed
     */
    public function getName()
    {
        return $this->is_recurring ? trans('texts.recurring') : $this->invoice_number;
    }
    /**
     * @return string
     */
    public function getFileName()
    {
        $entityType = $this->getEntityType();
        return trans("texts.$entityType") . '_' . $this->invoice_number . '.pdf';
    }
    /**
     * @return string
     */
    public function getPDFPath()
    {
        return storage_path() . '/pdfcache/cache-' . $this->id . '.pdf';
    }
    public function canBePaid()
    {
        return ! $this->isPaid() && ! $this->is_deleted && $this->isStandard();
    }
    public static function calcStatusLabel($status, $class, $entityType, $quoteInvoiceId)
    {
        if ($quoteInvoiceId) {
            $label = 'converted';
        } elseif ($class == 'danger') {
            $label = $entityType == ENTITY_INVOICE ? 'past_due' : 'expired';
        } else {
            $label = 'status_' . strtolower($status);
        }
        return trans("texts.{$label}");
    }
    public static function calcStatusClass($statusId, $balance, $dueDate, $isRecurring)
    {
        if ($statusId >= INVOICE_STATUS_SENT && ! $isRecurring && static::calcIsOverdue($balance, $dueDate)) {
            return 'danger';
        }
        if (isset(static::$statusClasses[$statusId])) {
            return static::$statusClasses[$statusId];
        }
        return 'default';
    }
    public static function calcIsOverdue($balance, $dueDate)
    {
        if (! Utils::parseFloat($balance) > 0) {
            return false;
        }
        if (! $dueDate || $dueDate == '0000-00-00') {
            return false;
        }
        // it isn't considered overdue until the end of the day
        return time() > (strtotime($dueDate) + (60 * 60 * 24));
    }
    public function statusClass()
    {
        $dueDate = $this->getOriginal('partial_due_date') ?: $this->getOriginal('due_date');
        return static::calcStatusClass($this->invoice_status_id, $this->balance, $dueDate, $this->is_recurring);
    }
    public function statusLabel()
    {
        return static::calcStatusLabel($this->invoice_status->name, $this->statusClass(), $this->getEntityType(), $this->quote_invoice_id);
    }
    /**
     * @param $invoice
     *
     * @return string
     */
    public static function calcLink($invoice)
    {
        return link_to('invoices/' . $invoice->public_id, $invoice->invoice_number);
    }
    /**
     * @return string
     */
    public function getLink()
    {
        return self::calcLink($this);
    }
    public function getInvitationLink($type = 'view', $forceOnsite = false, $forcePlain = false)
    {
        if (! $this->relationLoaded('invitations')) {
            $this->load('invitations');
        }
        return $this->invitations[0]->getLink($type, $forceOnsite, $forcePlain);
    }
    /**
     * @return mixed
     */
    public function getEntityType()
    {
        return $this->isType(INVOICE_TYPE_QUOTE) ? ENTITY_QUOTE : ENTITY_INVOICE;
    }
    public function subEntityType()
    {
        if ($this->is_recurring) {
            return ENTITY_RECURRING_INVOICE;
        } else {
            return $this->getEntityType();
        }
    }
    /**
     * @return bool
     */
    public function isSent()
    {
        return $this->invoice_status_id >= INVOICE_STATUS_SENT && $this->getOriginal('is_public');
    }
    /**
     * @return bool
     */
    public function isViewed()
    {
        return $this->invoice_status_id >= INVOICE_STATUS_VIEWED;
    }
    /**
     * @return bool
     */
    public function isPartial()
    {
        return $this->invoice_status_id >= INVOICE_STATUS_PARTIAL;
    }
    /**
     * @return bool
     */
    public function isPaid()
    {
        return $this->invoice_status_id >= INVOICE_STATUS_PAID;
    }
    /**
     * @return bool
     */
    public function isOverdue()
    {
        return static::calcIsOverdue($this->balance, $this->partial_due_date ?: $this->due_date);
    }
    /**
     * @return mixed
     */
    public function getRequestedAmount()
    {
        $fee = 0;
        if ($this->account->gateway_fee_enabled) {
            $fee = $this->getGatewayFee();
        }
        if ($this->partial > 0) {
            return $this->partial + $fee;
        } else {
            return $this->balance;
        }
    }
    /**
     * @return string
     */
    public function getCurrencyCode()
    {
        if ($this->client->currency) {
            return $this->client->currency->code;
        } elseif ($this->account->currency) {
            return $this->account->currency->code;
        } else {
            return 'USD';
        }
    }
    /**
     * @return $this
     */
    public function hidePrivateFields()
    {
        $this->setVisible([
            'invoice_number',
            'discount',
            'is_amount_discount',
            'po_number',
            'invoice_date',
            'due_date',
            'terms',
            'invoice_footer',
            'public_notes',
            'amount',
            'balance',
            'invoice_items',
            'documents',
            'expenses',
            'client',
            'invitations',
            'tax_name1',
            'tax_rate1',
            'tax_name2',
            'tax_rate2',
            'account',
            'invoice_design',
            'invoice_design_id',
            'invoice_fonts',
            'features',
            'invoice_type_id',
            'custom_value1',
            'custom_value2',
            'custom_taxes1',
            'custom_taxes2',
            'partial',
            'partial_due_date',
            'has_tasks',
            'custom_text_value1',
            'custom_text_value2',
            'has_expenses',
        ]);
        $this->client->setVisible([
            'name',
            'balance',
            'id_number',
            'vat_number',
            'address1',
            'address2',
            'city',
            'state',
            'postal_code',
            'work_phone',
            'payment_terms',
            'contacts',
            'country',
            'currency_id',
            'country_id',
            'custom_value1',
            'custom_value2',
        ]);
        $this->account->setVisible([
            'name',
            'website',
            'id_number',
            'vat_number',
            'address1',
            'address2',
            'city',
            'state',
            'postal_code',
            'work_phone',
            'work_email',
            'country',
            'country_id',
            'currency_id',
            'custom_label1',
            'custom_value1',
            'custom_label2',
            'custom_value2',
            'custom_client_label1',
            'custom_client_label2',
            'custom_contact_label1',
            'custom_contact_label2',
            'primary_color',
            'secondary_color',
            'hide_quantity',
            'hide_paid_to_date',
            'all_pages_header',
            'all_pages_footer',
            'custom_invoice_label1',
            'custom_invoice_label2',
            'pdf_email_attachment',
            'show_item_taxes',
            'custom_invoice_text_label1',
            'custom_invoice_text_label2',
            'custom_invoice_item_label1',
            'custom_invoice_item_label2',
            'invoice_embed_documents',
            'page_size',
            'include_item_taxes_inline',
            'invoice_fields',
            'show_currency_code',
            'inclusive_taxes',
            'date_format',
            'datetime_format',
            'timezone',
            'signature_on_pdf',
        ]);
        foreach ($this->invitations as $invitation) {
            $invitation->setVisible([
                'signature_base64',
                'signature_date',
            ]);
        }
        foreach ($this->invoice_items as $invoiceItem) {
            $invoiceItem->setVisible([
                'product_key',
                'notes',
                'custom_value1',
                'custom_value2',
                'cost',
                'qty',
                'tax_name1',
                'tax_rate1',
                'tax_name2',
                'tax_rate2',
                'invoice_item_type_id',
                'discount',
            ]);
        }
        foreach ($this->client->contacts as $contact) {
            $contact->setVisible([
                'first_name',
                'last_name',
                'email',
                'phone',
                'custom_value1',
                'custom_value2',
            ]);
        }
        foreach ($this->documents as $document) {
            $document->setVisible([
                'public_id',
                'name',
            ]);
        }
        foreach ($this->expenses as $expense) {
            $expense->setVisible([
                'documents',
            ]);
            foreach ($expense->documents as $document) {
                $document->setVisible([
                    'public_id',
                    'name',
                ]);
            }
        }
        return $this;
    }
    /**
     * @throws \Recurr\Exception\MissingData
     *
     * @return bool|\Recurr\RecurrenceCollection
     */
    public function getSchedule()
    {
        if (! $this->start_date || ! $this->is_recurring || ! $this->frequency_id) {
            return false;
        }
        $startDate = $this->getOriginal('last_sent_date') ?: $this->getOriginal('start_date');
        $startDate .= ' ' . $this->account->recurring_hour . ':00:00';
        $startDate = $this->account->getDateTime($startDate);
        $endDate = $this->end_date ? $this->account->getDateTime($this->getOriginal('end_date')) : null;
        $timezone = $this->account->getTimezone();
        $rule = $this->getRecurrenceRule();
        $rule = new \Recurr\Rule("{$rule}", $startDate, $endDate, $timezone);
        // Fix for months with less than 31 days
        $transformerConfig = new \Recurr\Transformer\ArrayTransformerConfig();
        $transformerConfig->enableLastDayOfMonthFix();
        $transformer = new \Recurr\Transformer\ArrayTransformer();
        $transformer->setConfig($transformerConfig);
        $dates = $transformer->transform($rule);
        if (count($dates) < 2) {
            return false;
        }
        return $dates;
    }
    /**
     * @return null
     */
    public function getNextSendDate()
    {
        if (! $this->is_public) {
            return null;
        }
        if ($this->start_date && ! $this->last_sent_date) {
            $startDate = $this->getOriginal('start_date') . ' ' . $this->account->recurring_hour . ':00:00';
            return $this->account->getDateTime($startDate);
        }
        if (! $schedule = $this->getSchedule()) {
            return null;
        }
        if (count($schedule) < 2) {
            return null;
        }
        return $schedule[1]->getStart();
    }
    /**
     * @param null $invoice_date
     *
     * @return mixed|null
     */
    public function getDueDate($invoice_date = null)
    {
        if (! $this->is_recurring) {
            return $this->due_date ? $this->due_date : null;
        } else {
            $now = time();
            if ($invoice_date) {
                // If $invoice_date is specified, all calculations are based on that date
                if (is_numeric($invoice_date)) {
                    $now = $invoice_date;
                } elseif (is_string($invoice_date)) {
                    $now = strtotime($invoice_date);
                } elseif ($invoice_date instanceof \DateTime) {
                    $now = $invoice_date->getTimestamp();
                }
            }
            if ($this->due_date && $this->due_date != '0000-00-00') {
                // This is a recurring invoice; we're using a custom format here.
                // The year is always 1998; January is 1st, 2nd, last day of the month.
                // February is 1st Sunday after, 1st Monday after, ..., through 4th Saturday after.
                $dueDateVal = strtotime($this->due_date);
                $monthVal = (int) date('n', $dueDateVal);
                $dayVal = (int) date('j', $dueDateVal);
                $dueDate = false;
                if ($monthVal == 1) {// January; day of month
                    $currentDay = (int) date('j', $now);
                    $lastDayOfMonth = (int) date('t', $now);
                    $dueYear = (int) date('Y', $now); // This year
                    $dueMonth = (int) date('n', $now); // This month
                    $dueDay = $dayVal; // The day specified for the invoice
                    if ($dueDay > $lastDayOfMonth) {
                        // No later than the end of the month
                        $dueDay = $lastDayOfMonth;
                    }
                    if ($currentDay >= $dueDay) {
                        // Wait until next month
                        // We don't need to handle the December->January wraparaound, since PHP handles month 13 as January of next year
                        $dueMonth++;
                        // Reset the due day
                        $dueDay = $dayVal;
                        $lastDayOfMonth = (int) date('t', mktime(0, 0, 0, $dueMonth, 1, $dueYear)); // The number of days in next month
                        // Check against the last day again
                        if ($dueDay > $lastDayOfMonth) {
                            // No later than the end of the month
                            $dueDay = $lastDayOfMonth;
                        }
                    }
                    $dueDate = mktime(0, 0, 0, $dueMonth, $dueDay, $dueYear);
                } elseif ($monthVal == 2) {// February; day of week
                    $ordinals = ['first', 'second', 'third', 'fourth'];
                    $daysOfWeek = ['sunday', 'monday', 'tuesday', 'wednesday', 'thursday', 'friday', 'saturday'];
                    $ordinalIndex = ceil($dayVal / 7) - 1; // 1-7 are "first"; 8-14 are "second", etc.
                    $dayOfWeekIndex = ($dayVal - 1) % 7; // 1,8,15,22 are Sunday, 2,9,16,23 are Monday, etc.
                    $dayStr = $ordinals[$ordinalIndex] . ' ' . $daysOfWeek[$dayOfWeekIndex]; // "first sunday", "first monday", etc.
                    $dueDate = strtotime($dayStr, $now);
                }
                if ($dueDate) {
                    return date('Y-m-d', $dueDate); // SQL format
                }
            } elseif ($this->client->payment_terms != 0) {
                // No custom due date set for this invoice; use the client's payment terms
                $days = $this->client->defaultDaysDue();
                return date('Y-m-d', strtotime('+'.$days.' day', $now));
            } elseif ($this->account->payment_terms != 0) {
                $days = $this->account->defaultDaysDue();
                return date('Y-m-d', strtotime('+'.$days.' day', $now));
            } elseif ($this->account->payment_terms != 0) {
                // No custom due date set for this invoice; use the client's payment terms
                $days = $this->account->payment_terms;
                if ($days == -1) {
                    $days = 0;
                }
                return date('Y-m-d', strtotime('+'.$days.' day', $now));
            }
        }
        // Couldn't calculate one
        return null;
    }
    /**
     * @param int $min
     * @param int $max
     *
     * @return null
     */
    public function getPrettySchedule($min = 1, $max = 10)
    {
        if (! $schedule = $this->getSchedule($max)) {
            return null;
        }
        $dates = [];
        for ($i = $min; $i < min($max, count($schedule)); $i++) {
            $date = $schedule[$i];
            $dateStart = $date->getStart();
            $date = $this->account->formatDate($dateStart);
            $dueDate = $this->getDueDate($dateStart);
            if ($dueDate) {
                $date .= ' (' . trans('texts.due') . ' ' . $this->account->formatDate($dueDate) . ')';
            }
            $dates[] = $date;
        }
        return implode('
', $dates);
    }
    /**
     * @return bool|string
     */
    public function getPDFString($decode = true)
    {
        if (! env('PHANTOMJS_CLOUD_KEY') && ! env('PHANTOMJS_BIN_PATH')) {
            return false;
        }
        if (Utils::isTravis()) {
            return false;
        }
        $invitation = $this->invitations[0];
        $link = $invitation->getLink('view', true, true);
        $pdfString = false;
        $phantomjsSecret = env('PHANTOMJS_SECRET');
        $phantomjsLink = $link . "?phantomjs=true&phantomjs_secret={$phantomjsSecret}";
        try {
            if (env('PHANTOMJS_BIN_PATH')) {
                // we see occasional 408 errors
                for ($i=1; $i<=5; $i++) {
                    $pdfString = CurlUtils::phantom('GET', $phantomjsLink);
                    $pdfString = strip_tags($pdfString);
                    if (strpos($pdfString, 'data') === 0) {
                        break;
                    } else {
                        $pdfString = false;
                        sleep(2);
                    }
                }
            }
            if (! $pdfString && ($key = env('PHANTOMJS_CLOUD_KEY'))) {
                $url = "http://api.phantomjscloud.com/api/browser/v2/{$key}/?request=%7Burl:%22{$link}?phantomjs=true%26phantomjs_secret={$phantomjsSecret}%22,renderType:%22html%22%7D";
                $pdfString = CurlUtils::get($url);
                $pdfString = strip_tags($pdfString);
            }
        } catch (\Exception $exception) {
            Utils::logError("PhantomJS - Failed to load {$phantomjsLink}: {$exception->getMessage()}");
            return false;
        }
        if (! $pdfString || strlen($pdfString) < 200) {
            Utils::logError("PhantomJS - Invalid response {$phantomjsLink}: {$pdfString}");
            return false;
        }
        if ($decode) {
            if ($pdf = Utils::decodePDF($pdfString)) {
                return $pdf;
            } else {
                Utils::logError("PhantomJS - Unable to decode {$phantomjsLink}");
                return false;
            }
        } else {
            return $pdfString;
        }
    }
    /**
     * @param $invoiceItem
     * @param $invoiceTotal
     *
     * @return float|int
     */
    public function getItemTaxable($invoiceItem, $invoiceTotal)
    {
        $total = $invoiceItem->qty * $invoiceItem->cost;
        if ($this->discount != 0) {
            if ($this->is_amount_discount) {
                $total -= $invoiceTotal ? ($total / ($invoiceTotal + $this->discount) * $this->discount) : 0;
            } else {
                $total *= (100 - $this->discount) / 100;
            }
        }
        if ($invoiceItem->discount != 0) {
            if ($this->is_amount_discount) {
                $total -= $invoiceItem->discount;
            } else {
                $total -= $total * $invoiceItem->discount / 100;
            }
        }
        return round($total, 2);
    }
    /**
     * @return float|int|mixed
     */
    public function getTaxable()
    {
        $total = 0;
        foreach ($this->invoice_items as $invoiceItem) {
            $lineTotal = $invoiceItem->qty * $invoiceItem->cost;
            if ($invoiceItem->discount != 0) {
                if ($this->is_amount_discount) {
                    $lineTotal -= $invoiceItem->discount;
                } else {
                    $lineTotal -= $lineTotal * $invoiceItem->discount / 100;
                }
            }
            $total += $lineTotal;
        }
        if ($this->discount > 0) {
            if ($this->is_amount_discount) {
                $total -= $this->discount;
            } else {
                $total *= (100 - $this->discount) / 100;
                $total = round($total, 2);
            }
        }
        if ($this->custom_value1 && $this->custom_taxes1) {
            $total += $this->custom_value1;
        }
        if ($this->custom_value2 && $this->custom_taxes2) {
            $total += $this->custom_value2;
        }
        return $total;
    }
    // if $calculatePaid is true we'll loop through each payment to
    // determine the sum, otherwise we'll use the cached paid_to_date amount
    /**
     * @param bool $calculatePaid
     *
     * @return array
     */
    public function getTaxes($calculatePaid = false)
    {
        $taxes = [];
        $account = $this->account;
        $taxable = $this->getTaxable();
        $paidAmount = $this->getAmountPaid($calculatePaid);
        if ($this->tax_name1) {
            if ($account->inclusive_taxes) {
                $invoiceTaxAmount = round($taxable - ($taxable / (1 + ($this->tax_rate1 / 100))), 2);
            } else {
                $invoiceTaxAmount = round($taxable * ($this->tax_rate1 / 100), 2);
            }
            $invoicePaidAmount = floatval($this->amount) && $invoiceTaxAmount ? ($paidAmount / $this->amount * $invoiceTaxAmount) : 0;
            $this->calculateTax($taxes, $this->tax_name1, $this->tax_rate1, $invoiceTaxAmount, $invoicePaidAmount);
        }
        if ($this->tax_name2) {
            if ($account->inclusive_taxes) {
                $invoiceTaxAmount = round($taxable - ($taxable / (1 + ($this->tax_rate2 / 100))), 2);
            } else {
                $invoiceTaxAmount = round($taxable * ($this->tax_rate2 / 100), 2);
            }
            $invoicePaidAmount = floatval($this->amount) && $invoiceTaxAmount ? ($paidAmount / $this->amount * $invoiceTaxAmount) : 0;
            $this->calculateTax($taxes, $this->tax_name2, $this->tax_rate2, $invoiceTaxAmount, $invoicePaidAmount);
        }
        foreach ($this->invoice_items as $invoiceItem) {
            $itemTaxable = $this->getItemTaxable($invoiceItem, $taxable);
            if ($invoiceItem->tax_name1) {
                if ($account->inclusive_taxes) {
                    $itemTaxAmount = round($taxable - ($taxable / (1 + ($invoiceItem->tax_rate1 / 100))), 2);
                } else {
                    $itemTaxAmount = round($itemTaxable * ($invoiceItem->tax_rate1 / 100), 2);
                }
                $itemPaidAmount = floatval($this->amount) && $itemTaxAmount ? ($paidAmount / $this->amount * $itemTaxAmount) : 0;
                $this->calculateTax($taxes, $invoiceItem->tax_name1, $invoiceItem->tax_rate1, $itemTaxAmount, $itemPaidAmount);
            }
            if ($invoiceItem->tax_name2) {
                if ($account->inclusive_taxes) {
                    $itemTaxAmount = round($taxable - ($taxable / (1 + ($invoiceItem->tax_rate2 / 100))), 2);
                } else {
                    $itemTaxAmount = round($itemTaxable * ($invoiceItem->tax_rate2 / 100), 2);
                }
                $itemPaidAmount = floatval($this->amount) && $itemTaxAmount ? ($paidAmount / $this->amount * $itemTaxAmount) : 0;
                $this->calculateTax($taxes, $invoiceItem->tax_name2, $invoiceItem->tax_rate2, $itemTaxAmount, $itemPaidAmount);
            }
        }
        return $taxes;
    }
    /**
     * @param $taxes
     * @param $name
     * @param $rate
     * @param $amount
     * @param $paid
     */
    private function calculateTax(&$taxes, $name, $rate, $amount, $paid)
    {
        if (! $amount) {
            return;
        }
        $amount = round($amount, 2);
        $paid = round($paid, 2);
        $key = $rate . ' ' . $name;
        if (! isset($taxes[$key])) {
            $taxes[$key] = [
                'name' => $name,
                'rate' => $rate + 0,
                'amount' => 0,
                'paid' => 0,
            ];
        }
        $taxes[$key]['amount'] += $amount;
        $taxes[$key]['paid'] += $paid;
    }
    /**
     * @return int
     */
    public function countDocuments($expenses = false)
    {
        $count = count($this->documents);
        foreach ($this->expenses as $expense) {
            if ($expense->invoice_documents) {
                $count += count($expense->documents);
            }
        }
        if ($expenses) {
            foreach ($expenses as $expense) {
                if ($expense->invoice_documents) {
                    $count += count($expense->documents);
                }
            }
        }
        return $count;
    }
    /**
     * @return bool
     */
    public function hasDocuments()
    {
        if ($this->documents->count()) {
            return true;
        }
        if ($this->account->defaultDocuments->count()) {
            return true;
        }
        return $this->hasExpenseDocuments();
    }
    /**
     * @return bool
     */
    public function hasExpenseDocuments()
    {
        foreach ($this->expenses as $expense) {
            if ($expense->invoice_documents && count($expense->documents)) {
                return true;
            }
        }
        return false;
    }
    /**
     * @return bool
     */
    public function getAutoBillEnabled()
    {
        if (! $this->is_recurring) {
            $recurInvoice = $this->recurring_invoice;
        } else {
            $recurInvoice = $this;
        }
        if (! $recurInvoice) {
            return false;
        }
        return $recurInvoice->auto_bill == AUTO_BILL_ALWAYS || ($recurInvoice->auto_bill != AUTO_BILL_OFF && $recurInvoice->client_enable_auto_bill);
    }
    public static function getStatuses($entityType = false)
    {
        $statuses = [];
        if ($entityType == ENTITY_RECURRING_INVOICE) {
            return $statuses;
        }
        foreach (\Cache::get('invoiceStatus') as $status) {
            if ($entityType == ENTITY_QUOTE) {
                if (in_array($status->id, [INVOICE_STATUS_PAID, INVOICE_STATUS_PARTIAL])) {
                    continue;
                }
            } elseif ($entityType == ENTITY_INVOICE) {
                if (in_array($status->id, [INVOICE_STATUS_APPROVED])) {
                    continue;
                }
            }
            $statuses[$status->id] = trans('texts.status_' . strtolower($status->name));
        }
        if ($entityType == ENTITY_INVOICE) {
            $statuses[INVOICE_STATUS_UNPAID] = trans('texts.unpaid');
            $statuses[INVOICE_STATUS_OVERDUE] = trans('texts.past_due');
        }
        return $statuses;
    }
    public function emailHistory()
    {
        return Activity::scope()
                ->with(['contact'])
                ->whereInvoiceId($this->id)
                ->whereIn('activity_type_id', [ACTIVITY_TYPE_EMAIL_INVOICE, ACTIVITY_TYPE_EMAIL_QUOTE])
                ->orderBy('id', 'desc')
                ->get();
    }
    public function getDueDateLabel()
    {
        return $this->isQuote() ? 'valid_until' : 'due_date';
    }
    public function onlyHasTasks()
    {
        foreach ($this->invoice_items as $item) {
            if ($item->invoice_item_type_id != INVOICE_ITEM_TYPE_TASK) {
                return false;
            }
        }
        return true;
    }
}
Invoice::creating(function ($invoice) {
    if (! $invoice->is_recurring) {
        $account = $invoice->account;
        if ($invoice->amount >= 0) {
            $account->incrementCounter($invoice);
        } elseif ($account->credit_number_counter > 0) {
            $account->incrementCounter(new Credit());
        }
    }
});
Invoice::created(function ($invoice) {
    if ($invoice->isType(INVOICE_TYPE_QUOTE)) {
        event(new QuoteWasCreated($invoice));
    } else {
        event(new InvoiceWasCreated($invoice));
    }
});
Invoice::updating(function ($invoice) {
    if ($invoice->isType(INVOICE_TYPE_QUOTE)) {
        event(new QuoteWasUpdated($invoice));
    } else {
        event(new InvoiceWasUpdated($invoice));
    }
});