UserController.php 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405
  1. <?php
  2. namespace App\Http\Controllers;
  3. use App\Http\Requests\User\DeleteUser;
  4. use App\Http\Requests\User\StoreProfile;
  5. use App\Http\Requests\User\StoreUser;
  6. use App\Models\Order;
  7. use App\Models\Permission;
  8. use App\Models\Reclamation;
  9. use App\Models\ReclamationStatus;
  10. use App\Models\Role;
  11. use App\Models\User;
  12. use App\Models\UserNotificationSetting;
  13. use App\Services\Access\AccessService;
  14. use Illuminate\Http\Request;
  15. use Illuminate\Support\Facades\Auth;
  16. use Illuminate\Support\Facades\DB;
  17. use Illuminate\Support\Facades\Hash;
  18. class UserController extends Controller
  19. {
  20. protected array $data = [
  21. 'active' => 'users',
  22. 'title' => 'Пользователи',
  23. 'id' => 'users',
  24. 'header' => [
  25. 'id' => 'ID',
  26. 'email' => 'Логин/email',
  27. 'name' => 'ФИО',
  28. 'phone' => 'Телефон',
  29. 'role' => 'Роль',
  30. 'app_installed' => 'Приложение',
  31. 'created_at' => 'Дата создания',
  32. 'deleted_at' => 'Дата Удаления',
  33. ],
  34. 'searchFields' => [
  35. 'name',
  36. 'phone',
  37. 'email',
  38. ],
  39. 'ranges' => [],
  40. 'filters' => [],
  41. ];
  42. /**
  43. * Display a listing of the resource.
  44. */
  45. public function index(Request $request)
  46. {
  47. session(['gp_users' => $request->query()]);
  48. $nav = $this->startNavigationContext($request);
  49. $model = new User;
  50. $this->createFilters($model, 'role');
  51. $this->createDateFilters($model, 'created_at');
  52. if (isset($this->data['filters']['role']['values'])) {
  53. foreach ($this->data['filters']['role']['values'] as $key => $value) {
  54. $this->data['filters']['role']['values'][$key] = \App\Models\Role::NAMES[$key] ?? $value;
  55. }
  56. }
  57. $q = $model::query();
  58. $this->acceptFilters($q, $request);
  59. $this->acceptSearch($q, $request);
  60. $this->setSortAndOrderBy($model, $request);
  61. // $q->withTrashed();
  62. $this->applyStableSorting($q);
  63. $this->data['users'] = $q->paginate($this->data['per_page'])->withQueryString();
  64. $this->data['nav'] = $nav;
  65. return view('users.index', $this->data);
  66. }
  67. /**
  68. * Show the form for creating a new resource.
  69. */
  70. public function create(Request $request)
  71. {
  72. $nav = $this->resolveNavToken($request);
  73. $this->rememberNavigation($request, $nav);
  74. $this->data['nav'] = $nav;
  75. $this->data['back_url'] = $this->navigationBackUrl(
  76. $request,
  77. $nav,
  78. route('user.index', session('gp_users', []))
  79. );
  80. $this->data['user'] = null;
  81. $this->prepareNotificationSettingsData(null);
  82. $this->preparePermissionSettingsData(null);
  83. return view('users.edit', $this->data);
  84. }
  85. /**
  86. * Store a newly or update existing created resource in storage.
  87. */
  88. public function store(StoreUser $request)
  89. {
  90. $validated = $request->validated();
  91. $settingsData = $this->extractNotificationSettings($request);
  92. $hasPermissionEffects = array_key_exists('permission_effects', $validated);
  93. $permissionEffects = $validated['permission_effects'] ?? [];
  94. unset($validated['notification_settings']);
  95. unset($validated['permission_effects']);
  96. if (!empty($validated['role_id'])) {
  97. $role = Role::query()->find($validated['role_id']);
  98. $validated['role'] = $role?->slug ?? $validated['role'] ?? Role::MANAGER;
  99. } elseif (!empty($validated['role'])) {
  100. $role = Role::query()->where('slug', $validated['role'])->first();
  101. $validated['role_id'] = $role?->id;
  102. }
  103. if(!empty($validated['password'])) {
  104. $validated['password'] = Hash::make($validated['password']);
  105. } else {
  106. unset($validated['password']);
  107. }
  108. $user = null;
  109. DB::transaction(function () use ($validated, $settingsData, $hasPermissionEffects, $permissionEffects, &$user) {
  110. if(isset($validated['id'])) {
  111. User::query()
  112. ->where('id', $validated['id'])
  113. ->update($validated);
  114. $user = User::query()->find($validated['id']);
  115. } else {
  116. $user = User::query()->create($validated);
  117. }
  118. if ($user) {
  119. UserNotificationSetting::query()->updateOrCreate(
  120. ['user_id' => $user->id],
  121. $settingsData,
  122. );
  123. if ($hasPermissionEffects) {
  124. $this->syncUserPermissionOverrides($user, $permissionEffects);
  125. }
  126. }
  127. });
  128. app(AccessService::class)->bumpCacheVersion();
  129. return redirect()->route('user.index')->with(['success' => 'Пользователь ' . $validated['name'] . ' сохранён!']);
  130. }
  131. /**
  132. * Display the specified resource.
  133. */
  134. public function show(Request $request, int $userId)
  135. {
  136. $nav = $this->resolveNavToken($request);
  137. $this->rememberNavigation($request, $nav);
  138. $this->data['nav'] = $nav;
  139. $this->data['back_url'] = $this->navigationBackUrl(
  140. $request,
  141. $nav,
  142. route('user.index', session('gp_users', []))
  143. );
  144. $this->data['user'] = User::query()
  145. ->where('id', $userId)
  146. ->withTrashed()
  147. ->first();
  148. $this->prepareNotificationSettingsData($this->data['user']);
  149. $this->preparePermissionSettingsData($this->data['user']);
  150. return view('users.edit', $this->data);
  151. }
  152. /**
  153. * Remove the specified resource from storage.
  154. */
  155. public function destroy(User $user, DeleteUser $request)
  156. {
  157. if($user->is($request->user())) {
  158. return redirect()->route('user.index')->with(['danger' => 'Нельзя удалить самого себя!']);
  159. } else {
  160. $user->delete();
  161. return redirect()->route('user.index')->with(['success' => 'Пользователь ' . $user->name . ' удалён!']);
  162. }
  163. }
  164. public function profile(Request $request)
  165. {
  166. $this->data['current_menu'] = 'profile';
  167. $this->data['user'] = $request->user();
  168. return view('users.profile', $this->data);
  169. }
  170. public function storeProfile(StoreProfile $request)
  171. {
  172. $data = $request->validated();
  173. unset($data['current_password'], $data['password']);
  174. if(
  175. isset($request->current_password)
  176. && isset($request->password)
  177. && (Hash::check($request->current_password, $request->user()->password))) {
  178. $data['password'] = Hash::make($request->password);
  179. }
  180. User::query()->where('id', '=', $request->user()->id)->update($data);
  181. return redirect()->route('user.profile')->with(['success' => 'Профиль обновлён!']);
  182. }
  183. public function deleteProfile(Request $request)
  184. {
  185. User::query()->where('id', '=', $request->user()->id)->delete();
  186. User::clearFcmToken((int)$request->user()->id);
  187. Auth::logout();
  188. return redirect()->route('login')->with(['success' => 'Профиль удалён!']);
  189. }
  190. public function undelete(int $userId)
  191. {
  192. User::query()->where('id', '=', $userId)->restore();
  193. return redirect()->route('user.show', $userId)->with(['success' => 'Пользователь восстановлен!']);
  194. }
  195. public function impersonate(Request $request, User $user)
  196. {
  197. $currentUser = $request->user();
  198. if ($currentUser->id === $user->id) {
  199. return redirect()->back()->with(['danger' => 'Нельзя войти от имени самого себя!']);
  200. }
  201. if ($user->trashed()) {
  202. return redirect()->back()->with(['danger' => 'Нельзя войти от имени удалённого пользователя!']);
  203. }
  204. if (session()->has('impersonator_id')) {
  205. return redirect()->back()->with(['danger' => 'Вы уже вошли от имени другого пользователя.']);
  206. }
  207. $impersonatorId = $currentUser->id;
  208. Auth::login($user);
  209. $request->session()->put('impersonator_id', $impersonatorId);
  210. $request->session()->regenerate();
  211. return redirect()->route('home')->with(['success' => 'Вы вошли от имени пользователя ' . $user->name . '.']);
  212. }
  213. public function leaveImpersonation(Request $request)
  214. {
  215. $impersonatorId = (int) session('impersonator_id');
  216. if (!$impersonatorId) {
  217. return redirect()->back()->with(['danger' => 'Режим impersonate не активен.']);
  218. }
  219. $impersonator = User::query()->find($impersonatorId);
  220. if (!$impersonator) {
  221. if ($request->user()) {
  222. User::clearFcmToken((int)$request->user()->id);
  223. }
  224. Auth::logout();
  225. $request->session()->invalidate();
  226. $request->session()->regenerateToken();
  227. return redirect()->route('login')->with(['danger' => 'Не удалось вернуться к исходному пользователю.']);
  228. }
  229. abort_unless($impersonator->resolvedRoleSlug() === Role::ADMIN, 403);
  230. Auth::login($impersonator);
  231. $request->session()->forget('impersonator_id');
  232. $request->session()->regenerate();
  233. return redirect()->route('user.index')->with(['success' => 'Вы вернулись в аккаунт администратора.']);
  234. }
  235. private function prepareNotificationSettingsData(?User $user): void
  236. {
  237. $this->data['orderStatusOptions'] = Order::STATUS_NAMES;
  238. $this->data['orderStatusColors'] = Order::STATUS_COLOR;
  239. $this->data['reclamationStatusOptions'] = Reclamation::STATUS_NAMES;
  240. $this->data['reclamationStatusColors'] = ReclamationStatus::STATUS_COLOR;
  241. $this->data['scheduleSourceOptions'] = ['platform' => 'Площадки', 'reclamation' => 'Рекламации'];
  242. $this->data['chatSourceOptions'] = ['platform' => 'Площадки', 'reclamation' => 'Рекламации'];
  243. $this->data['notificationChannels'] = ['browser' => 'Браузер', 'push' => 'Push', 'email' => 'Email'];
  244. $this->data['disabledChannels'] = [
  245. 'browser' => false,
  246. 'push' => !$user || !$user->token_fcm,
  247. 'email' => !$user || !$user->notification_email || !filter_var($user->notification_email, FILTER_VALIDATE_EMAIL),
  248. ];
  249. if (!$user) {
  250. $this->data['notificationSettings'] = UserNotificationSetting::defaultsForUser(0);
  251. return;
  252. }
  253. $settings = UserNotificationSetting::query()->firstOrCreate(
  254. ['user_id' => $user->id],
  255. UserNotificationSetting::defaultsForUser($user->id),
  256. );
  257. $this->data['notificationSettings'] = $settings->toArray();
  258. }
  259. private function preparePermissionSettingsData(?User $user): void
  260. {
  261. $this->data['roles'] = Role::query()
  262. ->where('is_active', true)
  263. ->orderBy('sort')
  264. ->orderBy('name')
  265. ->pluck('name', 'id')
  266. ->all();
  267. $this->data['permissionGroups'] = Permission::getGroupedForUi();
  268. $this->data['permissionEffects'] = [];
  269. $this->data['rolePermissionEffects'] = [];
  270. if (!$user) {
  271. return;
  272. }
  273. $this->data['permissionEffects'] = $user->permissions()
  274. ->pluck('user_permissions.effect', 'permissions.id')
  275. ->all();
  276. if ($user->roleModel) {
  277. $this->data['rolePermissionEffects'] = $user->roleModel
  278. ->permissions()
  279. ->pluck('role_permissions.effect', 'permissions.id')
  280. ->all();
  281. }
  282. }
  283. private function syncUserPermissionOverrides(User $user, array $effects): void
  284. {
  285. if ($user->resolvedRoleSlug() === Role::ADMIN) {
  286. $user->permissions()->detach();
  287. return;
  288. }
  289. $permissions = Permission::query()
  290. ->whereIn('id', array_filter(array_keys($effects), 'is_numeric'))
  291. ->get();
  292. $sync = [];
  293. foreach ($permissions as $permission) {
  294. $effect = $effects[$permission->id] ?? null;
  295. if (in_array($effect, ['allow', 'deny'], true)) {
  296. $sync[$permission->id] = ['effect' => $effect];
  297. }
  298. }
  299. $user->permissions()->sync($sync);
  300. }
  301. private function extractNotificationSettings(Request $request): array
  302. {
  303. $input = $request->input('notification_settings', []);
  304. $settings = [
  305. 'order_settings' => [],
  306. 'reclamation_settings' => [],
  307. 'schedule_settings' => [],
  308. 'chat_settings' => [],
  309. ];
  310. $orderStatuses = array_keys(Order::STATUS_NAMES);
  311. $reclamationStatuses = array_keys(Reclamation::STATUS_NAMES);
  312. $scheduleSources = ['platform', 'reclamation'];
  313. $chatSources = ['platform', 'reclamation'];
  314. $channels = ['browser', 'push', 'email'];
  315. foreach ($orderStatuses as $statusId) {
  316. foreach ($channels as $channel) {
  317. $settings['order_settings'][$statusId][$channel] = isset($input['orders'][$statusId][$channel]);
  318. }
  319. }
  320. foreach ($reclamationStatuses as $statusId) {
  321. foreach ($channels as $channel) {
  322. $settings['reclamation_settings'][$statusId][$channel] = isset($input['reclamations'][$statusId][$channel]);
  323. }
  324. }
  325. foreach ($scheduleSources as $source) {
  326. foreach ($channels as $channel) {
  327. $settings['schedule_settings'][$source][$channel] = isset($input['schedule'][$source][$channel]);
  328. }
  329. }
  330. foreach ($chatSources as $source) {
  331. foreach ($channels as $channel) {
  332. $settings['chat_settings'][$source][$channel] = isset($input['chat'][$source][$channel]);
  333. }
  334. }
  335. return $settings;
  336. }
  337. }