From 28cf22aaff7f092256db2ad6df699e17426f62ea Mon Sep 17 00:00:00 2001
From: chenhj <1263187585@qq.com>
Date: 星期四, 30 四月 2026 16:34:27 +0800
Subject: [PATCH] Merge branch 'dev_New_pro' of http://114.132.189.42:9002/r/product-inventory-management-after into dev_New_pro

---
 src/main/java/com/ruoyi/production/service/impl/ProductionOperationTaskServiceImpl.java |  337 ++++++++++++++++++++++++++++++++++++++++++++++++++++++-
 1 files changed, 329 insertions(+), 8 deletions(-)

diff --git a/src/main/java/com/ruoyi/production/service/impl/ProductionOperationTaskServiceImpl.java b/src/main/java/com/ruoyi/production/service/impl/ProductionOperationTaskServiceImpl.java
index ee245d6..e202ec2 100644
--- a/src/main/java/com/ruoyi/production/service/impl/ProductionOperationTaskServiceImpl.java
+++ b/src/main/java/com/ruoyi/production/service/impl/ProductionOperationTaskServiceImpl.java
@@ -1,20 +1,341 @@
 package com.ruoyi.production.service.impl;
 
+import cn.hutool.core.bean.BeanUtil;
+import com.alibaba.fastjson2.JSON;
+import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
+import com.baomidou.mybatisplus.core.metadata.IPage;
+import com.baomidou.mybatisplus.core.toolkit.CollectionUtils;
+import com.baomidou.mybatisplus.core.toolkit.Wrappers;
+import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
 import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
+import com.deepoove.poi.XWPFTemplate;
+import com.deepoove.poi.data.PictureRenderData;
+import com.deepoove.poi.data.PictureType;
+import com.deepoove.poi.data.Pictures;
+import com.ruoyi.basic.dto.StorageAttachmentDTO;
+import com.ruoyi.basic.dto.StorageBlobVO;
+import com.ruoyi.basic.enums.RecordTypeEnum;
+import com.ruoyi.basic.utils.FileUtil;
+import com.ruoyi.common.config.FileProperties;
+import com.ruoyi.common.exception.ServiceException;
+import com.ruoyi.common.utils.MatrixToImageWriter;
+import com.ruoyi.common.utils.StringUtils;
+import com.ruoyi.production.bean.dto.ProductionOperationTaskDto;
+import com.ruoyi.production.bean.vo.ProductionOperationTaskVo;
 import com.ruoyi.production.mapper.ProductionOperationTaskMapper;
 import com.ruoyi.production.pojo.ProductionOperationTask;
 import com.ruoyi.production.service.ProductionOperationTaskService;
+import com.ruoyi.project.system.domain.SysUser;
+import com.ruoyi.project.system.mapper.SysUserMapper;
+import jakarta.servlet.http.HttpServletResponse;
+import lombok.RequiredArgsConstructor;
+import org.springframework.beans.factory.annotation.Value;
 import org.springframework.stereotype.Service;
 
-/**
- * <p>
- * 鐢熶骇宸ュ崟琛� 鏈嶅姟瀹炵幇绫�
- * </p>
- *
- * @author 鑺杞欢锛堟睙鑻忥級鏈夐檺鍏徃
- * @since 2026-04-21 03:55:52
- */
+import java.io.File;
+import java.io.FileInputStream;
+import java.io.InputStream;
+import java.io.OutputStream;
+import java.net.URLEncoder;
+import java.util.*;
+import java.util.stream.Collectors;
+
 @Service
