package route import ( "fmt" "github.com/gin-gonic/gin" "github/fthvgb1/wp-go/helper" "github/fthvgb1/wp-go/models" "math" "net/http" "regexp" "strconv" "strings" "sync" ) var PostsCache sync.Map type IndexHandle struct { c *gin.Context page int pageSize int title string titleL string titleR string search string totalPage int category string categoryType string where models.SqlBuilder orderBy models.SqlBuilder order string join models.SqlBuilder postType []interface{} status []interface{} header string } func NewIndexHandle(ctx *gin.Context) *IndexHandle { return &IndexHandle{ c: ctx, page: 1, pageSize: 10, titleL: models.Options["blogname"], titleR: models.Options["blogdescription"], where: models.SqlBuilder{ {"post_type", "in", ""}, {"post_status", "in", ""}, }, orderBy: models.SqlBuilder{}, join: models.SqlBuilder{}, postType: []interface{}{"post"}, status: []interface{}{"publish"}, } } func (h *IndexHandle) setTitleLR(l, r string) { h.titleL = l h.titleR = r } func (h *IndexHandle) getTitle() string { h.title = fmt.Sprintf("%s-%s", h.titleL, h.titleR) return h.title } func (h *IndexHandle) parseParams() { h.order = h.c.Query("order") if !helper.IsContainInArr(h.order, []string{"asc", "desc"}) { h.order = "asc" } year := h.c.Param("year") if year != "" { h.where = append(h.where, []string{ "year(post_date)", year, }) } month := h.c.Param("month") if month != "" { h.where = append(h.where, []string{ "month(post_date)", month, }) ss := fmt.Sprintf("%s年%s月", year, month) h.header = fmt.Sprintf("月度归档: %s", ss) h.setTitleLR(ss, models.Options["blogname"]) } category := h.c.Param("category") if category == "" { category = h.c.Param("tag") if category != "" { h.categoryType = "post_tag" h.header = fmt.Sprintf("标签: %s", category) } } else { h.categoryType = "category" h.header = fmt.Sprintf("分类: %s", category) } h.category = category if category != "" { h.where = append(h.where, []string{ "d.name", category, }, []string{"taxonomy", h.categoryType}) h.join = append(h.join, []string{ "a", "left join", "wp_term_relationships b", "a.Id=b.object_id", }, []string{ "left join", "wp_term_taxonomy c", "b.term_taxonomy_id=c.term_taxonomy_id", }, []string{ "left join", "wp_terms d", "c.term_id=d.term_id", }) h.setTitleLR(category, models.Options["blogname"]) } s := h.c.Query("s") if s != "" && strings.Replace(s, " ", "", -1) != "" { q := helper.StrJoin("%", s, "%") h.where = append(h.where, []string{ "and", "post_title", "like", q, "", "or", "post_content", "like", q, "", "or", "post_excerpt", "like", q, "", }, []string{"post_password", ""}) h.postType = append(h.postType, "page", "attachment") h.header = fmt.Sprintf("%s的搜索结果", s) h.setTitleLR(helper.StrJoin(`"`, s, `"`, "的搜索结果"), models.Options["blogname"]) h.search = s } else { h.status = append(h.status, "private") } p := h.c.Query("paged") if p == "" { p = h.c.Param("page") } if p != "" { if pa, err := strconv.Atoi(p); err == nil { h.page = pa } } if h.page > 1 && (h.category != "" || h.search != "" || month != "") { h.setTitleLR(fmt.Sprintf("%s-第%d页", h.titleL, h.page), models.Options["blogname"]) } } func (h *IndexHandle) getTotalPage(totalRaws int) int { h.totalPage = int(math.Ceil(float64(totalRaws) / float64(h.pageSize))) return h.totalPage } func (h *IndexHandle) queryAndSetPostCache(postIds []models.WpPosts) (err error) { var all []uint64 var needQuery []interface{} for _, wpPosts := range postIds { all = append(all, wpPosts.Id) if _, ok := PostsCache.Load(wpPosts.Id); !ok { needQuery = append(needQuery, wpPosts.Id) } } if len(needQuery) > 0 { rawPosts, er := models.Find[models.WpPosts](models.SqlBuilder{{ "Id", "in", "", }}, "a.*,ifnull(d.name,'') category_name,ifnull(taxonomy,'') `taxonomy`", "", nil, models.SqlBuilder{{ "a", "left join", "wp_term_relationships b", "a.Id=b.object_id", }, { "left join", "wp_term_taxonomy c", "b.term_taxonomy_id=c.term_taxonomy_id", }, { "left join", "wp_terms d", "c.term_id=d.term_id", }}, 0, needQuery) if er != nil { err = er return } postsMap := make(map[uint64]*models.WpPosts) for i, post := range rawPosts { v, ok := postsMap[post.Id] if !ok { v = &rawPosts[i] } if post.Taxonomy == "category" { v.Categories = append(v.Categories, post.CategoryName) } else if post.Taxonomy == "post_tag" { v.Tags = append(v.Tags, post.CategoryName) } postsMap[post.Id] = v } for _, pp := range postsMap { if len(pp.Categories) > 0 { t := make([]string, 0, len(pp.Categories)) for _, cat := range pp.Categories { t = append(t, fmt.Sprintf(`%s`, cat, cat)) } pp.CategoriesHtml = strings.Join(t, "、") } if len(pp.Tags) > 0 { t := make([]string, 0, len(pp.Tags)) for _, cat := range pp.Tags { t = append(t, fmt.Sprintf(``, cat, cat)) } pp.TagsHtml = strings.Join(t, "、") } PostsCache.Store(pp.Id, pp) } } return } func index(c *gin.Context) { h := NewIndexHandle(c) h.parseParams() postIds, totalRaw, err := models.SimplePagination[models.WpPosts](h.where, "ID", "", h.page, h.pageSize, h.orderBy, h.join, h.postType, h.status) defer func() { if err != nil { c.Error(err) } }() if err != nil { return } if len(postIds) < 1 && h.category != "" { h.titleL = "未找到页面" } err = h.queryAndSetPostCache(postIds) for i, v := range postIds { post, _ := PostsCache.Load(v.Id) pp := post.(*models.WpPosts) postIds[i] = *pp } recent, err := recentPosts() archive, err := archives() categoryItems, err := categories() q := c.Request.URL.Query().Encode() if q != "" { q = fmt.Sprintf("?%s", q) } c.HTML(http.StatusOK, "index.html", gin.H{ "posts": postIds, "options": models.Options, "recentPosts": recent, "archives": archive, "categories": categoryItems, "totalPage": h.getTotalPage(totalRaw), "queryRaw": q, "pagination": pagination(h.page, h.totalPage, 1, c.Request.URL.Path, q), "search": h.search, "header": h.header, "title": h.getTitle(), }) } func recentPosts() (r []models.WpPosts, err error) { r, err = models.Find[models.WpPosts](models.SqlBuilder{{ "post_type", "post", }, {"post_status", "publish"}}, "ID,post_title", "", models.SqlBuilder{{"post_date", "desc"}}, nil, 5) return } func categories() (terms []models.WpTermsMy, err error) { var in = []interface{}{"category"} terms, err = models.Find[models.WpTermsMy](models.SqlBuilder{ {"tt.count", ">", "0", "int"}, {"tt.taxonomy", "in", ""}, }, "t.term_id", "", models.SqlBuilder{ {"t.name", "asc"}, }, models.SqlBuilder{ {"t", "inner join", "wp_term_taxonomy tt", "t.term_id = tt.term_id"}, }, 0, in) for i := 0; i < len(terms); i++ { if v, ok := models.Terms[terms[i].WpTerms.TermId]; ok { terms[i].WpTerms = v } if v, ok := models.TermTaxonomy[terms[i].WpTerms.TermId]; ok { terms[i].WpTermTaxonomy = v } } return } func archives() (r []models.PostArchive, err error) { r, err = models.Find[models.PostArchive](models.SqlBuilder{ {"post_type", "post"}, {"post_status", "publish"}, }, "YEAR(post_date) AS `year`, MONTH(post_date) AS `month`, count(ID) as posts", "year,month", models.SqlBuilder{{"year", "desc"}, {"month", "desc"}}, nil, 0) return } func pagination(currentPage, totalPage, step int, path, query string) (html string) { if totalPage < 2 { return } pathx := path if !strings.Contains(path, "/page/") { pathx = fmt.Sprintf("%s%s", path, "/page/1") } s := strings.Builder{} if currentPage > totalPage { currentPage = totalPage } r := regexp.MustCompile(`(/page)/(\d+)`) start := currentPage - step end := currentPage + step if start < 1 { start = 1 } if currentPage > 1 { pp := "" if currentPage >= 2 { pp = replacePage(r, pathx, currentPage-1) } s.WriteString(fmt.Sprintf(``, pp, query)) } if currentPage >= step+2 { d := "" if currentPage > step+2 { d = `` } e := replacePage(r, path, 1) s.WriteString(fmt.Sprintf(` 1 %s `, e, query, d)) } if totalPage < end { end = totalPage } for page := start; page <= end; page++ { h := "" if currentPage == page { h = fmt.Sprintf(` %d `, page) } else { d := replacePage(r, pathx, page) h = fmt.Sprintf(` %d `, d, query, page) } s.WriteString(h) } if totalPage >= currentPage+step+1 { if totalPage > currentPage+step+1 { s.WriteString(``) } dd := replacePage(r, pathx, totalPage) s.WriteString(fmt.Sprintf(` %d`, dd, query, totalPage)) } if currentPage < totalPage { dd := replacePage(r, pathx, currentPage+1) s.WriteString(fmt.Sprintf(``, dd, query)) } html = s.String() return } func replacePage(r *regexp.Regexp, path string, page int) (src string) { if page == 1 { src = r.ReplaceAllString(path, "") } else { s := fmt.Sprintf("$1/%d", page) src = r.ReplaceAllString(path, s) } src = strings.Replace(src, "//", "/", -1) if src == "" { src = "/" } return }