UserController.php 11 KB

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