InputInfoFragment.java 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441
  1. package com.emato.ich.fragment;
  2. import android.os.Bundle;
  3. import android.os.CountDownTimer;
  4. import android.text.Editable;
  5. import com.emato.ich.utils.Log;
  6. import android.view.LayoutInflater;
  7. import android.view.View;
  8. import android.view.ViewGroup;
  9. import android.widget.TextView;
  10. import androidx.annotation.NonNull;
  11. import androidx.annotation.Nullable;
  12. import androidx.fragment.app.Fragment;
  13. import androidx.navigation.fragment.NavHostFragment;
  14. import com.cherry.sdk.controller.utils.ScanGunKeyEventHelper;
  15. import com.emato.ich.MainActivity;
  16. import com.emato.ich.R;
  17. import com.emato.ich.api.ICSPClient;
  18. import com.emato.ich.api.ICSPResponseCodeEnum;
  19. import com.emato.ich.contant.ScanGunConstant;
  20. import com.emato.ich.databinding.FragmentInputInfoBinding;
  21. import com.emato.ich.entity.vo.ResponseData;
  22. import com.emato.ich.entity.vo.PreparedOrderResponseVo;
  23. import com.emato.ich.entity.vo.PreparedOrderVo;
  24. import com.emato.ich.local.LocalStorage;
  25. import com.emato.ich.utils.BaseUtils;
  26. import com.emato.ich.utils.JacksonUtils;
  27. import com.emato.ich.utils.LoggingUtils;
  28. import com.emato.ich.utils.StringUtils;
  29. import com.emato.ich.utils.TimeOutUtils;
  30. import com.emato.ich.utils.ToastUtils;
  31. import com.fasterxml.jackson.core.JsonProcessingException;
  32. import com.fasterxml.jackson.core.type.TypeReference;
  33. import com.fasterxml.jackson.databind.ObjectMapper;
  34. import com.google.android.material.snackbar.Snackbar;
  35. import org.jetbrains.annotations.NotNull;
  36. import java.io.IOException;
  37. import java.util.concurrent.atomic.AtomicBoolean;
  38. import java.util.concurrent.atomic.AtomicInteger;
  39. import java.util.regex.Matcher;
  40. import java.util.regex.Pattern;
  41. import okhttp3.Call;
  42. import okhttp3.Callback;
  43. import okhttp3.Response;
  44. /**
  45. * 输入投递信息
  46. */
  47. public class InputInfoFragment extends Fragment {
  48. private static final String TAG = "InputInfoFragment";
  49. private FragmentInputInfoBinding binding;
  50. private PreparedOrderResponseVo data;
  51. private volatile AtomicInteger mailNoLock = new AtomicInteger(1);
  52. private volatile AtomicInteger submitLock = new AtomicInteger(1);
  53. private CountDownTimer timer;
  54. @Nullable
  55. @org.jetbrains.annotations.Nullable
  56. @Override
  57. public View onCreateView(@NonNull @NotNull LayoutInflater inflater, @Nullable @org.jetbrains.annotations.Nullable ViewGroup container, @Nullable @org.jetbrains.annotations.Nullable Bundle savedInstanceState) {
  58. binding = FragmentInputInfoBinding.inflate(inflater, container, false);
  59. return binding.getRoot();
  60. }
  61. public void onViewCreated(@NonNull View view, Bundle savedInstanceState) {
  62. // BaseUtils.disableEditText(binding.delivererPhone);
  63. BaseUtils.disableEditText(binding.delivererPhoneConfirm);
  64. BaseUtils.disableEditText(binding.mailNo);
  65. try {
  66. MainActivity activity = (MainActivity) getActivity();
  67. timer = TimeOutUtils.timeout(activity, InputInfoFragment.this, binding.timeout, 120);
  68. timer.start();
  69. } catch (Exception e) {
  70. Log.e(TAG, "onViewCreated: 倒计时出现异常! ", e);
  71. LoggingUtils.sendErrorLog("业务异常: 输入投递信息页面倒计时出现异常! ", e);
  72. }
  73. Bundle arguments = null;
  74. String sectionType = null;
  75. try {
  76. MainActivity activity = (MainActivity) getActivity();
  77. arguments = activity.getBundleMap().get(ChooseCabinetFragment.class.getName());
  78. sectionType = arguments.getString("section_type");
  79. // 扫描枪监听 扫描运单号
  80. activity.scanGunKeyEventHelper = new ScanGunKeyEventHelper(s -> {
  81. // 扫码成功回调
  82. Log.d(TAG, "onScanSuccess: " + s);
  83. if (!StringUtils.isNullOrEmpty(s)) {
  84. for (String pattern : ScanGunConstant.getPatterns()) {
  85. Matcher matcher = Pattern.compile(pattern).matcher(s);
  86. if (!matcher.matches()) {
  87. ToastUtils.make(getContext(), "请将条形码置于扫码枪前! ");
  88. return;
  89. }
  90. }
  91. }
  92. if (mailNoLock.get() == 1) {
  93. Editable text = binding.mailNo.getText();
  94. text.clear();
  95. binding.mailNo.setText(s);
  96. mailNoLock.incrementAndGet();
  97. // TODO 预定义,实际参数可能不一致 oms获取用户手机号 自动填充
  98. ICSPClient.getUserPhoneNumber(LocalStorage.getInstance().getSession().getToken(), BaseUtils.getClientId(), s, new Callback() {
  99. @Override
  100. public void onFailure(@NotNull Call call, @NotNull IOException e) {
  101. // 请求失败
  102. Log.e(TAG, "onFailure: 请求oms获取手机号失败! 网络异常! ", e);
  103. LoggingUtils.sendErrorLog("业务异常: 请求oms获取手机号失败! 网络异常! ", e);
  104. ToastUtils.make(getContext(), "服务器异常! 请手动输入!");
  105. mailNoLock.set(1);
  106. }
  107. @Override
  108. public void onResponse(@NotNull Call call, @NotNull Response response) throws IOException {
  109. getActivity().runOnUiThread(() -> {
  110. String parseResponse = ICSPClient.isSuccessfulAndParseResponse(response);
  111. if (!StringUtils.isNullOrEmpty(parseResponse)) {
  112. try {
  113. ResponseData<PreparedOrderResponseVo> responseData = new ObjectMapper().readValue(parseResponse, new TypeReference<ResponseData<PreparedOrderResponseVo>>() {
  114. });
  115. PreparedOrderResponseVo data = responseData.getData();
  116. if (null != responseData && ICSPResponseCodeEnum.OK.getCode().equals(responseData.getCode())) {
  117. String collectorPhone = data.getCollectorPhone();
  118. binding.delivererPhoneConfirm.setText(collectorPhone);
  119. }
  120. ICSPResponseCodeEnum.responseHint(getContext(), responseData);
  121. } catch (JsonProcessingException e) {
  122. Log.e(TAG, "onResponse: 服务器响应错误! 自动获取用户手机号失败! ", e);
  123. ToastUtils.make(getContext(), "网络异常! 请手动输入手机号! ");
  124. LoggingUtils.sendErrorLog("业务异常: 服务器响应错误! 自动获取用户手机号失败! ", e);
  125. } catch (RuntimeException e) {
  126. Log.e(TAG, "onResponse: 获取用户手机号未知错误! ", e);
  127. ToastUtils.make(getContext(), "未查询到手机号! 请手动输入! ");
  128. LoggingUtils.sendErrorLog("业务异常: 获取用户手机号未知错误! ", e);
  129. }
  130. } else {
  131. ToastUtils.make(getContext(), "服务器异常!请稍后重试!");
  132. }
  133. mailNoLock.set(1);
  134. });
  135. }
  136. });
  137. }
  138. });
  139. } catch (RuntimeException e) {
  140. Log.e(TAG, "onViewCreated: InputInfoFragment页面获取ChooseCabinetFragment传值错误! ", e);
  141. LoggingUtils.sendErrorLog("业务异常: InputInfoFragment页面获取ChooseCabinetFragment传值错误! ", e);
  142. }
  143. // TODO 离线逻辑, 暂时不做
  144. Pattern phonePattern = Pattern.compile("^((13[0-9])|(14[0,1,4-9])|(15[0-3,5-9])|(16[2,5,6,7])|(17[0-8])|(18[0-9])|(19[0-3,5-9]))\\d{8}$");
  145. // 下单逻辑
  146. String finalSectionType = sectionType;
  147. if (submitLock.get() == 1) {
  148. submitLock.incrementAndGet();
  149. binding.buttonConfirm.setOnClickListener(view1 -> {
  150. Bundle bundle = new Bundle();
  151. String mailNo = binding.mailNo.getText().toString();
  152. // String phone = binding.delivererPhone.getText().toString();
  153. String phoneConfirm = binding.delivererPhoneConfirm.getText().toString();
  154. if (!phonePattern.matcher(phoneConfirm).matches()) {
  155. ToastUtils.make(getContext(), "请输入正确的手机号!");
  156. } else if (StringUtils.isNullOrEmpty(mailNo)) {
  157. ToastUtils.make(getContext(), "运单号不能为空!请扫描运单号!");
  158. } else {
  159. // 验证完 预下单
  160. PreparedOrderVo preparedOrderVo = new PreparedOrderVo();
  161. preparedOrderVo.setClientId(BaseUtils.getClientId());
  162. preparedOrderVo.setMailNo(mailNo);
  163. preparedOrderVo.setCollectorPhone(phoneConfirm);
  164. preparedOrderVo.setLockerType(finalSectionType);
  165. preparedOrderVo.setDelivererPhone(LocalStorage.getInstance().getSession().getPhone());
  166. // 预下单
  167. ICSPClient.preparedOrder(LocalStorage.getInstance().getSession().getToken(), preparedOrderVo, new Callback() {
  168. @Override
  169. public void onFailure(@NotNull Call call, @NotNull IOException e) {
  170. // TODO 预下单调用失败
  171. ToastUtils.make(getContext(), "请求服务器失败!网络异常!");
  172. Log.e(TAG, "onFailure: 预下单失败! call: " + call.timeout().toString(), e);
  173. LoggingUtils.sendErrorLog("业务异常: 预下单失败! call: " + call.timeout().toString(), e);
  174. }
  175. @Override
  176. public void onResponse(@NotNull Call call, @NotNull Response response) {
  177. ((MainActivity) getActivity()).runOnUiThread(() -> {
  178. try {
  179. String responseStr = ICSPClient.isSuccessfulAndParseResponse(response);
  180. if (!StringUtils.isNullOrEmpty(responseStr)) {
  181. ObjectMapper objectMapper = JacksonUtils.objectmapper;
  182. ResponseData<PreparedOrderResponseVo> responseData = objectMapper.readValue(responseStr, new TypeReference<ResponseData<PreparedOrderResponseVo>>() {});
  183. data = responseData.getData();
  184. // 响应信息
  185. if (responseData.getCode().equals(ICSPResponseCodeEnum.OK.getCode())) {
  186. bundle.putString("preparedOrderResponse", objectMapper.writeValueAsString(data));
  187. bundle.putString("takeNo", mailNo);
  188. bundle.putString("sectionType", finalSectionType);
  189. // 跳转到确认投递
  190. try {
  191. MainActivity activity = (MainActivity) getActivity();
  192. activity.getBundleMap().put(InputInfoFragment.class.getName(), bundle);
  193. } catch (RuntimeException e) {
  194. Log.e(TAG, "onResponse: InputInfoFragment页面获取SendInfoConfirmFragment传值错误! ", e);
  195. LoggingUtils.sendErrorLog("业务异常: InputInfoFragment页面获取SendInfoConfirmFragment传值错误! ", e);
  196. }
  197. timer.cancel();
  198. NavHostFragment.findNavController(InputInfoFragment.this).navigate(R.id.action_inputInfoFragment_to_sendInfoConfirmFragment);
  199. } else {
  200. RuntimeException exception = new RuntimeException("系统异常");
  201. Log.e(TAG, "onResponse: ICSP返回码: " + responseData.getCode() + ", 返回信息: " + responseData.getMsg(), exception);
  202. LoggingUtils.sendErrorLog("业务异常: ICSP返回码: " + responseData.getCode() + ", 返回信息: " + responseData.getMsg(), exception);
  203. }
  204. ToastUtils.make(getContext(), responseData.getMsg());
  205. } else {
  206. ToastUtils.make(getContext(), "服务器异常!请稍后重试!");
  207. }
  208. } catch (JsonProcessingException e) {
  209. Log.e(TAG, "onResponse: 预下单转换成JSON出错! ", e);
  210. LoggingUtils.sendErrorLog("业务异常: 预下单转换成JSON出错! ", e);
  211. } catch (RuntimeException e) {
  212. Log.e(TAG, "onResponse: 预下单失败! ", e);
  213. LoggingUtils.sendErrorLog("业务异常: 预下单失败! ", e);
  214. }
  215. });
  216. }
  217. });
  218. }
  219. });
  220. }
  221. // 数字输入 5 手机号 4 密码
  222. binding.delivererPhoneConfirm.setOnFocusChangeListener((view1, hasFocus) -> {
  223. if (hasFocus) {
  224. binding.inputButton0.setOnClickListener(v -> {
  225. int length = binding.delivererPhoneConfirm.getText().length();
  226. if (length >= 11) {
  227. ToastUtils.make(getContext(), "手机号不能超过11位! ");
  228. return;
  229. }
  230. int start = binding.delivererPhoneConfirm.getSelectionStart();
  231. binding.delivererPhoneConfirm.getText().insert(start, "0");
  232. });
  233. binding.inputButton1.setOnClickListener(v -> {
  234. int length = binding.delivererPhoneConfirm.getText().length();
  235. if (length >= 11) {
  236. ToastUtils.make(getContext(), "手机号不能超过11位! ");
  237. return;
  238. }
  239. int start = binding.delivererPhoneConfirm.getSelectionStart();
  240. binding.delivererPhoneConfirm.getText().insert(start, "1");
  241. });
  242. binding.inputButton2.setOnClickListener(v -> {
  243. int length = binding.delivererPhoneConfirm.getText().length();
  244. if (length >= 11) {
  245. ToastUtils.make(getContext(), "手机号不能超过11位! ");
  246. return;
  247. }
  248. int start = binding.delivererPhoneConfirm.getSelectionStart();
  249. binding.delivererPhoneConfirm.getText().insert(start, "2");
  250. });
  251. binding.inputButton3.setOnClickListener(v -> {
  252. int length = binding.delivererPhoneConfirm.getText().length();
  253. if (length >= 11) {
  254. ToastUtils.make(getContext(), "手机号不能超过11位! ");
  255. return;
  256. }
  257. int start = binding.delivererPhoneConfirm.getSelectionStart();
  258. binding.delivererPhoneConfirm.getText().insert(start, "3");
  259. });
  260. binding.inputButton4.setOnClickListener(v -> {
  261. int length = binding.delivererPhoneConfirm.getText().length();
  262. if (length >= 11) {
  263. ToastUtils.make(getContext(), "手机号不能超过11位! ");
  264. return;
  265. }
  266. int start = binding.delivererPhoneConfirm.getSelectionStart();
  267. binding.delivererPhoneConfirm.getText().insert(start, "4");
  268. });
  269. binding.inputButton5.setOnClickListener(v -> {
  270. int length = binding.delivererPhoneConfirm.getText().length();
  271. if (length >= 11) {
  272. ToastUtils.make(getContext(), "手机号不能超过11位! ");
  273. return;
  274. }
  275. int start = binding.delivererPhoneConfirm.getSelectionStart();
  276. binding.delivererPhoneConfirm.getText().insert(start, "5");
  277. });
  278. binding.inputButton6.setOnClickListener(v -> {
  279. int length = binding.delivererPhoneConfirm.getText().length();
  280. if (length >= 11) {
  281. ToastUtils.make(getContext(), "手机号不能超过11位! ");
  282. return;
  283. }
  284. int start = binding.delivererPhoneConfirm.getSelectionStart();
  285. binding.delivererPhoneConfirm.getText().insert(start, "6");
  286. });
  287. binding.inputButton7.setOnClickListener(v -> {
  288. int length = binding.delivererPhoneConfirm.getText().length();
  289. if (length >= 11) {
  290. ToastUtils.make(getContext(), "手机号不能超过11位! ");
  291. return;
  292. }
  293. int start = binding.delivererPhoneConfirm.getSelectionStart();
  294. binding.delivererPhoneConfirm.getText().insert(start, "7");
  295. });
  296. binding.inputButton8.setOnClickListener(v -> {
  297. int length = binding.delivererPhoneConfirm.getText().length();
  298. if (length >= 11) {
  299. ToastUtils.make(getContext(), "手机号不能超过11位! ");
  300. return;
  301. }
  302. int start = binding.delivererPhoneConfirm.getSelectionStart();
  303. binding.delivererPhoneConfirm.getText().insert(start, "8");
  304. });
  305. binding.inputButton9.setOnClickListener(v -> {
  306. int length = binding.delivererPhoneConfirm.getText().length();
  307. if (length >= 11) {
  308. ToastUtils.make(getContext(), "手机号不能超过11位");
  309. return;
  310. }
  311. int start = binding.delivererPhoneConfirm.getSelectionStart();
  312. binding.delivererPhoneConfirm.getText().insert(start, "9");
  313. });
  314. binding.inputButtonCancel.setOnClickListener(v -> {
  315. Editable text = binding.delivererPhoneConfirm.getText();
  316. if (text.length() > 0) {
  317. text.delete(text.length()-1, text.length());
  318. binding.delivererPhoneConfirm.setText(text, TextView.BufferType.EDITABLE);
  319. binding.delivererPhoneConfirm.setSelection(binding.delivererPhoneConfirm.getText().length());
  320. }
  321. });
  322. // 长按删除
  323. binding.inputButtonCancel.setOnLongClickListener(v -> {
  324. binding.delivererPhoneConfirm.getText().clear();
  325. return false;
  326. });
  327. }
  328. });
  329. binding.mailNo.setOnFocusChangeListener((view1, hasFocus) -> {
  330. if (hasFocus) {
  331. binding.inputButton0.setOnClickListener(v -> {
  332. int start = binding.mailNo.getSelectionStart();
  333. binding.mailNo.getText().insert(start, "0");
  334. });
  335. binding.inputButton1.setOnClickListener(v -> {
  336. int start = binding.mailNo.getSelectionStart();
  337. binding.mailNo.getText().insert(start, "1");
  338. });
  339. binding.inputButton2.setOnClickListener(v -> {
  340. int start = binding.mailNo.getSelectionStart();
  341. binding.mailNo.getText().insert(start, "2");
  342. });
  343. binding.inputButton3.setOnClickListener(v -> {
  344. int start = binding.mailNo.getSelectionStart();
  345. binding.mailNo.getText().insert(start, "3");
  346. });
  347. binding.inputButton4.setOnClickListener(v -> {
  348. int start = binding.mailNo.getSelectionStart();
  349. binding.mailNo.getText().insert(start, "4");
  350. });
  351. binding.inputButton5.setOnClickListener(v -> {
  352. int start = binding.mailNo.getSelectionStart();
  353. binding.mailNo.getText().insert(start, "5");
  354. });
  355. binding.inputButton6.setOnClickListener(v -> {
  356. int start = binding.mailNo.getSelectionStart();
  357. binding.mailNo.getText().insert(start, "6");
  358. });
  359. binding.inputButton7.setOnClickListener(v -> {
  360. int start = binding.mailNo.getSelectionStart();
  361. binding.mailNo.getText().insert(start, "7");
  362. });
  363. binding.inputButton8.setOnClickListener(v -> {
  364. int start = binding.mailNo.getSelectionStart();
  365. binding.mailNo.getText().insert(start, "8");
  366. });
  367. binding.inputButton9.setOnClickListener(v -> {
  368. int start = binding.mailNo.getSelectionStart();
  369. binding.mailNo.getText().insert(start, "9");
  370. });
  371. binding.inputButtonCancel.setOnClickListener(v -> {
  372. Editable text = binding.mailNo.getText();
  373. if (text.length() > 0) {
  374. text.delete(text.length()-1, text.length());
  375. binding.mailNo.setText(text, TextView.BufferType.EDITABLE);
  376. binding.mailNo.setSelection(binding.mailNo.getText().length());
  377. }
  378. });
  379. // 长按删除
  380. binding.inputButtonCancel.setOnLongClickListener(v -> {
  381. binding.mailNo.getText().clear();
  382. return false;
  383. });
  384. }
  385. });
  386. binding.returnBtn.setOnClickListener(view1 -> {
  387. timer.cancel();
  388. NavHostFragment.findNavController(InputInfoFragment.this)
  389. .navigate(R.id.action_inputInfoFragment_to_chooseCabinetFragment);
  390. });
  391. }
  392. public PreparedOrderResponseVo getData() {
  393. return data;
  394. }
  395. @Override
  396. public void onDestroy() {
  397. if (timer != null) {
  398. timer.cancel();
  399. timer = null;
  400. }
  401. super.onDestroy();
  402. binding = null;
  403. }
  404. }