// Package fortune 每日运势 package fortune import ( "archive/zip" "crypto/md5" "encoding/hex" "encoding/json" "image" "io" "math/rand" "os" "strconv" "sync" "time" "github.com/fogleman/gg" "github.com/sirupsen/logrus" zero "github.com/wdvxdr1123/ZeroBot" "github.com/wdvxdr1123/ZeroBot/message" "github.com/wdvxdr1123/ZeroBot/utils/helper" control "github.com/FloatTech/zbputils/control" "github.com/FloatTech/zbputils/file" "github.com/FloatTech/zbputils/math" "github.com/FloatTech/zbputils/process" "github.com/FloatTech/zbputils/txt2img" ) var ( // 底图缓存位置 images = "data/Fortune/" // 基础文件位置 omikujson = "data/Fortune/text.json" // 字体文件位置 font = "data/Font/sakura.ttf" // 生成图缓存位置 cache = images + "cache/" // 底图类型列表:车万 DC4 爱因斯坦 星空列车 樱云之恋 富婆妹 李清歌 // 公主连结 原神 明日方舟 碧蓝航线 碧蓝幻想 战双 阴阳师 table = [...]string{"车万", "DC4", "爱因斯坦", "星空列车", "樱云之恋", "富婆妹", "李清歌", "公主连结", "原神", "明日方舟", "碧蓝航线", "碧蓝幻想", "战双", "阴阳师"} // 映射底图与 index index = make(map[string]uint8) // 下载锁 dlmu sync.Mutex // 签文 omikujis []map[string]string ) func init() { for i, s := range table { index[s] = uint8(i) } err := os.MkdirAll(images, 0755) if err != nil { panic(err) } err = os.MkdirAll(cache, 0755) if err != nil { panic(err) } process.SleepAbout1sTo2s() data, err := file.GetLazyData(omikujson, true, false) if err != nil { panic(err) } err = json.Unmarshal(data, &omikujis) if err != nil { panic(err) } _, err = file.GetLazyData(font, false, true) if err != nil { panic(err) } // 插件主体 en := control.Register("fortune", &control.Options{ DisableOnDefault: false, Help: "每日运势: \n" + "- 运势|抽签\n" + "- 设置底图[车万 DC4 爱因斯坦 星空列车 樱云之恋 富婆妹 李清歌 公主连结 原神 明日方舟 碧蓝航线 碧蓝幻想 战双 阴阳师]", }) en.OnRegex(`^设置底图(.*)`).SetBlock(true).SecondPriority(). Handle(func(ctx *zero.Ctx) { gid := ctx.Event.GroupID if gid <= 0 { // 个人用户设为负数 gid = -ctx.Event.UserID } i, ok := index[ctx.State["regex_matched"].([]string)[1]] if ok { c, ok := control.Lookup("fortune") if ok { err = c.SetData(gid, int64(i)&0xff) if err != nil { ctx.SendChain(message.Text("设置失败:", err)) return } ctx.SendChain(message.Text("设置成功~")) return } ctx.SendChain(message.Text("设置失败: 找不到插件")) return } ctx.SendChain(message.Text("没有这个底图哦~")) }) en.OnFullMatchGroup([]string{"运势", "抽签"}).SetBlock(true).SecondPriority(). Handle(func(ctx *zero.Ctx) { // 获取该群背景类型,默认车万 kind := "车万" gid := ctx.Event.GroupID if gid <= 0 { // 个人用户设为负数 gid = -ctx.Event.UserID } logrus.Debugln("[fortune]gid:", ctx.Event.GroupID, "uid:", ctx.Event.UserID) c, ok := control.Lookup("fortune") if ok { v := uint8(c.GetData(gid) & 0xff) if int(v) < len(table) { kind = table[v] } } // 检查背景图片是否存在 zipfile := images + kind + ".zip" _, err = file.GetLazyData(zipfile, false, false) if err != nil { ctx.SendChain(message.Text("ERROR: ", err)) return } // 生成种子 t, _ := strconv.ParseInt(time.Now().Format("20060102"), 10, 64) seed := ctx.Event.UserID + t // 随机获取背景 background, index, err := randimage(zipfile, seed) if err != nil { ctx.SendChain(message.Text("ERROR: ", err)) return } // 随机获取签文 title, text, err := randtext(seed) if err != nil { ctx.SendChain(message.Text("ERROR: ", err)) return } digest := md5.Sum(helper.StringToBytes(zipfile + strconv.Itoa(index) + title + text)) cachefile := cache + hex.EncodeToString(digest[:]) if file.IsExist(cachefile) { ctx.SendChain(message.Image("file:///" + file.BOTPATH + "/" + cachefile)) return } dlmu.Lock() f, err := os.Create(cachefile) if err != nil { ctx.SendChain(message.Text("ERROR: ", err)) _ = f.Close() _ = os.Remove(cachefile) dlmu.Unlock() return } // 绘制背景 err = draw(background, title, text, f) _ = f.Close() dlmu.Unlock() if err != nil { ctx.SendChain(message.Text("ERROR: ", err)) return } // 发送图片 ctx.SendChain(message.Image("file:///" + file.BOTPATH + "/" + cachefile)) }) } // @function randimage 随机选取zip内的文件 // @param path zip路径 // @param seed 随机数种子 // @return 文件路径 & 错误信息 func randimage(path string, seed int64) (im image.Image, index int, err error) { reader, err := zip.OpenReader(path) if err != nil { return } defer reader.Close() rand.Seed(seed) index = rand.Intn(len(reader.File)) file := reader.File[index] f, err := file.Open() if err != nil { return } defer f.Close() im, _, err = image.Decode(f) return } // @function randtext 随机选取签文 // @param file 文件路径 // @param seed 随机数种子 // @return 签名 & 签文 & 错误信息 func randtext(seed int64) (string, string, error) { rand.Seed(seed) r := rand.Intn(len(omikujis)) return omikujis[r]["title"], omikujis[r]["content"], nil } // @function draw 绘制运势图 // @param background 背景图片路径 // @param seed 随机数种子 // @param title 签名 // @param text 签文 // @return 错误信息 func draw(back image.Image, title, text string, dst io.Writer) error { var txtc txt2img.TxtCanvas txtc.Canvas = gg.NewContext(back.Bounds().Size().Y, back.Bounds().Size().X) txtc.Canvas.DrawImage(back, 0, 0) // 写标题 txtc.Canvas.SetRGB(1, 1, 1) if err := txtc.Canvas.LoadFontFace(font, 45); err != nil { return err } sw, _ := txtc.Canvas.MeasureString(title) txtc.Canvas.DrawString(title, 140-sw/2, 112) // 写正文 txtc.Canvas.SetRGB(0, 0, 0) if err := txtc.Canvas.LoadFontFace(font, 23); err != nil { return err } tw, th := txtc.Canvas.MeasureString("测") tw, th = tw+10, th+10 r := []rune(text) xsum := rowsnum(len(r), 9) switch xsum { default: for i, o := range r { xnow := rowsnum(i+1, 9) ysum := math.Min(len(r)-(xnow-1)*9, 9) ynow := i%9 + 1 txtc.Canvas.DrawString(string(o), -offest(xsum, xnow, tw)+115, offest(ysum, ynow, th)+320.0) } case 2: div := rowsnum(len(r), 2) for i, o := range r { xnow := rowsnum(i+1, div) ysum := math.Min(len(r)-(xnow-1)*div, div) ynow := i%div + 1 switch xnow { case 1: txtc.Canvas.DrawString(string(o), -offest(xsum, xnow, tw)+115, offest(9, ynow, th)+320.0) case 2: txtc.Canvas.DrawString(string(o), -offest(xsum, xnow, tw)+115, offest(9, ynow+(9-ysum), th)+320.0) } } } _, err := txtc.WriteTo(dst) return err } func offest(total, now int, distance float64) float64 { if total%2 == 0 { return (float64(now-total/2) - 1) * distance } return (float64(now-total/2) - 1.5) * distance } func rowsnum(total, div int) int { temp := total / div if total%div != 0 { temp++ } return temp }