automngr/app/Http/Controllers/ContactController.php

306 lines
10 KiB
PHP

<?php
namespace App\Http\Controllers;
use Inertia\Inertia;
use App\Exports\Export;
use App\Models\Contact;
use App\Models\Contract;
use Illuminate\Http\Request;
use Maatwebsite\Excel\Facades\Excel;
use Illuminate\Support\Facades\Redirect;
class ContactController extends Controller
{
public function index(Request $request)
{
return $this->renderContactsList($request, Contact::query(), 'Contacts/Index');
}
public function sellers(Request $request)
{
return $this->renderContactsList($request, Contact::has('buyContracts'), 'Contacts/Sellers');
}
public function buyers(Request $request)
{
return $this->renderContactsList($request, Contact::has('sellContracts'), 'Contacts/Buyers');
}
private function renderContactsList(Request $request, $contacts, string $renderPage) {
$direction = $this->getDirection($request);
$sortBy = $this->getSortBy($request);
$contacts = $this->getWithCustomSort($contacts, $sortBy, $direction);
return Inertia::render($renderPage, [
'filters' => $request->all('search', 'trashed'),
'sort' => [
'by' => $sortBy,
'direction' => $direction,
],
'contacts' => $contacts->filter($request->only('search', 'trashed'))
->paginate(50)
->withQueryString()
->through(fn ($contact) => [
'id' => $contact->id,
'name' => $contact->name,
'company' => $contact->company,
'phone' => $contact->phone,
'email' => $contact->email,
'address' => $contact->address,
'fullCity' => $contact->fullCity,
'link' => route('contacts.show', $contact),
'deleted_at' => $contact->deleted_at,
]),
]);
}
public function letter(Contact $contact)
{
$templateProcessor = new \PhpOffice\PhpWord\TemplateProcessor(resource_path() . '/docx/letter.docx');
$templateProcessor->setValue('date', date("d.m.Y"));
$templateProcessor->setValue('company', $contact->company);
$templateProcessor->setValue('name', $contact->name);
$templateProcessor->setValue('address', $contact->address);
$templateProcessor->setValue('city', $contact->full_city);
$templateProcessor->setValue('country', $contact->country !== 'CH' ? $contact->country : '');
ob_start();
$templateProcessor->saveAs("php://output");
$contents = ob_get_contents();
ob_end_clean();
return response()->streamDownload(function () use ($contents) {
echo $contents;
}, 'Briefvorlage ' . $contact->title . '.docx');
}
public function print(Request $request)
{
return $this->printList($request, Contact::query(), date('Y-m-d') . '-Alle-Kontakte.xlsx');
}
public function buyersPrint(Request $request)
{
return $this->printList($request, Contact::has('buyContracts'), date('Y-m-d') . '-Verkäufer.xlsx');
}
public function sellersPrint(Request $request)
{
return $this->printList($request, Contact::has('sellContracts'), date('Y-m-d') . '-Käufer.xlsx');
}
private function printList(Request $request, $contacts, $title)
{
$headings = [
'Nachname',
'Vorname',
'Firma',
'Adresse',
'PLZ',
'Ort',
'Land',
'Email',
'Telefon',
'Bemerkungen',
];
$direction = $this->getDirection($request);
$sortBy = $this->getSortBy($request);
$contacts = $this->getWithCustomSort($contacts, $sortBy, $direction)
->filter($request->only('search', 'trashed'))
->get()
->map(function ($contact) {
return [
'lastname' => $contact->lastname,
'firstname' => $contact->firstname,
'company' => $contact->company,
'address' => $contact->address,
'zip' => $contact->zip,
'city' => $contact->city,
'country' => $contact->country,
'email' => $contact->email,
'phone' => $contact->phone,
'notes' => $contact->notes,
];
});
return Excel::download(new Export($contacts, $headings), $title);
}
private function getWithCustomSort($contacts, string $sortBy, string $direction)
{
return match ($sortBy) {
'company' => $contacts->orderBy('company', $direction),
'fullCity' => $contacts->orderBy('city', $direction),
'email' => $contacts->orderBy('email', $direction),
'address' => $contacts->orderBy('address', $direction),
default => $contacts->orderByName($direction),
};
}
private function getSortBy(Request $request)
{
if ($request->has('sortby')) {
return $request->get('sortby');
}
return 'name';
}
public function create()
{
return Inertia::render('Contacts/Create');
}
public function store(Request $request)
{
$contact = Contact::create(
$request->validate($this->getValidationRules())
);
session()->flash('flash.banner', 'Kontakt erstellt.');
return Redirect::route('contacts.show', $contact);
}
public function storeForContract(Request $request)
{
$contact = Contact::create(
$request->validate($this->getValidationRules())
);
return response()->json([
'id' => $contact->id,
'title' => $contact->full_title,
'name' => $contact->name,
'firstname' => $contact->firstname,
'lastname' => $contact->lastname,
'phone' => $contact->phone,
'address' => $contact->address,
'zip' => $contact->zip,
'city' => $contact->city,
'country' => $contact->country,
'company' => $contact->company,
'email' => $contact->email,
]);
}
public function edit(Contact $contact)
{
return Inertia::render('Contacts/Edit', [
'contact' => [
'id' => $contact->id,
'firstname' => $contact->firstname,
'lastname' => $contact->lastname,
'company' => $contact->company,
'title' => $contact->title,
'email' => $contact->email,
'notes' => $contact->notes,
'phone' => $contact->phone,
'address' => $contact->address,
'zip' => $contact->zip,
'city' => $contact->city,
'country' => $contact->country,
'deleted_at' => $contact->deleted_at,
]
]);
}
public function show(Contact $contact)
{
return Inertia::render('Contacts/Show', [
'contact' => [
'id' => $contact->id,
'firstname' => $contact->firstname,
'lastname' => $contact->lastname,
'company' => $contact->company,
'title' => $contact->title,
'name' => $contact->name,
'email' => $contact->email,
'notes' => $contact->notes,
'phone' => $contact->phone,
'address' => $contact->address,
'zip' => $contact->zip,
'city' => $contact->city,
'country' => $contact->country,
'deleted_at' => $contact->deleted_at,
'documents' => $contact->documents()->orderBy('created_at', 'asc')->get()
->map(function ($document) {
return [
'id' => $document->id,
'name' => $document->name,
'size' => $document->size,
'extension' => $document->extension,
'link' => $document->link,
'created_at' => $document->created_at,
];
}),
'buy_contracts' => $contact->buyContracts()
->orderBy('date', 'desc')
->with('car')
->get()
->map(function ($contract) { return $this->getContractFields($contract); }),
'sell_contracts' => $contact->sellContracts()
->orderBy('date', 'desc')
->with('car')
->get()
->map(function ($contract) { return $this->getContractFields($contract); }),
]
]);
}
private function getContractFields(?Contract $contract) {
if (!$contract) {
return null;
}
$car = $contract->car;
return [
'id' => $contract->id,
'date' => $contract->date_formatted,
'price' => $contract->price->format(),
'car' => $car->name_with_year,
'link' => route('contracts.show', $contract),
];
}
public function update(Request $request, Contact $contact)
{
$contact->update(
$request->validate($this->getValidationRules())
);
session()->flash('flash.banner', 'Kontakt geändert.');
return Redirect::route('contacts.show', $contact);
}
private function getValidationRules()
{
return [
'firstname' => ['max:75'],
'lastname' => ['max:75'],
'email' => ['nullable', 'max:75', 'email'],
'phone' => ['nullable', 'max:75'],
'address' => ['nullable', 'max:150'],
'zip' => ['nullable', 'max:6'],
'city' => ['nullable', 'max:75'],
'country' => ['nullable', 'max:3'],
'company' => ['nullable', 'max:75'],
'notes' => ['nullable'],
];
}
public function destroy(Contact $contact)
{
$contact->delete();
session()->flash('flash.banner', 'Kontakt gelöscht.');
return Redirect::back();
}
public function restore(Contact $contact)
{
$contact->restore();
session()->flash('flash.banner', 'Kontakt wiederhergestellt.');
return Redirect::back();
}
}