1
0
mirror of https://github.com/invoiceninja/invoiceninja.git synced 2024-11-10 21:22:58 +01:00
invoiceninja/app/Http/Controllers/QuoteController.php

859 lines
30 KiB
PHP
Raw Normal View History

2019-04-02 08:36:49 +02:00
<?php
2019-05-11 05:32:07 +02:00
/**
* Invoice Ninja (https://invoiceninja.com).
2019-05-11 05:32:07 +02:00
*
* @link https://github.com/invoiceninja/invoiceninja source repository
*
2022-04-27 05:20:41 +02:00
* @copyright Copyright (c) 2022. Invoice Ninja LLC (https://invoiceninja.com)
2019-05-11 05:32:07 +02:00
*
2021-06-16 08:58:16 +02:00
* @license https://www.elastic.co/licensing/elastic-license
2019-05-11 05:32:07 +02:00
*/
2019-04-02 08:36:49 +02:00
namespace App\Http\Controllers;
2020-11-03 11:19:32 +01:00
use App\Events\Quote\QuoteWasCreated;
use App\Events\Quote\QuoteWasUpdated;
use App\Factory\CloneQuoteFactory;
use App\Factory\CloneQuoteToInvoiceFactory;
2022-08-15 04:31:20 +02:00
use App\Factory\CloneQuoteToProjectFactory;
2019-05-02 13:07:38 +02:00
use App\Factory\QuoteFactory;
use App\Filters\QuoteFilters;
use App\Http\Requests\Quote\ActionQuoteRequest;
use App\Http\Requests\Quote\BulkActionQuoteRequest;
2019-05-02 13:07:38 +02:00
use App\Http\Requests\Quote\CreateQuoteRequest;
use App\Http\Requests\Quote\DestroyQuoteRequest;
use App\Http\Requests\Quote\EditQuoteRequest;
use App\Http\Requests\Quote\ShowQuoteRequest;
use App\Http\Requests\Quote\StoreQuoteRequest;
use App\Http\Requests\Quote\UpdateQuoteRequest;
2021-02-15 21:58:19 +01:00
use App\Http\Requests\Quote\UploadQuoteRequest;
use App\Jobs\Invoice\ZipInvoices;
2022-03-02 03:51:38 +01:00
use App\Jobs\Quote\ZipQuotes;
2021-03-07 22:32:38 +01:00
use App\Models\Account;
use App\Models\Client;
use App\Models\Invoice;
2022-08-15 04:31:20 +02:00
use App\Models\Project;
2019-05-02 13:07:38 +02:00
use App\Models\Quote;
use App\Repositories\QuoteRepository;
use App\Transformers\InvoiceTransformer;
2022-08-15 04:31:20 +02:00
use App\Transformers\ProjectTransformer;
2019-05-02 13:07:38 +02:00
use App\Transformers\QuoteTransformer;
2020-11-03 11:19:32 +01:00
use App\Utils\Ninja;
use App\Utils\TempFile;
2022-08-15 04:50:45 +02:00
use App\Utils\Traits\GeneratesCounter;
2019-05-02 13:07:38 +02:00
use App\Utils\Traits\MakesHash;
2021-02-15 21:58:19 +01:00
use App\Utils\Traits\SavesDocuments;
2019-04-02 08:36:49 +02:00
use Illuminate\Http\Request;
2020-10-28 11:10:49 +01:00
use Illuminate\Http\Response;
2021-07-07 13:39:49 +02:00
use Illuminate\Support\Facades\Storage;
2019-04-02 08:36:49 +02:00
2019-05-02 13:07:38 +02:00
/**
* Class QuoteController.
2019-05-02 13:07:38 +02:00
*/
2019-04-02 08:36:49 +02:00
class QuoteController extends BaseController
{
2019-05-02 13:07:38 +02:00
use MakesHash;
2021-02-15 21:58:19 +01:00
use SavesDocuments;
2022-08-15 04:50:45 +02:00
use GeneratesCounter;
2019-05-02 13:07:38 +02:00
protected $entity_type = Quote::class;
protected $entity_transformer = QuoteTransformer::class;
/**
* @var QuoteRepository
*/
protected $quote_repo;
protected $base_repo;
/**
* QuoteController constructor.
*
2020-10-28 11:10:49 +01:00
* @param QuoteRepository $quote_repo
2019-05-02 13:07:38 +02:00
*/
public function __construct(QuoteRepository $quote_repo)
{
parent::__construct();
$this->quote_repo = $quote_repo;
}
2019-04-02 08:36:49 +02:00
/**
* Display a listing of the resource.
*
2020-10-28 11:10:49 +01:00
* @param QuoteFilters $filters
* @return Response
2019-10-07 06:29:16 +02:00
*
*
2019-10-07 06:29:16 +02:00
* @OA\Get(
* path="/api/v1/quotes",
* operationId="getQuotes",
* tags={"quotes"},
* summary="Gets a list of quotes",
* description="Lists quotes, search and filters allow fine grained lists to be generated.
*
* Query parameters can be added to performed more fine grained filtering of the quotes, these are handled by the QuoteFilters class which defines the methods available",
2019-10-07 06:29:16 +02:00
* @OA\Parameter(ref="#/components/parameters/X-Api-Secret"),
* @OA\Parameter(ref="#/components/parameters/X-Api-Token"),
* @OA\Parameter(ref="#/components/parameters/X-Requested-With"),
* @OA\Parameter(ref="#/components/parameters/include"),
* @OA\Response(
* response=200,
* description="A list of quotes",
* @OA\Header(header="X-MINIMUM-CLIENT-VERSION", ref="#/components/headers/X-MINIMUM-CLIENT-VERSION"),
2019-10-07 06:29:16 +02:00
* @OA\Header(header="X-RateLimit-Remaining", ref="#/components/headers/X-RateLimit-Remaining"),
* @OA\Header(header="X-RateLimit-Limit", ref="#/components/headers/X-RateLimit-Limit"),
* @OA\JsonContent(ref="#/components/schemas/Quote"),
* ),
* @OA\Response(
* response=422,
* description="Validation error",
* @OA\JsonContent(ref="#/components/schemas/ValidationError"),
* ),
* @OA\Response(
* response="default",
2019-10-07 06:29:16 +02:00
* description="Unexpected Error",
* @OA\JsonContent(ref="#/components/schemas/Error"),
* ),
* )
2019-04-02 08:36:49 +02:00
*/
2019-05-02 13:07:38 +02:00
public function index(QuoteFilters $filters)
2019-04-02 08:36:49 +02:00
{
2019-05-02 13:07:38 +02:00
$quotes = Quote::filter($filters);
2019-05-02 13:07:38 +02:00
return $this->listResponse($quotes);
2019-04-02 08:36:49 +02:00
}
/**
* Show the form for creating a new resource.
*
2020-10-28 11:10:49 +01:00
* @param CreateQuoteRequest $request
* @return Response
2019-10-07 06:29:16 +02:00
*
*
*
2019-10-07 06:29:16 +02:00
* @OA\Get(
* path="/api/v1/quotes/create",
* operationId="getQuotesCreate",
* tags={"quotes"},
* summary="Gets a new blank Quote object",
* description="Returns a blank object with default values",
* @OA\Parameter(ref="#/components/parameters/X-Api-Secret"),
* @OA\Parameter(ref="#/components/parameters/X-Api-Token"),
* @OA\Parameter(ref="#/components/parameters/X-Requested-With"),
* @OA\Parameter(ref="#/components/parameters/include"),
* @OA\Response(
* response=200,
* description="A blank Quote object",
* @OA\Header(header="X-MINIMUM-CLIENT-VERSION", ref="#/components/headers/X-MINIMUM-CLIENT-VERSION"),
2019-10-07 06:29:16 +02:00
* @OA\Header(header="X-RateLimit-Remaining", ref="#/components/headers/X-RateLimit-Remaining"),
* @OA\Header(header="X-RateLimit-Limit", ref="#/components/headers/X-RateLimit-Limit"),
* @OA\JsonContent(ref="#/components/schemas/Quote"),
* ),
* @OA\Response(
* response=422,
* description="Validation error",
* @OA\JsonContent(ref="#/components/schemas/ValidationError"),
*
* ),
* @OA\Response(
* response="default",
2019-10-07 06:29:16 +02:00
* description="Unexpected Error",
* @OA\JsonContent(ref="#/components/schemas/Error"),
* ),
* )
2019-04-02 08:36:49 +02:00
*/
2019-05-02 13:07:38 +02:00
public function create(CreateQuoteRequest $request)
2019-04-02 08:36:49 +02:00
{
2019-05-02 13:07:38 +02:00
$quote = QuoteFactory::create(auth()->user()->company()->id, auth()->user()->id);
return $this->itemResponse($quote);
2019-04-02 08:36:49 +02:00
}
/**
* Store a newly created resource in storage.
*
2020-10-28 11:10:49 +01:00
* @param StoreQuoteRequest $request The request
2019-05-02 13:07:38 +02:00
*
2020-10-28 11:10:49 +01:00
* @return Response
2019-10-07 06:29:16 +02:00
*
*
*
* @OA\Post(
* path="/api/v1/quotes",
* operationId="storeQuote",
* tags={"quotes"},
* summary="Adds a Quote",
* description="Adds an Quote to the system",
* @OA\Parameter(ref="#/components/parameters/X-Api-Secret"),
* @OA\Parameter(ref="#/components/parameters/X-Api-Token"),
* @OA\Parameter(ref="#/components/parameters/X-Requested-With"),
* @OA\Parameter(ref="#/components/parameters/include"),
* @OA\Response(
* response=200,
* description="Returns the saved Quote object",
* @OA\Header(header="X-MINIMUM-CLIENT-VERSION", ref="#/components/headers/X-MINIMUM-CLIENT-VERSION"),
2019-10-07 06:29:16 +02:00
* @OA\Header(header="X-RateLimit-Remaining", ref="#/components/headers/X-RateLimit-Remaining"),
* @OA\Header(header="X-RateLimit-Limit", ref="#/components/headers/X-RateLimit-Limit"),
* @OA\JsonContent(ref="#/components/schemas/Quote"),
* ),
* @OA\Response(
* response=422,
* description="Validation error",
* @OA\JsonContent(ref="#/components/schemas/ValidationError"),
*
* ),
* @OA\Response(
* response="default",
2019-10-07 06:29:16 +02:00
* description="Unexpected Error",
* @OA\JsonContent(ref="#/components/schemas/Error"),
* ),
* )
2019-04-02 08:36:49 +02:00
*/
2019-05-02 13:07:38 +02:00
public function store(StoreQuoteRequest $request)
2019-04-02 08:36:49 +02:00
{
$client = Client::find($request->input('client_id'));
$quote = $this->quote_repo->save($request->all(), QuoteFactory::create(auth()->user()->company()->id, auth()->user()->id));
2019-05-02 13:07:38 +02:00
2021-09-15 03:20:31 +02:00
$quote = $quote->service()
->fillDefaults()
->triggeredActions($request)
->save();
2021-05-06 23:12:07 +02:00
event(new QuoteWasCreated($quote, $quote->company, Ninja::eventVars(auth()->user() ? auth()->user()->id : null)));
2020-11-03 11:19:32 +01:00
2019-05-02 13:07:38 +02:00
return $this->itemResponse($quote);
2019-04-02 08:36:49 +02:00
}
/**
* Display the specified resource.
*
2020-10-28 11:10:49 +01:00
* @param ShowQuoteRequest $request The request
* @param Quote $quote The quote
2019-05-02 13:07:38 +02:00
*
2020-10-28 11:10:49 +01:00
* @return Response
2019-10-07 06:29:16 +02:00
*
*
* @OA\Get(
* path="/api/v1/quotes/{id}",
* operationId="showQuote",
* tags={"quotes"},
* summary="Shows an Quote",
* description="Displays an Quote by id",
* @OA\Parameter(ref="#/components/parameters/X-Api-Secret"),
* @OA\Parameter(ref="#/components/parameters/X-Api-Token"),
* @OA\Parameter(ref="#/components/parameters/X-Requested-With"),
* @OA\Parameter(ref="#/components/parameters/include"),
* @OA\Parameter(
* name="id",
* in="path",
* description="The Quote Hashed ID",
* example="D2J234DFA",
* required=true,
* @OA\Schema(
* type="string",
* format="string",
* ),
* ),
* @OA\Response(
* response=200,
* description="Returns the Quote object",
* @OA\Header(header="X-MINIMUM-CLIENT-VERSION", ref="#/components/headers/X-MINIMUM-CLIENT-VERSION"),
2019-10-07 06:29:16 +02:00
* @OA\Header(header="X-RateLimit-Remaining", ref="#/components/headers/X-RateLimit-Remaining"),
* @OA\Header(header="X-RateLimit-Limit", ref="#/components/headers/X-RateLimit-Limit"),
* @OA\JsonContent(ref="#/components/schemas/Quote"),
* ),
* @OA\Response(
* response=422,
* description="Validation error",
* @OA\JsonContent(ref="#/components/schemas/ValidationError"),
*
* ),
* @OA\Response(
* response="default",
2019-10-07 06:29:16 +02:00
* description="Unexpected Error",
* @OA\JsonContent(ref="#/components/schemas/Error"),
* ),
* )
2019-04-02 08:36:49 +02:00
*/
2019-05-02 13:07:38 +02:00
public function show(ShowQuoteRequest $request, Quote $quote)
2019-04-02 08:36:49 +02:00
{
2019-05-02 13:07:38 +02:00
return $this->itemResponse($quote);
2019-04-02 08:36:49 +02:00
}
/**
* Show the form for editing the specified resource.
*
2020-10-28 11:10:49 +01:00
* @param EditQuoteRequest $request The request
* @param Quote $quote The quote
2019-05-02 13:07:38 +02:00
*
2020-10-28 11:10:49 +01:00
* @return Response
2019-10-07 06:29:16 +02:00
*
*
2019-10-07 06:29:16 +02:00
* @OA\Get(
* path="/api/v1/quotes/{id}/edit",
* operationId="editQuote",
* tags={"quotes"},
* summary="Shows an Quote for editting",
* description="Displays an Quote by id",
* @OA\Parameter(ref="#/components/parameters/X-Api-Secret"),
* @OA\Parameter(ref="#/components/parameters/X-Api-Token"),
* @OA\Parameter(ref="#/components/parameters/X-Requested-With"),
* @OA\Parameter(ref="#/components/parameters/include"),
* @OA\Parameter(
* name="id",
* in="path",
* description="The Quote Hashed ID",
* example="D2J234DFA",
* required=true,
* @OA\Schema(
* type="string",
* format="string",
* ),
* ),
* @OA\Response(
* response=200,
* description="Returns the Quote object",
* @OA\Header(header="X-MINIMUM-CLIENT-VERSION", ref="#/components/headers/X-MINIMUM-CLIENT-VERSION"),
2019-10-07 06:29:16 +02:00
* @OA\Header(header="X-RateLimit-Remaining", ref="#/components/headers/X-RateLimit-Remaining"),
* @OA\Header(header="X-RateLimit-Limit", ref="#/components/headers/X-RateLimit-Limit"),
* @OA\JsonContent(ref="#/components/schemas/Quote"),
* ),
* @OA\Response(
* response=422,
* description="Validation error",
* @OA\JsonContent(ref="#/components/schemas/ValidationError"),
*
* ),
* @OA\Response(
* response="default",
2019-10-07 06:29:16 +02:00
* description="Unexpected Error",
* @OA\JsonContent(ref="#/components/schemas/Error"),
* ),
* )
*/
2019-05-02 13:07:38 +02:00
public function edit(EditQuoteRequest $request, Quote $quote)
2019-04-02 08:36:49 +02:00
{
2019-05-02 13:07:38 +02:00
return $this->itemResponse($quote);
}
2019-04-02 08:36:49 +02:00
/**
* Update the specified resource in storage.
*
2020-10-28 11:10:49 +01:00
* @param UpdateQuoteRequest $request The request
* @param Quote $quote The quote
2019-05-02 13:07:38 +02:00
*
2020-10-28 11:10:49 +01:00
* @return Response
2019-10-07 06:29:16 +02:00
*
*
2019-10-07 06:29:16 +02:00
* @OA\Put(
* path="/api/v1/quotes/{id}",
* operationId="updateQuote",
* tags={"quotes"},
* summary="Updates an Quote",
* description="Handles the updating of an Quote by id",
* @OA\Parameter(ref="#/components/parameters/X-Api-Secret"),
* @OA\Parameter(ref="#/components/parameters/X-Api-Token"),
* @OA\Parameter(ref="#/components/parameters/X-Requested-With"),
* @OA\Parameter(ref="#/components/parameters/include"),
* @OA\Parameter(
* name="id",
* in="path",
* description="The Quote Hashed ID",
* example="D2J234DFA",
* required=true,
* @OA\Schema(
* type="string",
* format="string",
* ),
* ),
* @OA\Response(
* response=200,
* description="Returns the Quote object",
* @OA\Header(header="X-MINIMUM-CLIENT-VERSION", ref="#/components/headers/X-MINIMUM-CLIENT-VERSION"),
2019-10-07 06:29:16 +02:00
* @OA\Header(header="X-RateLimit-Remaining", ref="#/components/headers/X-RateLimit-Remaining"),
* @OA\Header(header="X-RateLimit-Limit", ref="#/components/headers/X-RateLimit-Limit"),
* @OA\JsonContent(ref="#/components/schemas/Quote"),
* ),
* @OA\Response(
* response=422,
* description="Validation error",
* @OA\JsonContent(ref="#/components/schemas/ValidationError"),
*
* ),
* @OA\Response(
* response="default",
2019-10-07 06:29:16 +02:00
* description="Unexpected Error",
* @OA\JsonContent(ref="#/components/schemas/Error"),
* ),
* )
2019-04-02 08:36:49 +02:00
*/
2019-05-02 13:07:38 +02:00
public function update(UpdateQuoteRequest $request, Quote $quote)
2019-04-02 08:36:49 +02:00
{
if ($request->entityIsDeleted($quote)) {
return $request->disallowUpdate();
}
$quote = $this->quote_repo->save($request->all(), $quote);
2019-05-02 13:07:38 +02:00
$quote->service()
->triggeredActions($request)
->deletePdf();
2021-05-06 23:12:07 +02:00
event(new QuoteWasUpdated($quote, $quote->company, Ninja::eventVars(auth()->user() ? auth()->user()->id : null)));
2020-11-03 11:19:32 +01:00
2019-05-02 13:07:38 +02:00
return $this->itemResponse($quote);
2019-04-02 08:36:49 +02:00
}
/**
* Remove the specified resource from storage.
*
2020-10-28 11:10:49 +01:00
* @param DestroyQuoteRequest $request
* @param Quote $quote
2019-05-02 13:07:38 +02:00
*
2020-10-28 11:10:49 +01:00
* @return Response
2019-10-07 06:29:16 +02:00
*
*
2020-10-28 11:10:49 +01:00
* @throws \Exception
2019-10-07 06:29:16 +02:00
* @OA\Delete(
* path="/api/v1/quotes/{id}",
* operationId="deleteQuote",
* tags={"quotes"},
* summary="Deletes a Quote",
* description="Handles the deletion of an Quote by id",
* @OA\Parameter(ref="#/components/parameters/X-Api-Secret"),
* @OA\Parameter(ref="#/components/parameters/X-Api-Token"),
* @OA\Parameter(ref="#/components/parameters/X-Requested-With"),
* @OA\Parameter(ref="#/components/parameters/include"),
* @OA\Parameter(
* name="id",
* in="path",
* description="The Quote Hashed ID",
* example="D2J234DFA",
* required=true,
* @OA\Schema(
* type="string",
* format="string",
* ),
* ),
* @OA\Response(
* response=200,
* description="Returns a HTTP status",
* @OA\Header(header="X-MINIMUM-CLIENT-VERSION", ref="#/components/headers/X-MINIMUM-CLIENT-VERSION"),
2019-10-07 06:29:16 +02:00
* @OA\Header(header="X-RateLimit-Remaining", ref="#/components/headers/X-RateLimit-Remaining"),
* @OA\Header(header="X-RateLimit-Limit", ref="#/components/headers/X-RateLimit-Limit"),
* ),
* @OA\Response(
* response=422,
* description="Validation error",
* @OA\JsonContent(ref="#/components/schemas/ValidationError"),
*
* ),
* @OA\Response(
* response="default",
2019-10-07 06:29:16 +02:00
* description="Unexpected Error",
* @OA\JsonContent(ref="#/components/schemas/Error"),
* ),
* )
2019-04-02 08:36:49 +02:00
*/
2019-05-02 13:07:38 +02:00
public function destroy(DestroyQuoteRequest $request, Quote $quote)
2019-04-02 08:36:49 +02:00
{
$this->quote_repo->delete($quote);
2019-05-02 13:07:38 +02:00
return $this->itemResponse($quote->fresh());
2019-05-02 13:07:38 +02:00
}
/**
* Perform bulk actions on the list view.
*
2019-05-02 13:07:38 +02:00
* @return Collection
2019-10-07 06:29:16 +02:00
*
*
2019-10-07 06:29:16 +02:00
* @OA\Post(
* path="/api/v1/quotes/bulk",
* operationId="bulkQuotes",
* tags={"quotes"},
* summary="Performs bulk actions on an array of quotes",
* description="",
* @OA\Parameter(ref="#/components/parameters/X-Api-Secret"),
* @OA\Parameter(ref="#/components/parameters/X-Api-Token"),
* @OA\Parameter(ref="#/components/parameters/X-Requested-With"),
* @OA\Parameter(ref="#/components/parameters/index"),
* @OA\RequestBody(
2019-10-07 06:57:14 +02:00
* description="Hashed ids",
2019-10-07 06:29:16 +02:00
* required=true,
* @OA\MediaType(
* mediaType="application/json",
* @OA\Schema(
* type="array",
* @OA\Items(
* type="integer",
* description="Array of hashed IDs to be bulk 'actioned",
* example="[0,1,2,3]",
* ),
* )
* )
* ),
* @OA\Response(
* response=200,
2019-10-07 06:57:14 +02:00
* description="The Quote response",
* @OA\Header(header="X-MINIMUM-CLIENT-VERSION", ref="#/components/headers/X-MINIMUM-CLIENT-VERSION"),
2019-10-07 06:29:16 +02:00
* @OA\Header(header="X-RateLimit-Remaining", ref="#/components/headers/X-RateLimit-Remaining"),
* @OA\Header(header="X-RateLimit-Limit", ref="#/components/headers/X-RateLimit-Limit"),
2019-10-07 06:57:14 +02:00
* @OA\JsonContent(ref="#/components/schemas/Quote"),
2019-10-07 06:29:16 +02:00
* ),
* @OA\Response(
* response=422,
* description="Validation error",
* @OA\JsonContent(ref="#/components/schemas/ValidationError"),
* ),
* @OA\Response(
* response="default",
2019-10-07 06:29:16 +02:00
* description="Unexpected Error",
* @OA\JsonContent(ref="#/components/schemas/Error"),
* ),
* )
2019-05-02 13:07:38 +02:00
*/
public function bulk(BulkActionQuoteRequest $request)
2019-05-02 13:07:38 +02:00
{
$action = request()->input('action');
2019-05-02 13:07:38 +02:00
$ids = request()->input('ids');
2022-07-27 06:57:17 +02:00
if(Ninja::isHosted() && (stripos($action, 'email') !== false) && !auth()->user()->company()->account->account_sms_verified)
2022-07-27 03:21:12 +02:00
return response(['message' => 'Please verify your account to send emails.'], 400);
$quotes = Quote::withTrashed()->whereIn('id', $this->transformKeys($ids))->company()->get();
if (! $quotes) {
2021-01-25 00:04:50 +01:00
return response()->json(['message' => ctrans('texts.quote_not_found')]);
}
/*
* Download Invoice/s
*/
2022-03-02 03:26:30 +01:00
if ($action == 'bulk_download' && $quotes->count() >= 1) {
$quotes->each(function ($quote) {
if (auth()->user()->cannot('view', $quote)) {
2021-01-25 00:04:50 +01:00
return response()->json(['message'=> ctrans('texts.access_denied')]);
}
});
2022-03-02 03:51:38 +01:00
ZipQuotes::dispatch($quotes, $quotes->first()->company, auth()->user());
2021-01-25 00:04:50 +01:00
return response()->json(['message' => ctrans('texts.sent_message')], 200);
}
2019-05-02 13:07:38 +02:00
if ($action == 'convert' || $action == 'convert_to_invoice') {
2020-05-28 13:22:25 +02:00
$this->entity_type = Quote::class;
$this->entity_transformer = QuoteTransformer::class;
$quotes->each(function ($quote, $key) use ($action) {
if (auth()->user()->can('edit', $quote) && $quote->service()->isConvertable()) {
$quote->service()->convertToInvoice();
}
});
return $this->listResponse(Quote::withTrashed()->whereIn('id', $this->transformKeys($ids))->company());
}
2022-08-15 05:49:47 +02:00
if($action == 'convert_to_project')
{
$quotes->each(function ($quote, $key) use ($action) {
if (auth()->user()->can('edit', $quote))
{
$project = CloneQuoteToProjectFactory::create($quote, auth()->user()->id);
if (empty($project->number)) {
$project->number = $this->getNextProjectNumber($project);
}
$project->save();
$quote->project_id = $project->id;
$quote->save();
}
});
return $this->listResponse(Quote::withTrashed()->whereIn('id', $this->transformKeys($ids))->company());
}
/*
* Send the other actions to the switch
*/
$quotes->each(function ($quote, $key) use ($action) {
if (auth()->user()->can('edit', $quote)) {
$this->performAction($quote, $action, true);
}
2019-05-02 13:07:38 +02:00
});
/* Need to understand which permission are required for the given bulk action ie. view / edit */
return $this->listResponse(Quote::withTrashed()->whereIn('id', $this->transformKeys($ids))->company());
2019-05-02 13:07:38 +02:00
}
2019-10-07 06:29:16 +02:00
/**
* Quote Actions.
2019-10-07 06:29:16 +02:00
*
*
*
2019-10-07 06:29:16 +02:00
* @OA\Get(
* path="/api/v1/quotes/{id}/{action}",
* operationId="actionQuote",
* tags={"quotes"},
* summary="Performs a custom action on an Quote",
* description="Performs a custom action on an Quote.
2020-10-28 11:10:49 +01:00
The current range of actions are as follows
- clone_to_quote
- history
- delivery_note
- mark_paid
- download
- archive
- delete
- convert
- convert_to_invoice
2020-10-28 11:10:49 +01:00
- email",
2019-10-07 06:29:16 +02:00
* @OA\Parameter(ref="#/components/parameters/X-Api-Secret"),
* @OA\Parameter(ref="#/components/parameters/X-Api-Token"),
* @OA\Parameter(ref="#/components/parameters/X-Requested-With"),
* @OA\Parameter(ref="#/components/parameters/include"),
* @OA\Parameter(
* name="id",
* in="path",
* description="The Quote Hashed ID",
* example="D2J234DFA",
* required=true,
* @OA\Schema(
* type="string",
* format="string",
* ),
* ),
* @OA\Parameter(
* name="action",
* in="path",
* description="The action string to be performed",
* example="clone_to_quote",
* required=true,
* @OA\Schema(
* type="string",
* format="string",
* ),
* ),
2019-10-07 06:29:16 +02:00
* @OA\Response(
* response=200,
* description="Returns the Quote object",
* @OA\Header(header="X-MINIMUM-CLIENT-VERSION", ref="#/components/headers/X-MINIMUM-CLIENT-VERSION"),
2019-10-07 06:29:16 +02:00
* @OA\Header(header="X-RateLimit-Remaining", ref="#/components/headers/X-RateLimit-Remaining"),
* @OA\Header(header="X-RateLimit-Limit", ref="#/components/headers/X-RateLimit-Limit"),
* @OA\JsonContent(ref="#/components/schemas/Quote"),
* ),
* @OA\Response(
* response=422,
* description="Validation error",
* @OA\JsonContent(ref="#/components/schemas/ValidationError"),
*
* ),
* @OA\Response(
* response="default",
2019-10-07 06:29:16 +02:00
* description="Unexpected Error",
* @OA\JsonContent(ref="#/components/schemas/Error"),
* ),
* )
2020-10-28 11:10:49 +01:00
* @param ActionQuoteRequest $request
* @param Quote $quote
* @param $action
* @return \Illuminate\Http\JsonResponse|Response|mixed|\Symfony\Component\HttpFoundation\StreamedResponse
*/
public function action(ActionQuoteRequest $request, Quote $quote, $action)
{
return $this->performAction($quote, $action);
}
private function performAction(Quote $quote, $action, $bulk = false)
2019-05-02 13:07:38 +02:00
{
switch ($action) {
case 'convert':
case 'convert_to_invoice':
$this->entity_type = Invoice::class;
$this->entity_transformer = InvoiceTransformer::class;
return $this->itemResponse($quote->service()->convertToInvoice());
break;
2022-08-15 04:31:20 +02:00
2019-05-02 13:07:38 +02:00
case 'clone_to_invoice':
$this->entity_type = Invoice::class;
$this->entity_transformer = InvoiceTransformer::class;
$invoice = CloneQuoteToInvoiceFactory::create($quote, auth()->user()->id);
return $this->itemResponse($invoice);
2019-05-02 13:07:38 +02:00
break;
case 'clone_to_quote':
$quote = CloneQuoteFactory::create($quote, auth()->user()->id);
return $this->itemResponse($quote);
2019-05-02 13:07:38 +02:00
break;
case 'approve':
if (! in_array($quote->status_id, [Quote::STATUS_SENT, Quote::STATUS_DRAFT])) {
2021-01-25 00:04:50 +01:00
return response()->json(['message' => ctrans('texts.quote_unapprovable')], 400);
}
return $this->itemResponse($quote->service()->approveWithNoCoversion()->save());
break;
2019-05-02 13:07:38 +02:00
case 'history':
// code...
2019-05-02 13:07:38 +02:00
break;
case 'download':
2021-05-15 04:19:36 +02:00
2021-07-07 13:39:49 +02:00
//$file = $quote->pdf_file_path();
$file = $quote->service()->getQuotePdf();
return response()->streamDownload(function () use ($file) {
echo Storage::get($file);
}, basename($file), ['Content-Type' => 'application/pdf']);
2021-07-07 13:39:49 +02:00
2021-05-15 04:19:36 +02:00
break;
2020-11-03 11:19:32 +01:00
case 'restore':
$this->quote_repo->restore($quote);
if (! $bulk) {
2020-11-03 11:19:32 +01:00
return $this->listResponse($quote);
2020-11-25 15:19:52 +01:00
}
2020-11-03 11:19:32 +01:00
break;
2019-05-02 13:07:38 +02:00
case 'archive':
2020-08-23 03:08:18 +02:00
$this->quote_repo->archive($quote);
if (! $bulk) {
2020-11-03 11:19:32 +01:00
return $this->listResponse($quote);
2020-11-25 15:19:52 +01:00
}
2020-11-03 11:19:32 +01:00
2019-05-02 13:07:38 +02:00
break;
case 'delete':
$this->quote_repo->delete($quote);
if (! $bulk) {
2020-11-03 11:19:32 +01:00
return $this->listResponse($quote);
2020-11-25 15:19:52 +01:00
}
2020-11-03 11:19:32 +01:00
2019-05-02 13:07:38 +02:00
break;
case 'email':
2020-08-14 00:08:10 +02:00
$quote->service()->sendEmail();
2021-01-25 00:04:50 +01:00
return response()->json(['message'=> ctrans('texts.sent_message')], 200);
2019-05-02 13:07:38 +02:00
break;
2022-07-08 01:00:36 +02:00
case 'send_email':
$quote->service()->sendEmail();
return response()->json(['message'=> ctrans('texts.sent_message')], 200);
break;
case 'mark_sent':
$quote->service()->markSent()->save();
if (! $bulk) {
return $this->itemResponse($quote);
}
2020-11-03 11:19:32 +01:00
break;
2019-05-02 13:07:38 +02:00
default:
return response()->json(['message' => ctrans('texts.action_unavailable', ['action' => $action])], 400);
2019-05-02 13:07:38 +02:00
break;
}
2019-04-02 08:36:49 +02:00
}
public function downloadPdf($invitation_key)
{
$invitation = $this->quote_repo->getInvitationByKey($invitation_key);
$contact = $invitation->contact;
$quote = $invitation->quote;
2021-07-07 13:39:49 +02:00
$file = $quote->service()->getQuotePdf($contact);
2021-09-20 13:16:28 +02:00
$headers = ['Content-Type' => 'application/pdf'];
if (request()->input('inline') == 'true') {
2021-09-20 13:16:28 +02:00
$headers = array_merge($headers, ['Content-Disposition' => 'inline']);
}
2021-09-20 13:16:28 +02:00
return response()->streamDownload(function () use ($file) {
echo Storage::get($file);
}, basename($file), $headers);
}
2021-02-15 21:58:19 +01:00
/**
* Update the specified resource in storage.
*
* @param UploadQuoteRequest $request
* @param Quote $quote
* @return Response
*
*
*
* @OA\Put(
* path="/api/v1/quotes/{id}/upload",
* operationId="uploadQuote",
* tags={"quotes"},
* summary="Uploads a document to a quote",
* description="Handles the uploading of a document to a quote",
* @OA\Parameter(ref="#/components/parameters/X-Api-Secret"),
* @OA\Parameter(ref="#/components/parameters/X-Api-Token"),
* @OA\Parameter(ref="#/components/parameters/X-Requested-With"),
* @OA\Parameter(ref="#/components/parameters/include"),
* @OA\Parameter(
* name="id",
* in="path",
* description="The Quote Hashed ID",
* example="D2J234DFA",
* required=true,
* @OA\Schema(
* type="string",
* format="string",
* ),
* ),
* @OA\Response(
* response=200,
* description="Returns the Quote object",
* @OA\Header(header="X-MINIMUM-CLIENT-VERSION", ref="#/components/headers/X-MINIMUM-CLIENT-VERSION"),
* @OA\Header(header="X-RateLimit-Remaining", ref="#/components/headers/X-RateLimit-Remaining"),
* @OA\Header(header="X-RateLimit-Limit", ref="#/components/headers/X-RateLimit-Limit"),
* @OA\JsonContent(ref="#/components/schemas/Quote"),
* ),
* @OA\Response(
* response=422,
* description="Validation error",
* @OA\JsonContent(ref="#/components/schemas/ValidationError"),
*
* ),
* @OA\Response(
* response="default",
* description="Unexpected Error",
* @OA\JsonContent(ref="#/components/schemas/Error"),
* ),
* )
*/
public function upload(UploadQuoteRequest $request, Quote $quote)
{
if (! $this->checkFeature(Account::FEATURE_DOCUMENTS)) {
2021-03-07 22:32:38 +01:00
return $this->featureFailure();
}
2021-03-07 22:32:38 +01:00
if ($request->has('documents')) {
2021-02-15 21:58:19 +01:00
$this->saveDocuments($request->file('documents'), $quote);
}
2021-02-15 21:58:19 +01:00
return $this->itemResponse($quote->fresh());
}
}