invoiceRepo = $invoiceRepo; $this->clientRepo = $clientRepo; $this->paymentRepo = $paymentRepo; $this->invoiceService = $invoiceService; $this->mailer = $mailer; } /** * @SWG\Get( * path="/invoices", * summary="List of invoices", * tags={"invoice"}, * @SWG\Response( * response=200, * description="A list with invoices", * @SWG\Schema(type="array", @SWG\Items(ref="#/definitions/Invoice")) * ), * @SWG\Response( * response="default", * description="an ""unexpected"" error" * ) * ) */ public function index() { $paginator = Invoice::scope()->withTrashed(); $invoices = Invoice::scope()->withTrashed() ->with(array_merge(['invoice_items'], $this->getIncluded())); if ($clientPublicId = Input::get('client_id')) { $filter = function($query) use ($clientPublicId) { $query->where('public_id', '=', $clientPublicId); }; $invoices->whereHas('client', $filter); $paginator->whereHas('client', $filter); } $invoices = $invoices->orderBy('created_at', 'desc')->paginate(); /* // Add the first invitation link to the data foreach ($invoices as $key => $invoice) { foreach ($invoice->invitations as $subKey => $invitation) { $invoices[$key]['link'] = $invitation->getLink(); } unset($invoice['invitations']); } */ $transformer = new InvoiceTransformer(Auth::user()->account, Input::get('serializer')); $paginator = $paginator->paginate(); $data = $this->createCollection($invoices, $transformer, 'invoices', $paginator); return $this->response($data); } /** * @SWG\Get( * path="/invoices/{invoice_id}", * summary="Individual Invoice", * tags={"invoice"}, * @SWG\Response( * response=200, * description="A single invoice", * @SWG\Schema(type="object", @SWG\Items(ref="#/definitions/Invoice")) * ), * @SWG\Response( * response="default", * description="an ""unexpected"" error" * ) * ) */ public function show($publicId) { $invoice = Invoice::scope($publicId)->withTrashed()->first(); if(!$invoice) return $this->errorResponse(['message'=>'Invoice does not exist!'], 404); $transformer = new InvoiceTransformer(\Auth::user()->account, Input::get('serializer')); $data = $this->createItem($invoice, $transformer, 'invoice'); return $this->response($data); } /** * @SWG\Post( * path="/invoices", * tags={"invoice"}, * summary="Create an invoice", * @SWG\Parameter( * in="body", * name="body", * @SWG\Schema(ref="#/definitions/Invoice") * ), * @SWG\Response( * response=200, * description="New invoice", * @SWG\Schema(type="object", @SWG\Items(ref="#/definitions/Invoice")) * ), * @SWG\Response( * response="default", * description="an ""unexpected"" error" * ) * ) */ public function store(CreateInvoiceAPIRequest $request) { $data = Input::all(); $error = null; if (isset($data['email'])) { $email = $data['email']; $client = Client::scope()->whereHas('contacts', function($query) use ($email) { $query->where('email', '=', $email); })->first(); if (!$client) { $validator = Validator::make(['email'=>$email], ['email' => 'email']); if ($validator->fails()) { $messages = $validator->messages(); return $messages->first(); } $clientData = ['contact' => ['email' => $email]]; foreach ([ 'name', 'address1', 'address2', 'city', 'state', 'postal_code', 'private_notes', 'currency_code', ] as $field) { if (isset($data[$field])) { $clientData[$field] = $data[$field]; } } foreach ([ 'first_name', 'last_name', 'phone', ] as $field) { if (isset($data[$field])) { $clientData['contact'][$field] = $data[$field]; } } $client = $this->clientRepo->save($clientData); } } else if (isset($data['client_id'])) { $client = Client::scope($data['client_id'])->firstOrFail(); } $data = self::prepareData($data, $client); $data['client_id'] = $client->id; $invoice = $this->invoiceService->save($data); $payment = false; // Optionally create payment with invoice if (isset($data['paid']) && $data['paid']) { $payment = $this->paymentRepo->save([ 'invoice_id' => $invoice->id, 'client_id' => $client->id, 'amount' => $data['paid'] ]); } if (isset($data['email_invoice']) && $data['email_invoice']) { if ($payment) { $this->mailer->sendPaymentConfirmation($payment); } else { $this->mailer->sendInvoice($invoice); } } $invoice = Invoice::scope($invoice->public_id)->with('client', 'invoice_items', 'invitations')->first(); $transformer = new InvoiceTransformer(\Auth::user()->account, Input::get('serializer')); $data = $this->createItem($invoice, $transformer, 'invoice'); return $this->response($data); } private function prepareData($data, $client) { $account = Auth::user()->account; $account->loadLocalizationSettings($client); // set defaults for optional fields $fields = [ 'discount' => 0, 'is_amount_discount' => false, 'terms' => '', 'invoice_footer' => '', 'public_notes' => '', 'po_number' => '', 'invoice_design_id' => $account->invoice_design_id, 'invoice_items' => [], 'custom_value1' => 0, 'custom_value2' => 0, 'custom_taxes1' => false, 'custom_taxes2' => false, 'partial' => 0 ]; if (!isset($data['invoice_status_id']) || $data['invoice_status_id'] == 0) { $data['invoice_status_id'] = INVOICE_STATUS_DRAFT; } if (!isset($data['invoice_date'])) { $fields['invoice_date_sql'] = date_create()->format('Y-m-d'); } if (!isset($data['due_date'])) { $fields['due_date_sql'] = false; } foreach ($fields as $key => $val) { if (!isset($data[$key])) { $data[$key] = $val; } } // initialize the line items if (isset($data['product_key']) || isset($data['cost']) || isset($data['notes']) || isset($data['qty'])) { $data['invoice_items'] = [self::prepareItem($data)]; // make sure the tax isn't applied twice (for the invoice and the line item) unset($data['invoice_items'][0]['tax_name1']); unset($data['invoice_items'][0]['tax_rate1']); unset($data['invoice_items'][0]['tax_name2']); unset($data['invoice_items'][0]['tax_rate2']); } else { foreach ($data['invoice_items'] as $index => $item) { $data['invoice_items'][$index] = self::prepareItem($item); } } return $data; } private function prepareItem($item) { // if only the product key is set we'll load the cost and notes if (!empty($item['product_key']) && empty($item['cost']) && empty($item['notes'])) { $product = Product::findProductByKey($item['product_key']); if ($product) { if (empty($item['cost'])) { $item['cost'] = $product->cost; } if (empty($item['notes'])) { $item['notes'] = $product->notes; } } } $fields = [ 'cost' => 0, 'product_key' => '', 'notes' => '', 'qty' => 1 ]; foreach ($fields as $key => $val) { if (!isset($item[$key])) { $item[$key] = $val; } } return $item; } public function emailInvoice() { $data = Input::all(); $error = null; $invoice = Invoice::scope($data['id'])->withTrashed()->first(); if(!$invoice) return $this->errorResponse(['message'=>'Invoice does not exist.'], 400); $this->mailer->sendInvoice($invoice, false, false); if($error) { return $this->errorResponse(['message'=>'There was an error sending the invoice'], 400); } else { $response = json_encode(RESULT_SUCCESS, JSON_PRETTY_PRINT); } $headers = Utils::getApiHeaders(); return Response::make($response, $error ? 400 : 200, $headers); } /** * @SWG\Put( * path="/invoices", * tags={"invoice"}, * summary="Update an invoice", * @SWG\Parameter( * in="body", * name="body", * @SWG\Schema(ref="#/definitions/Invoice") * ), * @SWG\Response( * response=200, * description="Update invoice", * @SWG\Schema(type="object", @SWG\Items(ref="#/definitions/Invoice")) * ), * @SWG\Response( * response="default", * description="an ""unexpected"" error" * ) * ) */ public function update(UpdateInvoiceAPIRequest $request, $publicId) { if ($request->action == ACTION_ARCHIVE) { $invoice = Invoice::scope($publicId)->firstOrFail(); $this->invoiceRepo->archive($invoice); $transformer = new InvoiceTransformer(\Auth::user()->account, Input::get('serializer')); $data = $this->createItem($invoice, $transformer, 'invoice'); return $this->response($data); } else if ($request->action == ACTION_CONVERT) { $quote = Invoice::scope($publicId)->firstOrFail(); $invoice = $this->invoiceRepo->cloneInvoice($quote, $quote->id); $transformer = new InvoiceTransformer(\Auth::user()->account, Input::get('serializer')); $data = $this->createItem($invoice, $transformer, 'invoice'); return $this->response($data); } else if ($request->action == ACTION_RESTORE) { $invoice = Invoice::scope($publicId)->withTrashed()->firstOrFail(); $this->invoiceRepo->restore($invoice); $transformer = new InvoiceTransformer(\Auth::user()->account, Input::get('serializer')); $data = $this->createItem($invoice, $transformer, 'invoice'); return $this->response($data); } $data = $request->input(); $data['public_id'] = $publicId; $this->invoiceService->save($data); $invoice = Invoice::scope($publicId)->with('client', 'invoice_items', 'invitations')->firstOrFail(); $transformer = new InvoiceTransformer(\Auth::user()->account, Input::get('serializer')); $data = $this->createItem($invoice, $transformer, 'invoice'); return $this->response($data); } /** * @SWG\Delete( * path="/invoices", * tags={"invoice"}, * summary="Delete an invoice", * @SWG\Parameter( * in="body", * name="body", * @SWG\Schema(ref="#/definitions/Invoice") * ), * @SWG\Response( * response=200, * description="Delete invoice", * @SWG\Schema(type="object", @SWG\Items(ref="#/definitions/Invoice")) * ), * @SWG\Response( * response="default", * description="an ""unexpected"" error" * ) * ) */ public function destroy($publicId) { $data['public_id'] = $publicId; $invoice = Invoice::scope($publicId)->firstOrFail(); $this->invoiceRepo->delete($invoice); $transformer = new InvoiceTransformer(\Auth::user()->account, Input::get('serializer')); $data = $this->createItem($invoice, $transformer, 'invoice'); return $this->response($data); } }