excel.go 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302
  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. fmt.Println(tbills)
  113. // 如果tbills为空,说明该部件没有订单
  114. if len(tbills) == 0 {
  115. // 该部件没有订单,跳过
  116. planCompStatus = append(planCompStatus, compStatus)
  117. continue
  118. }
  119. // 查询数据库获取bill详细信息
  120. // 最后工序订单号
  121. lastBillType := tbills[len(tbills)-1]
  122. for _, billType := range tbills {
  123. compStatus["下单"] = "√"
  124. bt := strings.Split(billType, "_")[0]
  125. billId, _ := primitive.ObjectIDFromHex(strings.Split(billType, "_")[1])
  126. if bt == "1" {
  127. // 查询采购单
  128. bill := &model.PurchaseBill{}
  129. err := client.Database("box-cost").Collection(PURCHASE_COLLECTION).FindOne(context.Background(), bson.M{"_id": billId}).Decode(bill)
  130. if err != nil {
  131. log.Fatal(err)
  132. }
  133. // if bill.Type == "纸张" {
  134. // compStatus["纸张"] = "〇"
  135. // if bill.Status == "complete" {
  136. // compStatus["纸张"] = "√"
  137. // }
  138. // }
  139. compStatus["纸张"] = "〇"
  140. if bill.Status == "complete" {
  141. compStatus["纸张"] = "√"
  142. }
  143. if lastBillType == billType {
  144. for _, paper := range bill.Paper {
  145. compStatus["交货"] = fmt.Sprintf("%d", paper.ConfirmCount)
  146. }
  147. }
  148. }
  149. if bt == "2" {
  150. // 查询工艺单
  151. bill := &model.ProduceBill{}
  152. err := client.Database("box-cost").Collection(PRODUCE_COLLECTION).FindOne(context.Background(), bson.M{"_id": billId}).Decode(bill)
  153. if err != nil {
  154. log.Fatal(err)
  155. }
  156. for _, produce := range bill.Produces {
  157. for k := range compStatus {
  158. if k == "下单" || k == "纸张" || k == "交货" || k == "部件" || k == "行数" {
  159. continue
  160. }
  161. if MatchString(produce.Name, k) {
  162. compStatus[k] = "〇"
  163. if bill.Status == "complete" {
  164. compStatus[k] = "√"
  165. }
  166. }
  167. compStatus["交货"] = fmt.Sprintf("%d", produce.ConfirmCount)
  168. }
  169. }
  170. }
  171. // 暂时没有状态标定
  172. if bt == "3" {
  173. // 查询成品单
  174. bill := &model.ProductBill{}
  175. err := client.Database("box-cost").Collection(PRODUCT_COLLECTION).FindOne(context.Background(), bson.M{"_id": billId}).Decode(bill)
  176. if err != nil {
  177. log.Fatal(err)
  178. }
  179. // fmt.Println(bill)
  180. if lastBillType == billType {
  181. for _, product := range bill.Products {
  182. compStatus["交货"] = fmt.Sprintf("%d", product.ConfirmCount)
  183. }
  184. }
  185. }
  186. }
  187. planCompStatus = append(planCompStatus, compStatus)
  188. }
  189. }
  190. // fmt.Printf("%#v\n", planCompStatus)
  191. UpdateCell(planCompStatus)
  192. }
  193. func UpdateCell(planCompStatus []map[string]string) {
  194. tmpfile, err := excelize.OpenFile(EXCEL_TMPLATE_FILE)
  195. if err != nil {
  196. log.Fatal(err)
  197. }
  198. excelIndex := tmpfile.GetActiveSheetIndex()
  199. sheetName := tmpfile.GetSheetName(excelIndex)
  200. // 定义样式
  201. style, err := tmpfile.NewStyle(&excelize.Style{
  202. Border: []excelize.Border{
  203. {
  204. Type: "left",
  205. Color: "FF000000",
  206. Style: 1,
  207. },
  208. {
  209. Type: "right",
  210. Color: "FF000000",
  211. Style: 1,
  212. },
  213. {
  214. Type: "top",
  215. Color: "FF000000",
  216. Style: 1,
  217. },
  218. {
  219. Type: "bottom",
  220. Color: "FF000000",
  221. Style: 1,
  222. },
  223. },
  224. Fill: excelize.Fill{
  225. Type: "pattern",
  226. Pattern: 1, // 1 表示实心填充
  227. Color: []string{CELL_BACKGROUND},
  228. },
  229. })
  230. if err != nil {
  231. log.Fatal(err)
  232. }
  233. for _, compStatus := range planCompStatus {
  234. // fmt.Println("-----------------------------------------------")
  235. // fmt.Println(compStatus)
  236. row := compStatus["行数"]
  237. for colk, col := range needChangeCol {
  238. for csk, csv := range compStatus {
  239. if colk == csk {
  240. cell := fmt.Sprintf("%s%s", col, row)
  241. // fmt.Println(cell)
  242. // 设置背景
  243. if csv == CELL_BACKGROUND {
  244. // 设置单元格的值
  245. err = tmpfile.SetCellStyle(sheetName, cell, cell, style)
  246. if err != nil {
  247. log.Fatal(err)
  248. }
  249. } else {
  250. // 设置字符
  251. err = tmpfile.SetCellValue(sheetName, cell, csv)
  252. if err != nil {
  253. log.Fatal(err)
  254. }
  255. }
  256. }
  257. }
  258. }
  259. }
  260. // 获取当前日期
  261. date := time.Now().Format("2006年01月02日_150405")
  262. fileName := fmt.Sprintf("礼盒加工追踪表_%s.xlsx", date)
  263. // 以新的文件名保存文件
  264. err = tmpfile.SaveAs(fileName)
  265. if err != nil {
  266. log.Fatal(err)
  267. }
  268. }