excel.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448
  1. package el
  2. import (
  3. "box-cost/model"
  4. "context"
  5. "fmt"
  6. "log"
  7. "regexp"
  8. "strings"
  9. "time"
  10. "github.com/xuri/excelize/v2"
  11. "go.mongodb.org/mongo-driver/bson"
  12. "go.mongodb.org/mongo-driver/bson/primitive"
  13. "go.mongodb.org/mongo-driver/mongo"
  14. )
  15. const (
  16. EXCEL_TMPLATE_FILE = "tmplate/tmplate.xlsx"
  17. PURCHASE_COLLECTION = "bill-purchase"
  18. PRODUCE_COLLECTION = "bill-produce"
  19. PRODUCT_COLLECTION = "bill-product"
  20. CELL_BACKGROUND = "808080"
  21. )
  22. var needChangeCol = map[string]string{
  23. "部件": "E",
  24. "下单": "F",
  25. "纸张": "G",
  26. "印刷": "H",
  27. "覆膜": "I",
  28. "烫金": "J",
  29. "丝印": "K",
  30. "对裱": "L",
  31. "压纹": "M",
  32. "裱瓦": "N",
  33. "模切": "O",
  34. "粘盒": "P",
  35. "组装": "Q",
  36. "交货": "R",
  37. }
  38. func MatchString(targetStr string, regexPattern string) bool {
  39. // 编译正则表达式
  40. re := regexp.MustCompile(regexPattern)
  41. // 检查目标字符串是否包含匹配的子串
  42. return re.MatchString(targetStr)
  43. }
  44. type PlanStatusInfo struct {
  45. PlanName string `json:"planName"`
  46. PlanUnit string `json:"planUnit"`
  47. PlanCount int `json:"planCount"`
  48. PlanRowStart int `json:"planRowStart"`
  49. PlanRowEnd int `json:"planRowEnd"`
  50. PlanCompStatus []map[string]string `json:"planCompStatus"`
  51. }
  52. func UpdateExcel(client *mongo.Client, plans []*model.ProductPlan) {
  53. row := 5
  54. planStatusInfos := make([]*PlanStatusInfo, 0)
  55. planCompStatus := []map[string]string{}
  56. for _, plan := range plans {
  57. startRow := row
  58. for _, comp := range plan.Pack.Components {
  59. // ""代表该部件没有该工艺 "〇"代表正在进行的工艺
  60. // "808080"背景颜色代表部件所含未进行工艺 √代表已完成工序
  61. compStatus := map[string]string{
  62. "部件": " ",
  63. "行数": "5",
  64. // 部件中只要有一个订单就说明下单了
  65. "下单": " ",
  66. // 采购单中type为纸张 订单对应状态为完成
  67. "纸张": " ",
  68. // 遍历工艺单,工序中包含印刷
  69. "印刷": " ",
  70. // 遍历工艺单,工序中包含覆膜
  71. "覆膜": " ",
  72. // 遍历工艺单,工序中包含烫金
  73. "烫金": " ",
  74. // 遍历工艺单,工序中包含丝印
  75. "丝印": " ",
  76. // 遍历工艺单,工序中包含对裱
  77. "对裱": " ",
  78. // 遍历工艺单,工序中包含压纹
  79. "压纹": " ",
  80. // 遍历工艺单,工序中包含裱瓦
  81. "裱瓦": " ",
  82. // 遍历工艺单,工序中包含模切
  83. "模切": " ",
  84. // 遍历工艺单,工序中包含粘盒
  85. "粘盒": " ",
  86. // 遍历工艺单,工序中包含组装
  87. "组装": " ",
  88. "交货": " ",
  89. }
  90. fmt.Println(plan.Name)
  91. fmt.Println(comp.Name)
  92. fmt.Println(row)
  93. fmt.Println("------------------------------------")
  94. compStatus["部件"] = comp.Name
  95. compStatus["行数"] = fmt.Sprintf("%d", row)
  96. row++
  97. // 去重获取所有订单
  98. seen := make(map[string]bool)
  99. tbills := make([]string, 0, len(comp.Stages)) // 结果数组,容量初始化为原数组的长度
  100. // ???:1 最后一个工序没下单
  101. // isBill := true
  102. // lastStage := comp.Stages[len(comp.Stages)-1]
  103. // if len(lastStage.BillId) < 24 {
  104. // isBill = false
  105. // }
  106. for _, stage := range comp.Stages {
  107. if len(stage.BillId) > 0 {
  108. value := fmt.Sprintf("%d_%s", stage.BillType, stage.BillId)
  109. if _, ok := seen[value]; !ok {
  110. // 标记为已出现
  111. seen[value] = true
  112. // 添加到结果数组
  113. tbills = append(tbills, value)
  114. }
  115. } else {
  116. // 产品中填写了这个工序但是没有下单 黑色背景
  117. for k := range compStatus {
  118. if k == "下单" || k == "交货" || k == "部件" || k == "行数" {
  119. continue
  120. }
  121. // 纸张 只要是采购单就设置纸张状态
  122. if stage.Type == 1 {
  123. compStatus["纸张"] = CELL_BACKGROUND
  124. }
  125. // 匹配工艺关键字 匹配到就设置状态为黑背景
  126. if MatchString(stage.Name, k) {
  127. compStatus[k] = CELL_BACKGROUND
  128. }
  129. }
  130. }
  131. }
  132. // fmt.Println(tbills)
  133. // 如果tbills为空,说明该部件没有订单
  134. if len(tbills) == 0 {
  135. // 该部件没有订单,跳过
  136. planCompStatus = append(planCompStatus, compStatus)
  137. continue
  138. }
  139. // 查询数据库获取bill详细信息
  140. // 最后工序订单号
  141. lastBillType := tbills[len(tbills)-1]
  142. for _, billType := range tbills {
  143. compStatus["下单"] = "√"
  144. bt := strings.Split(billType, "_")[0]
  145. billId, _ := primitive.ObjectIDFromHex(strings.Split(billType, "_")[1])
  146. if bt == "1" {
  147. // 查询采购单
  148. bill := &model.PurchaseBill{}
  149. err := client.Database("box-cost").Collection(PURCHASE_COLLECTION).FindOne(context.Background(), bson.M{"_id": billId}).Decode(bill)
  150. if err != nil {
  151. log.Fatal(err)
  152. }
  153. // if bill.Type == "纸张" {
  154. // compStatus["纸张"] = "〇"
  155. // if bill.Status == "complete" {
  156. // compStatus["纸张"] = "√"
  157. // }
  158. // }
  159. compStatus["纸张"] = "〇"
  160. if bill.Status == "complete" {
  161. compStatus["纸张"] = "√"
  162. }
  163. // if !isBill {
  164. // continue
  165. // }
  166. if lastBillType == billType {
  167. for _, paper := range bill.Paper {
  168. compStatus["交货"] = fmt.Sprintf("%d", paper.ConfirmCount)
  169. }
  170. }
  171. }
  172. if bt == "2" {
  173. // 查询工艺单
  174. bill := &model.ProduceBill{}
  175. err := client.Database("box-cost").Collection(PRODUCE_COLLECTION).FindOne(context.Background(), bson.M{"_id": billId}).Decode(bill)
  176. if err != nil {
  177. log.Fatal(err)
  178. }
  179. for _, produce := range bill.Produces {
  180. for k := range compStatus {
  181. if k == "下单" || k == "纸张" || k == "交货" || k == "部件" || k == "行数" {
  182. continue
  183. }
  184. if MatchString(produce.Name, k) {
  185. compStatus[k] = "〇"
  186. if bill.Status == "complete" {
  187. compStatus[k] = "√"
  188. }
  189. }
  190. // 直接赋值,如果这个订单是最后一个,则状态覆盖
  191. // ???思考:如果最后一个工序没有生成订单的话,是按最后一个工序还是最后一个订单?这里是最后一个订单
  192. // ???:1
  193. // if !isBill {
  194. // continue
  195. // }
  196. compStatus["交货"] = fmt.Sprintf("%d", produce.ConfirmCount)
  197. }
  198. }
  199. }
  200. // 暂时没有状态标定
  201. if bt == "3" {
  202. // 查询成品单
  203. bill := &model.ProductBill{}
  204. err := client.Database("box-cost").Collection(PRODUCT_COLLECTION).FindOne(context.Background(), bson.M{"_id": billId}).Decode(bill)
  205. if err != nil {
  206. log.Fatal(err)
  207. }
  208. // fmt.Println(bill)
  209. // ?? 这里需不需要 影响正确数据吗?
  210. // if !isBill {
  211. // continue
  212. // }
  213. if lastBillType == billType {
  214. for _, product := range bill.Products {
  215. compStatus["交货"] = fmt.Sprintf("%d", product.ConfirmCount)
  216. }
  217. }
  218. }
  219. }
  220. planCompStatus = append(planCompStatus, compStatus)
  221. }
  222. // fmt.Println(plan.Name)
  223. // fmt.Println("rowstart:", startRow)
  224. // fmt.Println("rowend:", row-1)
  225. planStatusInfos = append(planStatusInfos, &PlanStatusInfo{
  226. PlanName: plan.Name,
  227. // !这个规格好像没有
  228. PlanUnit: "",
  229. PlanCount: plan.Total,
  230. PlanRowStart: startRow,
  231. PlanRowEnd: row - 1,
  232. PlanCompStatus: planCompStatus,
  233. })
  234. }
  235. // fmt.Printf("%#v\n", planStatusInfos[len(planStatusInfos)-1])
  236. // UpdateCell(planCompStatus)
  237. UpdateCell1(planStatusInfos)
  238. }
  239. func UpdateCell(planCompStatus []map[string]string) {
  240. tmpfile, err := excelize.OpenFile(EXCEL_TMPLATE_FILE)
  241. if err != nil {
  242. log.Fatal(err)
  243. }
  244. excelIndex := tmpfile.GetActiveSheetIndex()
  245. sheetName := tmpfile.GetSheetName(excelIndex)
  246. // 定义样式
  247. style, err := tmpfile.NewStyle(&excelize.Style{
  248. Border: []excelize.Border{
  249. {
  250. Type: "left",
  251. Color: "FF000000",
  252. Style: 1,
  253. },
  254. {
  255. Type: "right",
  256. Color: "FF000000",
  257. Style: 1,
  258. },
  259. {
  260. Type: "top",
  261. Color: "FF000000",
  262. Style: 1,
  263. },
  264. {
  265. Type: "bottom",
  266. Color: "FF000000",
  267. Style: 1,
  268. },
  269. },
  270. Fill: excelize.Fill{
  271. Type: "pattern",
  272. Pattern: 1, // 1 表示实心填充
  273. Color: []string{CELL_BACKGROUND},
  274. },
  275. })
  276. if err != nil {
  277. log.Fatal(err)
  278. }
  279. for _, compStatus := range planCompStatus {
  280. // fmt.Println("-----------------------------------------------")
  281. // fmt.Println(compStatus)
  282. row := compStatus["行数"]
  283. for colk, col := range needChangeCol {
  284. for csk, csv := range compStatus {
  285. if colk == csk {
  286. cell := fmt.Sprintf("%s%s", col, row)
  287. // fmt.Println(cell)
  288. // 设置背景
  289. if csv == CELL_BACKGROUND {
  290. // 设置单元格的值
  291. err = tmpfile.SetCellStyle(sheetName, cell, cell, style)
  292. if err != nil {
  293. log.Fatal(err)
  294. }
  295. } else {
  296. // 设置字符
  297. err = tmpfile.SetCellValue(sheetName, cell, csv)
  298. if err != nil {
  299. log.Fatal(err)
  300. }
  301. }
  302. }
  303. }
  304. }
  305. }
  306. // 获取当前日期
  307. date := time.Now().Format("2006年01月02日_150405")
  308. fileName := fmt.Sprintf("礼盒加工追踪表_%s.xlsx", date)
  309. // 以新的文件名保存文件
  310. err = tmpfile.SaveAs(fileName)
  311. if err != nil {
  312. log.Fatal(err)
  313. }
  314. }
  315. func UpdateCell1(planStatusInfos []*PlanStatusInfo) {
  316. tmpfile, err := excelize.OpenFile(EXCEL_TMPLATE_FILE)
  317. if err != nil {
  318. log.Fatal(err)
  319. }
  320. excelIndex := tmpfile.GetActiveSheetIndex()
  321. sheetName := tmpfile.GetSheetName(excelIndex)
  322. // 定义样式
  323. style, err := tmpfile.NewStyle(&excelize.Style{
  324. Border: []excelize.Border{
  325. {
  326. Type: "left",
  327. Color: "FF000000",
  328. Style: 1,
  329. },
  330. {
  331. Type: "right",
  332. Color: "FF000000",
  333. Style: 1,
  334. },
  335. {
  336. Type: "top",
  337. Color: "FF000000",
  338. Style: 1,
  339. },
  340. {
  341. Type: "bottom",
  342. Color: "FF000000",
  343. Style: 1,
  344. },
  345. },
  346. Fill: excelize.Fill{
  347. Type: "pattern",
  348. Pattern: 1, // 1 表示实心填充
  349. Color: []string{CELL_BACKGROUND},
  350. },
  351. })
  352. if err != nil {
  353. log.Fatal(err)
  354. }
  355. for _, planStatusInfo := range planStatusInfos {
  356. // 填充品名/箱规/数量
  357. // 品名
  358. err = tmpfile.SetCellValue(sheetName, fmt.Sprintf("%s%d", "B", planStatusInfo.PlanRowStart), planStatusInfo.PlanName)
  359. if err != nil {
  360. log.Fatal(err)
  361. }
  362. // 数量
  363. err = tmpfile.SetCellValue(sheetName, fmt.Sprintf("%s%d", "D", planStatusInfo.PlanRowStart), planStatusInfo.PlanCount)
  364. if err != nil {
  365. log.Fatal(err)
  366. }
  367. planCompStatus := planStatusInfo.PlanCompStatus
  368. for _, compStatus := range planCompStatus {
  369. row := compStatus["行数"]
  370. for colk, col := range needChangeCol {
  371. for csk, csv := range compStatus {
  372. if colk == csk {
  373. cell := fmt.Sprintf("%s%s", col, row)
  374. // fmt.Println(cell)
  375. // 设置背景
  376. if csv == CELL_BACKGROUND {
  377. // 设置单元格的值
  378. err = tmpfile.SetCellStyle(sheetName, cell, cell, style)
  379. if err != nil {
  380. log.Fatal(err)
  381. }
  382. } else {
  383. // 设置字符
  384. err = tmpfile.SetCellValue(sheetName, cell, csv)
  385. if err != nil {
  386. log.Fatal(err)
  387. }
  388. }
  389. }
  390. }
  391. }
  392. }
  393. }
  394. // 获取当前日期
  395. date := time.Now().Format("2006年01月02日_150405")
  396. fileName := fmt.Sprintf("礼盒加工追踪表_%s.xlsx", date)
  397. // 以新的文件名保存文件
  398. err = tmpfile.SaveAs(fileName)
  399. if err != nil {
  400. log.Fatal(err)
  401. }
  402. }