Geen omschrijving

views.py 18KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485
  1. from django.shortcuts import render, redirect, get_object_or_404
  2. from django.core.paginator import Paginator
  3. from django.contrib import messages
  4. from core.models import Report
  5. from core.forms import ReportForm
  6. from core.utils import ConfigurableCRUDView, queryFromMaster
  7. from .filters import ReportFilter
  8. from .forms import ExportOptionsForm
  9. from pprint import pprint
  10. from legacy.models import Data, DataMs, DataRl, DataWb, LotSummary, LotSummaryRl, LotSummaryWb, PressCal, RotateData, TbFgPressinfoLotlist,\
  11. Manualsize
  12. from .gen_report import gen_xlsx
  13. from django.core.files.base import File
  14. from pathlib import Path
  15. from django.views.decorators.csrf import csrf_exempt
  16. from django.http import JsonResponse, HttpResponseBadRequest
  17. import json
  18. from django.contrib.auth.decorators import login_required
  19. from django.contrib.auth.models import User
  20. from legacy.models import Data
  21. from django.conf import settings
  22. def index(request):
  23. reports = Report.objects.all()
  24. report_filter = ReportFilter(request.GET, queryset=reports)
  25. # Paginate the filtered queryset
  26. paginator = Paginator(report_filter.qs, 10) # Show 10 reports per page
  27. page_number = request.GET.get('page')
  28. page_obj = paginator.get_page(page_number)
  29. context = {
  30. 'filter': report_filter,
  31. 'page_obj': page_obj,
  32. }
  33. return render(request, 'report/index.html', context)
  34. def report_create_view(request):
  35. if request.method == "POST":
  36. form = ReportForm(request.POST)
  37. if form.is_valid():
  38. form.save()
  39. messages.success(request, "Report created successfully!")
  40. return redirect("report:report_index") # Adjust with your report list view name
  41. else:
  42. form = ReportForm()
  43. return render(request, "report/create.html", {"form": form})
  44. class ReportCRUDView(ConfigurableCRUDView):
  45. model = Report
  46. list_template_name = 'legacy/datacrud_list.html'
  47. detail_template_name = 'legacy/datacrud_detail.html'
  48. form_template_name = 'report/report_form.html'
  49. confirm_delete_template_name = 'legacy/datacrud_confirm_delete.html'
  50. filterset_class = ReportFilter
  51. page_title = "Reports"
  52. # URL name mappings
  53. list_url_name = 'report:report-list'
  54. create_url_name = 'report:report-create'
  55. update_url_name = 'report:report-update'
  56. delete_url_name = 'report:report-delete'
  57. config_fields = ["name", "file", "created_by", "created_at"]
  58. config_field_orders = ["id", "name", "created_by"]
  59. # config_readonly_fields = ["lot_no"]
  60. # config_edit_fields = ["lot_no", "code"]
  61. ordering = ["-created_at", "-id",]
  62. def convert_sheet_data(sheet_data):
  63. """
  64. Convert sheet_data to the required form with prefixed keys.
  65. :param sheet_data: Dictionary with sheet names as keys and their data as values.
  66. :return: Dictionary in the required key-value format.
  67. """
  68. converted_data = {}
  69. for sheet_name, data in sheet_data.items():
  70. for key, value in data.items():
  71. # Prefix each key with the sheet name
  72. converted_key = f"{sheet_name}.{key}"
  73. converted_data[converted_key] = value
  74. return converted_data
  75. def hide_con(placeholders, mark_value, hide_rows):
  76. """
  77. Updates the 'placeholders' dictionary with a mark value and hide rows range.
  78. :param placeholders: The dictionary to update.
  79. :param mark_value: The key to check or update in the placeholders.
  80. :param hide_rows: The row range to append in the format '[start:end]'.
  81. """
  82. if mark_value in placeholders:
  83. placeholders[mark_value] = f"{placeholders[mark_value]}[{hide_rows}]"
  84. else:
  85. placeholders[mark_value] = f"0[{hide_rows}]"
  86. def generate_hardness_out_values(lot_no):
  87. """
  88. Generate a dictionary of placeholder values for a given lot_no.
  89. :param lot_no: The lot number to query data for.
  90. :return: A dictionary with placeholders (e.g., v1_1, v1_2, ...) as keys and their respective values.
  91. """
  92. # Query the Data model for records matching the given lot_no
  93. records = Data.objects.filter(lot_no=lot_no).order_by('row_no')
  94. print(f"records {lot_no} = {records.values()}")
  95. # Initialize an empty dictionary to store placeholder values
  96. placeholders = {}
  97. # Iterate over the records to populate placeholder values
  98. for record_idx, record in enumerate(records, start=1):
  99. placeholders[f'v{record_idx}_1'] = record.p1 # Checkpoint 1 value
  100. placeholders[f'v{record_idx}_2'] = record.p2 # Checkpoint 2 value
  101. placeholders[f'v{record_idx}_3'] = record.p3 # Checkpoint 3 value
  102. placeholders[f'v{record_idx}_4'] = record.avg # Average value
  103. placeholders[f'v{record_idx}_5'] = record.rgrade # Judgment value
  104. return placeholders
  105. def generate_hardness_out_in_values(lot_no):
  106. # Fetch records from the Data model
  107. records = Data.objects.filter(lot_no=lot_no).order_by('row_no')
  108. out_data = []
  109. in_data = []
  110. # Separate OUT and IN data
  111. for record in records:
  112. if record.r_type.upper() in ["OUT", "TOP", "FA1", "UPP", "UPPE", "RIM"]:
  113. out_data.append(record)
  114. elif record.r_type.upper() in ["IN", "UNDER", "UND", "FA2", "LOW", "LOWE", "BASE", "SOKO", "IN*"]:
  115. in_data.append(record)
  116. # Prepare placeholders
  117. placeholders = {}
  118. for idx, record in enumerate(out_data, start=1):
  119. placeholders[f'v{idx}_1'] = record.p1
  120. placeholders[f'v{idx}_2'] = record.p2
  121. placeholders[f'v{idx}_3'] = record.p3
  122. placeholders[f'v{idx}_4'] = record.avg
  123. placeholders[f'v{idx}_5'] = record.rgrade
  124. for idx, record in enumerate(in_data, start=1):
  125. placeholders[f'v{len(out_data) + idx}_1'] = record.p1
  126. placeholders[f'v{len(out_data) + idx}_2'] = record.p2
  127. placeholders[f'v{len(out_data) + idx}_3'] = record.p3
  128. placeholders[f'v{len(out_data) + idx}_4'] = record.avg
  129. placeholders[f'v{len(out_data) + idx}_5'] = record.rgrade
  130. # if "v3_1" in placeholders:
  131. # placeholders["v3_1"] = f"{placeholders['v3_1']}[25:28]"
  132. # else:
  133. # placeholders[f"v3_1"] = "0[25:28]"
  134. hide_con(placeholders, "v3_1", "25:28")
  135. return placeholders
  136. def generate_dimension_values(lot_no):
  137. """
  138. Fetch dimension records from manualSize and DataMs models
  139. and generate placeholder values for Standard, Actual, and Judgement.
  140. Supports two row_no entries per lot.
  141. """
  142. # Fetch standard values from manualSize (limit to 2 rows)
  143. manual_size_records = Manualsize.objects.filter(lotno=lot_no)
  144. # Fetch actual and judgement values from DataMs ordered by row_no (limit to 2 rows)
  145. data_ms_records = DataMs.objects.filter(lot_no=lot_no).order_by('row_no')[:2]
  146. # Prepare placeholders
  147. placeholders = {}
  148. for i in range(1,7):
  149. for j in range(1,4):
  150. placeholders[f'v{i}_{j}'] = 0
  151. pprint(placeholders)
  152. pprint(manual_size_records)
  153. for m in manual_size_records:
  154. if m.size_name == "D":
  155. placeholders['v1_1'] = placeholders['v4_1'] = f'{m.std} +{m.tolup} {m.tolun}'
  156. if m.size_name == "T":
  157. placeholders['v2_1'] = placeholders['v5_1'] = f'{m.std} +{m.tolup} {m.tolun}'
  158. if m.size_name == "H":
  159. placeholders['v3_1'] = placeholders['v6_1'] = f'{m.std} +{m.tolup} {m.tolun}'
  160. # Ensure that we map each manualSize entry to a corresponding DataMs entry
  161. for r in data_ms_records:
  162. if r.row_no == 1:
  163. placeholders[f'v1_2'] = r.dsize
  164. placeholders[f'v1_3'] = r.dsizeok
  165. placeholders[f'v2_2'] = r.tsize
  166. placeholders[f'v2_3'] = r.tsizeok
  167. placeholders[f'v3_2'] = r.hsize
  168. placeholders[f'v3_3'] = r.hsizeok
  169. if r.row_no == 2:
  170. placeholders[f'v4_2'] = r.dsize
  171. placeholders[f'v4_3'] = r.dsizeok
  172. placeholders[f'v5_2'] = r.tsize
  173. placeholders[f'v5_3'] = r.tsizeok
  174. placeholders[f'v6_2'] = r.hsize
  175. placeholders[f'v6_3'] = r.hsizeok
  176. hide_con(placeholders, "v4_2", "24:28")
  177. return placeholders
  178. def merge_sheet_data_with_data(sheet_data, data):
  179. """
  180. Merge `sheet_data` with `data`.
  181. :param sheet_data: Dictionary containing the sheet-specific data.
  182. :param data: Dictionary containing general data.
  183. :return: A merged dictionary combining both `sheet_data` and `data`.
  184. """
  185. # Merge dictionaries using unpacking
  186. merged_data = {**data, **sheet_data}
  187. return merged_data
  188. def create_coi_file(lot_no, sheets, user, md):
  189. pprint("---- create_coi_file ---")
  190. pprint(md)
  191. qa1 = User.objects.get(pk=md['qa1'])
  192. qa2 = User.objects.get(pk=md['qa2'])
  193. accept = specialAccept = False
  194. if md['acceptStatus'] == "accepted":
  195. accept = True
  196. if md['acceptStatus'] == "special_accepted":
  197. specialAccept = True
  198. pprint(qa1)
  199. pprint(qa2)
  200. sheet_data = {}
  201. for sheet_name in sheets:
  202. match sheet_name:
  203. case 'hardness_out':
  204. sheet_data[sheet_name] = generate_hardness_out_values(lot_no)
  205. case 'hardness_out_in':
  206. sheet_data[sheet_name] = generate_hardness_out_in_values(lot_no)
  207. case 'dimension':
  208. sheet_data[sheet_name] = generate_dimension_values(lot_no)
  209. converted_data = convert_sheet_data(sheet_data)
  210. print(f"sheet_data \n {sheet_data}")
  211. print(f"converted_data \n {converted_data}")
  212. results = queryFromMaster(lot_no)
  213. first_result = results[0] if results else None
  214. try:
  215. pcs = int(first_result.PRO5) - int(first_result.PRO27)
  216. except:
  217. pcs = 0
  218. if first_result:
  219. size_str = f"{first_result.PRO10}x{first_result.PRO11}x{first_result.PRO12}";
  220. spec = f"{first_result.PRO13} {first_result.PRO14} {first_result.PRO15} {first_result.PRO16} {first_result.PRO17} {first_result.PRO18}"
  221. else:
  222. size_str = ""
  223. spec = ""
  224. data = {
  225. "customer": first_result.PRO1C,
  226. "inspect_date": "2025-01-15",
  227. "lot_no": lot_no,
  228. "size": size_str,
  229. "pcs": pcs,
  230. "spec": spec,
  231. # "hardness_out.acc": True, # Hide rows 24 to 28 if the prefix is "0"
  232. # "hardness_out.spe_acc": False, # Hide rows 24 to 28 if the prefix is "0"
  233. "acc": accept, # Hide rows 24 to 28 if the prefix is "0"
  234. "spe_acc": specialAccept, # Hide rows 24 to 28 if the prefix is "0"
  235. # "hardness_out.qa1": f"{qa1.first_name} {qa1.last_name}",
  236. # "hardness_out.qa2": f"{qa2.first_name} {qa2.last_name}",
  237. "qa1": f"{qa1.first_name} {qa1.last_name}",
  238. "qa2": f"{qa2.first_name} {qa2.last_name}",
  239. "sign1": qa1.profile.signed_picture,
  240. "sign2": qa2.profile.signed_picture,
  241. "pos1": qa1.profile.get_position_display(),
  242. "pos2": qa2.profile.get_position_display()
  243. }
  244. merged_data = merge_sheet_data_with_data(converted_data, data)
  245. pprint(f"---- merged_data ---")
  246. pprint(merged_data)
  247. output_file = gen_xlsx(
  248. template_file=f"{settings.BASE_DIR}/report/coi_templates.xlsx",
  249. selected_sheets=sheets, # Replace with your actual sheet names
  250. prefix_filename=f"{settings.BASE_DIR}/media/coi",
  251. data=merged_data
  252. )
  253. report = Report.objects.create(
  254. name=lot_no,
  255. created_by=user,
  256. file=None # Leave this as None or assign a file if required
  257. )
  258. output_file_path = Path(output_file) # Convert to a Path object for convenience
  259. with open(output_file_path, "rb") as f:
  260. report.file.save(output_file_path.name, File(f), save=True)
  261. pprint(f"outputfile = {output_file}")
  262. return report
  263. SHEET_NAMES = {
  264. 'hardness_out': 'Hardness Out',
  265. 'hardness_out_in': 'Hardness Out/In',
  266. 'hardness_both_size': 'Hardness Both Size',
  267. 'dimension': 'Dimension',
  268. 'dimension_app': 'Dimension Appearance',
  269. 'dimension_bal_weight': 'Dimension Balance/Weight',
  270. 'dim_bal_app_hard': 'Dimension Balance/Appearance/Hardness',
  271. 'dim_bal_app_rot_hard': 'Dimension Balance/Appearance/Rotation/Hardness',
  272. 'thickness_8_point': 'Thickness 8 Points',
  273. 'centering': 'Centering',
  274. }
  275. def get_fields(model):
  276. # model_fields = {f.name: f for f in model._meta.get_fields()}
  277. # fields = list(model_fields.values())
  278. # return fields
  279. fields = [f for f in model._meta.get_fields() if not f.auto_created]
  280. return fields
  281. def filter_by_lot_no(lot_no):
  282. models = [Data, DataMs, DataRl, DataWb, LotSummary, LotSummaryRl, LotSummaryWb, PressCal, RotateData ] # List of models to process
  283. results = {}
  284. fields = {}
  285. for model in models:
  286. model_fields = [f.name for f in model._meta.get_fields()]
  287. # Check if "id" and "row_no" are in the model's fields
  288. order_fields = []
  289. if "id" in model_fields:
  290. order_fields.append("id")
  291. if "row_no" in model_fields:
  292. order_fields.append("row_no")
  293. # Dynamically filter and order results
  294. model_name = model.__name__
  295. if order_fields:
  296. results[model_name] = model.objects.filter(lot_no=lot_no).order_by(*order_fields)
  297. else:
  298. results[model_name] = model.objects.filter(lot_no=lot_no) # No
  299. fields[model_name] = get_fields(model)
  300. return results, fields
  301. def coi_view(request):
  302. pprint(f"xxxx method = xxx {request.method}")
  303. users = User.objects.all()
  304. if request.method == "POST":
  305. pprint(request.POST)
  306. exports = request.POST.getlist("exports") # Retrieve the list of selected values
  307. pprint(f"Selected Export Options: {exports}")
  308. if 'export' in request.POST:
  309. data = {
  310. "customer": "Tum Coder",
  311. "inspect_date": "2025-01-15",
  312. "lot_no": "12345",
  313. "staff_name": "Tum 8888",
  314. "man_name": "Tum 999",
  315. "size": "Large",
  316. "lot_size": "10 pcs",
  317. "spec": "Spec-A",
  318. "hardness_out.d1_act": "10",
  319. "hardness_out.d2_act": "0[24:28]", # Hide rows 24 to 28 if the prefix is "0"
  320. "hardness_out.acc": True, # Hide rows 24 to 28 if the prefix is "0"
  321. "hardness_out.spe_acc": False, # Hide rows 24 to 28 if the prefix is "0"
  322. "dimension_app.d1_act": "33",
  323. "dimension_app.d2_act": "0[26:32]", # Hide rows 24 to 28 if the prefix is "0"
  324. "dimension_app.acc": True, # Hide rows 24 to 28 if the prefix is "0"
  325. "dimension_app.spe_acc": True, # Hide rows 24 to 28 if the prefix is "0"
  326. }
  327. output_file = gen_xlsx(
  328. template_file="/app/report/coi_templates.xlsx",
  329. selected_sheets=exports, # Replace with your actual sheet names
  330. prefix_filename="/app/media/coi",
  331. data=data
  332. )
  333. report = Report.objects.create(
  334. name=request.POST.get('lot_no','Untitled'),
  335. created_by=request.user,
  336. file=None # Leave this as None or assign a file if required
  337. )
  338. output_file_path = Path(output_file) # Convert to a Path object for convenience
  339. with open(output_file_path, "rb") as f:
  340. report.file.save(output_file_path.name, File(f), save=True)
  341. pprint(f"outputfile = {output_file}")
  342. if 'search_lot' in request.POST:
  343. lot_no = request.POST.get('lot_no', None)
  344. lot_no = lot_no.strip()
  345. if lot_no:
  346. results = queryFromMaster(lot_no)
  347. first_result = results[0] if results else None
  348. try:
  349. pcs = int(first_result.PRO5) - int(first_result.PRO27)
  350. except:
  351. pcs = 0
  352. if first_result:
  353. size_str = f"{first_result.PRO10}x{first_result.PRO11}x{first_result.PRO12}";
  354. spec = f"{first_result.PRO13} {first_result.PRO14} {first_result.PRO15} {first_result.PRO16} {first_result.PRO17} {first_result.PRO18}"
  355. else:
  356. size_str = ""
  357. spec = ""
  358. results, fields = filter_by_lot_no(lot_no)
  359. # results1 = Data.objects.filter(lot_no=lot_no).order_by("id", "row_no")
  360. # fields1 = get_fields(Data)
  361. # results2 = DataMs.objects.filter(lot_no=lot_no).order_by("id", "row_no")
  362. # fields2 = get_fields(DataMs)
  363. return render(request, 'report/coi.html', {'result': first_result,
  364. 'pcs':pcs,
  365. 'size_str': size_str,
  366. 'lot_no': lot_no,
  367. 'spec': spec, 'users': users, 'SHEET_NAMES': SHEET_NAMES,
  368. 'results': results, 'fields': fields})
  369. messages.success(request, "Request Sent")
  370. return redirect(request.path_info)
  371. return render(request, 'report/coi.html', {'SHEET_NAMES': SHEET_NAMES, 'users': users})
  372. @csrf_exempt # Disable CSRF for API requests (ensure this is secure in production)
  373. @login_required
  374. def gen_report_view(request):
  375. if request.method == "POST":
  376. # try:
  377. # Parse JSON data from the request body
  378. data = json.loads(request.body)
  379. lot_no = data.get("lot_no").strip()
  380. exports = data.get("exports")
  381. qa1 = data.get('qa1')
  382. qa2 = data.get('qa2')
  383. print(f"data = {data}")
  384. if not lot_no:
  385. return HttpResponseBadRequest("Missing 'lot_no' in request data")
  386. # Call the `create_coi_file` function with the provided lot_no
  387. report = create_coi_file(lot_no, exports, request.user, {'qa1': qa1, 'qa2': qa2, \
  388. 'acceptStatus': data.get('acceptStatus')})
  389. # Return a success response with the report details
  390. return JsonResponse({
  391. "message": "Report generated successfully",
  392. "report_id": report.id,
  393. "file_url": report.file.url if report.file else None,
  394. })
  395. # except json.JSONDecodeError:
  396. # return HttpResponseBadRequest("Invalid JSON data")
  397. # except Exception as e:
  398. # pprint(e)
  399. # return JsonResponse({"error": str(e)}, status=500)
  400. else:
  401. return HttpResponseBadRequest("Only POST requests are allowed")