2025-12-11 15:34:42 +08:00

515 lines
11 KiB
Go
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

package controller
import (
"fmt"
"github.com/gin-gonic/gin"
"io"
"math/rand"
"mime/multipart"
"strconv"
"time"
"vote-admin-video-api/api/dao"
"vote-admin-video-api/api/dto"
"vote-admin-video-api/api/model"
"vote-admin-video-api/api/requests"
"vote-admin-video-api/api/responses"
"vote-admin-video-api/extend/aliyun"
"vote-admin-video-api/global"
"vote-admin-video-api/utils"
)
type Video struct{}
// GetVideoPage 获取视频列表-分页
func (r *Video) GetVideoPage(c *gin.Context) {
VideoRequest := requests.VideoRequest{}
req := VideoRequest.GetVideoPage
if err := c.ShouldBindJSON(&req); err != nil {
responses.FailWithMessage(err.Error(), c)
return
}
// 参数验证
if err := global.Validate.Struct(req); err != nil {
responses.FailWithMessage(utils.Translate(err), c)
return
}
if req.Page == 0 {
req.Page = 1
}
if req.PageSize == 0 {
req.PageSize = 20
}
if req.Order != nil {
if req.Order.VoteNum != "" {
if req.Order.VoteNum != "desc" && req.Order.VoteNum != "asc" {
responses.FailWithMessage("排序字段错误", c)
return
}
}
}
// 获取数据
VideoDao := dao.VideoDao{}
Videos, total, err := VideoDao.GetVideoPageSearch(req, req.Page, req.PageSize)
if err != nil {
responses.FailWithMessage(err.Error(), c)
return
}
// 处理返回值
g := dto.GetVideoListDto(Videos)
result := make(map[string]interface{})
result["page"] = req.Page
result["page_size"] = req.PageSize
result["total"] = total
result["data"] = g
responses.OkWithData(result, c)
}
// GetVideo 获取视频详情
func (r *Video) GetVideo(c *gin.Context) {
id := c.Param("video_id")
if id == "" {
responses.FailWithMessage("缺少参数", c)
return
}
// 将 id 转换为 int64 类型
videoId, err := strconv.ParseInt(id, 10, 64)
if err != nil {
responses.Fail(c)
return
}
// 获取数据
VideoDao := dao.VideoDao{}
video, err := VideoDao.GetVideoById(videoId)
if err != nil {
responses.FailWithMessage("视频错误", c)
return
}
// 获取作者数据
VideoAuthorDao := dao.VideoAuthorDao{}
maps := make(map[string]interface{})
maps["video_id"] = videoId
VideoAuthors, err := VideoAuthorDao.GetVideoAuthorPreloadList(maps)
if err != nil {
responses.FailWithMessage("视频错误", c)
return
}
// 获取排名
rank, _ := VideoDao.GetVideoRank(video.VideoId)
// 处理返回值
g := dto.GetVideoDto(video)
// 加载数据-作者
g.LoadVideoAuthor(VideoAuthors)
// 加载数据-作者排名
g.LoadRank(rank)
responses.OkWithData(g, c)
}
// PutVideo 修改视频详情
func (r *Video) PutVideo(c *gin.Context) {
VideoRequest := requests.VideoRequest{}
req := VideoRequest.PutVideo
if err := c.ShouldBindJSON(&req); err != nil {
responses.FailWithMessage(err.Error(), c)
return
}
// 参数验证
if err := global.Validate.Struct(req); err != nil {
responses.FailWithMessage(utils.Translate(err), c)
return
}
id := c.Param("video_id")
if id == "" {
responses.FailWithMessage("缺少参数", c)
return
}
// 将 id 转换为 int64 类型
videoId, err := strconv.ParseInt(id, 10, 64)
if err != nil {
responses.Fail(c)
return
}
if req.VoteNum != nil {
if *req.VoteNum < 0 {
responses.FailWithMessage("票数需大于0", c)
return
}
}
// 获取订单数据
videoDao := dao.VideoDao{}
video, err := videoDao.GetVideoById(videoId)
if err != nil {
responses.FailWithMessage("视频异常", c)
return
}
// 开始事务
tx := global.Db.Begin()
defer func() {
if r := recover(); r != nil {
tx.Rollback()
fmt.Println(r)
}
}()
videoData := make(map[string]interface{})
// 视频标题
if video.VideoTitle != req.VideoTitle {
videoData["video_title"] = req.VideoTitle
}
// 视频状态
if video.VideoStatus != req.VideoStatus {
videoData["video_status"] = req.VideoStatus
}
if req.VoteNum != nil {
if video.VoteNum != *req.VoteNum {
videoData["vote_num"] = *req.VoteNum
}
}
// 视频编号
if video.VideoNo != req.VideoNo {
videoData["video_no"] = req.VideoNo
}
// 视频内容
if video.VideoContent != req.VideoContent {
videoData["video_content"] = req.VideoContent
}
// 修改
if len(videoData) > 0 {
err = videoDao.EditVideoById(tx, videoId, videoData)
if err != nil {
tx.Rollback()
responses.FailWithMessage(err.Error(), c)
return
}
}
// 删除作者
VideoAuthorDao := dao.VideoAuthorDao{}
maps := make(map[string]interface{})
maps["video_id"] = videoId
err = VideoAuthorDao.DeleteVideoAuthor(tx, maps)
if err != nil {
tx.Rollback()
responses.FailWithMessage(err.Error(), c)
return
}
baseHospitalDao := dao.BaseHospitalDao{}
// 新增作者
for _, author := range req.VideoAuthor {
if author.HospitalId == "" {
tx.Rollback()
responses.FailWithMessage("请选择作者所属医院", c)
return
}
if author.AuthorName == "" {
tx.Rollback()
responses.FailWithMessage("请输入作者名称", c)
return
}
// 检测医院
hospitalId, err := strconv.ParseInt(author.HospitalId, 10, 64)
if err != nil {
tx.Rollback()
responses.FailWithMessage(err.Error(), c)
return
}
baseHospital, err := baseHospitalDao.GetBaseHospitalById(hospitalId)
if err != nil || baseHospital == nil {
tx.Rollback()
responses.FailWithMessage("医院错误", c)
return
}
videoAuthor := &model.VideoAuthor{
VideoId: videoId,
AuthorName: author.AuthorName,
AuthorAvatar: author.AuthorAvatar,
HospitalId: hospitalId,
}
videoAuthor, err = VideoAuthorDao.AddVideoAuthor(tx, videoAuthor)
if err != nil {
tx.Rollback()
responses.FailWithMessage(err.Error(), c)
return
}
}
tx.Commit()
responses.Ok(c)
}
// AddVideo 新增视频详情
func (r *Video) AddVideo(c *gin.Context) {
videoRequest := requests.VideoRequest{}
req := videoRequest.AddVideo
if err := c.ShouldBindJSON(&req); err != nil {
responses.FailWithMessage(err.Error(), c)
return
}
// 参数验证
if err := global.Validate.Struct(req); err != nil {
responses.FailWithMessage(utils.Translate(err), c)
return
}
if req.VoteNum != nil {
if *req.VoteNum < 0 {
responses.FailWithMessage("票数需大于0", c)
return
}
}
// 开始事务
tx := global.Db.Begin()
defer func() {
if r := recover(); r != nil {
tx.Rollback()
responses.FailWithMessage("操作失败", c)
return
}
}()
videoDao := dao.VideoDao{}
video := &model.Video{
VideoTitle: req.VideoTitle,
VideoStatus: req.VideoStatus,
VideoNo: req.VideoNo,
VideoContent: req.VideoContent,
}
if req.VoteNum != nil {
video.VoteNum = *req.VoteNum
}
// 获取总数量
maps := make(map[string]interface{})
total, err := videoDao.GetVideoCount(maps)
if err != nil {
tx.Rollback()
responses.FailWithMessage(err.Error(), c)
return
}
// 生成视频编号
video.VideoNumber = fmt.Sprintf("%03d", total+1)
video, err = videoDao.AddVideo(tx, video)
if err != nil {
tx.Rollback()
responses.FailWithMessage(err.Error(), c)
return
}
// 新增作者
baseHospitalDao := dao.BaseHospitalDao{}
videoAuthorDao := dao.VideoAuthorDao{}
for _, author := range req.VideoAuthor {
if author.HospitalId == "" {
tx.Rollback()
responses.FailWithMessage("请选择作者所属医院", c)
return
}
if author.AuthorName == "" {
tx.Rollback()
responses.FailWithMessage("请输入作者名称", c)
return
}
// 检测医院
hospitalId, err := strconv.ParseInt(author.HospitalId, 10, 64)
if err != nil {
tx.Rollback()
responses.FailWithMessage(err.Error(), c)
return
}
baseHospital, err := baseHospitalDao.GetBaseHospitalById(hospitalId)
if err != nil || baseHospital == nil {
tx.Rollback()
responses.FailWithMessage("医院错误", c)
return
}
videoAuthor := &model.VideoAuthor{
VideoId: video.VideoId,
AuthorName: author.AuthorName,
AuthorAvatar: author.AuthorAvatar,
HospitalId: hospitalId,
}
videoAuthor, err = videoAuthorDao.AddVideoAuthor(tx, videoAuthor)
if err != nil {
tx.Rollback()
responses.FailWithMessage(err.Error(), c)
return
}
}
tx.Commit()
responses.Ok(c)
}
// PutVideoStatus 操作视频状态
func (r *Video) PutVideoStatus(c *gin.Context) {
videoRequest := requests.VideoRequest{}
req := videoRequest.PutVideoStatus
if err := c.ShouldBind(&req); err != nil {
responses.FailWithMessage(err.Error(), c)
return
}
// 参数验证
if err := global.Validate.Struct(req); err != nil {
responses.FailWithMessage(utils.Translate(err), c)
return
}
id := c.Param("video_id")
if id == "" {
responses.FailWithMessage("缺少参数", c)
return
}
// 将 id 转换为 int64 类型
videoId, err := strconv.ParseInt(id, 10, 64)
if err != nil {
responses.Fail(c)
return
}
// 获取订单数据
videoDao := dao.VideoDao{}
video, err := videoDao.GetVideoById(videoId)
if err != nil {
responses.FailWithMessage("视频异常", c)
return
}
// 开始事务
tx := global.Db.Begin()
defer func() {
if r := recover(); r != nil {
tx.Rollback()
}
}()
videoData := make(map[string]interface{})
if req.VideoStatus != video.VideoStatus {
videoData["video_status"] = req.VideoStatus
}
if len(videoData) > 0 {
err = videoDao.EditVideoById(tx, videoId, videoData)
if err != nil {
tx.Rollback()
responses.FailWithMessage("修改失败", c)
return
}
}
tx.Commit()
responses.Ok(c)
}
// UploadAuthorAvatar 上传作者头像
func (r *Video) UploadAuthorAvatar(c *gin.Context) {
file, err := c.FormFile("file")
if err != nil {
responses.FailWithMessage("文件错误", c)
return
}
f, err := file.Open()
if err != nil {
responses.FailWithMessage("文件错误", c)
return
}
defer func(f multipart.File) {
err := f.Close()
if err != nil {
fmt.Println(err)
}
}(f)
// 读取文件内容到字节切片
fileBytes, err := io.ReadAll(f)
if err != nil {
responses.FailWithMessage("文件错误", c)
return
}
// 验证文件类型(只允许图片)
fileType := "jpg"
if file.Filename != "" {
fileType = utils.GetExtension(file.Filename)
if fileType == "" {
fileType = "jpg"
}
}
// 验证是否为图片格式
allowedTypes := map[string]bool{
"jpg": true,
"jpeg": true,
"png": true,
"gif": true,
"webp": true,
}
if !allowedTypes[fileType] {
responses.FailWithMessage("只支持图片格式jpg、jpeg、png、gif、webp", c)
return
}
// 生成OSS路径
now := time.Now()
dateTimeString := now.Format("20060102150405")
rand.New(rand.NewSource(time.Now().UnixNano()))
ossPath := "static/author_avatar/" + fmt.Sprintf("%d", rand.Intn(9000)+1000) + dateTimeString + "." + fileType
// 上传到OSS
_, err = aliyun.PutObjectByte(ossPath, fileBytes)
if err != nil {
responses.FailWithMessage(err.Error(), c)
return
}
// 返回上传后的URL
result := make(map[string]interface{})
result["url"] = utils.AddOssDomain("/" + ossPath)
result["path"] = "/" + ossPath
responses.OkWithData(result, c)
}