excel.go 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303
  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. func UpdateExcel(client *mongo.Client, plans []*model.ProductPlan) {
  45. row := 5
  46. planCompStatus := []map[string]string{}
  47. for _, plan := range plans {
  48. for _, comp := range plan.Pack.Components {
  49. // ""代表该部件没有该工艺 "〇"代表正在进行的工艺
  50. // "808080"背景颜色代表部件所含未进行工艺 √代表已完成工序
  51. compStatus := map[string]string{
  52. "部件": " ",
  53. "行数": "5",
  54. // 部件中只要有一个订单就说明下单了
  55. "下单": " ",
  56. // 采购单中type为纸张 订单对应状态为完成
  57. "纸张": " ",
  58. // 遍历工艺单,工序中包含印刷
  59. "印刷": " ",
  60. // 遍历工艺单,工序中包含覆膜
  61. "覆膜": " ",
  62. // 遍历工艺单,工序中包含烫金
  63. "烫金": " ",
  64. // 遍历工艺单,工序中包含丝印
  65. "丝印": " ",
  66. // 遍历工艺单,工序中包含对裱
  67. "对裱": " ",
  68. // 遍历工艺单,工序中包含压纹
  69. "压纹": " ",
  70. // 遍历工艺单,工序中包含裱瓦
  71. "裱瓦": " ",
  72. // 遍历工艺单,工序中包含模切
  73. "模切": " ",
  74. // 遍历工艺单,工序中包含粘盒
  75. "粘盒": " ",
  76. // 遍历工艺单,工序中包含组装
  77. "组装": " ",
  78. "交货": " ",
  79. }
  80. compStatus["部件"] = comp.Name
  81. compStatus["行数"] = fmt.Sprintf("%d", row)
  82. row++
  83. // 去重获取所有订单
  84. seen := make(map[string]bool)
  85. tbills := make([]string, 0, len(comp.Stages)) // 结果数组,容量初始化为原数组的长度
  86. for _, stage := range comp.Stages {
  87. if len(stage.BillId) > 0 {
  88. value := fmt.Sprintf("%d_%s", stage.BillType, stage.BillId)
  89. if _, ok := seen[value]; !ok {
  90. // 标记为已出现
  91. seen[value] = true
  92. // 添加到结果数组
  93. tbills = append(tbills, value)
  94. }
  95. } else {
  96. // 产品中填写了这个工序但是没有下单 黑色背景
  97. for k := range compStatus {
  98. if k == "下单" || k == "交货" || k == "部件" || k == "行数" {
  99. continue
  100. }
  101. // 纸张 只要是采购单就设置纸张状态
  102. if stage.Type == 1 {
  103. compStatus["纸张"] = CELL_BACKGROUND
  104. }
  105. // 匹配工艺关键字 匹配到就设置状态为黑背景
  106. if MatchString(stage.Name, k) {
  107. compStatus[k] = CELL_BACKGROUND
  108. }
  109. }
  110. }
  111. }
  112. // 如果tbills为空,说明该部件没有订单
  113. if len(tbills) == 0 {
  114. // 该部件没有订单,跳过
  115. planCompStatus = append(planCompStatus, compStatus)
  116. continue
  117. }
  118. // 查询数据库获取bill详细信息
  119. // 最后工序订单号
  120. lastBillType := tbills[len(tbills)-1]
  121. for _, billType := range tbills {
  122. compStatus["下单"] = "√"
  123. bt := strings.Split(billType, "_")[0]
  124. billId, _ := primitive.ObjectIDFromHex(strings.Split(billType, "_")[1])
  125. if bt == "1" {
  126. // 查询采购单
  127. bill := &model.PurchaseBill{}
  128. err := client.Database("box-cost").Collection(PURCHASE_COLLECTION).FindOne(context.Background(), bson.M{"_id": billId}).Decode(bill)
  129. if err != nil {
  130. log.Fatal(err)
  131. }
  132. // if bill.Type == "纸张" {
  133. // compStatus["纸张"] = "〇"
  134. // if bill.Status == "complete" {
  135. // compStatus["纸张"] = "√"
  136. // }
  137. // }
  138. compStatus["纸张"] = "〇"
  139. if bill.Status == "complete" {
  140. compStatus["纸张"] = "√"
  141. }
  142. if lastBillType == billType {
  143. for _, paper := range bill.Paper {
  144. compStatus["交货"] = fmt.Sprintf("%d", paper.ConfirmCount)
  145. }
  146. }
  147. }
  148. if bt == "2" {
  149. // 查询工艺单
  150. bill := &model.ProduceBill{}
  151. err := client.Database("box-cost").Collection(PRODUCE_COLLECTION).FindOne(context.Background(), bson.M{"_id": billId}).Decode(bill)
  152. if err != nil {
  153. log.Fatal(err)
  154. }
  155. for _, produce := range bill.Produces {
  156. for k := range compStatus {
  157. if k == "下单" || k == "纸张" || k == "交货" || k == "部件" || k == "行数" {
  158. continue
  159. }
  160. if MatchString(produce.Name, k) {
  161. compStatus[k] = "〇"
  162. if bill.Status == "complete" {
  163. compStatus[k] = "√"
  164. }
  165. }
  166. compStatus["交货"] = fmt.Sprintf("%d", produce.ConfirmCount)
  167. }
  168. }
  169. }
  170. // 暂时没有状态标定
  171. if bt == "3" {
  172. // 查询成品单
  173. bill := &model.ProductBill{}
  174. err := client.Database("box-cost").Collection(PRODUCT_COLLECTION).FindOne(context.Background(), bson.M{"_id": billId}).Decode(bill)
  175. if err != nil {
  176. log.Fatal(err)
  177. }
  178. // fmt.Println(bill)
  179. if lastBillType == billType {
  180. for _, product := range bill.Products {
  181. compStatus["交货"] = fmt.Sprintf("%d", product.ConfirmCount)
  182. }
  183. }
  184. }
  185. }
  186. planCompStatus = append(planCompStatus, compStatus)
  187. }
  188. }
  189. // fmt.Printf("%#v\n", planCompStatus)
  190. UpdateCell(planCompStatus)
  191. }
  192. func UpdateCell(planCompStatus []map[string]string) {
  193. tmpfile, err := excelize.OpenFile(EXCEL_TMPLATE_FILE)
  194. if err != nil {
  195. log.Fatal(err)
  196. }
  197. excelIndex := tmpfile.GetActiveSheetIndex()
  198. sheetName := tmpfile.GetSheetName(excelIndex)
  199. for _, compStatus := range planCompStatus {
  200. // fmt.Println("-----------------------------------------------")
  201. // fmt.Println(compStatus)
  202. row := compStatus["行数"]
  203. for colk, col := range needChangeCol {
  204. for csk, csv := range compStatus {
  205. if colk == csk {
  206. cell := fmt.Sprintf("%s%s", col, row)
  207. // fmt.Println(cell)
  208. // 设置背景
  209. if csv == CELL_BACKGROUND {
  210. // 设置单元格的值
  211. // 定义样式
  212. style, err := tmpfile.NewStyle(&excelize.Style{
  213. Border: []excelize.Border{
  214. {
  215. Type: "left",
  216. Color: "FF000000",
  217. Style: 1,
  218. },
  219. {
  220. Type: "right",
  221. Color: "FF000000",
  222. Style: 1,
  223. },
  224. {
  225. Type: "top",
  226. Color: "FF000000",
  227. Style: 1,
  228. },
  229. {
  230. Type: "bottom",
  231. Color: "FF000000",
  232. Style: 1,
  233. },
  234. },
  235. Fill: excelize.Fill{
  236. Type: "pattern",
  237. Pattern: 1, // 1 表示实心填充
  238. Color: []string{CELL_BACKGROUND},
  239. },
  240. })
  241. if err != nil {
  242. log.Fatal(err)
  243. }
  244. err = tmpfile.SetCellStyle(sheetName, cell, cell, style)
  245. if err != nil {
  246. log.Fatal(err)
  247. }
  248. } else {
  249. // 设置字符
  250. err = tmpfile.SetCellValue(sheetName, cell, csv)
  251. if err != nil {
  252. log.Fatal(err)
  253. }
  254. }
  255. }
  256. }
  257. }
  258. }
  259. // 获取当前日期
  260. date := time.Now().Format("2006年01月02日_150405")
  261. fileName := fmt.Sprintf("礼盒加工追踪表_%s.xlsx", date)
  262. // 以新的文件名保存文件
  263. err = tmpfile.SaveAs(fileName)
  264. if err != nil {
  265. log.Fatal(err)
  266. }
  267. }