UserDutyCheckController.php 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361
  1. <?php
  2. namespace App\Http\Controllers;
  3. use App\Events\ClockinEvent;
  4. use App\Events\ClockoutEvent;
  5. use App\Events\ExportEvent;
  6. use App\Events\ImportEvent;
  7. use App\LaborCompany;
  8. use App\LaborReport;
  9. use App\Services\LaborReportService;
  10. use App\Services\UserDutyCheckService;
  11. use App\User;
  12. use App\UserDetail;
  13. use App\UserDutyCheck;
  14. use App\UserLabor;
  15. use App\UserWorkgroup;
  16. use Carbon\Carbon;
  17. use Illuminate\Http\Request;
  18. use Illuminate\Support\Facades\Auth;
  19. use Illuminate\Support\Facades\Cache;
  20. use Illuminate\Support\Facades\Validator;
  21. use Ramsey\Uuid\Uuid;
  22. class UserDutyCheckController extends Controller
  23. {
  24. function __construct()
  25. {
  26. app()->bind('LaborReportService', LaborReportService::class);
  27. }
  28. //校验二维码进入进出场打卡页面
  29. public function importAndExportClock(Request $request)
  30. {
  31. $importAndExportQRCodeType = $request->input('importAndExportQRCodeType');
  32. $qrCode_refresh_everyday = config('hr.qrCode_refresh_everyday');
  33. $qrCode_refresh = config('hr.qrCode_refresh');
  34. if ($qrCode_refresh_everyday) {
  35. $key = $request->input('key');
  36. if (!$key) return "<h1 style='color: red;text-align:center'>当前二维码失效,请重新扫码!</h1>";
  37. $date = md5(date('Y-m-d'));
  38. if ($key != $date) return "<h1 style='color: red;text-align:center'>当前二维码失效,请重新扫码!</h1>";
  39. }
  40. if ($qrCode_refresh) {
  41. $key = $request->input('key');
  42. if (!$key) return "<h1 style='color: red;text-align:center'>当前二维码失效,请重新扫码!</h1>";
  43. $date = date('Y-m-d H:i:s');
  44. if ($key < $date) return "<h1 style='color: red;text-align:center'>当前二维码失效,请重新扫码!</h1>";
  45. }
  46. $userLaborToken = $request->cookie('userLaborToken');
  47. if ($userLaborToken && Cache::has('dutyCheckTokenStr_' . $userLaborToken)) {
  48. $user_id = Cache::get('dutyCheckTokenStr_' . $userLaborToken);
  49. $userDetail = UserDetail::find($user_id);
  50. if (!$userDetail) return redirect('personnel/checking-in/userDutyCheck/createUserDetail/' . $userDetail->mobile_phone)->with("importAndExportQRCodeType", $importAndExportQRCodeType);
  51. $userDutyCheck = $this->importAndExportDutyCheck($user_id, $importAndExportQRCodeType);
  52. if ($userDutyCheck && $userDutyCheck->isNotImport == true) return "<h1 style='color: darkred;text-align:center'>当前还未入场,不可出场!</h1>";
  53. if ($userDutyCheck && $userDutyCheck->import == true) return "<h1 style='color: darkred;text-align:center'>您已入场,不可再次入场!</h1>";
  54. if ($userDutyCheck && $userDutyCheck->export == true) return "<h1 style='color: darkred;text-align:center'>您已出场,不可再次出场!!</h1>";
  55. if (!$userDutyCheck) return "<h1 style='color: red;text-align:center'>打卡异常,请联系相关人员!</h1>";
  56. $laborReportService = app('LaborReportService');
  57. $errorMessage = $laborReportService->makeOrChangeLaborReportByUserDutyCheck($userDutyCheck, $importAndExportQRCodeType);
  58. if ($errorMessage) return $errorMessage;
  59. $laravelEchoPrefix = config('database.redis.options.prefix');
  60. return response()->view('personnel/checking-in/importAndExportSuccess', ['full_name' => $userDetail->full_name, 'checked_at' => $userDutyCheck->checked_at, 'type' => $userDutyCheck->type, 'laravelEchoPrefix' => $laravelEchoPrefix]);
  61. }
  62. return view("personnel/checking-in/importAndExportClock", compact('importAndExportQRCodeType'));
  63. }
  64. //提交进出场打卡
  65. public function storeClock(Request $request)
  66. {
  67. $importAndExportQRCodeType = $request->input('importAndExportQRCodeType');
  68. $mobile_phone = $request->input('mobile_phone');
  69. $this->validator($request)->validate();
  70. $userDetail = UserDetail::where('mobile_phone', $mobile_phone)->first();
  71. if (!$userDetail) return redirect('personnel/checking-in/userDutyCheck/createUserDetail/' . $mobile_phone)->with("importAndExportQRCodeType", $importAndExportQRCodeType);
  72. if (Cache::has('dutyCheckTokenUser_' . $userDetail->user_id)) return "<h1 style='color: red;text-align:center'>请使用原有设备进行打卡,如无法使用原有设备请联系管理人员解绑设备!</h1>";
  73. $userDutyCheckOld = UserDutyCheck::where('user_id', $userDetail->user_id)->orderBy('checked_at', 'DESC')->limit(10)->first();
  74. $date = date('Y-m-d H:i:s');
  75. $lastDate = Carbon::parse($userDutyCheckOld['checked_at']);
  76. $thisDate = Carbon::parse($date);
  77. $diffDate = $thisDate->diffInMinutes($lastDate);
  78. //七天未登录从新选择劳务所
  79. if ($diffDate > 7 * 1440) {
  80. return redirect('personnel/checking-in/userDutyCheck/updateUserDetail/' . $mobile_phone)->with("importAndExportQRCodeType", $importAndExportQRCodeType);
  81. }
  82. $userDutyCheck = $this->importAndExportDutyCheck($userDetail->user_id, $importAndExportQRCodeType);
  83. if ($userDutyCheck && $userDutyCheck->isNotImport == true) return "<h1 style='color: darkred;text-align:center'>当前还未入场,不可出场!</h1>";
  84. if ($userDutyCheck && $userDutyCheck->import == true) return "<h1 style='color: darkred;text-align:center'>您已入场,不可再次入场!</h1>";
  85. if ($userDutyCheck && $userDutyCheck->export == true) return "<h1 style='color: darkred;text-align:center'>您已出场,不可再次出场!!</h1>";
  86. if (!$userDutyCheck) return "<h1 style='color: red;text-align:center'>打卡异常,请联系相关人员!</h1>";
  87. $userLaborToken = $userDetail->getDutyCheckToken(config('users.token_check_in_expire_minutes'));
  88. $laborReportService = app('LaborReportService');
  89. $errorMessage = $laborReportService->makeOrChangeLaborReportByUserDutyCheck($userDutyCheck, $importAndExportQRCodeType);
  90. if ($errorMessage) return $errorMessage;
  91. $laravelEchoPrefix = config('database.redis.options.prefix');
  92. return response()->view('personnel/checking-in/importAndExportSuccess', ['full_name' => $userDetail->full_name, 'checked_at' => $userDutyCheck->checked_at, 'type' => $userDutyCheck->type, 'laravelEchoPrefix' => $laravelEchoPrefix])
  93. ->cookie('userLaborToken', $userLaborToken, config('users.token_check_in_expire_minutes'), '/');
  94. }
  95. //根据用户提交进出场打卡记录
  96. public function importAndExportDutyCheck($user_id, $importAndExportQRCodeType)
  97. {
  98. $dateNow = Carbon::now()->format('Y-m-d');
  99. $userDutyCheckOld = UserDutyCheck::where('user_id', $user_id)->where('checked_at', 'like', $dateNow . '%')->orderBy('id', 'desc')->limit(10)->first();
  100. $date = date('Y-m-d H:i:s');
  101. $this->log(__METHOD__, "提交打卡记录__" . __FUNCTION__, json_encode([$user_id, $date]));
  102. $userDutyCheck = new UserDutyCheck([
  103. 'user_id' => $user_id,
  104. 'checked_at' => $date,
  105. 'source' => '正常',
  106. ]);
  107. if (!$userDutyCheckOld) {
  108. //跨天登出情况
  109. if ($importAndExportQRCodeType && $importAndExportQRCodeType == 'export') {
  110. //当前日期的前一天
  111. $yesterday = Carbon::yesterday()->format('Y-m-d');
  112. $userDutyCheckYesterday = UserDutyCheck::where('user_id', $user_id)->where('checked_at', 'like', $yesterday . '%')->orderBy('id', 'desc')->limit(10)->first();
  113. if ($userDutyCheckYesterday && $userDutyCheckYesterday->type == '登入') {
  114. $userDutyCheck->verify_user_id = $userDutyCheckYesterday->verify_user_id;
  115. $userDutyCheck->type = '登出';
  116. $userDutyCheck->save();
  117. return $userDutyCheck;
  118. }
  119. }
  120. if ($importAndExportQRCodeType == 'export') {
  121. $userDutyCheck->isNotImport = true;//还未进场,不可出场
  122. return $userDutyCheck;
  123. }
  124. $userDutyCheck->type = "登入";
  125. $userDutyCheck->save();
  126. return $userDutyCheck;
  127. }
  128. if ($importAndExportQRCodeType && $importAndExportQRCodeType == 'import') {
  129. if ($userDutyCheckOld->type == '登入') {
  130. $userDutyCheck->import = true;//已入场的不能再入场
  131. return $userDutyCheck;
  132. }
  133. $userDutyCheck->type = '登入';
  134. }
  135. if ($importAndExportQRCodeType && $importAndExportQRCodeType == 'export') {
  136. if ($userDutyCheckOld->type == '登出') {
  137. $userDutyCheck->export = true;//已出场的不能再出场
  138. return $userDutyCheck;
  139. }
  140. $userDutyCheck->verify_user_id = $userDutyCheckOld->verify_user_id;
  141. $userDutyCheck->type = '登出';
  142. }
  143. $userDutyCheck->save();
  144. return $userDutyCheck;
  145. }
  146. //去往登记资料页面
  147. public function createUserDetail($mobile_phone)
  148. {
  149. $importAndExportQRCodeType = session("importAndExportQRCodeType");
  150. $laborCompanies = LaborCompany::select('id', 'name')->get();
  151. return view('personnel/checking-in/createUserDetail', ['mobile_phone' => $mobile_phone, 'importAndExportQRCodeType' => $importAndExportQRCodeType, 'laborCompanies' => $laborCompanies]);
  152. }
  153. //七天未打卡去往选择劳务所页面
  154. public function updateUserDetail($mobile_phone)
  155. {
  156. $importAndExportQRCodeType = session("importAndExportQRCodeType");
  157. $laborCompanies = LaborCompany::select('id', 'name')->get();
  158. return view('personnel/checking-in/updateUserDetail', ['mobile_phone' => $mobile_phone, 'importAndExportQRCodeType' => $importAndExportQRCodeType, 'laborCompanies' => $laborCompanies]);
  159. }
  160. //打卡修改临时工资料并生成进场记录 AndMakeEnterRecord
  161. public function storeUpdateUserDetail(Request $request)
  162. {
  163. $this->updateValidator($request)->validate();
  164. $importAndExportQRCodeType = $request->input('importAndExportQRCodeType');
  165. $mobile_phone = $request->input('mobile_phone');
  166. $userDetail = UserDetail::where('mobile_phone', $mobile_phone)->first();
  167. if (!$userDetail) return view('exception.404', ['error' => '临时工信息不存在!']);
  168. $labor_company_id = $request->input('labor_company_id');
  169. $userLabor = UserLabor::where('user_id', $userDetail->user_id)->first();
  170. $userLabor->update([
  171. 'labor_company_id' => $labor_company_id,
  172. ]);
  173. $this->log(__METHOD__, "更改临时工劳务所信息" . __FUNCTION__, json_encode($userLabor), Auth::user()['id']);
  174. $laborReportService = app('LaborReportService');
  175. $laborReportService->includeEnterRecordToCreateLaborReport($userDetail, $importAndExportQRCodeType);
  176. }
  177. public function updateValidator(Request $request)
  178. {
  179. $updateValidator = Validator::make($request->input(), [
  180. 'mobile_phone' => ['required', 'integer', 'digits:11'],
  181. 'labor_company_id' => ['filled'],
  182. ], [
  183. 'filled' => ':attribute 不能为空',
  184. 'required' => ':attribute 为必填项',
  185. 'integer' => ':attribute 数字类型',
  186. 'digits' => ':attribute 必须是11位',
  187. ], [
  188. 'mobile_phone' => '手机号',
  189. 'labor_company_id' => '劳务所',
  190. ]);
  191. return $updateValidator;
  192. }
  193. //打卡创建临时工资料并生成进场记录
  194. public function storeUserDetail(Request $request)
  195. {
  196. $this->validator($request)->validate();
  197. $importAndExportQRCodeType = $request->input('importAndExportQRCodeType');
  198. $mobile_phone = $request->input('mobile_phone');
  199. $full_name = $request->input('full_name');
  200. $identity_number = $request->input('identity_number');
  201. $gender = $request->input('gender');
  202. $labor_company_id = $request->input('labor_company_id');
  203. $user = new User([
  204. 'name' => $mobile_phone,
  205. 'password' => Uuid::uuid1(),
  206. ]);
  207. $user->save();
  208. $userDetail = new UserDetail([
  209. 'user_id' => $user->id,
  210. 'full_name' => $full_name,
  211. 'identity_number' => $identity_number,
  212. 'gender' => $gender,
  213. 'mobile_phone' => $mobile_phone,
  214. 'type' => '临时工',
  215. ]);
  216. $userDetail->save();
  217. $userLabor = new UserLabor([
  218. 'user_id' => $user->id,
  219. 'labor_company_id' => $labor_company_id,
  220. ]);
  221. $userLabor->save();
  222. $laborReportService = app('LaborReportService');
  223. $laborReportService->includeEnterRecordToCreateLaborReport($userDetail, $importAndExportQRCodeType);
  224. }
  225. public function validator(Request $request)
  226. {
  227. $validator = Validator::make($request->input(), [
  228. 'full_name' => ['filled'],
  229. 'gender' => ['filled'],
  230. 'mobile_phone' => ['required', 'integer', 'digits:11'],
  231. 'identity_number' => ['filled', 'identity_cards', 'unique:user_details,identity_number'],
  232. 'labor_company_id' => ['filled'],
  233. ], [
  234. 'unique' => ':attribute 已经存在,不能重复!',
  235. 'filled' => ':attribute 不能为空',
  236. 'required' => ':attribute 为必填项',
  237. 'integer' => ':attribute 数字类型',
  238. 'digits' => ':attribute 必须是11位',
  239. ], [
  240. 'full_name' => '姓名',
  241. 'gender' => '性别',
  242. 'mobile_phone' => '手机号',
  243. 'identity_number' => '身份证号',
  244. 'labor_company_id' => '劳务所',
  245. ]);
  246. return $validator;
  247. }
  248. //扫工作组二维码
  249. public function clock(Request $request)
  250. {
  251. $userWorkgroupID = $request->input('userWorkgroupID');
  252. $userLaborToken = $request->cookie('userLaborToken');
  253. $laborReportService = app('LaborReportService');
  254. if ($userLaborToken && Cache::has('dutyCheckTokenStr_' . $userLaborToken)) {
  255. $user_id = Cache::get('dutyCheckTokenStr_' . $userLaborToken);
  256. $userDetail = UserDetail::find($user_id);
  257. $dateNow = Carbon::now()->format('Y-m-d');
  258. $group_name = UserWorkgroup::where('id', $userWorkgroupID)->value('name');
  259. $laravelEchoPrefix = config('database.redis.options.prefix');
  260. $userDutyCheck = UserDutyCheck::where('user_id', $userDetail->user_id)->where('checked_at', 'like', $dateNow . '%')->orderBy('id', 'desc')->first();
  261. if (!$userDutyCheck->verify_user_id) return "<h1 style='color: red;text-align:center'>进场门卫还未审核,暂无法进组!</h1>";
  262. if ($userDutyCheck->type == '登出') return "<h1 style='color: red;text-align:center'>进场是否未打卡?如若未打,请先返回打进场卡!</h1>";
  263. //判断是否是第一次进组
  264. $laborReport = LaborReport::where('user_id', $user_id)->where('user_workgroup_id', null)->where('created_at', 'like', $dateNow . '%')->orderBy('id', 'desc')->first();
  265. if ($laborReport) {
  266. $newLaborReport = $laborReportService->enterGroupAndUpdateLaborReport($laborReport, $userWorkgroupID);
  267. $this->log(__METHOD__, "当日首次进组更新临时工报表记录__" . __FUNCTION__, json_encode($request->all()));
  268. if (!$newLaborReport) return "<h1 style='color: red;text-align:center'>打卡异常,请联系相关人员!</h1>";
  269. if ($newLaborReport) {
  270. event(new ClockinEvent($newLaborReport));
  271. }
  272. return response()->view('personnel/checking-in/success', ['group_name' => $group_name ?? '', 'full_name' => $userDetail->full_name, 'check_in_at' => $newLaborReport->check_in_at, 'laravelEchoPrefix' => $laravelEchoPrefix]);
  273. } else {
  274. //换组判断上一条临时工报表信息是否退组
  275. $laborReportNow = LaborReport::where('user_id', $user_id)->whereNotNull('group_user_id')->orderBy('id', 'desc')->first();
  276. if ($laborReportNow && !$laborReportNow['check_out_at']) return "<h1 style='color: darkred;text-align:center'>您还未退组,暂不可换组,请联系之前组长!</h1>";
  277. $laborReport = LaborReport::where('user_id', $user_id)->where('user_workgroup_id', $userWorkgroupID)->where('created_at', 'like', $dateNow . '%')->orderBy('id', 'desc')->first();
  278. //此处表示临时工当日换组操作
  279. if (!$laborReport) {
  280. $newLaborReport = $laborReportService->changeGroupAndMakeLaborReport($user_id, $userWorkgroupID);
  281. $this->log(__METHOD__, "当日临时工换组新建临时工报表记录__" . __FUNCTION__, json_encode([$user_id, $userWorkgroupID]));
  282. if (!$newLaborReport) return "<h1 style='color: red;text-align:center'>打卡异常,请联系相关人员!</h1>";
  283. if ($newLaborReport) {
  284. event(new ClockinEvent($newLaborReport));
  285. }
  286. return response()->view('personnel/checking-in/success', ['group_name' => $group_name ?? '', 'full_name' => $userDetail->full_name, 'check_in_at' => $newLaborReport->check_in_at, 'laravelEchoPrefix' => $laravelEchoPrefix]);
  287. } else if ($laborReport['enter_number'] != $laborReportNow['enter_number']) {
  288. $newLaborReport = $laborReportService->changeGroupAndMakeLaborReport($user_id, $userWorkgroupID);
  289. $this->log(__METHOD__, "当日临时工换组新建临时工报表记录__" . __FUNCTION__, json_encode([$user_id, $userWorkgroupID]));
  290. if (!$newLaborReport) return "<h1 style='color: red;text-align:center'>打卡异常,请联系相关人员!</h1>";
  291. if ($newLaborReport) {
  292. event(new ClockinEvent($newLaborReport));
  293. }
  294. return response()->view('personnel/checking-in/success', ['group_name' => $group_name ?? '', 'full_name' => $userDetail->full_name, 'check_in_at' => $newLaborReport->check_in_at, 'laravelEchoPrefix' => $laravelEchoPrefix]);
  295. } else {
  296. return "<h1 style='color: darkred;text-align:center'>重复打卡!</h1>";
  297. }
  298. }
  299. }
  300. return view("personnel/checking-in/clock", compact('userWorkgroupID'));
  301. }
  302. //进组提交打卡
  303. // public function importGroupClock(Request $request)
  304. // {
  305. // $mobile_phone = $request->input('mobile_phone');
  306. // $userWorkgroupID = $request->input('userWorkgroupID');
  307. // $laborReportService = app('LaborReportService');
  308. // $this->validator($request)->validate();
  309. // $userDetail = UserDetail::where('mobile_phone', $mobile_phone)->first();
  310. // if (Cache::has('dutyCheckTokenUser_' . $userDetail->user_id)) return "<h1 style='color: red;text-align:center'>请使用原有设备进行打卡,如无法使用原有设备请联系管理人员解绑设备!</h1>";
  311. // $dateNow = Carbon::now()->format('Y-m-d');
  312. // $group_name = UserWorkgroup::where('id', $userWorkgroupID)->value('name');
  313. // $laravelEchoPrefix = config('database.redis.options.prefix');
  314. // $userDutyCheck = UserDutyCheck::where('user_id', $userDetail->user_id)->where('checked_at', 'like', $dateNow . '%')->orderBy('id', 'desc')->first();
  315. // if (!$userDutyCheck->verify_user_id) return "<h1 style='color: red;text-align:center'>进场门卫还未审核,暂无法进组!</h1>";
  316. // if ($userDutyCheck->type == '登出') return "<h1 style='color: red;text-align:center'>进场是否未打卡?如若未打,请先返回打进场卡!</h1>";
  317. // $userLaborToken = $userDetail->getDutyCheckToken(config('users.token_check_in_expire_minutes'));
  318. // //判断是否是第一次进组
  319. // $laborReport = LaborReport::where('user_id', $userDetail->user_id)->where('user_workgroup_id', null)->where('created_at', 'like', $dateNow . '%')->orderBy('id', 'desc')->first();
  320. // if ($laborReport) {
  321. // $newLaborReport = $laborReportService->enterGroupAndUpdateLaborReport($laborReport, $userWorkgroupID);
  322. // $this->log(__METHOD__, "当日首次进组更新临时工报表记录__" . __FUNCTION__, json_encode($request->all()));
  323. // if (!$newLaborReport) return "<h1 style='color: red;text-align:center'>打卡异常,请联系相关人员!</h1>";
  324. // if ($newLaborReport) {
  325. // event(new ClockinEvent($newLaborReport));
  326. // }
  327. // return response()->view('personnel/checking-in/success', ['group_name' => $group_name ?? '', 'full_name' => $userDetail->full_name, 'check_in_at' => $newLaborReport->check_in_at, 'laravelEchoPrefix' => $laravelEchoPrefix]);
  328. // } else {
  329. // $laborReport = LaborReport::where('user_id', $userDetail->user_id)->where('user_workgroup_id', $userWorkgroupID)->where('created_at', 'like', $dateNow . '%')->orderBy('id', 'desc')->first();
  330. // //此处表示临时工当日换组操作
  331. // if (!$laborReport) {
  332. // $newLaborReport = $laborReportService->changeGroupAndMakeLaborReport($userDetail->user_id, $userWorkgroupID);
  333. // $this->log(__METHOD__, "当日临时工换组新建临时工报表记录__" . __FUNCTION__, json_encode([$userDetail->user_id, $userWorkgroupID]));
  334. // if (!$newLaborReport) return "<h1 style='color: red;text-align:center'>打卡异常,请联系相关人员!</h1>";
  335. // if ($newLaborReport) {
  336. // event(new ClockinEvent($newLaborReport));
  337. // }
  338. // return response()->view('personnel/checking-in/success', ['group_name' => $group_name ?? '', 'full_name' => $userDetail->full_name, 'check_in_at' => $newLaborReport->check_in_at, 'laravelEchoPrefix' => $laravelEchoPrefix])
  339. // ->cookie('userLaborToken', $userLaborToken, config('users.token_check_in_expire_minutes'), '/');
  340. // } else {
  341. // return "<h1 style='color: darkred;text-align:center'>重复打卡!</h1>";
  342. // }
  343. // }
  344. // }
  345. }