Version: 0.1.1.dev.260207 - 新增并测试通过将任务块排进日程接口 ✅ - 批量导入课程接口增加单双周功能,支持只在单双周上课的课程 📚 - 任务块时间定位逻辑调整为「第几周-周几」模式 🧭 refactor: 🔨 重构时间与日程数据结构 - 完成绝对日期与相对时间的转换逻辑 🔄 - 后续可根据需求灵活决定时间的传入与输出类型 - 再次重构 schedule 表单结构 - 拆分为 schedule_event(单)与 schedule(多) - 建立前者对后者的一对多关系 🧩 fix: 🐛 大幅调整表结构与业务逻辑,修复大量历史遗留 bug 🔥
193 lines
5.5 KiB
Go
193 lines
5.5 KiB
Go
package conv
|
|
|
|
import (
|
|
"encoding/json"
|
|
"errors"
|
|
"time"
|
|
|
|
"github.com/LoveLosita/smartflow/backend/model"
|
|
"github.com/LoveLosita/smartflow/backend/respond"
|
|
)
|
|
|
|
const dateLayout = "2006-01-02"
|
|
|
|
func parseDatePtr(s string) (*time.Time, error) {
|
|
if s == "" {
|
|
return nil, nil
|
|
}
|
|
t, err := time.ParseInLocation(dateLayout, s, time.Local)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &t, nil
|
|
}
|
|
|
|
func ProcessUserAddTaskClassRequest(req *model.UserAddTaskClassRequest, userID int) (*model.TaskClass, []model.TaskClassItem, error) {
|
|
startDate, err := parseDatePtr(req.StartDate)
|
|
if err != nil {
|
|
return nil, nil, respond.WrongParamType
|
|
}
|
|
endDate, err := parseDatePtr(req.EndDate)
|
|
if err != nil {
|
|
return nil, nil, respond.WrongParamType
|
|
}
|
|
//1.填充section1,2
|
|
taskClass := model.TaskClass{
|
|
Name: &req.Name,
|
|
Mode: &req.Mode,
|
|
StartDate: startDate,
|
|
EndDate: endDate,
|
|
UserID: &userID,
|
|
}
|
|
//2.填充section3
|
|
taskClass.TotalSlots = &req.Config.TotalSlots
|
|
taskClass.AllowFillerCourse = &req.Config.AllowFillerCourse
|
|
taskClass.Strategy = &req.Config.Strategy
|
|
//处理 ExcludedSlots 切片为 JSON 字符串
|
|
if len(req.Config.ExcludedSlots) > 0 {
|
|
//转换为 JSON 字符串
|
|
excludedSlotsJSON := "["
|
|
for i, slot := range req.Config.ExcludedSlots {
|
|
excludedSlotsJSON += string(rune(slot + '0')) //简单转换为字符
|
|
if i != len(req.Config.ExcludedSlots)-1 {
|
|
excludedSlotsJSON += ","
|
|
}
|
|
}
|
|
excludedSlotsJSON += "]"
|
|
taskClass.ExcludedSlots = &excludedSlotsJSON
|
|
} else {
|
|
emptyJSON := "[]"
|
|
taskClass.ExcludedSlots = &emptyJSON
|
|
}
|
|
//3.开始构建 items
|
|
var items []model.TaskClassItem
|
|
for _, itemReq := range req.Items {
|
|
item := model.TaskClassItem{ //填充section 2
|
|
Order: &itemReq.Order,
|
|
Content: &itemReq.Content,
|
|
EmbeddedTime: itemReq.EmbeddedTime,
|
|
Status: nil,
|
|
}
|
|
items = append(items, item)
|
|
}
|
|
return &taskClass, items, nil
|
|
}
|
|
|
|
func timeOrZero(t *time.Time) time.Time {
|
|
if t == nil {
|
|
return time.Time{}
|
|
}
|
|
return *t
|
|
}
|
|
|
|
func TaskClassModelToResponse(taskClasses []model.TaskClass) *model.UserGetTaskClassesResponse {
|
|
var resp model.UserGetTaskClassesResponse
|
|
for _, tc := range taskClasses {
|
|
tcResp := model.TaskClassSummary{
|
|
ID: tc.ID,
|
|
Name: *tc.Name,
|
|
Mode: *tc.Mode,
|
|
StartDate: timeOrZero(tc.StartDate),
|
|
EndDate: timeOrZero(tc.EndDate),
|
|
TotalSlots: *tc.TotalSlots,
|
|
Strategy: *tc.Strategy,
|
|
}
|
|
resp.TaskClasses = append(resp.TaskClasses, tcResp)
|
|
}
|
|
return &resp
|
|
}
|
|
|
|
func ProcessUserGetCompleteTaskClassRequest(taskClass *model.TaskClass) (*model.UserAddTaskClassRequest, error) {
|
|
if taskClass == nil {
|
|
return nil, errors.New("源数据对象不可为空")
|
|
}
|
|
// 1. 映射基础信息 (处理指针解引用)
|
|
req := &model.UserAddTaskClassRequest{
|
|
Name: safeStr(taskClass.Name),
|
|
Mode: safeStr(taskClass.Mode),
|
|
StartDate: formatTime(taskClass.StartDate),
|
|
EndDate: formatTime(taskClass.EndDate),
|
|
}
|
|
// 2. 映射配置信息 (Config Section)
|
|
req.Config = model.UserAddTaskClassConfig{
|
|
TotalSlots: safeInt(taskClass.TotalSlots),
|
|
AllowFillerCourse: safeBool(taskClass.AllowFillerCourse),
|
|
Strategy: safeStr(taskClass.Strategy),
|
|
}
|
|
// 3. 处理 ExcludedSlots JSON 字符串 -> []int
|
|
if taskClass.ExcludedSlots != nil && *taskClass.ExcludedSlots != "" {
|
|
var excluded []int
|
|
// 直接使用标准反序列化,比手动处理 rune 字符要健壮得多
|
|
if err := json.Unmarshal([]byte(*taskClass.ExcludedSlots), &excluded); err == nil {
|
|
req.Config.ExcludedSlots = excluded
|
|
}
|
|
}
|
|
// 4. 映射子项信息 (Items Section)
|
|
// 此时 items 已经通过 Preload 加载到了 taskClass.Items 中
|
|
req.Items = make([]model.UserAddTaskClassItemRequest, 0, len(taskClass.Items))
|
|
for _, item := range taskClass.Items {
|
|
itemReq := model.UserAddTaskClassItemRequest{
|
|
Order: safeInt(item.Order),
|
|
Content: safeStr(item.Content),
|
|
EmbeddedTime: item.EmbeddedTime, // 结构体指针直接复用
|
|
}
|
|
req.Items = append(req.Items, itemReq)
|
|
}
|
|
return req, nil
|
|
}
|
|
|
|
// UserInsertTaskItemRequestToModel 用于将填入空闲时段日程的请求转换为 Schedule 模型
|
|
func UserInsertTaskItemRequestToModel(req *model.UserInsertTaskClassItemToScheduleRequest, item *model.TaskClassItem, taskID, userID, startSection, endSection int) ([]model.Schedule, *model.ScheduleEvent) {
|
|
var schedules []model.Schedule
|
|
for section := startSection; section <= endSection; section++ {
|
|
req1 := &model.Schedule{
|
|
UserID: userID, // 由调用方填充
|
|
EmbeddedTaskID: &taskID,
|
|
Week: req.Week,
|
|
DayOfWeek: req.DayOfWeek,
|
|
Section: section,
|
|
Status: "normal",
|
|
}
|
|
schedules = append(schedules, *req1)
|
|
}
|
|
req2 := &model.ScheduleEvent{
|
|
UserID: userID, // 由调用方填充
|
|
Name: safeStr(item.Content), // 任务内容作为事件名称
|
|
Type: "task",
|
|
RelID: &item.ID, // 关联到 TaskClassItem 的 ID
|
|
CanBeEmbedded: false, // 任务事件允许嵌入其他任务(如果需要的话)
|
|
}
|
|
return schedules, req2
|
|
}
|
|
|
|
// --- 🛡️ 辅助工具函数:保持代码清爽并防止 Panic ---
|
|
|
|
func safeStr(s *string) string {
|
|
if s == nil {
|
|
return ""
|
|
}
|
|
return *s
|
|
}
|
|
|
|
func safeInt(i *int) int {
|
|
if i == nil {
|
|
return 0
|
|
}
|
|
return *i
|
|
}
|
|
|
|
func safeBool(b *bool) bool {
|
|
if b == nil {
|
|
return true
|
|
}
|
|
return *b
|
|
}
|
|
|
|
func formatTime(t *time.Time) string {
|
|
if t == nil {
|
|
return ""
|
|
}
|
|
// 务必使用 2006-01-02 格式以匹配前端校验
|
|
return t.Format("2006-01-02")
|
|
}
|