+@RequiredArgsConstructor
 public class ProductionOperationTaskServiceImpl extends ServiceImpl<ProductionOperationTaskMapper, ProductionOperationTask> implements ProductionOperationTaskService {
 
+    private final SysUserMapper sysUserMapper;
+
+    private final FileUtil fileUtil;
+
+    private final FileProperties fileProperties;
+
+    @Value("${file.temp-dir}")
+    private String tempDir;
+
+    @Override
+    public IPage<ProductionOperationTaskVo> pageProductionOperationTask(Page<ProductionOperationTaskDto> page, ProductionOperationTaskDto dto) {
+        Page<ProductionOperationTaskVo> voPage = new Page<>(page.getCurrent(), page.getSize(), page.getTotal());
+        IPage<ProductionOperationTaskVo> result = baseMapper.pageProductionOperationTask(voPage, dto);
+        fillUserNames(result.getRecords());
+        return result;
+    }
+
+    @Override
+    public List<ProductionOperationTaskVo> listProductionOperationTask(ProductionOperationTaskDto dto) {
+        List<ProductionOperationTaskVo> result = BeanUtil.copyToList(this.list(buildQueryWrapper(dto)), ProductionOperationTaskVo.class);
+        fillUserNames(result);
+        return result;
+    }
+
+    @Override
+    public ProductionOperationTaskVo getProductionOperationTaskInfo(Long id) {
+        ProductionOperationTask item = this.getById(id);
+        if (item == null) {
+            return null;
+        }
+        ProductionOperationTaskVo vo = BeanUtil.copyProperties(item, ProductionOperationTaskVo.class);
+        fillUserNames(Collections.singletonList(vo));
+        return vo;
+    }
+
+    @Override
+    public boolean saveProductionOperationTask(ProductionOperationTask productionOperationTask) {
+        return this.saveOrUpdate(productionOperationTask);
+    }
+
+    @Override
+    public boolean removeProductionOperationTask(List<Long> ids) {
+        return ids != null && !ids.isEmpty() && this.removeByIds(ids);
+    }
+
+    @Override
+    public int updateProductWorkOrder(ProductionOperationTaskDto dto) {
+        return baseMapper.updateById(dto);
+    }
+
+    @Override
+    public boolean assign(ProductionOperationTaskDto dto) {
+        if (dto == null || dto.getId() == null) {
+            throw new ServiceException("宸ュ崟ID涓嶈兘涓虹┖");
+        }
+
+        ProductionOperationTask update = new ProductionOperationTask();
+        update.setId(dto.getId());
+        update.setUserIds(dto.getUserIds());
+        int rows = baseMapper.updateById(update);
+        if (rows <= 0) {
+            throw new ServiceException("宸ュ崟涓嶅瓨鍦ㄦ垨宸插垹闄�");
+        }
+        return true;
+    }
+
+    private LambdaQueryWrapper<ProductionOperationTask> buildQueryWrapper(ProductionOperationTaskDto dto) {
+        ProductionOperationTask query = dto == null ? new ProductionOperationTask() : dto;
+        return Wrappers.<ProductionOperationTask>lambdaQuery()
+                .eq(query.getId() != null, ProductionOperationTask::getId, query.getId())
+                .eq(query.getProductionOrderId() != null, ProductionOperationTask::getProductionOrderId, query.getProductionOrderId())
+                .eq(query.getProductionOrderRoutingOperationId() != null,
+                        ProductionOperationTask::getProductionOrderRoutingOperationId, query.getProductionOrderRoutingOperationId())
+                .eq(query.getStatus() != null, ProductionOperationTask::getStatus, query.getStatus())
+                .like(query.getWorkOrderNo() != null && !query.getWorkOrderNo().trim().isEmpty(),
+                        ProductionOperationTask::getWorkOrderNo, query.getWorkOrderNo())
+                .orderByDesc(ProductionOperationTask::getId);
+    }
+
+    private void fillUserNames(List<ProductionOperationTaskVo> voList) {
+        if (voList == null || voList.isEmpty()) {
+            return;
+        }
+        Set<Long> userIdSet = new LinkedHashSet<>();
+        for (ProductionOperationTaskVo vo : voList) {
+            if (vo == null) {
+                continue;
+            }
+            userIdSet.addAll(parseUserIdList(vo.getUserIds(), false));
+        }
+        if (userIdSet.isEmpty()) {
+            return;
+        }
+        List<SysUser> userList = sysUserMapper.selectUsersByIds(new ArrayList<>(userIdSet));
+        if (userList == null || userList.isEmpty()) {
+            return;
+        }
+        Map<Long, String> userNameById = userList.stream()
+                .filter(item -> item.getUserId() != null)
+                .collect(Collectors.toMap(SysUser::getUserId, SysUser::getNickName, (left, right) -> left));
+
+        for (ProductionOperationTaskVo vo : voList) {
+            if (vo == null) {
+                continue;
+            }
+            List<Long> userIds = parseUserIdList(vo.getUserIds(), false);
+            if (userIds.isEmpty()) {
+                vo.setUserNames(null);
+                continue;
+            }
+            String userNames = userIds.stream()
+                    .map(userNameById::get)
+                    .filter(StringUtils::isNotBlank)
+                    .collect(Collectors.joining(","));
+            vo.setUserNames(userNames);
+        }
+    }
+
+    private List<Long> parseUserIdList(String userIds, boolean strict) {
+        if (StringUtils.isBlank(userIds)) {
+            if (strict) {
+                throw new ServiceException("userIds鏍煎紡涓嶆纭紝蹇呴』涓篔SON鏁板瓧鏁扮粍");
+            }
+            return new ArrayList<>();
+        }
+
+        String text = userIds.trim();
+        try {
+            List<Long> parsed = JSON.parseArray(text, Long.class);
+            LinkedHashSet<Long> idSet = parsed == null ? new LinkedHashSet<>() : parsed.stream()
+                    .filter(Objects::nonNull)
+                    .collect(Collectors.toCollection(LinkedHashSet::new));
+            if (strict && idSet.isEmpty()) {
+                throw new ServiceException("userIds鏍煎紡涓嶆纭紝蹇呴』涓篔SON鏁板瓧鏁扮粍");
+            }
+            return new ArrayList<>(idSet);
+        } catch (Exception e) {
+            if (strict) {
+                throw new ServiceException("userIds鏍煎紡涓嶆纭紝蹇呴』涓篔SON鏁板瓧鏁扮粍");
+            }
+            return new ArrayList<>();
+        }
+    }
+
+    @Override
+    public void down(HttpServletResponse response, ProductionOperationTaskDto dto) {
+        if (dto == null || dto.getId() == null) {
+            throw new ServiceException("宸ュ崟ID涓嶈兘涓虹┖");
+        }
+
+        ProductionOperationTaskDto taskDto = baseMapper.getProductWorkOrderFlowCard(dto.getId());
+        if (taskDto == null) {
+            throw new ServiceException("宸ュ崟涓嶅瓨鍦紝ID=" + dto.getId());
+        }
+        String codePath;
+        try {
+            codePath = new MatrixToImageWriter().code(taskDto.getId().toString(), tempDir);
+        } catch (Exception e) {
+            throw new ServiceException("鐢熸垚浜岀淮鐮佸け璐�");
+        }
+
+        List<Map<String, Object>> images = buildTaskAttachmentImages(taskDto.getId());
+        Map<String, Object> renderData = new HashMap<>();
+        renderData.put("process", taskDto.getProcessName());
+        renderData.put("workOrderNo", taskDto.getWorkOrderNo());
+        renderData.put("productOrderNpsNo", taskDto.getProductOrderNpsNo());
+        renderData.put("productName", taskDto.getProductName());
+        renderData.put("planQuantity", taskDto.getPlanQuantity());
+        renderData.put("model", taskDto.getModel());
+        renderData.put("completeQuantity", taskDto.getCompleteQuantity());
+        renderData.put("scrapQty", taskDto.getScrapQty());
+        renderData.put("planStartTime", taskDto.getPlanStartTime());
+        renderData.put("planEndTime", taskDto.getPlanEndTime());
+        renderData.put("actualStartTime", taskDto.getActualStartTime());
+        renderData.put("actualEndTime", taskDto.getActualEndTime());
+        renderData.put("twoCode", Pictures.ofLocal(codePath).create());
+        renderData.put("images", images.isEmpty() ? null : images);
+
+        try (InputStream inputStream = this.getClass().getResourceAsStream("/static/work-order-template.docx")) {
+            if (inputStream == null) {
+                throw new ServiceException("娴佽浆鍗℃ā鏉夸笉瀛樺湪");
+            }
+            XWPFTemplate template = XWPFTemplate.compile(inputStream).render(renderData);
+            response.setContentType("application/msword");
+            String fileName = URLEncoder.encode("娴佽浆鍗�", "UTF-8");
+            response.setHeader("Access-Control-Expose-Headers", "Content-Disposition");
+            response.setHeader("Content-disposition", "attachment;filename=" + fileName + ".docx");
+            try (OutputStream os = response.getOutputStream()) {
+                template.write(os);
+                os.flush();
+            }
+        } catch (Exception e) {
+            throw new RuntimeException("瀵煎嚭澶辫触");
+        }
+    }
+
+    private List<Map<String, Object>> buildTaskAttachmentImages(Long taskId) {
+        List<Map<String, Object>> images = new ArrayList<>();
+        StorageAttachmentDTO storageAttachmentDTO = new StorageAttachmentDTO();
+        storageAttachmentDTO.setRecordType(RecordTypeEnum.PRODUCTION_OPERATION_TASK.getType());
+        storageAttachmentDTO.setRecordId(taskId);
+        List<StorageBlobVO> taskWorkOrderFiles =
+                fileUtil.getStorageBlobVOsByApplicationAndRecordTypeAndRecordId(storageAttachmentDTO);
+        if (CollectionUtils.isEmpty(taskWorkOrderFiles)) {
+            return images;
+        }
+        for (StorageBlobVO blobVO : taskWorkOrderFiles) {
+            if (blobVO == null) {
+                continue;
+            }
+            PictureType pictureType = resolvePictureType(blobVO);
+            if (pictureType == null) {
+                continue;
+            }
+            File imageFile = resolveImageFile(blobVO);
+            if (imageFile == null || !imageFile.exists() || !imageFile.isFile()) {
+                continue;
+            }
+            try (InputStream imageInputStream = new FileInputStream(imageFile)) {
+                Map<String, Object> image = new HashMap<>();
+                PictureRenderData pictureRenderData = Pictures.ofStream(imageInputStream, pictureType)
+                        .sizeInCm(17, 20)
+                        .create();
+                image.put("url", pictureRenderData);
+                images.add(image);
+            } catch (Exception ignored) {
+                // 鍗曚釜闄勪欢瑙f瀽澶辫触鏃惰烦杩囷紝閬垮厤褰卞搷鏁翠釜娴佽浆鍗″鍑�
+            }
+        }
+        return images;
+    }
+
+    private File resolveImageFile(StorageBlobVO blobVO) {
+        if (blobVO == null || StringUtils.isBlank(blobVO.getUidFilename())) {
+            return null;
+        }
+        if (StringUtils.isBlank(blobVO.getPath())) {
+            return new File(fileProperties.getPath(), blobVO.getUidFilename());
+        }
+        return new File(new File(fileProperties.getPath(), blobVO.getPath()), blobVO.getUidFilename());
+    }
+
+    private PictureType resolvePictureType(StorageBlobVO blobVO) {
+        if (blobVO == null) {
+            return null;
+        }
+        PictureType type = parsePictureTypeByFileName(blobVO.getOriginalFilename());
+        if (type != null) {
+            return type;
+        }
+        type = parsePictureTypeByFileName(blobVO.getUidFilename());
+        if (type != null) {
+            return type;
+        }
+        return parsePictureTypeByContentType(blobVO.getContentType());
+    }
+
+    private PictureType parsePictureTypeByFileName(String fileName) {
+        if (StringUtils.isBlank(fileName) || !fileName.contains(".")) {
+            return null;
+        }
+        try {
+            return PictureType.suggestFileType(fileName);
+        } catch (Exception ex) {
+            return null;
+        }
+    }
+
+    private PictureType parsePictureTypeByContentType(String contentType) {
+        if (StringUtils.isBlank(contentType)) {
+            return null;
+        }
+        String normalized = contentType.trim().toLowerCase(Locale.ROOT);
+        switch (normalized) {
+            case "image/jpeg":
+            case "image/jpg":
+            case "image/pjpeg":
+                return PictureType.JPEG;
+            case "image/png":
+                return PictureType.PNG;
+            case "image/gif":
+                return PictureType.GIF;
+            case "image/bmp":
+            case "image/x-ms-bmp":
+                return PictureType.BMP;
+            case "image/tiff":
+            case "image/tif":
+                return PictureType.TIFF;
+            case "image/svg+xml":
+                return PictureType.SVG;
+            default:
+                return null;
+        }
+    }
 }

--
Gitblit v1.9.3