bindbox-game/internal/api/admin/rewards_admin.go
邹方成 45815bfb7d chore: 清理无用文件与优化代码结构
refactor(utils): 修复密码哈希比较逻辑错误
feat(user): 新增按状态筛选优惠券接口
docs: 添加虚拟发货与任务中心相关文档
fix(wechat): 修正Code2Session上下文传递问题
test: 补充订单折扣与积分转换测试用例
build: 更新配置文件与构建脚本
style: 清理多余的空行与注释
2025-12-18 17:35:55 +08:00

276 lines
9.0 KiB
Go

package admin
import (
"net/http"
"strconv"
"bindbox-game/internal/code"
"bindbox-game/internal/pkg/core"
"bindbox-game/internal/pkg/validation"
activitysvc "bindbox-game/internal/service/activity"
)
type rewardItem struct {
ID int64 `json:"id"`
ProductID int64 `json:"product_id"`
Name string `json:"name" binding:"required"`
Weight int32 `json:"weight" binding:"required"`
Quantity int64 `json:"quantity" binding:"required"`
OriginalQty int64 `json:"original_qty" binding:"required"`
Level int32 `json:"level" binding:"required"`
Sort int32 `json:"sort"`
IsBoss int32 `json:"is_boss"`
}
type createRewardsRequest struct {
Rewards []rewardItem `json:"rewards" binding:"required"`
}
type listRewardsResponse struct {
List []rewardItem `json:"list"`
}
// CreateIssueRewards 创建期数奖品
// @Summary 创建期数奖品
// @Description 为指定期数批量创建奖励配置
// @Tags 管理端.活动
// @Accept json
// @Produce json
// @Param activity_id path integer true "活动ID"
// @Param issue_id path integer true "期ID"
// @Param RequestBody body createRewardsRequest true "请求参数"
// @Success 200 {object} simpleMessageResponse
// @Failure 400 {object} code.Failure
// @Router /api/admin/activities/{activity_id}/issues/{issue_id}/rewards [post]
// @Security LoginVerifyToken
func (h *handler) CreateIssueRewards() core.HandlerFunc {
return func(ctx core.Context) {
req := new(createRewardsRequest)
res := new(simpleMessageResponse)
if err := ctx.ShouldBindJSON(req); err != nil {
ctx.AbortWithError(core.Error(http.StatusBadRequest, code.ParamBindError, validation.Error(err)))
return
}
if ctx.SessionUserInfo().IsSuper != 1 {
ctx.AbortWithError(core.Error(http.StatusBadRequest, code.CreateIssueRewardsError, "禁止操作"))
return
}
issueID, err := strconv.ParseInt(ctx.Param("issue_id"), 10, 64)
if err != nil {
ctx.AbortWithError(core.Error(http.StatusBadRequest, code.ParamBindError, "未传递期ID"))
return
}
var rewards []activitysvc.CreateRewardInput
for _, r := range req.Rewards {
rewards = append(rewards, activitysvc.CreateRewardInput{
ProductID: r.ProductID,
Name: r.Name,
Weight: r.Weight,
Quantity: r.Quantity,
OriginalQty: r.OriginalQty,
Level: r.Level,
Sort: r.Sort,
IsBoss: r.IsBoss,
})
}
if err := h.activity.CreateIssueRewards(ctx.RequestContext(), issueID, rewards); err != nil {
ctx.AbortWithError(core.Error(http.StatusBadRequest, code.CreateIssueRewardsError, err.Error()))
return
}
res.Message = "操作成功"
ctx.Payload(res)
}
}
// ListIssueRewards 查看期数奖品
// @Summary 查看期数奖品
// @Description 查看指定期数的奖励配置列表
// @Tags 管理端.活动
// @Accept json
// @Produce json
// @Param activity_id path integer true "活动ID"
// @Param issue_id path integer true "期ID"
// @Success 200 {object} listRewardsResponse
// @Failure 400 {object} code.Failure
// @Router /api/admin/activities/{activity_id}/issues/{issue_id}/rewards [get]
// @Security LoginVerifyToken
func (h *handler) ListIssueRewards() core.HandlerFunc {
return func(ctx core.Context) {
issueID, err := strconv.ParseInt(ctx.Param("issue_id"), 10, 64)
if err != nil {
ctx.AbortWithError(core.Error(http.StatusBadRequest, code.ParamBindError, "未传递期ID"))
return
}
items, err := h.activity.ListIssueRewards(ctx.RequestContext(), issueID)
if err != nil {
ctx.AbortWithError(core.Error(http.StatusBadRequest, code.ListIssueRewardsError, err.Error()))
return
}
res := new(listRewardsResponse)
res.List = make([]rewardItem, len(items))
for i, v := range items {
res.List[i] = rewardItem{
ID: v.ID,
ProductID: v.ProductID,
Name: v.Name,
Weight: v.Weight,
Quantity: v.Quantity,
OriginalQty: v.OriginalQty,
Level: v.Level,
Sort: v.Sort,
IsBoss: v.IsBoss,
}
}
ctx.Payload(res)
}
}
type modifyRewardRequest struct {
ProductID *int64 `json:"product_id"`
Name string `json:"name"`
Weight *int32 `json:"weight"`
Quantity *int64 `json:"quantity"`
OriginalQty *int64 `json:"original_qty"`
Level *int32 `json:"level"`
Sort *int32 `json:"sort"`
IsBoss *int32 `json:"is_boss"`
}
// ModifyIssueReward 更新期数奖励
// @Summary 更新期数奖励
// @Tags 管理端.活动
// @Accept json
// @Produce json
// @Param activity_id path integer true "活动ID"
// @Param issue_id path integer true "期ID"
// @Param reward_id path integer true "奖励ID"
// @Param RequestBody body modifyRewardRequest true "请求参数"
// @Success 200 {object} simpleMessageResponse
// @Failure 400 {object} code.Failure
// @Router /api/admin/activities/{activity_id}/issues/{issue_id}/rewards/{reward_id} [put]
// @Security LoginVerifyToken
func (h *handler) ModifyIssueReward() core.HandlerFunc {
return func(ctx core.Context) {
req := new(modifyRewardRequest)
res := new(simpleMessageResponse)
if err := ctx.ShouldBindJSON(req); err != nil {
ctx.AbortWithError(core.Error(http.StatusBadRequest, code.ParamBindError, validation.Error(err)))
return
}
if ctx.SessionUserInfo().IsSuper != 1 {
ctx.AbortWithError(core.Error(http.StatusBadRequest, code.CreateIssueRewardsError, "禁止操作"))
return
}
rewardID, err := strconv.ParseInt(ctx.Param("reward_id"), 10, 64)
if err != nil {
ctx.AbortWithError(core.Error(http.StatusBadRequest, code.ParamBindError, "未传递奖励ID"))
return
}
in := activitysvc.ModifyRewardInput{
ProductID: req.ProductID,
Name: req.Name,
Weight: req.Weight,
Quantity: req.Quantity,
OriginalQty: req.OriginalQty,
Level: req.Level,
Sort: req.Sort,
IsBoss: req.IsBoss,
}
if err := h.activity.ModifyIssueReward(ctx.RequestContext(), rewardID, in); err != nil {
ctx.AbortWithError(core.Error(http.StatusBadRequest, code.CreateIssueRewardsError, err.Error()))
return
}
res.Message = "操作成功"
ctx.Payload(res)
}
}
// DeleteIssueReward 删除期数奖励
// @Summary 删除期数奖励
// @Tags 管理端.活动
// @Accept json
// @Produce json
// @Param activity_id path integer true "活动ID"
// @Param issue_id path integer true "期ID"
// @Param reward_id path integer true "奖励ID"
// @Success 200 {object} simpleMessageResponse
// @Failure 400 {object} code.Failure
// @Router /api/admin/activities/{activity_id}/issues/{issue_id}/rewards/{reward_id} [delete]
// @Security LoginVerifyToken
func (h *handler) DeleteIssueReward() core.HandlerFunc {
return func(ctx core.Context) {
res := new(simpleMessageResponse)
if ctx.SessionUserInfo().IsSuper != 1 {
ctx.AbortWithError(core.Error(http.StatusBadRequest, code.CreateIssueRewardsError, "禁止操作"))
return
}
rewardID, err := strconv.ParseInt(ctx.Param("reward_id"), 10, 64)
if err != nil {
ctx.AbortWithError(core.Error(http.StatusBadRequest, code.ParamBindError, "未传递奖励ID"))
return
}
if err := h.activity.DeleteIssueReward(ctx.RequestContext(), rewardID); err != nil {
ctx.AbortWithError(core.Error(http.StatusBadRequest, code.CreateIssueRewardsError, err.Error()))
return
}
res.Message = "操作成功"
ctx.Payload(res)
}
}
type batchModifyRewardItem struct {
RewardID int64 `json:"reward_id" binding:"required"`
Weight *int32 `json:"weight" binding:"omitempty,gte=0"`
}
type batchModifyRewardsRequest struct {
Rewards []batchModifyRewardItem `json:"rewards" binding:"required,dive"`
}
// BatchModifyIssueRewards 批量更新期数奖励
// @Summary 批量更新期数奖励
// @Tags 管理端.活动
// @Accept json
// @Produce json
// @Param activity_id path integer true "活动ID"
// @Param issue_id path integer true "期ID"
// @Param RequestBody body batchModifyRewardsRequest true "请求参数"
// @Success 200 {object} simpleMessageResponse
// @Failure 400 {object} code.Failure
// @Router /api/admin/activities/{activity_id}/issues/{issue_id}/rewards/batch [put]
// @Security LoginVerifyToken
func (h *handler) BatchModifyIssueRewards() core.HandlerFunc {
return func(ctx core.Context) {
req := new(batchModifyRewardsRequest)
res := new(simpleMessageResponse)
if err := ctx.ShouldBindJSON(req); err != nil {
ctx.AbortWithError(core.Error(http.StatusBadRequest, code.ParamBindError, validation.Error(err)))
return
}
if ctx.SessionUserInfo().IsSuper != 1 {
ctx.AbortWithError(core.Error(http.StatusBadRequest, code.CreateIssueRewardsError, "禁止操作"))
return
}
for _, item := range req.Rewards {
in := activitysvc.ModifyRewardInput{
Weight: item.Weight,
}
if err := h.activity.ModifyIssueReward(ctx.RequestContext(), item.RewardID, in); err != nil {
// Log error but continue or abort? For batch, aborting on first error is usually safer or return partial errors.
// For simplicity here, we abort.
ctx.AbortWithError(core.Error(http.StatusBadRequest, code.CreateIssueRewardsError, err.Error()))
return
}
}
res.Message = "批量操作成功"
ctx.Payload(res)
}
}