Du kan inte välja fler än 25 ämnen Ämnen måste starta med en bokstav eller siffra, kan innehålla bindestreck ('-') och vara max 35 tecken långa.

ClientResource.php 20KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391
  1. <?php
  2. namespace App\Filament\Company\Resources\Sales;
  3. use App\Filament\Company\Resources\Sales\ClientResource\Pages;
  4. use App\Filament\Forms\Components\CreateCurrencySelect;
  5. use App\Filament\Forms\Components\CustomSection;
  6. use App\Filament\Forms\Components\PhoneBuilder;
  7. use App\Filament\Tables\Columns;
  8. use App\Models\Common\Address;
  9. use App\Models\Common\Client;
  10. use App\Models\Locale\Country;
  11. use App\Models\Locale\State;
  12. use App\Utilities\Currency\CurrencyConverter;
  13. use Filament\Forms;
  14. use Filament\Forms\Form;
  15. use Filament\Forms\Get;
  16. use Filament\Forms\Set;
  17. use Filament\Resources\Resource;
  18. use Filament\Tables;
  19. use Filament\Tables\Table;
  20. use Illuminate\Database\Eloquent\Builder;
  21. use Illuminate\Support\HtmlString;
  22. class ClientResource extends Resource
  23. {
  24. protected static ?string $model = Client::class;
  25. public static function form(Form $form): Form
  26. {
  27. return $form
  28. ->schema([
  29. Forms\Components\Section::make('General Information')
  30. ->schema([
  31. Forms\Components\Group::make()
  32. ->columns()
  33. ->schema([
  34. Forms\Components\TextInput::make('name')
  35. ->label('Client Name')
  36. ->required()
  37. ->maxLength(255),
  38. Forms\Components\TextInput::make('account_number')
  39. ->maxLength(255)
  40. ->columnStart(1),
  41. Forms\Components\TextInput::make('website')
  42. ->maxLength(255),
  43. Forms\Components\Textarea::make('notes')
  44. ->columnSpanFull(),
  45. ]),
  46. CustomSection::make('Primary Contact')
  47. ->relationship('primaryContact')
  48. ->contained(false)
  49. ->schema([
  50. Forms\Components\Hidden::make('is_primary')
  51. ->default(true),
  52. Forms\Components\TextInput::make('first_name')
  53. ->label('First Name')
  54. ->required()
  55. ->maxLength(255),
  56. Forms\Components\TextInput::make('last_name')
  57. ->label('Last Name')
  58. ->required()
  59. ->maxLength(255),
  60. Forms\Components\TextInput::make('email')
  61. ->label('Email')
  62. ->required()
  63. ->email()
  64. ->columnSpanFull()
  65. ->maxLength(255),
  66. PhoneBuilder::make('phones')
  67. ->hiddenLabel()
  68. ->blockLabels(false)
  69. ->default([
  70. ['type' => 'primary'],
  71. ])
  72. ->columnSpanFull()
  73. ->blocks([
  74. Forms\Components\Builder\Block::make('primary')
  75. ->schema([
  76. Forms\Components\TextInput::make('number')
  77. ->label('Phone')
  78. ->required()
  79. ->maxLength(15),
  80. ])->maxItems(1),
  81. Forms\Components\Builder\Block::make('mobile')
  82. ->schema([
  83. Forms\Components\TextInput::make('number')
  84. ->label('Mobile')
  85. ->required()
  86. ->maxLength(15),
  87. ])->maxItems(1),
  88. Forms\Components\Builder\Block::make('toll_free')
  89. ->schema([
  90. Forms\Components\TextInput::make('number')
  91. ->label('Toll Free')
  92. ->required()
  93. ->maxLength(15),
  94. ])->maxItems(1),
  95. Forms\Components\Builder\Block::make('fax')
  96. ->schema([
  97. Forms\Components\TextInput::make('number')
  98. ->label('Fax')
  99. ->live()
  100. ->maxLength(15),
  101. ])->maxItems(1),
  102. ])
  103. ->deletable(fn (PhoneBuilder $builder) => $builder->getItemsCount() > 1)
  104. ->reorderable(false)
  105. ->blockNumbers(false)
  106. ->addActionLabel('Add Phone'),
  107. ])->columns(),
  108. Forms\Components\Repeater::make('secondaryContacts')
  109. ->relationship()
  110. ->hiddenLabel()
  111. ->extraAttributes([
  112. 'class' => 'uncontained',
  113. ])
  114. ->columns()
  115. ->defaultItems(0)
  116. ->maxItems(3)
  117. ->itemLabel(function (Forms\Components\Repeater $component, array $state): ?string {
  118. if ($component->getItemsCount() === 1) {
  119. return 'Secondary Contact';
  120. }
  121. $firstName = $state['first_name'] ?? null;
  122. $lastName = $state['last_name'] ?? null;
  123. if ($firstName && $lastName) {
  124. return "{$firstName} {$lastName}";
  125. }
  126. if ($firstName) {
  127. return $firstName;
  128. }
  129. return 'Secondary Contact';
  130. })
  131. ->addActionLabel('Add Contact')
  132. ->schema([
  133. Forms\Components\TextInput::make('first_name')
  134. ->label('First Name')
  135. ->required()
  136. ->live(onBlur: true)
  137. ->maxLength(255),
  138. Forms\Components\TextInput::make('last_name')
  139. ->label('Last Name')
  140. ->required()
  141. ->live(onBlur: true)
  142. ->maxLength(255),
  143. Forms\Components\TextInput::make('email')
  144. ->label('Email')
  145. ->required()
  146. ->email()
  147. ->maxLength(255),
  148. PhoneBuilder::make('phones')
  149. ->hiddenLabel()
  150. ->blockLabels(false)
  151. ->default([
  152. ['type' => 'primary'],
  153. ])
  154. ->blocks([
  155. Forms\Components\Builder\Block::make('primary')
  156. ->schema([
  157. Forms\Components\TextInput::make('number')
  158. ->label('Phone')
  159. ->required()
  160. ->maxLength(255),
  161. ])->maxItems(1),
  162. ])
  163. ->addable(false)
  164. ->deletable(false)
  165. ->reorderable(false)
  166. ->blockNumbers(false),
  167. ]),
  168. ])->columns(1),
  169. Forms\Components\Section::make('Billing')
  170. ->schema([
  171. CreateCurrencySelect::make('currency_code'),
  172. CustomSection::make('Billing Address')
  173. ->relationship('billingAddress')
  174. ->saveRelationshipsUsing(null)
  175. ->dehydrated(true)
  176. ->contained(false)
  177. ->schema([
  178. Forms\Components\Hidden::make('type')
  179. ->default('billing'),
  180. Forms\Components\TextInput::make('address_line_1')
  181. ->label('Address Line 1')
  182. ->required()
  183. ->maxLength(255),
  184. Forms\Components\TextInput::make('address_line_2')
  185. ->label('Address Line 2')
  186. ->maxLength(255),
  187. Forms\Components\Select::make('country')
  188. ->searchable()
  189. ->localizeLabel()
  190. ->live()
  191. ->options(Country::getAvailableCountryOptions())
  192. ->afterStateUpdated(static function (Set $set) {
  193. $set('state_id', null);
  194. })
  195. ->required(),
  196. Forms\Components\Select::make('state_id')
  197. ->localizeLabel('State / Province')
  198. ->searchable()
  199. ->options(static fn (Get $get) => State::getStateOptions($get('country')))
  200. ->nullable(),
  201. Forms\Components\TextInput::make('city')
  202. ->label('City')
  203. ->required()
  204. ->maxLength(255),
  205. Forms\Components\TextInput::make('postal_code')
  206. ->label('Postal Code / Zip Code')
  207. ->required()
  208. ->maxLength(255),
  209. ])->columns(),
  210. ])
  211. ->columns(1),
  212. Forms\Components\Section::make('Shipping')
  213. ->relationship('shippingAddress')
  214. ->saveRelationshipsUsing(null)
  215. ->dehydrated(true)
  216. ->schema([
  217. Forms\Components\Hidden::make('type')
  218. ->default('shipping'),
  219. Forms\Components\TextInput::make('recipient')
  220. ->label('Recipient')
  221. ->required()
  222. ->maxLength(255),
  223. Forms\Components\TextInput::make('phone')
  224. ->label('Phone')
  225. ->required()
  226. ->maxLength(255),
  227. CustomSection::make('Shipping Address')
  228. ->contained(false)
  229. ->schema([
  230. Forms\Components\Checkbox::make('same_as_billing')
  231. ->label('Same as Billing Address')
  232. ->live()
  233. ->afterStateHydrated(function (?Address $record, Forms\Components\Checkbox $component) {
  234. if (! $record || $record->parent_address_id) {
  235. return $component->state(true);
  236. }
  237. return $component->state(false);
  238. })
  239. ->afterStateUpdated(static function (Get $get, Set $set, $state) {
  240. if ($state) {
  241. return;
  242. }
  243. $billingAddress = $get('../billingAddress');
  244. $fieldsToSync = [
  245. 'address_line_1',
  246. 'address_line_2',
  247. 'country',
  248. 'state_id',
  249. 'city',
  250. 'postal_code',
  251. ];
  252. foreach ($fieldsToSync as $field) {
  253. $set($field, $billingAddress[$field]);
  254. }
  255. })
  256. ->columnSpanFull(),
  257. Forms\Components\Grid::make()
  258. ->schema([
  259. Forms\Components\TextInput::make('address_line_1')
  260. ->label('Address Line 1')
  261. ->required()
  262. ->maxLength(255),
  263. Forms\Components\TextInput::make('address_line_2')
  264. ->label('Address Line 2')
  265. ->maxLength(255),
  266. Forms\Components\Select::make('country')
  267. ->searchable()
  268. ->localizeLabel()
  269. ->live()
  270. ->options(Country::getAvailableCountryOptions())
  271. ->afterStateUpdated(static function (Set $set) {
  272. $set('state_id', null);
  273. })
  274. ->required(),
  275. Forms\Components\Select::make('state_id')
  276. ->localizeLabel('State / Province')
  277. ->searchable()
  278. ->options(static fn (Get $get) => State::getStateOptions($get('country')))
  279. ->nullable(),
  280. Forms\Components\TextInput::make('city')
  281. ->label('City')
  282. ->required()
  283. ->maxLength(255),
  284. Forms\Components\TextInput::make('postal_code')
  285. ->label('Postal Code / Zip Code')
  286. ->required()
  287. ->maxLength(255),
  288. ])
  289. ->visible(fn (Get $get) => ! $get('same_as_billing')),
  290. Forms\Components\Textarea::make('notes')
  291. ->label('Delivery Instructions')
  292. ->maxLength(255)
  293. ->columnSpanFull(),
  294. ])->columns(),
  295. ])->columns(),
  296. ]);
  297. }
  298. public static function table(Table $table): Table
  299. {
  300. return $table
  301. ->columns([
  302. Columns::id(),
  303. Tables\Columns\TextColumn::make('name')
  304. ->searchable()
  305. ->sortable()
  306. ->description(fn (Client $client) => $client->primaryContact->full_name),
  307. Tables\Columns\TextColumn::make('primaryContact.email')
  308. ->label('Email')
  309. ->searchable()
  310. ->toggleable(),
  311. Tables\Columns\TextColumn::make('primaryContact.phones')
  312. ->label('Phone')
  313. ->toggleable()
  314. ->state(fn (Client $client) => $client->primaryContact->first_available_phone),
  315. Tables\Columns\TextColumn::make('billingAddress.address_string')
  316. ->label('Billing Address')
  317. ->searchable()
  318. ->toggleable(isToggledHiddenByDefault: true)
  319. ->listWithLineBreaks(),
  320. Tables\Columns\TextColumn::make('balance')
  321. ->label('Balance')
  322. ->getStateUsing(function (Client $client) {
  323. return $client->invoices()
  324. ->unpaid()
  325. ->get()
  326. ->sumMoneyInDefaultCurrency('amount_due');
  327. })
  328. ->description(function (Client $client) {
  329. $overdue = $client->invoices()
  330. ->overdue()
  331. ->get()
  332. ->sumMoneyInDefaultCurrency('amount_due');
  333. if ($overdue <= 0) {
  334. return null;
  335. }
  336. $formattedOverdue = CurrencyConverter::formatCentsToMoney($overdue);
  337. return new HtmlString("<span class='text-danger-700 dark:text-danger-400'>Overdue: {$formattedOverdue}</span>");
  338. })
  339. ->sortable(query: function (Builder $query, string $direction) {
  340. return $query
  341. ->withSum(['invoices' => fn (Builder $query) => $query->unpaid()], 'amount_due')
  342. ->orderBy('invoices_sum_amount_due', $direction);
  343. })
  344. ->currency(fn (Client $client) => $client->currency_code, false)
  345. ->alignEnd(),
  346. ])
  347. ->filters([
  348. //
  349. ])
  350. ->actions([
  351. Tables\Actions\EditAction::make(),
  352. ])
  353. ->bulkActions([
  354. Tables\Actions\BulkActionGroup::make([
  355. Tables\Actions\DeleteBulkAction::make(),
  356. ]),
  357. ]);
  358. }
  359. public static function getRelations(): array
  360. {
  361. return [
  362. //
  363. ];
  364. }
  365. public static function getPages(): array
  366. {
  367. return [
  368. 'index' => Pages\ListClients::route('/'),
  369. 'create' => Pages\CreateClient::route('/create'),
  370. 'edit' => Pages\EditClient::route('/{record}/edit'),
  371. ];
  372. }
  373. }