|
|
// Copyright 2022 ROC. All rights reserved.
|
|
|
// Use of this source code is governed by a MIT style
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
package jinzhu
|
|
|
|
|
|
import (
|
|
|
"errors"
|
|
|
"strings"
|
|
|
|
|
|
"github.com/rocboss/paopao-ce/internal/core"
|
|
|
"github.com/rocboss/paopao-ce/internal/dao/jinzhu/dbr"
|
|
|
"gorm.io/gorm"
|
|
|
)
|
|
|
|
|
|
var (
|
|
|
_ core.TopicService = (*topicServant)(nil)
|
|
|
)
|
|
|
|
|
|
type topicServant struct {
|
|
|
db *gorm.DB
|
|
|
ums core.UserManageService
|
|
|
tnTopicUser string
|
|
|
tnDotTopicUser string
|
|
|
}
|
|
|
|
|
|
type topicInfo struct {
|
|
|
TopicId int64
|
|
|
IsTop int8
|
|
|
}
|
|
|
|
|
|
func newTopicService(db *gorm.DB, ums core.UserManageService) core.TopicService {
|
|
|
return &topicServant{
|
|
|
db: db,
|
|
|
ums: ums,
|
|
|
tnTopicUser: db.NamingStrategy.TableName("TopicUser"),
|
|
|
tnDotTopicUser: db.NamingStrategy.TableName("TopicUser") + ".",
|
|
|
}
|
|
|
}
|
|
|
|
|
|
func (s *topicServant) CreateTag(tag *core.Tag) (*core.Tag, error) {
|
|
|
return createTag(s.db, tag)
|
|
|
}
|
|
|
|
|
|
func (s *topicServant) DeleteTag(tag *core.Tag) error {
|
|
|
return deleteTag(s.db, tag)
|
|
|
}
|
|
|
|
|
|
func (s *topicServant) GetTags(conditions *core.ConditionsT, offset, limit int) ([]*core.Tag, error) {
|
|
|
return (&dbr.Tag{}).List(s.db, conditions, offset, limit)
|
|
|
}
|
|
|
|
|
|
func (s *topicServant) GetHotTags(userId int64, limit int, offset int) ([]*core.TagFormated, error) {
|
|
|
tags, err := (&dbr.Tag{}).List(s.db, &core.ConditionsT{
|
|
|
"ORDER": "quote_num DESC",
|
|
|
}, offset, limit)
|
|
|
if err != nil {
|
|
|
return nil, err
|
|
|
}
|
|
|
return s.tagsFormat(userId, nil, tags)
|
|
|
}
|
|
|
|
|
|
func (s *topicServant) GetNewestTags(userId int64, limit int, offset int) ([]*core.TagFormated, error) {
|
|
|
tags, err := (&dbr.Tag{}).List(s.db, &core.ConditionsT{
|
|
|
"ORDER": "id DESC",
|
|
|
}, offset, limit)
|
|
|
if err != nil {
|
|
|
return nil, err
|
|
|
}
|
|
|
return s.tagsFormat(userId, nil, tags)
|
|
|
}
|
|
|
|
|
|
func (s *topicServant) GetFollowTags(userId int64, limit int, offset int) ([]*core.TagFormated, error) {
|
|
|
if userId < 0 {
|
|
|
return nil, nil
|
|
|
}
|
|
|
userTopics := []*topicInfo{}
|
|
|
err := s.db.Model(&dbr.TopicUser{}).
|
|
|
Where("user_id=?", userId).
|
|
|
Order("is_top DESC").
|
|
|
Limit(limit).
|
|
|
Offset(offset).
|
|
|
Find(&userTopics).Error
|
|
|
if err != nil {
|
|
|
return nil, err
|
|
|
}
|
|
|
userTopicsMap := make(map[int64]*topicInfo, len(userTopics))
|
|
|
topicIds := make([]int64, 0, len(userTopics))
|
|
|
topicIdsMap := make(map[int64]int, len(userTopics))
|
|
|
for idx, info := range userTopics {
|
|
|
userTopicsMap[info.TopicId] = info
|
|
|
topicIds = append(topicIds, info.TopicId)
|
|
|
topicIdsMap[info.TopicId] = idx
|
|
|
}
|
|
|
var tags []*core.Tag
|
|
|
err = s.db.Model(&dbr.Tag{}).Where("quote_num > 0 and id in ?", topicIds).Order("quote_num DESC").Find(&tags).Error
|
|
|
if err != nil {
|
|
|
return nil, err
|
|
|
}
|
|
|
formtedTags, err := s.tagsFormat(-1, userTopicsMap, tags)
|
|
|
if err != nil {
|
|
|
return nil, err
|
|
|
}
|
|
|
// 置顶排序后处理
|
|
|
// TODO: 垃圾办法,最好是topic_user join tag 一次查询,但是gorm的join真他喵的别扭,F*K
|
|
|
res := make([]*core.TagFormated, len(topicIds), len(topicIds))
|
|
|
for _, tag := range formtedTags {
|
|
|
res[topicIdsMap[tag.ID]] = tag
|
|
|
}
|
|
|
return res, nil
|
|
|
}
|
|
|
|
|
|
func (s *topicServant) tagsFormat(userId int64, userTopicsMap map[int64]*topicInfo, tags []*core.Tag) ([]*core.TagFormated, error) {
|
|
|
// 获取创建者User IDs
|
|
|
userIds := []int64{}
|
|
|
tagIds := []int64{}
|
|
|
for _, tag := range tags {
|
|
|
userIds = append(userIds, tag.UserID)
|
|
|
tagIds = append(tagIds, tag.ID)
|
|
|
}
|
|
|
users, err := s.ums.GetUsersByIDs(userIds)
|
|
|
if err != nil {
|
|
|
return nil, err
|
|
|
}
|
|
|
tagsFormated := []*core.TagFormated{}
|
|
|
for _, tag := range tags {
|
|
|
tagFormated := tag.Format()
|
|
|
for _, user := range users {
|
|
|
if user.ID == tagFormated.UserID {
|
|
|
tagFormated.User = user.Format()
|
|
|
}
|
|
|
}
|
|
|
tagsFormated = append(tagsFormated, tagFormated)
|
|
|
}
|
|
|
// 填充话题follow信息
|
|
|
if userId > -1 && len(userTopicsMap) <= 0 {
|
|
|
userTopics := []*topicInfo{}
|
|
|
err = s.db.Model(&dbr.TopicUser{}).Where("is_del=0 and user_id=? and topic_id in ?", userId, tagIds).Find(&userTopics).Error
|
|
|
if err != nil {
|
|
|
return nil, err
|
|
|
}
|
|
|
userTopicsMap = make(map[int64]*topicInfo, len(userTopics))
|
|
|
for _, info := range userTopics {
|
|
|
userTopicsMap[info.TopicId] = info
|
|
|
}
|
|
|
}
|
|
|
if len(userTopicsMap) > 0 {
|
|
|
for _, tag := range tagsFormated {
|
|
|
if info, exist := userTopicsMap[tag.ID]; exist {
|
|
|
tag.IsFollowing, tag.IsTop = 1, info.IsTop
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
return tagsFormated, nil
|
|
|
}
|
|
|
|
|
|
func (s *topicServant) GetTagsByKeyword(keyword string) ([]*core.Tag, error) {
|
|
|
tag := &dbr.Tag{}
|
|
|
|
|
|
keyword = "%" + strings.Trim(keyword, " ") + "%"
|
|
|
if keyword == "%%" {
|
|
|
return tag.List(s.db, &dbr.ConditionsT{
|
|
|
"ORDER": "quote_num DESC",
|
|
|
}, 0, 6)
|
|
|
} else {
|
|
|
return tag.List(s.db, &dbr.ConditionsT{
|
|
|
"tag LIKE ?": keyword,
|
|
|
"ORDER": "quote_num DESC",
|
|
|
}, 0, 6)
|
|
|
}
|
|
|
}
|
|
|
|
|
|
func (s *topicServant) FollowTopic(userId int64, topicId int64) (err error) {
|
|
|
return s.db.Create(&dbr.TopicUser{
|
|
|
UserID: userId,
|
|
|
TopicID: topicId,
|
|
|
IsTop: 0,
|
|
|
}).Error
|
|
|
}
|
|
|
|
|
|
func (s *topicServant) UnfollowTopic(userId int64, topicId int64) error {
|
|
|
return s.db.Exec("DELETE FROM "+s.tnTopicUser+" WHERE user_id=? AND topic_id=?", userId, topicId).Error
|
|
|
}
|
|
|
|
|
|
func (s *topicServant) StickTopic(userId int64, topicId int64) (status int8, err error) {
|
|
|
db := s.db.Begin()
|
|
|
defer db.Rollback()
|
|
|
|
|
|
m := &dbr.TopicUser{}
|
|
|
err = db.Model(m).
|
|
|
Where("user_id=? and topic_id=?", userId, topicId).
|
|
|
UpdateColumn("is_top", gorm.Expr("1-is_top")).Error
|
|
|
if err != nil {
|
|
|
return
|
|
|
}
|
|
|
status = -1
|
|
|
err = db.Model(m).Where("user_id=? and topic_id=?", userId, topicId).Select("is_top").Scan(&status).Error
|
|
|
if err != nil {
|
|
|
return
|
|
|
}
|
|
|
if status < 0 {
|
|
|
return -1, errors.New("topic not exist")
|
|
|
}
|
|
|
|
|
|
db.Commit()
|
|
|
return
|
|
|
}
|