2023-06-24 20:39:55 +08:00
|
|
|
|
// SiYuan - Refactor your thinking
|
2023-03-02 11:32:39 +08:00
|
|
|
|
// Copyright (c) 2020-present, b3log.org
|
|
|
|
|
//
|
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
|
// it under the terms of the GNU Affero General Public License as published by
|
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
|
// (at your option) any later version.
|
|
|
|
|
//
|
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
// GNU Affero General Public License for more details.
|
|
|
|
|
//
|
|
|
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
|
|
|
// along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
|
|
package model
|
|
|
|
|
|
|
|
|
|
import (
|
2023-10-01 10:58:46 +08:00
|
|
|
|
"bytes"
|
2024-04-17 19:54:33 +08:00
|
|
|
|
"fmt"
|
2024-12-12 11:22:37 +08:00
|
|
|
|
"math/rand"
|
2023-11-23 11:22:35 +08:00
|
|
|
|
"os"
|
|
|
|
|
"path/filepath"
|
2024-05-09 10:43:02 +08:00
|
|
|
|
"slices"
|
2023-07-13 09:37:52 +08:00
|
|
|
|
"sort"
|
2023-11-23 11:22:35 +08:00
|
|
|
|
"strconv"
|
2023-06-30 20:23:31 +08:00
|
|
|
|
"strings"
|
2023-10-08 10:44:23 +08:00
|
|
|
|
"time"
|
2023-06-30 20:23:31 +08:00
|
|
|
|
|
|
|
|
|
"github.com/88250/gulu"
|
2023-03-03 10:49:45 +08:00
|
|
|
|
"github.com/88250/lute/ast"
|
2023-03-02 14:47:57 +08:00
|
|
|
|
"github.com/88250/lute/parse"
|
2024-06-10 16:25:57 +08:00
|
|
|
|
"github.com/jinzhu/copier"
|
2023-11-06 22:13:04 +08:00
|
|
|
|
"github.com/siyuan-note/filelock"
|
2023-03-02 15:03:33 +08:00
|
|
|
|
"github.com/siyuan-note/logging"
|
2023-03-02 11:32:39 +08:00
|
|
|
|
"github.com/siyuan-note/siyuan/kernel/av"
|
2024-03-04 14:14:10 +08:00
|
|
|
|
"github.com/siyuan-note/siyuan/kernel/cache"
|
2024-05-23 23:57:10 +08:00
|
|
|
|
"github.com/siyuan-note/siyuan/kernel/filesys"
|
2024-05-15 00:47:25 +08:00
|
|
|
|
"github.com/siyuan-note/siyuan/kernel/sql"
|
2023-03-02 11:32:39 +08:00
|
|
|
|
"github.com/siyuan-note/siyuan/kernel/treenode"
|
2023-07-03 15:29:54 +08:00
|
|
|
|
"github.com/siyuan-note/siyuan/kernel/util"
|
2024-04-09 17:51:39 +08:00
|
|
|
|
"github.com/xrash/smetrics"
|
2023-03-02 11:32:39 +08:00
|
|
|
|
)
|
|
|
|
|
|
2025-08-06 12:08:55 +08:00
|
|
|
|
func GetAttrViewAddingBlockDefaultValues(avID, viewID, groupID, previousBlockID, addingBlockID string) (ret map[string]*av.Value, ignore bool) {
|
2025-08-02 11:25:39 +08:00
|
|
|
|
ret = map[string]*av.Value{}
|
|
|
|
|
|
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
logging.LogErrorf("parse attribute view [%s] failed: %s", avID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view := attrView.GetView(viewID)
|
|
|
|
|
if nil == view {
|
|
|
|
|
logging.LogErrorf("view [%s] not found in attribute view [%s]", viewID, avID)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-06 12:08:55 +08:00
|
|
|
|
if 1 > len(view.Filters) && nil == view.Group {
|
|
|
|
|
// 没有过滤条件也没有分组条件时忽略
|
|
|
|
|
ignore = true
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-02 11:25:39 +08:00
|
|
|
|
groupView := view
|
|
|
|
|
if "" != groupID {
|
2025-08-09 11:24:44 +08:00
|
|
|
|
groupView = view.GetGroupByID(groupID)
|
2025-08-02 11:25:39 +08:00
|
|
|
|
}
|
|
|
|
|
if nil == groupView {
|
|
|
|
|
logging.LogErrorf("group [%s] not found in view [%s] of attribute view [%s]", groupID, viewID, avID)
|
|
|
|
|
return
|
|
|
|
|
}
|
2025-08-06 12:08:55 +08:00
|
|
|
|
|
|
|
|
|
ret = getAttrViewAddingBlockDefaultValues(attrView, view, groupView, previousBlockID, addingBlockID)
|
|
|
|
|
return
|
2025-08-02 11:25:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-08-10 17:25:07 +08:00
|
|
|
|
func getAttrViewAddingBlockDefaultValues(attrView *av.AttributeView, view, groupView *av.View, previousItemID, addingItemID string) (ret map[string]*av.Value) {
|
2025-08-02 11:25:39 +08:00
|
|
|
|
ret = map[string]*av.Value{}
|
|
|
|
|
|
2025-08-08 09:40:20 +08:00
|
|
|
|
if 1 > len(view.Filters) && nil == view.Group {
|
|
|
|
|
// 没有过滤条件也没有分组条件时忽略
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-10 17:25:07 +08:00
|
|
|
|
nearItem := getNearItem(attrView, view, groupView, previousItemID)
|
2025-08-02 11:25:39 +08:00
|
|
|
|
filterKeyIDs := map[string]bool{}
|
|
|
|
|
for _, f := range view.Filters {
|
|
|
|
|
filterKeyIDs[f.Column] = true
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-04 11:15:23 +08:00
|
|
|
|
// 对库中存在模板字段和汇总字段的情况进行处理(尽量从临近项获取新值,获取不到的话直接返回)
|
|
|
|
|
existSpecialField := false
|
2025-08-04 10:47:21 +08:00
|
|
|
|
for _, keyValues := range attrView.KeyValues {
|
2025-08-04 11:15:23 +08:00
|
|
|
|
if av.KeyTypeTemplate == keyValues.Key.Type || av.KeyTypeRollup == keyValues.Key.Type {
|
|
|
|
|
existSpecialField = true
|
2025-08-04 10:47:21 +08:00
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-08-04 11:15:23 +08:00
|
|
|
|
if existSpecialField {
|
2025-08-04 10:47:21 +08:00
|
|
|
|
if nil != nearItem {
|
2025-08-04 11:15:23 +08:00
|
|
|
|
// 存在临近项时从临近项获取新值
|
2025-08-04 10:47:21 +08:00
|
|
|
|
for _, keyValues := range attrView.KeyValues {
|
2025-08-10 17:25:07 +08:00
|
|
|
|
newValue := getNewValueByNearItem(nearItem, keyValues.Key, addingItemID)
|
2025-08-04 10:47:21 +08:00
|
|
|
|
ret[keyValues.Key.ID] = newValue
|
|
|
|
|
}
|
2025-08-04 11:15:23 +08:00
|
|
|
|
} else { // 不存在临近项时不生成任何新值
|
2025-08-04 10:47:21 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-02 11:25:39 +08:00
|
|
|
|
for _, filter := range view.Filters {
|
|
|
|
|
keyValues, _ := attrView.GetKeyValues(filter.Column)
|
|
|
|
|
if nil == keyValues {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-03 11:02:02 +08:00
|
|
|
|
var newValue *av.Value
|
2025-08-02 11:25:39 +08:00
|
|
|
|
if nil != nearItem {
|
2025-08-04 10:47:21 +08:00
|
|
|
|
// 存在临近项时优先通过临近项获取新值
|
2025-08-10 17:25:07 +08:00
|
|
|
|
newValue = getNewValueByNearItem(nearItem, keyValues.Key, addingItemID)
|
2025-08-03 11:02:02 +08:00
|
|
|
|
} else {
|
2025-08-04 10:47:21 +08:00
|
|
|
|
// 不存在临近项时通过过滤条件计算新值
|
2025-08-10 17:25:07 +08:00
|
|
|
|
newValue = filter.GetAffectValue(keyValues.Key, addingItemID)
|
2025-08-02 11:25:39 +08:00
|
|
|
|
}
|
|
|
|
|
if nil != newValue {
|
|
|
|
|
ret[keyValues.Key.ID] = newValue
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
groupKey := view.GetGroupKey(attrView)
|
|
|
|
|
if nil != groupKey && !filterKeyIDs[groupKey.ID] /* 命中了过滤条件的话就不重复处理了 */ {
|
|
|
|
|
if keyValues, _ := attrView.GetKeyValues(groupKey.ID); nil != keyValues {
|
2025-08-10 17:25:07 +08:00
|
|
|
|
newValue := getNewValueByNearItem(nearItem, groupKey, addingItemID)
|
2025-08-02 11:25:39 +08:00
|
|
|
|
if av.KeyTypeSelect == groupKey.Type || av.KeyTypeMSelect == groupKey.Type {
|
|
|
|
|
// 因为单选或多选只能按选项分组,并且可能存在空白分组(前面可能找不到临近项) ,所以单选或多选类型的分组字段使用分组值内容对应的选项
|
2025-08-05 17:09:11 +08:00
|
|
|
|
if opt := groupKey.GetOption(groupView.GetGroupValue()); nil != opt && groupValueDefault != groupView.GetGroupValue() {
|
2025-08-05 16:27:44 +08:00
|
|
|
|
exists := false
|
|
|
|
|
for _, s := range newValue.MSelect {
|
2025-08-05 17:09:11 +08:00
|
|
|
|
if s.Content == groupView.GetGroupValue() {
|
2025-08-05 16:27:44 +08:00
|
|
|
|
exists = true
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if !exists {
|
|
|
|
|
newValue.MSelect = append(newValue.MSelect, &av.ValueSelect{Content: opt.Name, Color: opt.Color})
|
|
|
|
|
}
|
2025-08-02 11:25:39 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ret[groupKey.ID] = newValue
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-25 19:17:58 +08:00
|
|
|
|
func (tx *Transaction) doSortAttrViewGroup(operation *Operation) (ret *TxErr) {
|
|
|
|
|
if err := sortAttributeViewGroup(operation.AvID, operation.BlockID, operation.PreviousID, operation.ID); nil != err {
|
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func sortAttributeViewGroup(avID, blockID, previousGroupID, groupID string) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
logging.LogErrorf("parse attribute view [%s] failed: %s", avID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, blockID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-09 21:15:59 +08:00
|
|
|
|
sortGroupViews(attrView, view)
|
2025-08-09 17:54:45 +08:00
|
|
|
|
|
2025-07-27 20:26:32 +08:00
|
|
|
|
var groupView *av.View
|
2025-07-25 19:17:58 +08:00
|
|
|
|
var index, previousIndex int
|
|
|
|
|
for i, g := range view.Groups {
|
|
|
|
|
if g.ID == groupID {
|
2025-07-27 20:26:32 +08:00
|
|
|
|
groupView = g
|
2025-07-25 19:17:58 +08:00
|
|
|
|
index = i
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-07-27 20:26:32 +08:00
|
|
|
|
if nil == groupView {
|
2025-07-25 19:17:58 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
2025-07-27 20:26:32 +08:00
|
|
|
|
view.Group.Order = av.GroupOrderMan
|
2025-07-25 19:17:58 +08:00
|
|
|
|
|
|
|
|
|
view.Groups = append(view.Groups[:index], view.Groups[index+1:]...)
|
2025-07-26 09:40:39 +08:00
|
|
|
|
for i, g := range view.Groups {
|
2025-07-25 19:17:58 +08:00
|
|
|
|
if g.ID == previousGroupID {
|
|
|
|
|
previousIndex = i + 1
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-07-27 20:26:32 +08:00
|
|
|
|
view.Groups = util.InsertElem(view.Groups, previousIndex, groupView)
|
2025-07-25 19:17:58 +08:00
|
|
|
|
|
2025-08-09 11:47:18 +08:00
|
|
|
|
for i, g := range view.Groups {
|
|
|
|
|
g.GroupSort = i
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-25 19:17:58 +08:00
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-25 12:05:33 +08:00
|
|
|
|
func (tx *Transaction) doRemoveAttrViewGroup(operation *Operation) (ret *TxErr) {
|
|
|
|
|
if err := removeAttributeViewGroup(operation.AvID, operation.BlockID); nil != err {
|
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func removeAttributeViewGroup(avID, blockID string) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, blockID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-25 22:31:46 +08:00
|
|
|
|
removeAttributeViewGroup0(view)
|
2025-07-25 12:05:33 +08:00
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
if err != nil {
|
|
|
|
|
logging.LogErrorf("save attribute view [%s] failed: %s", avID, err)
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-25 22:31:46 +08:00
|
|
|
|
func removeAttributeViewGroup0(view *av.View) {
|
2025-08-05 17:42:53 +08:00
|
|
|
|
view.Group, view.Groups, view.GroupCreated = nil, nil, 0
|
2025-07-25 22:31:46 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-07-11 10:43:09 +08:00
|
|
|
|
func (tx *Transaction) doSyncAttrViewTableColWidth(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := syncAttrViewTableColWidth(operation)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func syncAttrViewTableColWidth(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view := attrView.GetView(operation.ID)
|
|
|
|
|
if nil == view {
|
|
|
|
|
err = av.ErrViewNotFound
|
|
|
|
|
logging.LogErrorf("view [%s] not found in attribute view [%s]", operation.ID, operation.AvID)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var width string
|
|
|
|
|
switch view.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
for _, column := range view.Table.Columns {
|
|
|
|
|
if column.ID == operation.KeyID {
|
2025-07-11 11:34:31 +08:00
|
|
|
|
width = column.Width
|
2025-07-11 10:43:09 +08:00
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, v := range attrView.Views {
|
|
|
|
|
if av.LayoutTypeTable == v.LayoutType {
|
|
|
|
|
for _, column := range v.Table.Columns {
|
|
|
|
|
if column.ID == operation.KeyID {
|
|
|
|
|
column.Width = width
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-06 17:41:43 +08:00
|
|
|
|
func (tx *Transaction) doHideAttrViewGroup(operation *Operation) (ret *TxErr) {
|
2025-07-25 17:57:51 +08:00
|
|
|
|
if err := hideAttributeViewGroup(operation.AvID, operation.BlockID, operation.ID, int(operation.Data.(float64))); nil != err {
|
2025-07-06 17:41:43 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-25 17:57:51 +08:00
|
|
|
|
func hideAttributeViewGroup(avID, blockID, groupID string, hidden int) (err error) {
|
2025-07-06 17:41:43 +08:00
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
|
|
|
|
if err != nil {
|
2025-07-29 15:52:30 +08:00
|
|
|
|
return
|
2025-07-06 17:41:43 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, blockID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, group := range view.Groups {
|
|
|
|
|
if group.ID == groupID {
|
|
|
|
|
group.GroupHidden = hidden
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
if err != nil {
|
|
|
|
|
logging.LogErrorf("save attribute view [%s] failed: %s", avID, err)
|
2025-07-29 15:52:04 +08:00
|
|
|
|
return
|
2025-07-25 22:36:02 +08:00
|
|
|
|
}
|
2025-07-29 15:52:04 +08:00
|
|
|
|
return
|
2025-07-25 22:36:02 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-08-06 09:18:40 +08:00
|
|
|
|
func (tx *Transaction) doHideAttrViewAllGroups(operation *Operation) (ret *TxErr) {
|
2025-08-06 09:51:30 +08:00
|
|
|
|
if err := hideAttributeViewAllGroups(operation.AvID, operation.BlockID, operation.Data.(bool)); nil != err {
|
2025-08-06 09:18:40 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-06 09:51:30 +08:00
|
|
|
|
func hideAttributeViewAllGroups(avID, blockID string, hidden bool) (err error) {
|
2025-08-06 09:18:40 +08:00
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, blockID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, group := range view.Groups {
|
2025-08-06 09:51:30 +08:00
|
|
|
|
if hidden {
|
|
|
|
|
group.GroupHidden = 2
|
|
|
|
|
} else {
|
|
|
|
|
group.GroupHidden = 0
|
|
|
|
|
}
|
2025-08-06 09:18:40 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
if err != nil {
|
|
|
|
|
logging.LogErrorf("save attribute view [%s] failed: %s", avID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-25 22:36:02 +08:00
|
|
|
|
func (tx *Transaction) doFoldAttrViewGroup(operation *Operation) (ret *TxErr) {
|
|
|
|
|
if err := foldAttrViewGroup(operation.AvID, operation.BlockID, operation.ID, operation.Data.(bool)); nil != err {
|
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func foldAttrViewGroup(avID, blockID, groupID string, folded bool) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, blockID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if nil == view.Group {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, group := range view.Groups {
|
|
|
|
|
if group.ID == groupID {
|
|
|
|
|
group.GroupFolded = folded
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
if err != nil {
|
|
|
|
|
logging.LogErrorf("save attribute view [%s] failed: %s", avID, err)
|
2025-07-06 17:41:43 +08:00
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-01 17:56:32 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewGroup(operation *Operation) (ret *TxErr) {
|
2025-07-05 10:07:36 +08:00
|
|
|
|
data, err := gulu.JSON.MarshalJSON(operation.Data)
|
|
|
|
|
if nil != err {
|
|
|
|
|
logging.LogErrorf("marshal operation data failed: %s", err)
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2025-07-05 10:07:36 +08:00
|
|
|
|
}
|
|
|
|
|
group := &av.ViewGroup{}
|
|
|
|
|
if err = gulu.JSON.UnmarshalJSON(data, &group); nil != err {
|
|
|
|
|
logging.LogErrorf("unmarshal operation data failed: %s", err)
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2025-07-05 10:07:36 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err = SetAttributeViewGroup(operation.AvID, operation.BlockID, group); nil != err {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2025-07-01 17:56:32 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-06 17:41:43 +08:00
|
|
|
|
func SetAttributeViewGroup(avID, blockID string, group *av.ViewGroup) (err error) {
|
2025-07-01 21:50:32 +08:00
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
2025-07-01 17:56:32 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-01 21:50:32 +08:00
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, blockID)
|
2025-07-01 17:56:32 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-10 11:33:59 +08:00
|
|
|
|
var oldHideEmpty, firstInit, changeGroupField bool
|
2025-08-08 18:18:44 +08:00
|
|
|
|
if nil != view.Group {
|
|
|
|
|
oldHideEmpty = view.Group.HideEmpty
|
2025-08-10 11:33:59 +08:00
|
|
|
|
changeGroupField = group.Field != view.Group.Field
|
2025-08-09 11:24:44 +08:00
|
|
|
|
} else {
|
|
|
|
|
firstInit = true
|
2025-08-08 18:18:44 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-08-05 11:26:25 +08:00
|
|
|
|
groupStates := getAttrViewGroupStates(view)
|
2025-07-25 11:56:32 +08:00
|
|
|
|
view.Group = group
|
2025-08-09 20:39:12 +08:00
|
|
|
|
regenAttrViewGroups(attrView, "force")
|
2025-08-05 11:26:25 +08:00
|
|
|
|
setAttrViewGroupStates(view, groupStates)
|
2025-07-25 17:57:51 +08:00
|
|
|
|
|
2025-08-08 18:18:44 +08:00
|
|
|
|
if view.Group.HideEmpty != oldHideEmpty {
|
|
|
|
|
if !oldHideEmpty && view.Group.HideEmpty { // 启用隐藏空分组
|
|
|
|
|
for _, g := range view.Groups {
|
|
|
|
|
if g.GroupHidden == 0 && 1 > len(g.GroupItemIDs) {
|
|
|
|
|
g.GroupHidden = 1
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if oldHideEmpty && !view.Group.HideEmpty { // 禁用隐藏空分组
|
|
|
|
|
for _, g := range view.Groups {
|
|
|
|
|
if g.GroupHidden == 1 && 1 > len(g.GroupItemIDs) {
|
|
|
|
|
g.GroupHidden = 0
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-10 11:33:59 +08:00
|
|
|
|
if firstInit || changeGroupField {
|
2025-08-09 11:24:44 +08:00
|
|
|
|
if groupKey := view.GetGroupKey(attrView); nil != groupKey && (av.KeyTypeSelect == groupKey.Type || av.KeyTypeMSelect == groupKey.Type) {
|
|
|
|
|
// 首次设置分组时,如果分组字段是单选或多选类型,则将分组方式改为手动排序,并按选项顺序排序分组视图 https://github.com/siyuan-note/siyuan/issues/15491
|
|
|
|
|
view.Group.Order = av.GroupOrderMan
|
2025-08-09 21:15:59 +08:00
|
|
|
|
sortGroupsBySelectOption(view, groupKey)
|
2025-08-09 13:28:56 +08:00
|
|
|
|
for i, g := range view.Groups {
|
|
|
|
|
g.GroupSort = i
|
|
|
|
|
}
|
2025-08-09 11:24:44 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-25 17:57:51 +08:00
|
|
|
|
err = av.SaveAttributeView(attrView)
|
2025-07-27 20:14:18 +08:00
|
|
|
|
ReloadAttrView(avID)
|
2025-07-25 11:56:32 +08:00
|
|
|
|
return
|
2025-07-01 17:56:32 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-06-17 17:46:32 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewCardAspectRatio(operation *Operation) (ret *TxErr) {
|
2025-06-17 20:15:42 +08:00
|
|
|
|
err := setAttrViewCardAspectRatio(operation)
|
2025-06-17 17:46:32 +08:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2025-06-17 17:46:32 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func setAttrViewCardAspectRatio(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, operation.BlockID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch view.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
return
|
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
view.Gallery.CardAspectRatio = av.CardAspectRatio(operation.Data.(float64))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-16 10:40:56 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewBlockView(operation *Operation) (ret *TxErr) {
|
2025-06-16 11:43:09 +08:00
|
|
|
|
err := SetDatabaseBlockView(operation.BlockID, operation.AvID, operation.ID)
|
2025-06-16 10:40:56 +08:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2025-06-16 10:40:56 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-10 12:23:07 +08:00
|
|
|
|
func (tx *Transaction) doChangeAttrViewLayout(operation *Operation) (ret *TxErr) {
|
2025-06-16 21:07:45 +08:00
|
|
|
|
err := ChangeAttrViewLayout(operation.BlockID, operation.AvID, operation.Layout)
|
2025-06-10 12:23:07 +08:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2025-06-10 12:23:07 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-16 21:07:45 +08:00
|
|
|
|
func ChangeAttrViewLayout(blockID, avID string, layout av.LayoutType) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
2025-06-10 12:23:07 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-16 21:07:45 +08:00
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, blockID)
|
2025-06-10 12:23:07 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-16 21:07:45 +08:00
|
|
|
|
newLayout := layout
|
2025-06-10 12:23:07 +08:00
|
|
|
|
if newLayout == view.LayoutType {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch newLayout {
|
|
|
|
|
case av.LayoutTypeTable:
|
2025-06-11 16:03:37 +08:00
|
|
|
|
if view.Name == av.GetAttributeViewI18n("gallery") {
|
|
|
|
|
view.Name = av.GetAttributeViewI18n("table")
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-10 12:23:07 +08:00
|
|
|
|
if nil != view.Table {
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view.Table = av.NewLayoutTable()
|
|
|
|
|
switch view.LayoutType {
|
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
for _, field := range view.Gallery.CardFields {
|
2025-06-30 15:06:43 +08:00
|
|
|
|
view.Table.Columns = append(view.Table.Columns, &av.ViewTableColumn{BaseField: &av.BaseField{ID: field.ID}})
|
2025-06-10 12:23:07 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
case av.LayoutTypeGallery:
|
2025-06-11 16:03:37 +08:00
|
|
|
|
if view.Name == av.GetAttributeViewI18n("table") {
|
|
|
|
|
view.Name = av.GetAttributeViewI18n("gallery")
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-10 12:23:07 +08:00
|
|
|
|
if nil != view.Gallery {
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view.Gallery = av.NewLayoutGallery()
|
|
|
|
|
switch view.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
for _, col := range view.Table.Columns {
|
2025-06-30 15:06:43 +08:00
|
|
|
|
view.Gallery.CardFields = append(view.Gallery.CardFields, &av.ViewGalleryCardField{BaseField: &av.BaseField{ID: col.ID}})
|
2025-06-10 12:23:07 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-11 15:32:12 +08:00
|
|
|
|
view.LayoutType = newLayout
|
2025-06-17 11:25:28 +08:00
|
|
|
|
|
2025-06-28 10:22:59 +08:00
|
|
|
|
blockIDs := treenode.GetMirrorAttrViewBlockIDs(avID)
|
2025-06-28 11:04:54 +08:00
|
|
|
|
for _, bID := range blockIDs {
|
|
|
|
|
node, tree, _ := getNodeByBlockID(nil, bID)
|
|
|
|
|
if nil == node || nil == tree {
|
|
|
|
|
logging.LogErrorf("get node by block ID [%s] failed", bID)
|
2025-06-28 10:22:59 +08:00
|
|
|
|
continue
|
|
|
|
|
}
|
2025-06-17 11:25:28 +08:00
|
|
|
|
|
2025-07-08 17:53:13 +08:00
|
|
|
|
changed := false
|
2025-06-28 10:22:59 +08:00
|
|
|
|
attrs := parse.IAL2Map(node.KramdownIAL)
|
2025-07-08 17:53:13 +08:00
|
|
|
|
if blockID == bID { // 当前操作的镜像库
|
|
|
|
|
attrs[av.NodeAttrView] = view.ID
|
|
|
|
|
node.AttributeViewType = string(view.LayoutType)
|
2025-07-11 10:23:13 +08:00
|
|
|
|
attrView.ViewID = view.ID
|
2025-07-08 17:53:13 +08:00
|
|
|
|
changed = true
|
|
|
|
|
} else {
|
|
|
|
|
if view.ID == attrs[av.NodeAttrView] {
|
|
|
|
|
// 仅更新和当前操作的镜像库指定的视图相同的镜像库
|
|
|
|
|
node.AttributeViewType = string(view.LayoutType)
|
|
|
|
|
changed = true
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if changed {
|
|
|
|
|
err = setNodeAttrs(node, tree, attrs)
|
|
|
|
|
if err != nil {
|
|
|
|
|
logging.LogWarnf("set node [%s] attrs failed: %s", bID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
2025-06-28 10:22:59 +08:00
|
|
|
|
}
|
2025-06-17 11:25:28 +08:00
|
|
|
|
}
|
2025-06-17 11:31:06 +08:00
|
|
|
|
|
2025-08-09 20:39:12 +08:00
|
|
|
|
regenAttrViewGroups(attrView, "force")
|
2025-08-05 17:32:37 +08:00
|
|
|
|
|
2025-06-28 11:04:54 +08:00
|
|
|
|
if err = av.SaveAttributeView(attrView); nil != err {
|
|
|
|
|
logging.LogErrorf("save attribute view [%s] failed: %s", avID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-17 11:31:06 +08:00
|
|
|
|
ReloadAttrView(avID)
|
2025-06-10 12:23:07 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-09 22:34:07 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewWrapField(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := setAttrViewWrapField(operation)
|
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2025-06-09 22:34:07 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func setAttrViewWrapField(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, operation.BlockID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-30 12:50:50 +08:00
|
|
|
|
allFieldWrap := operation.Data.(bool)
|
2025-06-09 22:34:07 +08:00
|
|
|
|
switch view.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
2025-06-30 12:50:50 +08:00
|
|
|
|
view.Table.WrapField = allFieldWrap
|
|
|
|
|
for _, col := range view.Table.Columns {
|
|
|
|
|
col.Wrap = allFieldWrap
|
|
|
|
|
}
|
2025-06-09 22:34:07 +08:00
|
|
|
|
case av.LayoutTypeGallery:
|
2025-06-30 12:50:50 +08:00
|
|
|
|
view.Gallery.WrapField = allFieldWrap
|
|
|
|
|
for _, field := range view.Gallery.CardFields {
|
|
|
|
|
field.Wrap = allFieldWrap
|
|
|
|
|
}
|
2025-06-09 22:34:07 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-09 22:33:25 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewShowIcon(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := setAttrViewShowIcon(operation)
|
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2025-06-09 22:33:25 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func setAttrViewShowIcon(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, operation.BlockID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch view.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
2025-06-30 12:50:50 +08:00
|
|
|
|
view.Table.ShowIcon = operation.Data.(bool)
|
2025-06-09 22:33:25 +08:00
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
view.Gallery.ShowIcon = operation.Data.(bool)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-09 22:31:29 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewFitImage(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := setAttrViewFitImage(operation)
|
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2025-06-09 22:31:29 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func setAttrViewFitImage(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, operation.BlockID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch view.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
return
|
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
view.Gallery.FitImage = operation.Data.(bool)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-30 10:40:04 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewDisplayFieldName(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := setAttrViewDisplayFieldName(operation)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func setAttrViewDisplayFieldName(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, operation.BlockID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch view.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
return
|
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
view.Gallery.DisplayFieldName = operation.Data.(bool)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-09 22:30:00 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewCardSize(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := setAttrViewCardSize(operation)
|
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2025-06-09 22:30:00 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func setAttrViewCardSize(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, operation.BlockID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch view.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
return
|
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
view.Gallery.CardSize = av.CardSize(operation.Data.(float64))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-09 22:27:51 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewCoverFromAssetKeyID(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := setAttrViewCoverFromAssetKeyID(operation)
|
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2025-06-09 22:27:51 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func setAttrViewCoverFromAssetKeyID(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, operation.BlockID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch view.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
return
|
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
view.Gallery.CoverFromAssetKeyID = operation.KeyID
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-09 22:26:08 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewCoverFrom(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := setAttrViewCoverFrom(operation)
|
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2025-06-09 22:26:08 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func setAttrViewCoverFrom(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, operation.BlockID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch view.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
return
|
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
view.Gallery.CoverFrom = av.CoverFrom(operation.Data.(float64))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-05-31 23:55:17 +08:00
|
|
|
|
func AppendAttributeViewDetachedBlocksWithValues(avID string, blocksValues [][]*av.Value) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2024-05-31 23:55:17 +08:00
|
|
|
|
logging.LogErrorf("parse attribute view [%s] failed: %s", avID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
now := util.CurrentTimeMillis()
|
|
|
|
|
var blockIDs []string
|
|
|
|
|
for _, blockValues := range blocksValues {
|
|
|
|
|
blockID := ast.NewNodeID()
|
|
|
|
|
blockIDs = append(blockIDs, blockID)
|
|
|
|
|
for _, v := range blockValues {
|
|
|
|
|
keyValues, _ := attrView.GetKeyValues(v.KeyID)
|
|
|
|
|
if nil == keyValues {
|
|
|
|
|
err = fmt.Errorf("key [%s] not found", v.KeyID)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
v.ID = ast.NewNodeID()
|
|
|
|
|
v.BlockID = blockID
|
|
|
|
|
v.Type = keyValues.Key.Type
|
|
|
|
|
if av.KeyTypeBlock == v.Type {
|
|
|
|
|
v.Block.Created = now
|
|
|
|
|
v.Block.Updated = now
|
2025-08-10 12:16:10 +08:00
|
|
|
|
v.Block.ID = ""
|
2024-05-31 23:55:17 +08:00
|
|
|
|
}
|
|
|
|
|
v.IsDetached = true
|
|
|
|
|
v.CreatedAt = now
|
|
|
|
|
v.UpdatedAt = now
|
|
|
|
|
|
|
|
|
|
keyValues.Values = append(keyValues.Values, v)
|
2024-09-14 17:53:46 +08:00
|
|
|
|
|
|
|
|
|
if av.KeyTypeSelect == v.Type || av.KeyTypeMSelect == v.Type {
|
|
|
|
|
// 保存选项 https://github.com/siyuan-note/siyuan/issues/12475
|
|
|
|
|
key, _ := attrView.GetKey(v.KeyID)
|
|
|
|
|
if nil != key && 0 < len(v.MSelect) {
|
|
|
|
|
for _, valOpt := range v.MSelect {
|
|
|
|
|
if opt := key.GetOption(valOpt.Content); nil == opt {
|
|
|
|
|
// 不存在的选项新建保存
|
|
|
|
|
opt = &av.SelectOption{Name: valOpt.Content, Color: valOpt.Color}
|
|
|
|
|
key.Options = append(key.Options, opt)
|
|
|
|
|
} else {
|
|
|
|
|
// 已经存在的选项颜色需要保持不变
|
|
|
|
|
valOpt.Color = opt.Color
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-05-31 23:55:17 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, v := range attrView.Views {
|
2025-07-02 17:02:40 +08:00
|
|
|
|
for _, addingBlockID := range blockIDs {
|
|
|
|
|
v.ItemIDs = append(v.ItemIDs, addingBlockID)
|
2024-05-31 23:55:17 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = av.SaveAttributeView(attrView); err != nil {
|
2024-05-31 23:55:17 +08:00
|
|
|
|
logging.LogErrorf("save attribute view [%s] failed: %s", avID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-09-05 12:10:42 +08:00
|
|
|
|
ReloadAttrView(avID)
|
2024-05-31 23:55:17 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-05-23 18:00:35 +08:00
|
|
|
|
func DuplicateDatabaseBlock(avID string) (newAvID, newBlockID string, err error) {
|
|
|
|
|
storageAvDir := filepath.Join(util.DataDir, "storage", "av")
|
|
|
|
|
oldAvPath := filepath.Join(storageAvDir, avID+".json")
|
|
|
|
|
newAvID, newBlockID = ast.NewNodeID(), ast.NewNodeID()
|
|
|
|
|
|
|
|
|
|
oldAv, err := av.ParseAttributeView(avID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2024-05-23 18:00:35 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
data, err := filelock.ReadFile(oldAvPath)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2024-05-23 18:00:35 +08:00
|
|
|
|
logging.LogErrorf("read attribute view [%s] failed: %s", avID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
data = bytes.ReplaceAll(data, []byte(avID), []byte(newAvID))
|
|
|
|
|
av.UpsertBlockRel(newAvID, newBlockID)
|
|
|
|
|
|
|
|
|
|
newAv := &av.AttributeView{}
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = gulu.JSON.UnmarshalJSON(data, newAv); err != nil {
|
2024-05-23 18:00:35 +08:00
|
|
|
|
logging.LogErrorf("unmarshal attribute view [%s] failed: %s", newAvID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
newAv.Name = oldAv.Name + " (Duplicated " + time.Now().Format("2006-01-02 15:04:05") + ")"
|
2024-05-27 16:52:13 +08:00
|
|
|
|
|
|
|
|
|
for _, keyValues := range newAv.KeyValues {
|
|
|
|
|
if nil != keyValues.Key.Relation && keyValues.Key.Relation.IsTwoWay {
|
|
|
|
|
// 断开双向关联
|
|
|
|
|
keyValues.Key.Relation.IsTwoWay = false
|
|
|
|
|
keyValues.Key.Relation.BackKeyID = ""
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-05-23 18:00:35 +08:00
|
|
|
|
data, err = gulu.JSON.MarshalJSON(newAv)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2024-05-23 18:00:35 +08:00
|
|
|
|
logging.LogErrorf("marshal attribute view [%s] failed: %s", newAvID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
newAvPath := filepath.Join(storageAvDir, newAvID+".json")
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = filelock.WriteFile(newAvPath, data); err != nil {
|
2024-05-23 18:00:35 +08:00
|
|
|
|
logging.LogErrorf("write attribute view [%s] failed: %s", newAvID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
2024-05-23 23:57:10 +08:00
|
|
|
|
|
|
|
|
|
updateBoundBlockAvsAttribute([]string{newAvID})
|
2024-05-23 18:00:35 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-05-17 16:19:56 +08:00
|
|
|
|
func GetAttributeViewKeysByAvID(avID string) (ret []*av.Key) {
|
|
|
|
|
ret = []*av.Key{}
|
2024-05-17 09:49:10 +08:00
|
|
|
|
|
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2024-05-17 09:49:10 +08:00
|
|
|
|
logging.LogErrorf("parse attribute view [%s] failed: %s", avID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-05-17 16:19:56 +08:00
|
|
|
|
for _, keyValues := range attrView.KeyValues {
|
|
|
|
|
key := keyValues.Key
|
|
|
|
|
ret = append(ret, key)
|
2024-05-17 09:49:10 +08:00
|
|
|
|
}
|
|
|
|
|
return ret
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-09 18:06:16 +08:00
|
|
|
|
func SetDatabaseBlockView(blockID, avID, viewID string) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
|
|
|
|
if nil != err {
|
|
|
|
|
logging.LogErrorf("parse attribute view [%s] failed: %s", avID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
if attrView.ViewID != viewID {
|
|
|
|
|
attrView.ViewID = viewID
|
|
|
|
|
if err = av.SaveAttributeView(attrView); err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view := attrView.GetView(viewID)
|
|
|
|
|
if nil == view {
|
|
|
|
|
err = av.ErrViewNotFound
|
|
|
|
|
logging.LogErrorf("view [%s] not found in attribute view [%s]", viewID, avID)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-04 11:51:03 +08:00
|
|
|
|
node, tree, err := getNodeByBlockID(nil, blockID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2024-03-04 11:51:03 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-09 18:06:16 +08:00
|
|
|
|
node.AttributeViewType = string(view.LayoutType)
|
2024-03-04 11:51:03 +08:00
|
|
|
|
attrs := parse.IAL2Map(node.KramdownIAL)
|
|
|
|
|
attrs[av.NodeAttrView] = viewID
|
|
|
|
|
err = setNodeAttrs(node, tree, attrs)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2024-03-04 11:51:03 +08:00
|
|
|
|
logging.LogWarnf("set node [%s] attrs failed: %s", blockID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-07 09:54:55 +08:00
|
|
|
|
func GetAttributeViewPrimaryKeyValues(avID, keyword string, page, pageSize int) (attributeViewName string, databaseBlockIDs []string, keyValues *av.KeyValues, err error) {
|
2024-02-23 22:22:54 +08:00
|
|
|
|
waitForSyncingStorages()
|
|
|
|
|
|
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2024-02-23 22:22:54 +08:00
|
|
|
|
logging.LogErrorf("parse attribute view [%s] failed: %s", avID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
2024-04-18 11:29:09 +08:00
|
|
|
|
attributeViewName = getAttrViewName(attrView)
|
2024-02-23 22:55:35 +08:00
|
|
|
|
|
2024-04-07 09:54:55 +08:00
|
|
|
|
databaseBlockIDs = treenode.GetMirrorAttrViewBlockIDs(avID)
|
|
|
|
|
|
2024-02-23 22:55:35 +08:00
|
|
|
|
keyValues = attrView.GetBlockKeyValues()
|
2024-10-19 12:25:30 +08:00
|
|
|
|
var values []*av.Value
|
2024-02-23 22:55:35 +08:00
|
|
|
|
for _, kv := range keyValues.Values {
|
2024-10-19 12:25:30 +08:00
|
|
|
|
if !kv.IsDetached && !treenode.ExistBlockTree(kv.BlockID) {
|
|
|
|
|
continue
|
2024-02-23 22:55:35 +08:00
|
|
|
|
}
|
2024-10-19 12:25:30 +08:00
|
|
|
|
|
|
|
|
|
if strings.Contains(strings.ToLower(kv.String(true)), strings.ToLower(keyword)) {
|
|
|
|
|
values = append(values, kv)
|
2024-03-06 09:40:04 +08:00
|
|
|
|
}
|
2024-02-23 22:55:35 +08:00
|
|
|
|
}
|
2024-10-19 12:25:30 +08:00
|
|
|
|
keyValues.Values = values
|
2024-02-23 22:22:54 +08:00
|
|
|
|
|
|
|
|
|
if 1 > pageSize {
|
2024-04-10 17:28:30 +08:00
|
|
|
|
pageSize = 16
|
2024-02-23 22:22:54 +08:00
|
|
|
|
}
|
|
|
|
|
start := (page - 1) * pageSize
|
|
|
|
|
end := start + pageSize
|
2024-02-23 22:55:35 +08:00
|
|
|
|
if len(keyValues.Values) < end {
|
|
|
|
|
end = len(keyValues.Values)
|
2024-02-23 22:22:54 +08:00
|
|
|
|
}
|
2024-02-23 22:55:35 +08:00
|
|
|
|
keyValues.Values = keyValues.Values[start:end]
|
2024-05-12 11:05:57 +08:00
|
|
|
|
|
|
|
|
|
sort.Slice(keyValues.Values, func(i, j int) bool {
|
|
|
|
|
return keyValues.Values[i].Block.Updated > keyValues.Values[j].Block.Updated
|
|
|
|
|
})
|
2024-02-23 22:22:54 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-06 00:00:43 +08:00
|
|
|
|
func GetAttributeViewFilterSort(avID, blockID string) (filters []*av.ViewFilter, sorts []*av.ViewSort) {
|
2024-01-07 22:53:05 +08:00
|
|
|
|
waitForSyncingStorages()
|
|
|
|
|
|
2024-03-06 00:00:43 +08:00
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2024-03-06 00:00:43 +08:00
|
|
|
|
logging.LogErrorf("parse attribute view [%s] failed: %s", avID, err)
|
2024-01-07 22:53:05 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-06 00:00:43 +08:00
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, blockID)
|
|
|
|
|
if nil == view {
|
|
|
|
|
view, err = attrView.GetCurrentView(attrView.ViewID)
|
2025-06-09 16:46:50 +08:00
|
|
|
|
if nil == view || err != nil {
|
2024-03-06 00:00:43 +08:00
|
|
|
|
logging.LogErrorf("get current view failed: %s", err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-29 11:11:20 +08:00
|
|
|
|
filters = view.Filters
|
|
|
|
|
sorts = view.Sorts
|
2025-06-30 17:02:48 +08:00
|
|
|
|
if 1 > len(filters) {
|
|
|
|
|
filters = []*av.ViewFilter{}
|
|
|
|
|
}
|
|
|
|
|
if 1 > len(sorts) {
|
|
|
|
|
sorts = []*av.ViewSort{}
|
|
|
|
|
}
|
2024-01-07 22:53:05 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-29 09:54:54 +08:00
|
|
|
|
func SearchAttributeViewNonRelationKey(avID, keyword string) (ret []*av.Key) {
|
|
|
|
|
waitForSyncingStorages()
|
|
|
|
|
|
|
|
|
|
ret = []*av.Key{}
|
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-12-29 09:54:54 +08:00
|
|
|
|
logging.LogErrorf("parse attribute view [%s] failed: %s", avID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, keyValues := range attrView.KeyValues {
|
2025-08-06 11:08:26 +08:00
|
|
|
|
if av.KeyTypeRelation != keyValues.Key.Type && av.KeyTypeRollup != keyValues.Key.Type && av.KeyTypeCreated != keyValues.Key.Type && av.KeyTypeUpdated != keyValues.Key.Type && av.KeyTypeLineNumber != keyValues.Key.Type {
|
2023-12-29 09:54:54 +08:00
|
|
|
|
if strings.Contains(strings.ToLower(keyValues.Key.Name), strings.ToLower(keyword)) {
|
|
|
|
|
ret = append(ret, keyValues.Key)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func SearchAttributeViewRelationKey(avID, keyword string) (ret []*av.Key) {
|
|
|
|
|
waitForSyncingStorages()
|
|
|
|
|
|
|
|
|
|
ret = []*av.Key{}
|
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-12-29 09:54:54 +08:00
|
|
|
|
logging.LogErrorf("parse attribute view [%s] failed: %s", avID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, keyValues := range attrView.KeyValues {
|
|
|
|
|
if av.KeyTypeRelation == keyValues.Key.Type && nil != keyValues.Key.Relation {
|
|
|
|
|
if strings.Contains(strings.ToLower(keyValues.Key.Name), strings.ToLower(keyword)) {
|
|
|
|
|
ret = append(ret, keyValues.Key)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-23 17:17:47 +08:00
|
|
|
|
func GetAttributeView(avID string) (ret *av.AttributeView) {
|
|
|
|
|
waitForSyncingStorages()
|
|
|
|
|
|
|
|
|
|
ret, _ = av.ParseAttributeView(avID)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-10 23:50:23 +08:00
|
|
|
|
type AvSearchResult struct {
|
|
|
|
|
AvID string `json:"avID"`
|
|
|
|
|
AvName string `json:"avName"`
|
|
|
|
|
BlockID string `json:"blockID"`
|
|
|
|
|
HPath string `json:"hPath"`
|
|
|
|
|
Children []*AvSearchResult `json:"children,omitempty"`
|
2023-12-23 12:04:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-08-10 23:50:23 +08:00
|
|
|
|
type AvSearchTempResult struct {
|
|
|
|
|
AvID string
|
|
|
|
|
AvName string
|
|
|
|
|
AvUpdated int64
|
|
|
|
|
Score float64
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func SearchAttributeView(keyword string, excludeAvIDs []string) (ret []*AvSearchResult) {
|
2023-12-23 12:04:54 +08:00
|
|
|
|
waitForSyncingStorages()
|
2024-04-20 23:10:05 +08:00
|
|
|
|
|
2025-08-10 23:50:23 +08:00
|
|
|
|
ret = []*AvSearchResult{}
|
2023-12-23 20:51:10 +08:00
|
|
|
|
keyword = strings.TrimSpace(keyword)
|
2024-06-05 23:25:16 +08:00
|
|
|
|
keywords := strings.Fields(keyword)
|
2024-04-08 21:58:57 +08:00
|
|
|
|
|
2025-08-10 23:50:23 +08:00
|
|
|
|
var avs []*AvSearchTempResult
|
2024-04-08 21:58:57 +08:00
|
|
|
|
avDir := filepath.Join(util.DataDir, "storage", "av")
|
|
|
|
|
entries, err := os.ReadDir(avDir)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2024-04-08 21:58:57 +08:00
|
|
|
|
logging.LogErrorf("read directory [%s] failed: %s", avDir, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
2024-04-09 22:26:37 +08:00
|
|
|
|
avBlockRels := av.GetBlockRels()
|
2024-04-08 21:58:57 +08:00
|
|
|
|
for _, entry := range entries {
|
|
|
|
|
if entry.IsDir() {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
id := strings.TrimSuffix(entry.Name(), ".json")
|
|
|
|
|
if !ast.IsNodeIDPattern(id) {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-09 22:26:37 +08:00
|
|
|
|
if nil == avBlockRels[id] {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-08 21:58:57 +08:00
|
|
|
|
name, _ := av.GetAttributeViewNameByPath(filepath.Join(avDir, entry.Name()))
|
2024-04-09 21:26:41 +08:00
|
|
|
|
info, _ := entry.Info()
|
|
|
|
|
if "" != keyword {
|
2024-06-05 23:25:16 +08:00
|
|
|
|
score := 0.0
|
|
|
|
|
hit := false
|
|
|
|
|
for _, k := range keywords {
|
|
|
|
|
if strings.Contains(strings.ToLower(name), strings.ToLower(k)) {
|
|
|
|
|
score += smetrics.JaroWinkler(name, k, 0.7, 4)
|
|
|
|
|
hit = true
|
|
|
|
|
} else {
|
|
|
|
|
hit = false
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if hit {
|
2025-08-10 23:50:23 +08:00
|
|
|
|
a := &AvSearchTempResult{AvID: id, AvName: name, Score: score}
|
2024-04-09 21:26:41 +08:00
|
|
|
|
if nil != info && !info.ModTime().IsZero() {
|
|
|
|
|
a.AvUpdated = info.ModTime().UnixMilli()
|
|
|
|
|
}
|
|
|
|
|
avs = append(avs, a)
|
|
|
|
|
}
|
|
|
|
|
} else {
|
2025-08-10 23:50:23 +08:00
|
|
|
|
a := &AvSearchTempResult{AvID: id, AvName: name}
|
2024-04-09 21:26:41 +08:00
|
|
|
|
if nil != info && !info.ModTime().IsZero() {
|
|
|
|
|
a.AvUpdated = info.ModTime().UnixMilli()
|
2024-04-08 21:58:57 +08:00
|
|
|
|
}
|
2024-04-09 21:26:41 +08:00
|
|
|
|
avs = append(avs, a)
|
2024-04-08 21:58:57 +08:00
|
|
|
|
}
|
2024-04-09 17:51:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
2024-04-09 21:26:41 +08:00
|
|
|
|
if "" == keyword {
|
|
|
|
|
sort.Slice(avs, func(i, j int) bool { return avs[i].AvUpdated > avs[j].AvUpdated })
|
|
|
|
|
} else {
|
2024-04-09 22:26:37 +08:00
|
|
|
|
sort.SliceStable(avs, func(i, j int) bool {
|
|
|
|
|
if avs[i].Score == avs[j].Score {
|
|
|
|
|
return avs[i].AvUpdated > avs[j].AvUpdated
|
|
|
|
|
}
|
|
|
|
|
return avs[i].Score > avs[j].Score
|
|
|
|
|
})
|
2024-04-09 21:26:41 +08:00
|
|
|
|
}
|
2024-04-09 22:26:37 +08:00
|
|
|
|
if 12 <= len(avs) {
|
|
|
|
|
avs = avs[:12]
|
2024-04-09 21:26:41 +08:00
|
|
|
|
}
|
2024-04-08 21:58:57 +08:00
|
|
|
|
var avIDs []string
|
2024-04-09 21:26:41 +08:00
|
|
|
|
for _, a := range avs {
|
|
|
|
|
avIDs = append(avIDs, a.AvID)
|
2024-04-08 21:58:57 +08:00
|
|
|
|
}
|
2024-04-09 21:26:41 +08:00
|
|
|
|
|
2024-04-09 22:26:37 +08:00
|
|
|
|
avBlocks := treenode.BatchGetMirrorAttrViewBlocks(avIDs)
|
|
|
|
|
var blockIDs []string
|
|
|
|
|
for _, avBlock := range avBlocks {
|
|
|
|
|
blockIDs = append(blockIDs, avBlock.BlockIDs...)
|
|
|
|
|
}
|
|
|
|
|
blockIDs = gulu.Str.RemoveDuplicatedElem(blockIDs)
|
|
|
|
|
|
2024-06-21 23:37:47 +08:00
|
|
|
|
trees := filesys.LoadTrees(blockIDs)
|
2025-08-10 23:50:23 +08:00
|
|
|
|
for _, avBlock := range avBlocks {
|
|
|
|
|
parentResult := buildSearchAttributeViewResult(avs, avBlock.BlockIDs[0], trees, excludeAvIDs)
|
|
|
|
|
ret = append(ret, parentResult)
|
|
|
|
|
if 1 < len(avBlock.BlockIDs) {
|
|
|
|
|
parentResult.BlockID = "" // 置空表示该结果为父级
|
|
|
|
|
for _, blockID := range avBlock.BlockIDs {
|
|
|
|
|
result := buildSearchAttributeViewResult(avs, blockID, trees, excludeAvIDs)
|
|
|
|
|
if nil != result {
|
|
|
|
|
parentResult.Children = append(parentResult.Children, result)
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-12-23 12:04:54 +08:00
|
|
|
|
}
|
2025-08-10 23:50:23 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
2023-12-23 12:04:54 +08:00
|
|
|
|
|
2025-08-10 23:50:23 +08:00
|
|
|
|
func buildSearchAttributeViewResult(avSearchResults []*AvSearchTempResult, blockID string, trees map[string]*parse.Tree, excludeAvIDs []string) (ret *AvSearchResult) {
|
|
|
|
|
tree := trees[blockID]
|
|
|
|
|
if nil == tree {
|
|
|
|
|
return
|
|
|
|
|
}
|
2023-12-23 12:04:54 +08:00
|
|
|
|
|
2025-08-10 23:50:23 +08:00
|
|
|
|
node := treenode.GetNodeInTree(tree, blockID)
|
|
|
|
|
if nil == node {
|
|
|
|
|
return
|
|
|
|
|
}
|
2023-12-23 12:04:54 +08:00
|
|
|
|
|
2025-08-10 23:50:23 +08:00
|
|
|
|
if "" == node.AttributeViewID {
|
|
|
|
|
return
|
|
|
|
|
}
|
2023-12-23 12:04:54 +08:00
|
|
|
|
|
2025-08-10 23:50:23 +08:00
|
|
|
|
avID := node.AttributeViewID
|
|
|
|
|
var existAv *AvSearchTempResult
|
|
|
|
|
for _, av := range avSearchResults {
|
|
|
|
|
if av.AvID == avID {
|
|
|
|
|
existAv = av
|
|
|
|
|
break
|
2023-12-23 21:05:52 +08:00
|
|
|
|
}
|
2025-08-10 23:50:23 +08:00
|
|
|
|
}
|
|
|
|
|
if nil == existAv {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var hPath string
|
|
|
|
|
baseBlock := treenode.GetBlockTreeRootByPath(node.Box, node.Path)
|
|
|
|
|
if nil != baseBlock {
|
|
|
|
|
hPath = baseBlock.HPath
|
|
|
|
|
}
|
|
|
|
|
box := Conf.Box(node.Box)
|
|
|
|
|
if nil != box {
|
|
|
|
|
hPath = box.Name + hPath
|
|
|
|
|
}
|
2023-12-23 21:02:31 +08:00
|
|
|
|
|
2025-08-10 23:50:23 +08:00
|
|
|
|
if !gulu.Str.Contains(avID, excludeAvIDs) {
|
|
|
|
|
ret = &AvSearchResult{
|
|
|
|
|
AvID: avID,
|
|
|
|
|
AvName: existAv.AvName,
|
|
|
|
|
BlockID: blockID,
|
|
|
|
|
HPath: hPath,
|
2023-12-23 12:04:54 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-30 20:35:03 +08:00
|
|
|
|
type BlockAttributeViewKeys struct {
|
|
|
|
|
AvID string `json:"avID"`
|
|
|
|
|
AvName string `json:"avName"`
|
2023-11-02 16:24:50 +08:00
|
|
|
|
BlockIDs []string `json:"blockIDs"`
|
2023-07-30 20:35:03 +08:00
|
|
|
|
KeyValues []*av.KeyValues `json:"keyValues"`
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-10 18:20:29 +08:00
|
|
|
|
func GetBlockAttributeViewKeys(nodeID string) (ret []*BlockAttributeViewKeys) {
|
2023-07-30 20:35:03 +08:00
|
|
|
|
waitForSyncingStorages()
|
|
|
|
|
|
|
|
|
|
ret = []*BlockAttributeViewKeys{}
|
2025-08-10 18:20:29 +08:00
|
|
|
|
attrs := sql.GetBlockAttrs(nodeID)
|
2023-10-05 12:02:17 +08:00
|
|
|
|
avs := attrs[av.NodeAttrNameAvs]
|
2023-07-30 20:35:03 +08:00
|
|
|
|
if "" == avs {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-10-20 10:35:49 +08:00
|
|
|
|
attrViewCache := map[string]*av.AttributeView{}
|
2023-07-30 20:35:03 +08:00
|
|
|
|
avIDs := strings.Split(avs, ",")
|
|
|
|
|
for _, avID := range avIDs {
|
2024-10-20 10:35:49 +08:00
|
|
|
|
attrView := attrViewCache[avID]
|
|
|
|
|
if nil == attrView {
|
|
|
|
|
attrView, _ = av.ParseAttributeView(avID)
|
|
|
|
|
if nil == attrView {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
attrViewCache[avID] = attrView
|
2023-07-30 20:35:03 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-08-10 18:20:29 +08:00
|
|
|
|
if !attrView.ExistBoundBlock(nodeID) {
|
2024-04-03 21:17:54 +08:00
|
|
|
|
// 比如剪切后粘贴,块 ID 会变,但是属性还在块上,这里做一次数据订正
|
|
|
|
|
// Auto verify the database name when clicking the block superscript icon https://github.com/siyuan-note/siyuan/issues/10861
|
2025-08-10 18:20:29 +08:00
|
|
|
|
unbindBlockAv(nil, avID, nodeID)
|
2024-04-03 21:17:54 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-10 18:20:29 +08:00
|
|
|
|
blockVal := attrView.GetBlockValueByBoundID(nodeID)
|
|
|
|
|
if nil == blockVal {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
itemID := blockVal.BlockID
|
2023-07-30 20:35:03 +08:00
|
|
|
|
var keyValues []*av.KeyValues
|
|
|
|
|
for _, kv := range attrView.KeyValues {
|
2024-05-09 11:26:23 +08:00
|
|
|
|
if av.KeyTypeLineNumber == kv.Key.Type {
|
|
|
|
|
// 属性面板中不显示行号字段
|
|
|
|
|
// The line number field no longer appears in the database attribute panel https://github.com/siyuan-note/siyuan/issues/11319
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-30 20:35:03 +08:00
|
|
|
|
kValues := &av.KeyValues{Key: kv.Key}
|
|
|
|
|
for _, v := range kv.Values {
|
2025-08-10 18:20:29 +08:00
|
|
|
|
if v.BlockID == itemID {
|
2023-07-30 20:35:03 +08:00
|
|
|
|
kValues.Values = append(kValues.Values, v)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-09 12:14:11 +08:00
|
|
|
|
switch kValues.Key.Type {
|
2023-12-30 23:34:53 +08:00
|
|
|
|
case av.KeyTypeRollup:
|
2025-08-10 18:20:29 +08:00
|
|
|
|
kValues.Values = append(kValues.Values, &av.Value{ID: ast.NewNodeID(), KeyID: kValues.Key.ID, BlockID: itemID, Type: av.KeyTypeRollup, Rollup: &av.ValueRollup{Contents: []*av.Value{}}})
|
2023-10-09 12:14:11 +08:00
|
|
|
|
case av.KeyTypeTemplate:
|
2025-08-10 18:20:29 +08:00
|
|
|
|
kValues.Values = append(kValues.Values, &av.Value{ID: ast.NewNodeID(), KeyID: kValues.Key.ID, BlockID: itemID, Type: av.KeyTypeTemplate, Template: &av.ValueTemplate{Content: ""}})
|
2023-10-09 12:14:11 +08:00
|
|
|
|
case av.KeyTypeCreated:
|
2025-08-10 18:20:29 +08:00
|
|
|
|
kValues.Values = append(kValues.Values, &av.Value{ID: ast.NewNodeID(), KeyID: kValues.Key.ID, BlockID: itemID, Type: av.KeyTypeCreated})
|
2023-10-09 12:14:11 +08:00
|
|
|
|
case av.KeyTypeUpdated:
|
2025-08-10 18:20:29 +08:00
|
|
|
|
kValues.Values = append(kValues.Values, &av.Value{ID: ast.NewNodeID(), KeyID: kValues.Key.ID, BlockID: itemID, Type: av.KeyTypeUpdated})
|
2024-09-16 12:04:23 +08:00
|
|
|
|
case av.KeyTypeNumber:
|
|
|
|
|
for _, v := range kValues.Values {
|
|
|
|
|
if nil != v.Number {
|
|
|
|
|
v.Number.Format = kValues.Key.NumberFormat
|
|
|
|
|
v.Number.FormatNumber()
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-10-01 18:37:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-07-30 20:35:03 +08:00
|
|
|
|
if 0 < len(kValues.Values) {
|
2025-06-22 22:22:44 +08:00
|
|
|
|
for _, v := range kValues.Values {
|
|
|
|
|
sql.FillAttributeViewNilValue(v, v.Type)
|
|
|
|
|
}
|
2023-07-30 20:35:03 +08:00
|
|
|
|
keyValues = append(keyValues, kValues)
|
2024-01-01 21:48:23 +08:00
|
|
|
|
} else {
|
|
|
|
|
// 如果没有值,那么就补一个默认值
|
2025-08-10 18:20:29 +08:00
|
|
|
|
kValues.Values = append(kValues.Values, av.GetAttributeViewDefaultValue(ast.NewNodeID(), kv.Key.ID, itemID, kv.Key.Type))
|
2024-01-01 21:48:23 +08:00
|
|
|
|
keyValues = append(keyValues, kValues)
|
2023-07-30 20:35:03 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-03 11:44:06 +08:00
|
|
|
|
// 渲染自动生成的字段值,比如模板字段、关联字段、汇总字段、创建时间字段和更新时间字段
|
|
|
|
|
// 先处理关联字段、汇总字段、创建时间字段和更新时间字段
|
2023-10-03 11:46:25 +08:00
|
|
|
|
for _, kv := range keyValues {
|
2023-10-09 12:14:11 +08:00
|
|
|
|
switch kv.Key.Type {
|
2024-12-21 12:00:02 +08:00
|
|
|
|
case av.KeyTypeBlock: // 对于主键可能需要填充静态锚文本 Database-bound block primary key supports setting static anchor text https://github.com/siyuan-note/siyuan/issues/10049
|
|
|
|
|
if nil != kv.Values[0].Block {
|
2025-08-10 18:20:29 +08:00
|
|
|
|
ial := sql.GetBlockAttrs(nodeID)
|
2024-12-21 12:00:02 +08:00
|
|
|
|
if v := ial[av.NodeAttrViewStaticText+"-"+attrView.ID]; "" != v {
|
|
|
|
|
kv.Values[0].Block.Content = v
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-12-24 22:27:38 +08:00
|
|
|
|
case av.KeyTypeRollup:
|
|
|
|
|
if nil == kv.Key.Rollup {
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
relKey, _ := attrView.GetKey(kv.Key.Rollup.RelationKeyID)
|
|
|
|
|
if nil == relKey {
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
relVal := attrView.GetValue(kv.Key.Rollup.RelationKeyID, kv.Values[0].BlockID)
|
|
|
|
|
if nil != relVal && nil != relVal.Relation {
|
2024-10-20 10:35:49 +08:00
|
|
|
|
destAv := attrViewCache[relKey.Relation.AvID]
|
|
|
|
|
if nil == destAv {
|
|
|
|
|
destAv, _ = av.ParseAttributeView(relKey.Relation.AvID)
|
|
|
|
|
if nil == destAv {
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
attrViewCache[relKey.Relation.AvID] = destAv
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-01 16:27:01 +08:00
|
|
|
|
destKey, _ := destAv.GetKey(kv.Key.Rollup.KeyID)
|
2024-10-20 10:35:49 +08:00
|
|
|
|
if nil != destKey {
|
2023-12-30 23:34:53 +08:00
|
|
|
|
for _, bID := range relVal.Relation.BlockIDs {
|
|
|
|
|
destVal := destAv.GetValue(kv.Key.Rollup.KeyID, bID)
|
2024-01-01 14:32:32 +08:00
|
|
|
|
if nil == destVal {
|
2025-08-10 16:02:59 +08:00
|
|
|
|
if destAv.ExistItem(bID) { // 数据库中存在项目但是字段值不存在是数据未初始化,这里补一个默认值
|
2024-04-17 19:38:27 +08:00
|
|
|
|
destVal = av.GetAttributeViewDefaultValue(ast.NewNodeID(), kv.Key.Rollup.KeyID, bID, destKey.Type)
|
2024-04-03 09:37:42 +08:00
|
|
|
|
}
|
|
|
|
|
if nil == destVal {
|
|
|
|
|
continue
|
|
|
|
|
}
|
2023-12-30 23:34:53 +08:00
|
|
|
|
}
|
2024-01-01 16:08:46 +08:00
|
|
|
|
if av.KeyTypeNumber == destKey.Type {
|
|
|
|
|
destVal.Number.Format = destKey.NumberFormat
|
2024-01-01 14:32:32 +08:00
|
|
|
|
destVal.Number.FormatNumber()
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-01 15:14:52 +08:00
|
|
|
|
kv.Values[0].Rollup.Contents = append(kv.Values[0].Rollup.Contents, destVal.Clone())
|
2023-12-30 23:34:53 +08:00
|
|
|
|
}
|
2024-01-01 16:27:01 +08:00
|
|
|
|
kv.Values[0].Rollup.RenderContents(kv.Key.Rollup.Calc, destKey)
|
2023-12-23 21:47:01 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2023-12-24 22:27:38 +08:00
|
|
|
|
case av.KeyTypeRelation:
|
|
|
|
|
if nil == kv.Key.Relation {
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
|
2024-10-20 10:35:49 +08:00
|
|
|
|
destAv := attrViewCache[kv.Key.Relation.AvID]
|
2023-12-24 22:27:38 +08:00
|
|
|
|
if nil == destAv {
|
2024-10-20 10:35:49 +08:00
|
|
|
|
destAv, _ = av.ParseAttributeView(kv.Key.Relation.AvID)
|
|
|
|
|
if nil == destAv {
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
attrViewCache[kv.Key.Relation.AvID] = destAv
|
2023-12-24 22:27:38 +08:00
|
|
|
|
}
|
|
|
|
|
|
2024-03-07 16:11:58 +08:00
|
|
|
|
blocks := map[string]*av.Value{}
|
2023-12-24 22:27:38 +08:00
|
|
|
|
for _, blockValue := range destAv.GetBlockKeyValues().Values {
|
2024-03-07 16:11:58 +08:00
|
|
|
|
blocks[blockValue.BlockID] = blockValue
|
2023-12-24 22:27:38 +08:00
|
|
|
|
}
|
2024-03-20 22:30:21 +08:00
|
|
|
|
kv.Values[0].Relation.Contents = nil // 先清空 https://github.com/siyuan-note/siyuan/issues/10670
|
2023-12-24 22:27:38 +08:00
|
|
|
|
for _, bID := range kv.Values[0].Relation.BlockIDs {
|
|
|
|
|
kv.Values[0].Relation.Contents = append(kv.Values[0].Relation.Contents, blocks[bID])
|
|
|
|
|
}
|
2023-10-09 12:14:11 +08:00
|
|
|
|
case av.KeyTypeCreated:
|
2025-08-10 18:20:29 +08:00
|
|
|
|
createdStr := nodeID[:len("20060102150405")]
|
2023-10-09 16:10:15 +08:00
|
|
|
|
created, parseErr := time.ParseInLocation("20060102150405", createdStr, time.Local)
|
2023-10-09 12:14:11 +08:00
|
|
|
|
if nil == parseErr {
|
|
|
|
|
kv.Values[0].Created = av.NewFormattedValueCreated(created.UnixMilli(), 0, av.CreatedFormatNone)
|
2023-10-10 20:05:48 +08:00
|
|
|
|
kv.Values[0].Created.IsNotEmpty = true
|
2023-10-09 12:14:11 +08:00
|
|
|
|
} else {
|
|
|
|
|
logging.LogWarnf("parse created [%s] failed: %s", createdStr, parseErr)
|
|
|
|
|
kv.Values[0].Created = av.NewFormattedValueCreated(time.Now().UnixMilli(), 0, av.CreatedFormatNone)
|
|
|
|
|
}
|
|
|
|
|
case av.KeyTypeUpdated:
|
2025-08-10 18:20:29 +08:00
|
|
|
|
ial := sql.GetBlockAttrs(nodeID)
|
2023-10-09 12:14:11 +08:00
|
|
|
|
updatedStr := ial["updated"]
|
2023-10-09 16:10:15 +08:00
|
|
|
|
updated, parseErr := time.ParseInLocation("20060102150405", updatedStr, time.Local)
|
2023-10-09 12:14:11 +08:00
|
|
|
|
if nil == parseErr {
|
|
|
|
|
kv.Values[0].Updated = av.NewFormattedValueUpdated(updated.UnixMilli(), 0, av.UpdatedFormatNone)
|
2023-10-10 20:05:48 +08:00
|
|
|
|
kv.Values[0].Updated.IsNotEmpty = true
|
2023-10-09 12:14:11 +08:00
|
|
|
|
} else {
|
|
|
|
|
logging.LogWarnf("parse updated [%s] failed: %s", updatedStr, parseErr)
|
|
|
|
|
kv.Values[0].Updated = av.NewFormattedValueUpdated(time.Now().UnixMilli(), 0, av.UpdatedFormatNone)
|
|
|
|
|
}
|
2023-10-03 11:46:25 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2024-03-09 10:58:14 +08:00
|
|
|
|
|
2025-08-03 11:44:06 +08:00
|
|
|
|
// 再处理模板字段
|
2024-03-09 10:58:14 +08:00
|
|
|
|
|
|
|
|
|
// 渲染模板
|
2024-04-17 19:54:33 +08:00
|
|
|
|
var renderTemplateErr error
|
2023-10-10 21:55:43 +08:00
|
|
|
|
for _, kv := range keyValues {
|
|
|
|
|
switch kv.Key.Type {
|
|
|
|
|
case av.KeyTypeTemplate:
|
|
|
|
|
if 0 < len(kv.Values) {
|
2023-10-12 19:55:57 +08:00
|
|
|
|
ial := map[string]string{}
|
2024-05-31 23:55:17 +08:00
|
|
|
|
block := av.GetKeyBlockValue(keyValues)
|
2024-01-01 23:34:15 +08:00
|
|
|
|
if nil != block && !block.IsDetached {
|
2024-10-17 23:44:55 +08:00
|
|
|
|
ial = sql.GetBlockAttrs(block.BlockID)
|
2023-10-12 19:55:57 +08:00
|
|
|
|
}
|
2024-03-09 10:58:14 +08:00
|
|
|
|
|
2024-05-27 21:02:00 +08:00
|
|
|
|
if nil == kv.Values[0].Template {
|
2025-08-10 18:20:29 +08:00
|
|
|
|
kv.Values[0] = av.GetAttributeViewDefaultValue(kv.Values[0].ID, kv.Key.ID, nodeID, kv.Key.Type)
|
2024-05-27 21:02:00 +08:00
|
|
|
|
}
|
|
|
|
|
|
2024-04-17 19:54:33 +08:00
|
|
|
|
var renderErr error
|
2025-06-08 16:12:37 +08:00
|
|
|
|
kv.Values[0].Template.Content, renderErr = sql.RenderTemplateField(ial, keyValues, kv.Key.Template)
|
2024-04-17 19:54:33 +08:00
|
|
|
|
if nil != renderErr {
|
2024-04-18 11:29:09 +08:00
|
|
|
|
renderTemplateErr = fmt.Errorf("database [%s] template field [%s] rendering failed: %s", getAttrViewName(attrView), kv.Key.Name, renderErr)
|
2024-04-17 19:54:33 +08:00
|
|
|
|
}
|
2023-10-10 21:55:43 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-04-17 19:54:33 +08:00
|
|
|
|
if nil != renderTemplateErr {
|
|
|
|
|
util.PushErrMsg(fmt.Sprintf(Conf.Language(44), util.EscapeHTML(renderTemplateErr.Error())), 30000)
|
|
|
|
|
}
|
2023-10-03 11:46:25 +08:00
|
|
|
|
|
2024-05-16 22:24:15 +08:00
|
|
|
|
// 字段排序
|
2024-09-04 11:49:22 +08:00
|
|
|
|
refreshAttrViewKeyIDs(attrView, true)
|
2024-05-16 22:24:15 +08:00
|
|
|
|
sorts := map[string]int{}
|
|
|
|
|
for i, k := range attrView.KeyIDs {
|
|
|
|
|
sorts[k] = i
|
2023-09-30 11:57:37 +08:00
|
|
|
|
}
|
2024-05-16 22:24:15 +08:00
|
|
|
|
sort.Slice(keyValues, func(i, j int) bool {
|
|
|
|
|
return sorts[keyValues[i].Key.ID] < sorts[keyValues[j].Key.ID]
|
|
|
|
|
})
|
2023-09-30 11:57:37 +08:00
|
|
|
|
|
2024-03-10 23:27:13 +08:00
|
|
|
|
blockIDs := treenode.GetMirrorAttrViewBlockIDs(avID)
|
2023-11-03 11:15:33 +08:00
|
|
|
|
if 1 > len(blockIDs) {
|
|
|
|
|
// 老数据兼容处理
|
|
|
|
|
avBts := treenode.GetBlockTreesByType("av")
|
|
|
|
|
for _, avBt := range avBts {
|
|
|
|
|
if nil == avBt {
|
|
|
|
|
continue
|
|
|
|
|
}
|
2024-03-10 23:27:13 +08:00
|
|
|
|
tree, _ := LoadTreeByBlockID(avBt.ID)
|
2023-11-03 11:15:33 +08:00
|
|
|
|
if nil == tree {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
node := treenode.GetNodeInTree(tree, avBt.ID)
|
|
|
|
|
if nil == node {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
if avID == node.AttributeViewID {
|
|
|
|
|
blockIDs = append(blockIDs, avBt.ID)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if 1 > len(blockIDs) {
|
2025-08-10 18:20:29 +08:00
|
|
|
|
tree, _ := LoadTreeByBlockID(nodeID)
|
2024-02-16 16:02:22 +08:00
|
|
|
|
if nil != tree {
|
2025-08-10 18:20:29 +08:00
|
|
|
|
node := treenode.GetNodeInTree(tree, nodeID)
|
2024-02-16 16:02:22 +08:00
|
|
|
|
if nil != node {
|
|
|
|
|
if removeErr := removeNodeAvID(node, avID, nil, tree); nil != removeErr {
|
|
|
|
|
logging.LogErrorf("remove node avID [%s] failed: %s", avID, removeErr)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-11-03 11:15:33 +08:00
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
blockIDs = gulu.Str.RemoveDuplicatedElem(blockIDs)
|
|
|
|
|
for _, blockID := range blockIDs {
|
|
|
|
|
av.UpsertBlockRel(avID, blockID)
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-11-02 16:24:50 +08:00
|
|
|
|
|
2023-07-30 20:35:03 +08:00
|
|
|
|
ret = append(ret, &BlockAttributeViewKeys{
|
|
|
|
|
AvID: avID,
|
2024-05-20 23:04:03 +08:00
|
|
|
|
AvName: getAttrViewName(attrView),
|
2023-11-02 16:24:50 +08:00
|
|
|
|
BlockIDs: blockIDs,
|
2023-07-30 20:35:03 +08:00
|
|
|
|
KeyValues: keyValues,
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-09 20:39:12 +08:00
|
|
|
|
func genAttrViewGroups(view *av.View, attrView *av.AttributeView) {
|
2025-07-08 15:55:45 +08:00
|
|
|
|
if nil == view.Group {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-05 11:26:25 +08:00
|
|
|
|
groupStates := getAttrViewGroupStates(view)
|
2025-08-05 16:27:44 +08:00
|
|
|
|
|
2025-07-08 15:55:45 +08:00
|
|
|
|
group := view.Group
|
|
|
|
|
view.Groups = nil
|
|
|
|
|
viewable := sql.RenderView(attrView, view, "")
|
|
|
|
|
var items []av.Item
|
|
|
|
|
for _, item := range viewable.(av.Collection).GetItems() {
|
|
|
|
|
items = append(items, item)
|
|
|
|
|
}
|
2025-07-08 16:04:11 +08:00
|
|
|
|
|
2025-07-28 23:58:34 +08:00
|
|
|
|
groupKey := view.GetGroupKey(attrView)
|
2025-07-08 16:33:37 +08:00
|
|
|
|
if nil == groupKey {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-08 15:55:45 +08:00
|
|
|
|
var rangeStart, rangeEnd float64
|
|
|
|
|
switch group.Method {
|
|
|
|
|
case av.GroupMethodValue:
|
|
|
|
|
if av.GroupOrderMan != group.Order {
|
|
|
|
|
sort.SliceStable(items, func(i, j int) bool {
|
2025-07-27 19:50:14 +08:00
|
|
|
|
return items[i].GetValue(group.Field).String(false) < items[j].GetValue(group.Field).String(false)
|
2025-07-08 15:55:45 +08:00
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
case av.GroupMethodRangeNum:
|
|
|
|
|
if nil == group.Range {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
rangeStart, rangeEnd = group.Range.NumStart, group.Range.NumStart+group.Range.NumStep
|
|
|
|
|
sort.SliceStable(items, func(i, j int) bool {
|
2025-07-27 19:50:14 +08:00
|
|
|
|
return items[i].GetValue(group.Field).Number.Content < items[j].GetValue(group.Field).Number.Content
|
2025-07-08 15:55:45 +08:00
|
|
|
|
})
|
|
|
|
|
case av.GroupMethodDateDay, av.GroupMethodDateWeek, av.GroupMethodDateMonth, av.GroupMethodDateYear, av.GroupMethodDateRelative:
|
2025-07-28 23:58:34 +08:00
|
|
|
|
if av.KeyTypeCreated == groupKey.Type {
|
|
|
|
|
sort.SliceStable(items, func(i, j int) bool {
|
|
|
|
|
return items[i].GetValue(group.Field).Created.Content < items[j].GetValue(group.Field).Created.Content
|
|
|
|
|
})
|
|
|
|
|
} else if av.KeyTypeUpdated == groupKey.Type {
|
|
|
|
|
sort.SliceStable(items, func(i, j int) bool {
|
|
|
|
|
return items[i].GetValue(group.Field).Updated.Content < items[j].GetValue(group.Field).Updated.Content
|
|
|
|
|
})
|
|
|
|
|
} else if av.KeyTypeDate == groupKey.Type {
|
|
|
|
|
sort.SliceStable(items, func(i, j int) bool {
|
|
|
|
|
return items[i].GetValue(group.Field).Date.Content < items[j].GetValue(group.Field).Date.Content
|
|
|
|
|
})
|
|
|
|
|
}
|
2025-07-08 15:55:45 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-07-29 10:43:28 +08:00
|
|
|
|
todayStart := time.Now()
|
|
|
|
|
todayStart = time.Date(todayStart.Year(), todayStart.Month(), todayStart.Day(), 0, 0, 0, 0, time.Local)
|
|
|
|
|
|
2025-07-08 15:55:45 +08:00
|
|
|
|
groupItemsMap := map[string][]av.Item{}
|
|
|
|
|
for _, item := range items {
|
|
|
|
|
value := item.GetValue(group.Field)
|
|
|
|
|
if value.IsEmpty() {
|
2025-08-05 17:39:10 +08:00
|
|
|
|
groupItemsMap[groupValueDefault] = append(groupItemsMap[groupValueDefault], item)
|
2025-07-08 15:55:45 +08:00
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-05 17:39:10 +08:00
|
|
|
|
var groupVal string
|
2025-07-08 15:55:45 +08:00
|
|
|
|
switch group.Method {
|
|
|
|
|
case av.GroupMethodValue:
|
2025-08-05 16:27:44 +08:00
|
|
|
|
if av.KeyTypeMSelect == groupKey.Type {
|
|
|
|
|
for _, s := range value.MSelect {
|
|
|
|
|
groupItemsMap[s.Content] = append(groupItemsMap[s.Content], item)
|
|
|
|
|
}
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-29 11:59:15 +08:00
|
|
|
|
groupVal = value.String(false)
|
2025-07-08 15:55:45 +08:00
|
|
|
|
case av.GroupMethodRangeNum:
|
|
|
|
|
if group.Range.NumStart > value.Number.Content || group.Range.NumEnd < value.Number.Content {
|
2025-07-29 11:59:15 +08:00
|
|
|
|
groupVal = groupValueNotInRange
|
2025-07-08 15:55:45 +08:00
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-27 17:08:22 +08:00
|
|
|
|
for rangeEnd <= group.Range.NumEnd && rangeEnd <= value.Number.Content {
|
2025-07-08 15:55:45 +08:00
|
|
|
|
rangeStart += group.Range.NumStep
|
|
|
|
|
rangeEnd += group.Range.NumStep
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-27 17:08:22 +08:00
|
|
|
|
if rangeStart <= value.Number.Content && rangeEnd > value.Number.Content {
|
2025-07-29 11:59:15 +08:00
|
|
|
|
groupVal = fmt.Sprintf("%s - %s", strconv.FormatFloat(rangeStart, 'f', -1, 64), strconv.FormatFloat(rangeEnd, 'f', -1, 64))
|
2025-07-08 15:55:45 +08:00
|
|
|
|
}
|
|
|
|
|
case av.GroupMethodDateDay, av.GroupMethodDateWeek, av.GroupMethodDateMonth, av.GroupMethodDateYear, av.GroupMethodDateRelative:
|
|
|
|
|
var contentTime time.Time
|
|
|
|
|
switch value.Type {
|
|
|
|
|
case av.KeyTypeDate:
|
|
|
|
|
contentTime = time.UnixMilli(value.Date.Content)
|
|
|
|
|
case av.KeyTypeCreated:
|
|
|
|
|
contentTime = time.UnixMilli(value.Created.Content)
|
|
|
|
|
case av.KeyTypeUpdated:
|
|
|
|
|
contentTime = time.UnixMilli(value.Updated.Content)
|
|
|
|
|
}
|
|
|
|
|
switch group.Method {
|
|
|
|
|
case av.GroupMethodDateDay:
|
2025-07-29 11:59:15 +08:00
|
|
|
|
groupVal = contentTime.Format("2006-01-02")
|
2025-07-08 15:55:45 +08:00
|
|
|
|
case av.GroupMethodDateWeek:
|
|
|
|
|
year, week := contentTime.ISOWeek()
|
2025-07-29 11:59:15 +08:00
|
|
|
|
groupVal = fmt.Sprintf("%d-W%02d", year, week)
|
2025-07-08 15:55:45 +08:00
|
|
|
|
case av.GroupMethodDateMonth:
|
2025-07-29 11:59:15 +08:00
|
|
|
|
groupVal = contentTime.Format("2006-01")
|
2025-07-08 15:55:45 +08:00
|
|
|
|
case av.GroupMethodDateYear:
|
2025-07-29 11:59:15 +08:00
|
|
|
|
groupVal = contentTime.Format("2006")
|
2025-07-08 15:55:45 +08:00
|
|
|
|
case av.GroupMethodDateRelative:
|
|
|
|
|
// 过去 30 天之前的按月分组
|
|
|
|
|
// 过去 30 天、过去 7 天、昨天、今天、明天、未来 7 天、未来 30 天
|
|
|
|
|
// 未来 30 天之后的按月分组
|
2025-07-26 10:11:15 +08:00
|
|
|
|
if contentTime.Before(todayStart.AddDate(0, 0, -30)) {
|
2025-07-29 11:59:15 +08:00
|
|
|
|
groupVal = contentTime.Format("2006-01") // 开头的数字用于排序,下同
|
2025-07-26 10:11:15 +08:00
|
|
|
|
} else if contentTime.Before(todayStart.AddDate(0, 0, -7)) {
|
2025-07-29 11:59:15 +08:00
|
|
|
|
groupVal = groupValueLast30Days
|
2025-07-26 10:11:15 +08:00
|
|
|
|
} else if contentTime.Before(todayStart.AddDate(0, 0, -1)) {
|
2025-07-29 11:59:15 +08:00
|
|
|
|
groupVal = groupValueLast7Days
|
2025-07-26 10:11:15 +08:00
|
|
|
|
} else if contentTime.Before(todayStart) {
|
2025-07-29 11:59:15 +08:00
|
|
|
|
groupVal = groupValueYesterday
|
2025-07-26 10:45:40 +08:00
|
|
|
|
} else if (contentTime.After(todayStart) || contentTime.Equal(todayStart)) && contentTime.Before(todayStart.AddDate(0, 0, 1)) {
|
2025-07-29 11:59:15 +08:00
|
|
|
|
groupVal = groupValueToday
|
2025-07-26 10:11:15 +08:00
|
|
|
|
} else if contentTime.After(todayStart.AddDate(0, 0, 30)) {
|
2025-07-29 11:59:15 +08:00
|
|
|
|
groupVal = contentTime.Format("2006-01")
|
2025-07-26 10:11:15 +08:00
|
|
|
|
} else if contentTime.After(todayStart.AddDate(0, 0, 7)) {
|
2025-07-29 11:59:15 +08:00
|
|
|
|
groupVal = groupValueNext30Days
|
2025-07-29 11:04:42 +08:00
|
|
|
|
} else if contentTime.Equal(todayStart.AddDate(0, 0, 2)) || contentTime.After(todayStart.AddDate(0, 0, 2)) {
|
2025-07-29 11:59:15 +08:00
|
|
|
|
groupVal = groupValueNext7Days
|
2025-07-26 10:45:40 +08:00
|
|
|
|
} else {
|
2025-07-29 11:59:15 +08:00
|
|
|
|
groupVal = groupValueTomorrow
|
2025-07-08 15:55:45 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-08-05 16:27:44 +08:00
|
|
|
|
|
2025-07-29 11:59:15 +08:00
|
|
|
|
groupItemsMap[groupVal] = append(groupItemsMap[groupVal], item)
|
2025-07-08 15:55:45 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-07-25 15:20:56 +08:00
|
|
|
|
if av.KeyTypeSelect == groupKey.Type || av.KeyTypeMSelect == groupKey.Type {
|
|
|
|
|
for _, o := range groupKey.Options {
|
|
|
|
|
if _, ok := groupItemsMap[o.Name]; !ok {
|
|
|
|
|
groupItemsMap[o.Name] = []av.Item{}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-05 17:09:11 +08:00
|
|
|
|
for groupValue, groupItems := range groupItemsMap {
|
2025-07-08 15:55:45 +08:00
|
|
|
|
var v *av.View
|
|
|
|
|
switch view.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
v = av.NewTableView()
|
|
|
|
|
v.Table = av.NewLayoutTable()
|
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
v = av.NewGalleryView()
|
|
|
|
|
v.Gallery = av.NewLayoutGallery()
|
2025-07-08 16:35:26 +08:00
|
|
|
|
default:
|
|
|
|
|
logging.LogWarnf("unknown layout type [%s] for group view", view.LayoutType)
|
|
|
|
|
return
|
2025-07-08 15:55:45 +08:00
|
|
|
|
}
|
2025-07-25 16:34:30 +08:00
|
|
|
|
|
|
|
|
|
v.GroupItemIDs = []string{}
|
2025-07-08 15:55:45 +08:00
|
|
|
|
for _, item := range groupItems {
|
|
|
|
|
v.GroupItemIDs = append(v.GroupItemIDs, item.GetID())
|
|
|
|
|
}
|
2025-07-08 16:33:37 +08:00
|
|
|
|
|
2025-07-29 13:16:29 +08:00
|
|
|
|
v.Name = "" // 分组视图的名称在渲染时才填充
|
2025-08-05 17:13:00 +08:00
|
|
|
|
v.GroupVal = &av.Value{Type: av.KeyTypeText, Text: &av.ValueText{Content: groupValue}}
|
2025-08-05 17:09:11 +08:00
|
|
|
|
if av.KeyTypeSelect == groupKey.Type || av.KeyTypeMSelect == groupKey.Type {
|
2025-08-05 17:38:07 +08:00
|
|
|
|
if opt := groupKey.GetOption(groupValue); nil != opt {
|
2025-08-05 17:24:16 +08:00
|
|
|
|
v.GroupVal.Text = nil
|
|
|
|
|
v.GroupVal.Type = av.KeyTypeSelect
|
|
|
|
|
v.GroupVal.MSelect = []*av.ValueSelect{{Content: opt.Name, Color: opt.Color}}
|
|
|
|
|
}
|
2025-08-05 17:09:11 +08:00
|
|
|
|
}
|
2025-07-08 15:55:45 +08:00
|
|
|
|
view.Groups = append(view.Groups, v)
|
|
|
|
|
}
|
2025-07-08 17:06:52 +08:00
|
|
|
|
|
2025-08-05 17:42:53 +08:00
|
|
|
|
view.GroupCreated = time.Now().UnixMilli()
|
2025-07-09 10:42:10 +08:00
|
|
|
|
|
2025-08-05 11:26:25 +08:00
|
|
|
|
setAttrViewGroupStates(view, groupStates)
|
2025-07-08 17:06:52 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-08-05 11:26:25 +08:00
|
|
|
|
// GroupState 用于临时记录每个分组视图的状态,以便后面重新生成分组后可以恢复这些状态。
|
|
|
|
|
type GroupState struct {
|
|
|
|
|
ID string
|
|
|
|
|
Folded bool
|
|
|
|
|
Hidden int
|
2025-08-09 11:47:18 +08:00
|
|
|
|
Sort int
|
2025-08-05 11:26:25 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func getAttrViewGroupStates(view *av.View) (groupStates map[string]*GroupState) {
|
|
|
|
|
groupStates = map[string]*GroupState{}
|
|
|
|
|
if nil == view.Group {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-08 18:09:33 +08:00
|
|
|
|
for _, groupView := range view.Groups {
|
2025-08-05 17:09:11 +08:00
|
|
|
|
groupStates[groupView.GetGroupValue()] = &GroupState{
|
2025-08-05 11:26:25 +08:00
|
|
|
|
ID: groupView.ID,
|
|
|
|
|
Folded: groupView.GroupFolded,
|
|
|
|
|
Hidden: groupView.GroupHidden,
|
2025-08-09 11:47:18 +08:00
|
|
|
|
Sort: groupView.GroupSort,
|
2025-08-05 11:26:25 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func setAttrViewGroupStates(view *av.View, groupStates map[string]*GroupState) {
|
|
|
|
|
for _, groupView := range view.Groups {
|
2025-08-05 17:09:11 +08:00
|
|
|
|
if state, ok := groupStates[groupView.GetGroupValue()]; ok {
|
2025-08-05 11:26:25 +08:00
|
|
|
|
groupView.ID = state.ID
|
|
|
|
|
groupView.GroupFolded = state.Folded
|
|
|
|
|
groupView.GroupHidden = state.Hidden
|
2025-08-09 11:47:18 +08:00
|
|
|
|
groupView.GroupSort = state.Sort
|
2025-08-05 11:26:25 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-05-18 11:46:33 +08:00
|
|
|
|
func GetCurrentAttributeViewImages(avID, viewID, query string) (ret []string, err error) {
|
2025-05-18 11:31:22 +08:00
|
|
|
|
var attrView *av.AttributeView
|
|
|
|
|
attrView, err = av.ParseAttributeView(avID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
logging.LogErrorf("parse attribute view [%s] failed: %s", avID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
var view *av.View
|
2025-05-18 11:46:33 +08:00
|
|
|
|
|
2025-05-18 11:31:22 +08:00
|
|
|
|
if "" != viewID {
|
|
|
|
|
view, _ = attrView.GetCurrentView(viewID)
|
|
|
|
|
} else {
|
|
|
|
|
view = attrView.GetView(attrView.ViewID)
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-12 22:25:10 +08:00
|
|
|
|
table := getAttrViewTable(attrView, view, query)
|
2025-07-05 12:11:12 +08:00
|
|
|
|
av.Filter(table, attrView)
|
|
|
|
|
av.Sort(table, attrView)
|
2025-05-18 11:31:22 +08:00
|
|
|
|
|
|
|
|
|
for _, row := range table.Rows {
|
|
|
|
|
for _, cell := range row.Cells {
|
2025-05-18 11:46:33 +08:00
|
|
|
|
if nil != cell.Value && av.KeyTypeMAsset == cell.Value.Type && nil != cell.Value.MAsset {
|
|
|
|
|
for _, a := range cell.Value.MAsset {
|
|
|
|
|
if av.AssetTypeImage == a.Type {
|
|
|
|
|
ret = append(ret, a.Content)
|
2025-05-18 11:31:22 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-04 11:33:51 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewColDate(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := setAttributeViewColDate(operation)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2024-04-04 11:33:51 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func setAttributeViewColDate(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2024-04-04 11:33:51 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
keyID := operation.ID
|
|
|
|
|
key, _ := attrView.GetKey(keyID)
|
|
|
|
|
if nil == key || av.KeyTypeDate != key.Type {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if nil == key.Date {
|
|
|
|
|
key.Date = &av.Date{}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
key.Date.AutoFillNow = operation.Data.(bool)
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-01 20:38:53 +08:00
|
|
|
|
func (tx *Transaction) doHideAttrViewName(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := hideAttrViewName(operation)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2024-03-01 20:38:53 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func hideAttrViewName(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2024-03-01 20:38:53 +08:00
|
|
|
|
logging.LogErrorf("parse attribute view [%s] failed: %s", operation.AvID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-04 16:41:41 +08:00
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, operation.BlockID)
|
2024-03-01 20:38:53 +08:00
|
|
|
|
if nil == view {
|
2024-03-04 16:41:41 +08:00
|
|
|
|
logging.LogErrorf("get view [%s] failed: %s", operation.BlockID, err)
|
2024-03-01 20:38:53 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view.HideAttrViewName = operation.Data.(bool)
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-24 21:47:10 +08:00
|
|
|
|
func (tx *Transaction) doUpdateAttrViewColRollup(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := updateAttributeViewColRollup(operation)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-12-24 21:47:10 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func updateAttributeViewColRollup(operation *Operation) (err error) {
|
2025-08-03 11:44:06 +08:00
|
|
|
|
// operation.AvID 汇总字段所在 av
|
|
|
|
|
// operation.ID 汇总字段 ID
|
|
|
|
|
// operation.ParentID 汇总字段基于的关联字段 ID
|
|
|
|
|
// operation.KeyID 目标字段 ID
|
2023-12-24 21:47:10 +08:00
|
|
|
|
// operation.Data 计算方式
|
|
|
|
|
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-12-24 21:47:10 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
rollUpKey, _ := attrView.GetKey(operation.ID)
|
|
|
|
|
if nil == rollUpKey {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
rollUpKey.Rollup = &av.Rollup{
|
|
|
|
|
RelationKeyID: operation.ParentID,
|
|
|
|
|
KeyID: operation.KeyID,
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-30 20:49:57 +08:00
|
|
|
|
if nil != operation.Data {
|
2023-12-30 17:15:50 +08:00
|
|
|
|
data := operation.Data.(map[string]interface{})
|
|
|
|
|
if nil != data["calc"] {
|
|
|
|
|
calcData, jsonErr := gulu.JSON.MarshalJSON(data["calc"])
|
|
|
|
|
if nil != jsonErr {
|
|
|
|
|
err = jsonErr
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
if jsonErr = gulu.JSON.UnmarshalJSON(calcData, &rollUpKey.Rollup.Calc); nil != jsonErr {
|
|
|
|
|
err = jsonErr
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-12-24 21:47:10 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-23 11:35:53 +08:00
|
|
|
|
func (tx *Transaction) doUpdateAttrViewColRelation(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := updateAttributeViewColRelation(operation)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-12-23 11:35:53 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func updateAttributeViewColRelation(operation *Operation) (err error) {
|
2023-12-23 17:45:46 +08:00
|
|
|
|
// operation.AvID 源 avID
|
|
|
|
|
// operation.ID 目标 avID
|
2025-08-03 11:44:06 +08:00
|
|
|
|
// operation.KeyID 源 av 关联字段 ID
|
2023-12-23 17:45:46 +08:00
|
|
|
|
// operation.IsTwoWay 是否双向关联
|
2025-08-03 11:44:06 +08:00
|
|
|
|
// operation.BackRelationKeyID 双向关联的目标关联字段 ID
|
|
|
|
|
// operation.Name 双向关联的目标关联字段名称
|
|
|
|
|
// operation.Format 源 av 关联字段名称
|
2023-12-23 17:45:46 +08:00
|
|
|
|
|
|
|
|
|
srcAv, err := av.ParseAttributeView(operation.AvID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-12-23 11:35:53 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-23 17:45:46 +08:00
|
|
|
|
destAv, err := av.ParseAttributeView(operation.ID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-12-23 11:35:53 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-23 17:45:46 +08:00
|
|
|
|
isSameAv := srcAv.ID == destAv.ID
|
2023-12-24 09:40:01 +08:00
|
|
|
|
if isSameAv {
|
|
|
|
|
destAv = srcAv
|
|
|
|
|
}
|
2023-12-23 17:45:46 +08:00
|
|
|
|
|
|
|
|
|
for _, keyValues := range srcAv.KeyValues {
|
2023-12-24 11:24:40 +08:00
|
|
|
|
if keyValues.Key.ID != operation.KeyID {
|
|
|
|
|
continue
|
|
|
|
|
}
|
2023-12-23 17:57:45 +08:00
|
|
|
|
|
2023-12-24 11:24:40 +08:00
|
|
|
|
srcRel := keyValues.Key.Relation
|
|
|
|
|
// 已经设置过双向关联的话需要先断开双向关联
|
2023-12-31 10:30:39 +08:00
|
|
|
|
if nil != srcRel {
|
|
|
|
|
if srcRel.IsTwoWay {
|
|
|
|
|
oldDestAv, _ := av.ParseAttributeView(srcRel.AvID)
|
|
|
|
|
if nil != oldDestAv {
|
|
|
|
|
isOldSameAv := oldDestAv.ID == destAv.ID
|
|
|
|
|
if isOldSameAv {
|
|
|
|
|
oldDestAv = destAv
|
|
|
|
|
}
|
2023-12-23 17:57:45 +08:00
|
|
|
|
|
2023-12-31 10:30:39 +08:00
|
|
|
|
oldDestKey, _ := oldDestAv.GetKey(srcRel.BackKeyID)
|
|
|
|
|
if nil != oldDestKey && nil != oldDestKey.Relation && oldDestKey.Relation.AvID == srcAv.ID && oldDestKey.Relation.IsTwoWay {
|
|
|
|
|
oldDestKey.Relation.IsTwoWay = false
|
|
|
|
|
oldDestKey.Relation.BackKeyID = ""
|
|
|
|
|
}
|
2023-12-24 11:24:40 +08:00
|
|
|
|
|
2023-12-31 10:30:39 +08:00
|
|
|
|
if !isOldSameAv {
|
|
|
|
|
err = av.SaveAttributeView(oldDestAv)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-12-31 10:30:39 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
2023-12-23 17:57:45 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-12-31 10:30:39 +08:00
|
|
|
|
|
|
|
|
|
av.RemoveAvRel(srcAv.ID, srcRel.AvID)
|
2023-12-24 11:24:40 +08:00
|
|
|
|
}
|
2023-12-23 17:57:45 +08:00
|
|
|
|
|
2023-12-24 11:24:40 +08:00
|
|
|
|
srcRel = &av.Relation{
|
|
|
|
|
AvID: operation.ID,
|
|
|
|
|
IsTwoWay: operation.IsTwoWay,
|
2023-12-23 11:35:53 +08:00
|
|
|
|
}
|
2023-12-24 11:24:40 +08:00
|
|
|
|
if operation.IsTwoWay {
|
|
|
|
|
srcRel.BackKeyID = operation.BackRelationKeyID
|
|
|
|
|
} else {
|
|
|
|
|
srcRel.BackKeyID = ""
|
|
|
|
|
}
|
2023-12-24 12:44:51 +08:00
|
|
|
|
keyValues.Key.Relation = srcRel
|
2023-12-25 12:06:53 +08:00
|
|
|
|
keyValues.Key.Name = operation.Format
|
2023-12-24 11:24:40 +08:00
|
|
|
|
|
|
|
|
|
break
|
2023-12-23 11:35:53 +08:00
|
|
|
|
}
|
2023-12-23 17:45:46 +08:00
|
|
|
|
|
|
|
|
|
destAdded := false
|
2023-12-23 19:57:50 +08:00
|
|
|
|
backRelKey, _ := destAv.GetKey(operation.BackRelationKeyID)
|
|
|
|
|
if nil != backRelKey {
|
|
|
|
|
backRelKey.Relation = &av.Relation{
|
|
|
|
|
AvID: operation.AvID,
|
|
|
|
|
IsTwoWay: operation.IsTwoWay,
|
|
|
|
|
BackKeyID: operation.KeyID,
|
2023-12-23 17:45:46 +08:00
|
|
|
|
}
|
2023-12-23 19:57:50 +08:00
|
|
|
|
destAdded = true
|
2023-12-24 22:36:48 +08:00
|
|
|
|
if operation.IsTwoWay {
|
|
|
|
|
name := strings.TrimSpace(operation.Name)
|
|
|
|
|
if "" == name {
|
2023-12-25 12:06:53 +08:00
|
|
|
|
name = srcAv.Name + " " + operation.Format
|
2023-12-24 22:36:48 +08:00
|
|
|
|
}
|
2023-12-25 12:06:53 +08:00
|
|
|
|
backRelKey.Name = strings.TrimSpace(name)
|
2024-05-06 21:46:46 +08:00
|
|
|
|
} else {
|
|
|
|
|
backRelKey.Relation.BackKeyID = ""
|
2023-12-24 22:36:48 +08:00
|
|
|
|
}
|
2023-12-23 17:45:46 +08:00
|
|
|
|
}
|
2023-12-23 19:57:50 +08:00
|
|
|
|
|
2024-05-06 13:44:37 +08:00
|
|
|
|
if !destAdded && operation.IsTwoWay {
|
|
|
|
|
// 新建双向关联目标字段
|
|
|
|
|
name := strings.TrimSpace(operation.Name)
|
|
|
|
|
if "" == name {
|
|
|
|
|
name = srcAv.Name + " " + operation.Format
|
2024-05-12 18:04:10 +08:00
|
|
|
|
name = strings.TrimSpace(name)
|
2024-05-06 13:44:37 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
destKeyValues := &av.KeyValues{
|
|
|
|
|
Key: &av.Key{
|
|
|
|
|
ID: operation.BackRelationKeyID,
|
|
|
|
|
Name: name,
|
|
|
|
|
Type: av.KeyTypeRelation,
|
|
|
|
|
Relation: &av.Relation{AvID: operation.AvID, IsTwoWay: operation.IsTwoWay, BackKeyID: operation.KeyID},
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
destAv.KeyValues = append(destAv.KeyValues, destKeyValues)
|
|
|
|
|
|
|
|
|
|
for _, v := range destAv.Views {
|
|
|
|
|
switch v.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
2025-06-30 15:06:43 +08:00
|
|
|
|
v.Table.Columns = append(v.Table.Columns, &av.ViewTableColumn{BaseField: &av.BaseField{ID: operation.BackRelationKeyID}})
|
2025-06-09 16:46:50 +08:00
|
|
|
|
case av.LayoutTypeGallery:
|
2025-06-30 15:06:43 +08:00
|
|
|
|
v.Gallery.CardFields = append(v.Gallery.CardFields, &av.ViewGalleryCardField{BaseField: &av.BaseField{ID: operation.BackRelationKeyID}})
|
2023-12-24 14:40:05 +08:00
|
|
|
|
}
|
2024-05-06 13:44:37 +08:00
|
|
|
|
}
|
2023-12-24 14:40:05 +08:00
|
|
|
|
|
2024-05-06 13:44:37 +08:00
|
|
|
|
now := time.Now().UnixMilli()
|
|
|
|
|
// 和现有值进行关联
|
|
|
|
|
for _, keyValues := range srcAv.KeyValues {
|
|
|
|
|
if keyValues.Key.ID != operation.KeyID {
|
|
|
|
|
continue
|
|
|
|
|
}
|
2023-12-24 09:40:01 +08:00
|
|
|
|
|
2024-05-12 17:18:23 +08:00
|
|
|
|
for _, srcVal := range keyValues.Values {
|
2024-05-06 13:44:37 +08:00
|
|
|
|
for _, blockID := range srcVal.Relation.BlockIDs {
|
2024-05-12 17:18:23 +08:00
|
|
|
|
destVal := destAv.GetValue(destKeyValues.Key.ID, blockID)
|
|
|
|
|
if nil == destVal {
|
|
|
|
|
destVal = &av.Value{ID: ast.NewNodeID(), KeyID: destKeyValues.Key.ID, BlockID: blockID, Type: keyValues.Key.Type, Relation: &av.ValueRelation{}, CreatedAt: now, UpdatedAt: now + 1000}
|
|
|
|
|
} else {
|
|
|
|
|
destVal.Type = keyValues.Key.Type
|
|
|
|
|
if nil == destVal.Relation {
|
|
|
|
|
destVal.Relation = &av.ValueRelation{}
|
|
|
|
|
}
|
|
|
|
|
destVal.UpdatedAt = now
|
|
|
|
|
}
|
2024-05-06 13:44:37 +08:00
|
|
|
|
destVal.Relation.BlockIDs = append(destVal.Relation.BlockIDs, srcVal.BlockID)
|
|
|
|
|
destVal.Relation.BlockIDs = gulu.Str.RemoveDuplicatedElem(destVal.Relation.BlockIDs)
|
|
|
|
|
destKeyValues.Values = append(destKeyValues.Values, destVal)
|
2023-12-24 09:50:42 +08:00
|
|
|
|
}
|
2023-12-24 09:40:01 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2023-12-23 17:45:46 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-08-09 20:39:12 +08:00
|
|
|
|
regenAttrViewGroups(srcAv, "force")
|
2023-12-23 17:45:46 +08:00
|
|
|
|
err = av.SaveAttributeView(srcAv)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-12-23 17:45:46 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
if !isSameAv {
|
2025-08-09 20:39:12 +08:00
|
|
|
|
regenAttrViewGroups(destAv, "force")
|
2023-12-23 17:45:46 +08:00
|
|
|
|
err = av.SaveAttributeView(destAv)
|
2024-09-05 12:10:42 +08:00
|
|
|
|
ReloadAttrView(destAv.ID)
|
2023-12-23 17:45:46 +08:00
|
|
|
|
}
|
2023-12-31 10:30:39 +08:00
|
|
|
|
|
2023-12-31 10:58:40 +08:00
|
|
|
|
av.UpsertAvBackRel(srcAv.ID, destAv.ID)
|
2024-05-10 23:37:27 +08:00
|
|
|
|
if operation.IsTwoWay && !isSameAv {
|
2024-05-10 23:35:04 +08:00
|
|
|
|
av.UpsertAvBackRel(destAv.ID, srcAv.ID)
|
|
|
|
|
}
|
2023-12-23 11:35:53 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-01 15:39:21 +08:00
|
|
|
|
func (tx *Transaction) doSortAttrViewView(operation *Operation) (ret *TxErr) {
|
|
|
|
|
avID := operation.AvID
|
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-12-01 15:39:21 +08:00
|
|
|
|
logging.LogErrorf("parse attribute view [%s] failed: %s", operation.AvID, err)
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-12-01 15:39:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
2024-03-13 21:59:59 +08:00
|
|
|
|
view := attrView.GetView(operation.ID)
|
2024-03-04 16:41:41 +08:00
|
|
|
|
if nil == view {
|
|
|
|
|
logging.LogErrorf("get view failed: %s", operation.BlockID)
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2024-03-04 16:41:41 +08:00
|
|
|
|
}
|
|
|
|
|
viewID := view.ID
|
2024-03-13 21:59:59 +08:00
|
|
|
|
previousViewID := operation.PreviousID
|
|
|
|
|
if viewID == previousViewID {
|
2023-12-01 15:39:21 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-01 20:06:08 +08:00
|
|
|
|
var index, previousIndex int
|
|
|
|
|
for i, v := range attrView.Views {
|
|
|
|
|
if v.ID == viewID {
|
|
|
|
|
view = v
|
|
|
|
|
index = i
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-12-01 15:39:21 +08:00
|
|
|
|
|
2023-12-01 20:06:08 +08:00
|
|
|
|
attrView.Views = append(attrView.Views[:index], attrView.Views[index+1:]...)
|
|
|
|
|
for i, v := range attrView.Views {
|
2024-03-13 21:59:59 +08:00
|
|
|
|
if v.ID == previousViewID {
|
2023-12-01 20:06:08 +08:00
|
|
|
|
previousIndex = i + 1
|
|
|
|
|
break
|
2023-12-01 15:39:21 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2023-12-01 20:06:08 +08:00
|
|
|
|
attrView.Views = util.InsertElem(attrView.Views, previousIndex, view)
|
2023-12-01 15:39:21 +08:00
|
|
|
|
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = av.SaveAttributeView(attrView); err != nil {
|
2023-12-01 15:39:21 +08:00
|
|
|
|
logging.LogErrorf("save attribute view [%s] failed: %s", avID, err)
|
|
|
|
|
return &TxErr{code: TxErrCodeWriteTree, msg: err.Error(), id: avID}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-01 08:55:32 +08:00
|
|
|
|
func (tx *Transaction) doRemoveAttrViewView(operation *Operation) (ret *TxErr) {
|
|
|
|
|
var err error
|
|
|
|
|
avID := operation.AvID
|
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-12-01 08:55:32 +08:00
|
|
|
|
logging.LogErrorf("parse attribute view [%s] failed: %s", avID, err)
|
|
|
|
|
return &TxErr{code: TxErrCodeBlockNotFound, id: avID}
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-01 10:33:38 +08:00
|
|
|
|
if 1 >= len(attrView.Views) {
|
2024-03-04 16:41:41 +08:00
|
|
|
|
logging.LogWarnf("can't remove last view [%s] of attribute view [%s]", operation.AvID, avID)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, operation.BlockID)
|
|
|
|
|
if nil == view {
|
|
|
|
|
logging.LogWarnf("get view failed: %s", operation.BlockID)
|
2023-12-01 10:33:38 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-04 16:41:41 +08:00
|
|
|
|
viewID := view.ID
|
2023-12-01 10:44:10 +08:00
|
|
|
|
var index int
|
2023-12-01 08:55:32 +08:00
|
|
|
|
for i, view := range attrView.Views {
|
|
|
|
|
if viewID == view.ID {
|
|
|
|
|
attrView.Views = append(attrView.Views[:i], attrView.Views[i+1:]...)
|
2023-12-01 10:44:10 +08:00
|
|
|
|
index = i - 1
|
2023-12-01 08:55:32 +08:00
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-12-01 15:39:21 +08:00
|
|
|
|
if 0 > index {
|
|
|
|
|
index = 0
|
|
|
|
|
}
|
2024-03-04 23:07:32 +08:00
|
|
|
|
|
2025-06-09 18:34:20 +08:00
|
|
|
|
view = attrView.Views[index]
|
|
|
|
|
attrView.ViewID = view.ID
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = av.SaveAttributeView(attrView); err != nil {
|
2024-03-04 23:07:32 +08:00
|
|
|
|
logging.LogErrorf("save attribute view [%s] failed: %s", avID, err)
|
|
|
|
|
return &TxErr{code: TxErrCodeWriteTree, msg: err.Error(), id: avID}
|
|
|
|
|
}
|
2023-12-01 08:55:32 +08:00
|
|
|
|
|
2024-03-08 22:21:51 +08:00
|
|
|
|
trees, nodes := getMirrorBlocksNodes(avID)
|
|
|
|
|
for _, node := range nodes {
|
|
|
|
|
attrs := parse.IAL2Map(node.KramdownIAL)
|
|
|
|
|
blockViewID := attrs[av.NodeAttrView]
|
|
|
|
|
if blockViewID == viewID {
|
|
|
|
|
attrs[av.NodeAttrView] = attrView.ViewID
|
2025-06-09 18:34:20 +08:00
|
|
|
|
node.AttributeViewType = string(view.LayoutType)
|
2024-03-08 22:21:51 +08:00
|
|
|
|
oldAttrs, e := setNodeAttrs0(node, attrs)
|
|
|
|
|
if nil != e {
|
|
|
|
|
logging.LogErrorf("set node attrs failed: %s", e)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
cache.PutBlockIAL(node.ID, parse.IAL2Map(node.KramdownIAL))
|
|
|
|
|
pushBroadcastAttrTransactions(oldAttrs, node)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, tree := range trees {
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = indexWriteTreeUpsertQueue(tree); err != nil {
|
2024-03-08 22:21:51 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-06-10 11:14:13 +08:00
|
|
|
|
|
|
|
|
|
operation.RetData = view.LayoutType
|
2024-03-08 22:21:51 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func getMirrorBlocksNodes(avID string) (trees []*parse.Tree, nodes []*ast.Node) {
|
2024-06-21 23:37:47 +08:00
|
|
|
|
mirrorBlockIDs := treenode.GetMirrorAttrViewBlockIDs(avID)
|
2024-10-19 17:05:26 +08:00
|
|
|
|
mirrorBlockTrees := filesys.LoadTrees(mirrorBlockIDs)
|
|
|
|
|
for id, tree := range mirrorBlockTrees {
|
|
|
|
|
node := treenode.GetNodeInTree(tree, id)
|
2024-03-04 14:14:10 +08:00
|
|
|
|
if nil == node {
|
2024-10-19 17:05:26 +08:00
|
|
|
|
logging.LogErrorf("get node in tree by block ID [%s] failed", id)
|
2024-03-04 14:14:10 +08:00
|
|
|
|
continue
|
|
|
|
|
}
|
2024-03-08 22:21:51 +08:00
|
|
|
|
nodes = append(nodes, node)
|
2024-03-04 14:14:10 +08:00
|
|
|
|
}
|
|
|
|
|
|
2024-10-19 17:05:26 +08:00
|
|
|
|
for _, tree := range mirrorBlockTrees {
|
2024-03-08 22:21:51 +08:00
|
|
|
|
trees = append(trees, tree)
|
2024-03-04 14:14:10 +08:00
|
|
|
|
}
|
2023-12-01 08:55:32 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-01 09:48:20 +08:00
|
|
|
|
func (tx *Transaction) doDuplicateAttrViewView(operation *Operation) (ret *TxErr) {
|
|
|
|
|
var err error
|
|
|
|
|
avID := operation.AvID
|
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-12-01 09:48:20 +08:00
|
|
|
|
logging.LogErrorf("parse attribute view [%s] failed: %s", avID, err)
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: avID}
|
2023-12-01 09:48:20 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
masterView := attrView.GetView(operation.PreviousID)
|
|
|
|
|
if nil == masterView {
|
|
|
|
|
logging.LogErrorf("get master view failed: %s", avID)
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: avID}
|
2023-12-01 09:48:20 +08:00
|
|
|
|
}
|
|
|
|
|
|
2024-03-04 16:41:41 +08:00
|
|
|
|
node, tree, _ := getNodeByBlockID(nil, operation.BlockID)
|
|
|
|
|
if nil == node {
|
|
|
|
|
logging.LogErrorf("get node by block ID [%s] failed", operation.BlockID)
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID}
|
2024-03-04 16:41:41 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
attrs := parse.IAL2Map(node.KramdownIAL)
|
|
|
|
|
attrs[av.NodeAttrView] = operation.ID
|
2025-06-15 18:21:26 +08:00
|
|
|
|
node.AttributeViewType = string(masterView.LayoutType)
|
2024-03-04 16:41:41 +08:00
|
|
|
|
err = setNodeAttrs(node, tree, attrs)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2024-03-04 16:41:41 +08:00
|
|
|
|
logging.LogWarnf("set node [%s] attrs failed: %s", operation.BlockID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-10 11:36:30 +08:00
|
|
|
|
var view *av.View
|
|
|
|
|
switch masterView.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
view = av.NewTableView()
|
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
view = av.NewGalleryView()
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-04 16:41:41 +08:00
|
|
|
|
view.ID = operation.ID
|
2023-12-01 09:48:20 +08:00
|
|
|
|
attrView.Views = append(attrView.Views, view)
|
|
|
|
|
attrView.ViewID = view.ID
|
|
|
|
|
|
|
|
|
|
view.Icon = masterView.Icon
|
2024-06-10 16:25:57 +08:00
|
|
|
|
view.Name = util.GetDuplicateName(masterView.Name)
|
2024-03-01 22:40:56 +08:00
|
|
|
|
view.HideAttrViewName = masterView.HideAttrViewName
|
2025-06-10 11:36:30 +08:00
|
|
|
|
view.Desc = masterView.Desc
|
|
|
|
|
view.LayoutType = masterView.LayoutType
|
2023-12-01 09:48:20 +08:00
|
|
|
|
|
2025-06-29 11:11:20 +08:00
|
|
|
|
for _, filter := range masterView.Filters {
|
|
|
|
|
view.Filters = append(view.Filters, &av.ViewFilter{
|
|
|
|
|
Column: filter.Column,
|
|
|
|
|
Operator: filter.Operator,
|
|
|
|
|
Value: filter.Value,
|
|
|
|
|
RelativeDate: filter.RelativeDate,
|
|
|
|
|
RelativeDate2: filter.RelativeDate2,
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, s := range masterView.Sorts {
|
|
|
|
|
view.Sorts = append(view.Sorts, &av.ViewSort{
|
|
|
|
|
Column: s.Column,
|
|
|
|
|
Order: s.Order,
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-29 15:53:56 +08:00
|
|
|
|
if nil != masterView.Group {
|
|
|
|
|
if copyErr := copier.Copy(view.Group, masterView.Group); nil != copyErr {
|
|
|
|
|
logging.LogErrorf("copy group failed: %s", copyErr)
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: avID, msg: copyErr.Error()}
|
2025-06-29 15:53:56 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-29 11:11:20 +08:00
|
|
|
|
view.PageSize = masterView.PageSize
|
|
|
|
|
|
2025-06-10 11:36:30 +08:00
|
|
|
|
switch masterView.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
for _, col := range masterView.Table.Columns {
|
|
|
|
|
view.Table.Columns = append(view.Table.Columns, &av.ViewTableColumn{
|
2025-06-30 15:06:43 +08:00
|
|
|
|
BaseField: &av.BaseField{
|
|
|
|
|
ID: col.ID,
|
|
|
|
|
Wrap: col.Wrap,
|
|
|
|
|
Hidden: col.Hidden,
|
|
|
|
|
Desc: col.Desc,
|
|
|
|
|
},
|
|
|
|
|
Pin: col.Pin,
|
|
|
|
|
Width: col.Width,
|
|
|
|
|
Calc: col.Calc,
|
2025-06-10 11:36:30 +08:00
|
|
|
|
})
|
|
|
|
|
}
|
2023-12-01 09:48:20 +08:00
|
|
|
|
|
2025-06-30 12:50:50 +08:00
|
|
|
|
view.Table.ShowIcon = masterView.Table.ShowIcon
|
|
|
|
|
view.Table.WrapField = masterView.Table.WrapField
|
2025-06-10 11:36:30 +08:00
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
for _, field := range masterView.Gallery.CardFields {
|
|
|
|
|
view.Gallery.CardFields = append(view.Gallery.CardFields, &av.ViewGalleryCardField{
|
2025-06-30 15:06:43 +08:00
|
|
|
|
BaseField: &av.BaseField{
|
|
|
|
|
ID: field.ID,
|
|
|
|
|
Wrap: field.Wrap,
|
|
|
|
|
Hidden: field.Hidden,
|
|
|
|
|
Desc: field.Desc,
|
|
|
|
|
},
|
2025-06-10 11:36:30 +08:00
|
|
|
|
})
|
|
|
|
|
}
|
2023-12-01 09:48:20 +08:00
|
|
|
|
|
2025-06-10 11:36:30 +08:00
|
|
|
|
view.Gallery.CoverFrom = masterView.Gallery.CoverFrom
|
|
|
|
|
view.Gallery.CoverFromAssetKeyID = masterView.Gallery.CoverFromAssetKeyID
|
|
|
|
|
view.Gallery.CardSize = masterView.Gallery.CardSize
|
|
|
|
|
view.Gallery.FitImage = masterView.Gallery.FitImage
|
2025-07-30 10:40:04 +08:00
|
|
|
|
view.Gallery.DisplayFieldName = masterView.Gallery.DisplayFieldName
|
2025-06-10 11:36:30 +08:00
|
|
|
|
view.Gallery.ShowIcon = masterView.Gallery.ShowIcon
|
|
|
|
|
view.Gallery.WrapField = masterView.Gallery.WrapField
|
|
|
|
|
}
|
2023-12-08 22:00:23 +08:00
|
|
|
|
|
2025-07-02 17:02:40 +08:00
|
|
|
|
view.ItemIDs = masterView.ItemIDs
|
|
|
|
|
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = av.SaveAttributeView(attrView); err != nil {
|
2023-12-01 09:48:20 +08:00
|
|
|
|
logging.LogErrorf("save attribute view [%s] failed: %s", avID, err)
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, msg: err.Error(), id: avID}
|
2023-12-01 09:48:20 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-01 08:55:32 +08:00
|
|
|
|
func (tx *Transaction) doAddAttrViewView(operation *Operation) (ret *TxErr) {
|
2025-06-23 17:36:19 +08:00
|
|
|
|
err := addAttrViewView(operation.AvID, operation.ID, operation.BlockID, operation.Layout)
|
|
|
|
|
if nil != err {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2025-06-23 17:36:19 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func addAttrViewView(avID, viewID, blockID string, layout av.LayoutType) (err error) {
|
2023-12-01 08:55:32 +08:00
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-12-01 08:55:32 +08:00
|
|
|
|
logging.LogErrorf("parse attribute view [%s] failed: %s", avID, err)
|
2025-06-23 17:36:19 +08:00
|
|
|
|
return
|
2023-12-01 08:55:32 +08:00
|
|
|
|
}
|
|
|
|
|
|
2024-03-04 16:41:41 +08:00
|
|
|
|
if 1 > len(attrView.Views) {
|
|
|
|
|
logging.LogErrorf("no view in attribute view [%s]", avID)
|
2025-06-23 17:36:19 +08:00
|
|
|
|
return
|
2024-03-04 16:41:41 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-12-01 09:17:44 +08:00
|
|
|
|
firstView := attrView.Views[0]
|
|
|
|
|
if nil == firstView {
|
|
|
|
|
logging.LogErrorf("get first view failed: %s", avID)
|
2025-06-23 17:36:19 +08:00
|
|
|
|
return
|
2023-12-01 08:55:32 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-06-23 17:36:19 +08:00
|
|
|
|
if "" == layout {
|
|
|
|
|
layout = av.LayoutTypeTable
|
2025-06-08 17:22:03 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-06-08 11:01:21 +08:00
|
|
|
|
var view *av.View
|
2025-06-23 17:36:19 +08:00
|
|
|
|
switch layout {
|
2025-06-08 11:01:21 +08:00
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
view = av.NewTableView()
|
|
|
|
|
switch firstView.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
for _, col := range firstView.Table.Columns {
|
2025-07-12 11:57:39 +08:00
|
|
|
|
view.Table.Columns = append(view.Table.Columns, &av.ViewTableColumn{BaseField: &av.BaseField{ID: col.ID}, Width: col.Width})
|
2025-06-08 11:01:21 +08:00
|
|
|
|
}
|
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
for _, field := range firstView.Gallery.CardFields {
|
2025-06-30 15:06:43 +08:00
|
|
|
|
view.Table.Columns = append(view.Table.Columns, &av.ViewTableColumn{BaseField: &av.BaseField{ID: field.ID}})
|
2025-06-08 11:01:21 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
view = av.NewGalleryView()
|
|
|
|
|
switch firstView.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
for _, col := range firstView.Table.Columns {
|
2025-06-30 15:06:43 +08:00
|
|
|
|
view.Gallery.CardFields = append(view.Gallery.CardFields, &av.ViewGalleryCardField{BaseField: &av.BaseField{ID: col.ID}})
|
2025-06-08 11:01:21 +08:00
|
|
|
|
}
|
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
for _, field := range firstView.Gallery.CardFields {
|
2025-06-30 15:06:43 +08:00
|
|
|
|
view.Gallery.CardFields = append(view.Gallery.CardFields, &av.ViewGalleryCardField{BaseField: &av.BaseField{ID: field.ID}})
|
2025-06-08 11:01:21 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
err = av.ErrWrongLayoutType
|
2025-06-23 17:36:19 +08:00
|
|
|
|
logging.LogErrorf("wrong layout type [%s] for attribute view [%s]", layout, avID)
|
2025-06-08 11:01:21 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-02 17:02:40 +08:00
|
|
|
|
view.ItemIDs = firstView.ItemIDs
|
2025-06-23 17:36:19 +08:00
|
|
|
|
attrView.ViewID = viewID
|
|
|
|
|
view.ID = viewID
|
2023-12-01 08:55:32 +08:00
|
|
|
|
attrView.Views = append(attrView.Views, view)
|
2025-06-09 18:31:38 +08:00
|
|
|
|
|
2025-06-23 17:36:19 +08:00
|
|
|
|
node, tree, _ := getNodeByBlockID(nil, blockID)
|
2025-06-09 18:31:38 +08:00
|
|
|
|
if nil == node {
|
2025-06-23 17:36:19 +08:00
|
|
|
|
logging.LogErrorf("get node by block ID [%s] failed", blockID)
|
|
|
|
|
return
|
2025-06-09 18:31:38 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-06-09 18:15:00 +08:00
|
|
|
|
node.AttributeViewType = string(view.LayoutType)
|
2025-06-09 18:31:38 +08:00
|
|
|
|
attrs := parse.IAL2Map(node.KramdownIAL)
|
2025-06-23 17:36:19 +08:00
|
|
|
|
attrs[av.NodeAttrView] = viewID
|
2025-06-09 18:31:38 +08:00
|
|
|
|
err = setNodeAttrs(node, tree, attrs)
|
|
|
|
|
if err != nil {
|
2025-06-23 17:36:19 +08:00
|
|
|
|
logging.LogWarnf("set node [%s] attrs failed: %s", blockID, err)
|
2025-06-09 18:31:38 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
2023-12-01 09:17:44 +08:00
|
|
|
|
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = av.SaveAttributeView(attrView); err != nil {
|
2023-12-01 08:55:32 +08:00
|
|
|
|
logging.LogErrorf("save attribute view [%s] failed: %s", avID, err)
|
2025-06-23 17:36:19 +08:00
|
|
|
|
return
|
2023-12-01 08:55:32 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-01 09:00:14 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewViewName(operation *Operation) (ret *TxErr) {
|
|
|
|
|
var err error
|
|
|
|
|
avID := operation.AvID
|
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-12-01 09:00:14 +08:00
|
|
|
|
logging.LogErrorf("parse attribute view [%s] failed: %s", avID, err)
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: avID}
|
2023-12-01 09:00:14 +08:00
|
|
|
|
}
|
|
|
|
|
|
2024-03-04 16:41:41 +08:00
|
|
|
|
viewID := operation.ID
|
2023-12-01 09:00:14 +08:00
|
|
|
|
view := attrView.GetView(viewID)
|
|
|
|
|
if nil == view {
|
|
|
|
|
logging.LogErrorf("get view [%s] failed: %s", viewID, err)
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: viewID}
|
2023-12-01 09:00:14 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-12-01 11:43:49 +08:00
|
|
|
|
view.Name = strings.TrimSpace(operation.Data.(string))
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = av.SaveAttributeView(attrView); err != nil {
|
2023-12-01 09:00:14 +08:00
|
|
|
|
logging.LogErrorf("save attribute view [%s] failed: %s", avID, err)
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, msg: err.Error(), id: avID}
|
2023-12-01 09:00:14 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-01 09:48:20 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewViewIcon(operation *Operation) (ret *TxErr) {
|
|
|
|
|
var err error
|
|
|
|
|
avID := operation.AvID
|
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-12-01 09:48:20 +08:00
|
|
|
|
logging.LogErrorf("parse attribute view [%s] failed: %s", avID, err)
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: avID}
|
2023-12-01 09:48:20 +08:00
|
|
|
|
}
|
|
|
|
|
|
2024-03-04 16:41:41 +08:00
|
|
|
|
viewID := operation.ID
|
2023-12-01 09:48:20 +08:00
|
|
|
|
view := attrView.GetView(viewID)
|
|
|
|
|
if nil == view {
|
|
|
|
|
logging.LogErrorf("get view [%s] failed: %s", viewID, err)
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: viewID}
|
2023-12-01 09:48:20 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view.Icon = operation.Data.(string)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = av.SaveAttributeView(attrView); err != nil {
|
2023-12-01 09:48:20 +08:00
|
|
|
|
logging.LogErrorf("save attribute view [%s] failed: %s", avID, err)
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, msg: err.Error(), id: avID}
|
2023-12-01 09:48:20 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-09 14:09:40 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewViewDesc(operation *Operation) (ret *TxErr) {
|
|
|
|
|
var err error
|
|
|
|
|
avID := operation.AvID
|
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
logging.LogErrorf("parse attribute view [%s] failed: %s", avID, err)
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: avID}
|
2024-11-09 14:09:40 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
viewID := operation.ID
|
|
|
|
|
view := attrView.GetView(viewID)
|
|
|
|
|
if nil == view {
|
|
|
|
|
logging.LogErrorf("get view [%s] failed: %s", viewID, err)
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: viewID}
|
2024-11-09 14:09:40 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view.Desc = strings.TrimSpace(operation.Data.(string))
|
|
|
|
|
if err = av.SaveAttributeView(attrView); err != nil {
|
|
|
|
|
logging.LogErrorf("save attribute view [%s] failed: %s", avID, err)
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, msg: err.Error(), id: avID}
|
2024-11-09 14:09:40 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-11 22:36:02 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewName(operation *Operation) (ret *TxErr) {
|
2024-09-26 09:00:35 +08:00
|
|
|
|
err := tx.setAttributeViewName(operation)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-07-11 22:11:15 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-08 22:51:28 +08:00
|
|
|
|
const attrAvNameTpl = `<span data-av-id="${avID}" data-popover-url="/api/av/getMirrorDatabaseBlocks" class="popover__block">${avName}</span>`
|
|
|
|
|
|
2024-09-26 09:00:35 +08:00
|
|
|
|
func (tx *Transaction) setAttributeViewName(operation *Operation) (err error) {
|
2024-03-08 22:51:28 +08:00
|
|
|
|
avID := operation.ID
|
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-11 22:11:15 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-01 11:43:49 +08:00
|
|
|
|
attrView.Name = strings.TrimSpace(operation.Data.(string))
|
2025-08-07 10:35:40 +08:00
|
|
|
|
attrView.Name = strings.ReplaceAll(attrView.Name, "\n", " ")
|
2025-08-08 15:12:00 +08:00
|
|
|
|
if 512 < len(attrView.Name) {
|
|
|
|
|
attrView.Name = gulu.Str.SubStr(attrView.Name, 512)
|
2025-08-07 10:35:40 +08:00
|
|
|
|
}
|
2023-07-11 22:11:15 +08:00
|
|
|
|
err = av.SaveAttributeView(attrView)
|
2024-03-08 22:51:28 +08:00
|
|
|
|
|
2024-09-26 09:00:35 +08:00
|
|
|
|
_, nodes := tx.getAttrViewBoundNodes(attrView)
|
2024-03-08 22:51:28 +08:00
|
|
|
|
for _, node := range nodes {
|
2024-03-08 23:12:33 +08:00
|
|
|
|
avNames := getAvNames(node.IALAttr(av.NodeAttrNameAvs))
|
2024-03-08 22:51:28 +08:00
|
|
|
|
oldAttrs := parse.IAL2Map(node.KramdownIAL)
|
2024-03-10 23:00:45 +08:00
|
|
|
|
node.SetIALAttr(av.NodeAttrViewNames, avNames)
|
2024-03-08 22:59:28 +08:00
|
|
|
|
pushBroadcastAttrTransactions(oldAttrs, node)
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
2024-03-08 22:51:28 +08:00
|
|
|
|
|
2024-03-08 23:12:33 +08:00
|
|
|
|
func getAvNames(avIDs string) (ret string) {
|
2024-03-08 22:59:28 +08:00
|
|
|
|
if "" == avIDs {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
avNames := bytes.Buffer{}
|
|
|
|
|
nodeAvIDs := strings.Split(avIDs, ",")
|
|
|
|
|
for _, nodeAvID := range nodeAvIDs {
|
|
|
|
|
nodeAvName, getErr := av.GetAttributeViewName(nodeAvID)
|
|
|
|
|
if nil != getErr {
|
|
|
|
|
continue
|
2024-03-08 22:51:28 +08:00
|
|
|
|
}
|
2024-03-08 22:59:28 +08:00
|
|
|
|
if "" == nodeAvName {
|
2024-03-24 22:15:19 +08:00
|
|
|
|
nodeAvName = Conf.language(105)
|
2024-03-08 22:51:28 +08:00
|
|
|
|
}
|
2024-03-08 22:59:28 +08:00
|
|
|
|
|
|
|
|
|
tpl := strings.ReplaceAll(attrAvNameTpl, "${avID}", nodeAvID)
|
|
|
|
|
tpl = strings.ReplaceAll(tpl, "${avName}", nodeAvName)
|
|
|
|
|
avNames.WriteString(tpl)
|
|
|
|
|
avNames.WriteString(" ")
|
|
|
|
|
}
|
|
|
|
|
if 0 < avNames.Len() {
|
|
|
|
|
avNames.Truncate(avNames.Len() - 6)
|
|
|
|
|
ret = avNames.String()
|
2024-03-08 22:51:28 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-05 09:18:11 +08:00
|
|
|
|
func (tx *Transaction) getAttrViewBoundNodes(attrView *av.AttributeView) (trees map[string]*parse.Tree, nodes []*ast.Node) {
|
2024-03-08 22:51:28 +08:00
|
|
|
|
blockKeyValues := attrView.GetBlockKeyValues()
|
2024-11-05 09:18:11 +08:00
|
|
|
|
trees = map[string]*parse.Tree{}
|
2024-03-08 22:51:28 +08:00
|
|
|
|
for _, blockKeyValue := range blockKeyValues.Values {
|
|
|
|
|
if blockKeyValue.IsDetached {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var tree *parse.Tree
|
2025-08-10 16:02:59 +08:00
|
|
|
|
tree = trees[blockKeyValue.Block.ID]
|
2024-03-08 22:51:28 +08:00
|
|
|
|
if nil == tree {
|
2024-09-26 09:00:35 +08:00
|
|
|
|
if nil == tx {
|
2025-08-10 16:02:59 +08:00
|
|
|
|
tree, _ = LoadTreeByBlockID(blockKeyValue.Block.ID)
|
2024-09-26 09:00:35 +08:00
|
|
|
|
} else {
|
2025-08-10 16:02:59 +08:00
|
|
|
|
tree, _ = tx.loadTree(blockKeyValue.Block.ID)
|
2024-09-26 09:00:35 +08:00
|
|
|
|
}
|
2024-03-08 22:51:28 +08:00
|
|
|
|
}
|
|
|
|
|
if nil == tree {
|
|
|
|
|
continue
|
|
|
|
|
}
|
2025-08-10 16:02:59 +08:00
|
|
|
|
trees[blockKeyValue.Block.ID] = tree
|
2024-03-08 22:51:28 +08:00
|
|
|
|
|
2025-08-10 16:02:59 +08:00
|
|
|
|
node := treenode.GetNodeInTree(tree, blockKeyValue.Block.ID)
|
2024-03-08 22:51:28 +08:00
|
|
|
|
if nil == node {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-02 12:35:11 +08:00
|
|
|
|
nodes = append(nodes, node)
|
|
|
|
|
}
|
2023-07-11 22:11:15 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-11 22:36:02 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewFilters(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := setAttributeViewFilters(operation)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-07-11 22:36:02 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-11 22:11:15 +08:00
|
|
|
|
func setAttributeViewFilters(operation *Operation) (err error) {
|
2023-07-13 09:45:39 +08:00
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-11 22:11:15 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-04 16:41:41 +08:00
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, operation.BlockID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-11 22:11:15 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
operationData := operation.Data.([]interface{})
|
|
|
|
|
data, err := gulu.JSON.MarshalJSON(operationData)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-11 22:11:15 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-29 11:11:20 +08:00
|
|
|
|
if err = gulu.JSON.UnmarshalJSON(data, &view.Filters); err != nil {
|
|
|
|
|
return
|
2023-07-11 22:11:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-11 22:36:02 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewSorts(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := setAttributeViewSorts(operation)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-07-11 22:36:02 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-11 22:11:15 +08:00
|
|
|
|
func setAttributeViewSorts(operation *Operation) (err error) {
|
2023-07-13 09:45:39 +08:00
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-11 22:11:15 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-04 16:41:41 +08:00
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, operation.BlockID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-11 22:11:15 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
operationData := operation.Data.([]interface{})
|
|
|
|
|
data, err := gulu.JSON.MarshalJSON(operationData)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-11 22:11:15 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-29 11:11:20 +08:00
|
|
|
|
if err = gulu.JSON.UnmarshalJSON(data, &view.Sorts); err != nil {
|
|
|
|
|
return
|
2023-07-11 22:11:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-08 21:05:21 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewPageSize(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := setAttributeViewPageSize(operation)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-12-08 21:05:21 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func setAttributeViewPageSize(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-12-08 21:05:21 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-04 21:58:26 +08:00
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, operation.BlockID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-12-08 21:05:21 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-29 11:11:20 +08:00
|
|
|
|
view.PageSize = int(operation.Data.(float64))
|
2023-12-08 21:05:21 +08:00
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-15 21:36:35 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewColCalc(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := setAttributeViewColumnCalc(operation)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-07-15 21:36:35 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func setAttributeViewColumnCalc(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-15 21:36:35 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-04 16:41:41 +08:00
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, operation.BlockID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-15 21:36:35 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-15 22:27:32 +08:00
|
|
|
|
operationData := operation.Data.(interface{})
|
2023-07-15 21:36:35 +08:00
|
|
|
|
data, err := gulu.JSON.MarshalJSON(operationData)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-15 21:36:35 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-04 16:42:37 +08:00
|
|
|
|
calc := &av.FieldCalc{}
|
2023-07-15 21:36:35 +08:00
|
|
|
|
switch view.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = gulu.JSON.UnmarshalJSON(data, calc); err != nil {
|
2023-07-15 21:36:35 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, column := range view.Table.Columns {
|
|
|
|
|
if column.ID == operation.ID {
|
|
|
|
|
column.Calc = calc
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-06-09 16:46:50 +08:00
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
return
|
2023-07-15 21:36:35 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-11 22:36:02 +08:00
|
|
|
|
func (tx *Transaction) doInsertAttrViewBlock(operation *Operation) (ret *TxErr) {
|
2025-08-02 17:49:30 +08:00
|
|
|
|
err := AddAttributeViewBlock(tx, operation.Srcs, operation.AvID, operation.BlockID, operation.GroupID, operation.PreviousID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2024-01-18 21:01:36 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-10 17:25:07 +08:00
|
|
|
|
func AddAttributeViewBlock(tx *Transaction, srcs []map[string]interface{}, avID, dbBlockID, groupID, previousItemID string) (err error) {
|
2024-05-09 10:43:02 +08:00
|
|
|
|
slices.Reverse(srcs) // https://github.com/siyuan-note/siyuan/issues/11286
|
|
|
|
|
|
2024-05-12 10:57:36 +08:00
|
|
|
|
now := time.Now().UnixMilli()
|
2024-04-20 12:07:17 +08:00
|
|
|
|
for _, src := range srcs {
|
|
|
|
|
srcID := src["id"].(string)
|
2024-07-25 17:34:47 +08:00
|
|
|
|
if !ast.IsNodeIDPattern(srcID) {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-10 17:25:07 +08:00
|
|
|
|
srcItemID := ast.NewNodeID()
|
|
|
|
|
if nil != src["itemID"] {
|
|
|
|
|
srcItemID = src["itemID"].(string)
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-20 22:37:39 +08:00
|
|
|
|
isDetached := src["isDetached"].(bool)
|
2024-01-18 21:01:36 +08:00
|
|
|
|
var tree *parse.Tree
|
|
|
|
|
if !isDetached {
|
|
|
|
|
var loadErr error
|
|
|
|
|
if nil != tx {
|
2024-04-20 12:07:17 +08:00
|
|
|
|
tree, loadErr = tx.loadTree(srcID)
|
2024-01-18 21:01:36 +08:00
|
|
|
|
} else {
|
2024-04-20 12:07:17 +08:00
|
|
|
|
tree, loadErr = LoadTreeByBlockID(srcID)
|
2024-01-18 21:01:36 +08:00
|
|
|
|
}
|
|
|
|
|
if nil != loadErr {
|
2024-04-20 12:26:36 +08:00
|
|
|
|
logging.LogErrorf("load tree [%s] failed: %s", srcID, loadErr)
|
2024-01-18 21:01:36 +08:00
|
|
|
|
return loadErr
|
|
|
|
|
}
|
2023-07-31 21:16:25 +08:00
|
|
|
|
}
|
|
|
|
|
|
2024-04-20 12:22:28 +08:00
|
|
|
|
var srcContent string
|
|
|
|
|
if nil != src["content"] {
|
|
|
|
|
srcContent = src["content"].(string)
|
|
|
|
|
}
|
2025-08-10 17:25:07 +08:00
|
|
|
|
if avErr := addAttributeViewBlock(now, avID, dbBlockID, groupID, previousItemID, srcItemID, srcID, srcContent, isDetached, tree, tx); nil != avErr {
|
2024-01-18 21:01:36 +08:00
|
|
|
|
return avErr
|
2023-07-11 22:36:02 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-10 17:25:07 +08:00
|
|
|
|
func addAttributeViewBlock(now int64, avID, dbBlockID, groupID, previousItemID, addingItemID, addingBoundBlockID, addingBlockContent string, isDetached bool, tree *parse.Tree, tx *Transaction) (err error) {
|
2023-09-27 16:23:46 +08:00
|
|
|
|
var node *ast.Node
|
2024-01-18 21:01:36 +08:00
|
|
|
|
if !isDetached {
|
2025-08-10 17:25:07 +08:00
|
|
|
|
node = treenode.GetNodeInTree(tree, addingBoundBlockID)
|
2023-09-27 16:23:46 +08:00
|
|
|
|
if nil == node {
|
|
|
|
|
err = ErrBlockNotFound
|
|
|
|
|
return
|
|
|
|
|
}
|
2023-09-27 23:14:42 +08:00
|
|
|
|
} else {
|
2025-08-10 17:25:07 +08:00
|
|
|
|
if "" == addingItemID {
|
|
|
|
|
addingItemID = ast.NewNodeID()
|
|
|
|
|
logging.LogWarnf("detached block id is empty, generate a new one [%s]", addingItemID)
|
2023-11-10 17:10:28 +08:00
|
|
|
|
}
|
2023-07-11 22:36:02 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-08-10 17:25:07 +08:00
|
|
|
|
if addingItemID == addingBoundBlockID {
|
|
|
|
|
addingItemID = ast.NewNodeID()
|
|
|
|
|
logging.LogWarnf("the adding item ID is the same as the bound block ID [%s], generate a new one item id [%s]", addingBoundBlockID, addingItemID)
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-18 21:01:36 +08:00
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-11 22:36:02 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-12-09 11:55:29 +08:00
|
|
|
|
var blockIcon string
|
2024-04-08 09:25:19 +08:00
|
|
|
|
if !isDetached {
|
2024-12-09 11:55:29 +08:00
|
|
|
|
blockIcon, addingBlockContent = getNodeAvBlockText(node)
|
2025-01-12 17:57:56 +08:00
|
|
|
|
addingBlockContent = util.UnescapeHTML(addingBlockContent)
|
2024-04-08 09:25:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 检查是否重复添加相同的块
|
2023-07-12 19:10:05 +08:00
|
|
|
|
blockValues := attrView.GetBlockKeyValues()
|
|
|
|
|
for _, blockValue := range blockValues.Values {
|
2025-08-10 17:25:07 +08:00
|
|
|
|
if blockValue.Block.ID == addingBoundBlockID {
|
2024-03-28 21:27:31 +08:00
|
|
|
|
if !isDetached {
|
2024-04-08 09:25:19 +08:00
|
|
|
|
// 重复绑定一下,比如剪切数据库块、取消绑定块后再次添加的场景需要
|
|
|
|
|
bindBlockAv0(tx, avID, node, tree)
|
|
|
|
|
blockValue.IsDetached = isDetached
|
2024-12-09 11:55:29 +08:00
|
|
|
|
blockValue.Block.Icon = blockIcon
|
2024-04-20 12:07:17 +08:00
|
|
|
|
blockValue.Block.Content = addingBlockContent
|
2024-04-08 09:25:19 +08:00
|
|
|
|
blockValue.UpdatedAt = now
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
2024-03-28 21:27:31 +08:00
|
|
|
|
}
|
2023-07-11 22:36:02 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-03 16:12:44 +08:00
|
|
|
|
blockValue := &av.Value{
|
|
|
|
|
ID: ast.NewNodeID(),
|
|
|
|
|
KeyID: blockValues.Key.ID,
|
2025-08-10 17:25:07 +08:00
|
|
|
|
BlockID: addingItemID,
|
2024-03-03 16:12:44 +08:00
|
|
|
|
Type: av.KeyTypeBlock,
|
|
|
|
|
IsDetached: isDetached,
|
|
|
|
|
CreatedAt: now,
|
|
|
|
|
UpdatedAt: now,
|
2025-08-10 11:20:53 +08:00
|
|
|
|
Block: &av.ValueBlock{Icon: blockIcon, Content: addingBlockContent, Created: now, Updated: now}}
|
|
|
|
|
if !isDetached {
|
2025-08-10 17:25:07 +08:00
|
|
|
|
blockValue.Block.ID = addingBoundBlockID
|
2025-08-10 11:20:53 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-12-17 12:11:59 +08:00
|
|
|
|
blockValues.Values = append(blockValues.Values, blockValue)
|
2023-07-11 22:36:02 +08:00
|
|
|
|
|
2025-08-10 17:25:07 +08:00
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, dbBlockID)
|
2025-08-02 17:49:30 +08:00
|
|
|
|
if nil != err {
|
2025-08-10 17:25:07 +08:00
|
|
|
|
logging.LogErrorf("get view by block ID [%s] failed: %s", dbBlockID, err)
|
2025-08-02 17:49:30 +08:00
|
|
|
|
return
|
2025-07-27 17:08:22 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-08-02 17:49:30 +08:00
|
|
|
|
groupView := view
|
|
|
|
|
if "" != groupID {
|
2025-08-09 11:24:44 +08:00
|
|
|
|
groupView = view.GetGroupByID(groupID)
|
2025-07-27 17:08:22 +08:00
|
|
|
|
}
|
2024-03-06 11:20:20 +08:00
|
|
|
|
|
2025-08-10 17:25:07 +08:00
|
|
|
|
fillDefaultValue(attrView, view, groupView, previousItemID, addingItemID)
|
2023-12-17 12:00:55 +08:00
|
|
|
|
|
2024-04-04 11:33:51 +08:00
|
|
|
|
// 处理日期字段默认填充当前创建时间
|
|
|
|
|
// The database date field supports filling the current time by default https://github.com/siyuan-note/siyuan/issues/10823
|
|
|
|
|
for _, keyValues := range attrView.KeyValues {
|
|
|
|
|
if av.KeyTypeDate == keyValues.Key.Type && nil != keyValues.Key.Date && keyValues.Key.Date.AutoFillNow {
|
|
|
|
|
dateVal := &av.Value{
|
2025-08-10 17:25:07 +08:00
|
|
|
|
ID: ast.NewNodeID(), KeyID: keyValues.Key.ID, BlockID: addingItemID, Type: av.KeyTypeDate, IsDetached: isDetached, CreatedAt: now, UpdatedAt: now + 1000,
|
2024-04-04 11:33:51 +08:00
|
|
|
|
Date: &av.ValueDate{Content: now, IsNotEmpty: true},
|
|
|
|
|
}
|
|
|
|
|
keyValues.Values = append(keyValues.Values, dateVal)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-18 21:01:36 +08:00
|
|
|
|
if !isDetached {
|
2024-04-08 09:25:19 +08:00
|
|
|
|
bindBlockAv0(tx, avID, node, tree)
|
2023-07-11 22:36:02 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-07-27 17:08:22 +08:00
|
|
|
|
// 在所有视图上添加项目
|
2024-04-04 11:33:51 +08:00
|
|
|
|
for _, v := range attrView.Views {
|
2025-08-10 17:25:07 +08:00
|
|
|
|
if "" != previousItemID {
|
2025-07-02 17:02:40 +08:00
|
|
|
|
changed := false
|
|
|
|
|
for i, id := range v.ItemIDs {
|
2025-08-10 17:25:07 +08:00
|
|
|
|
if id == previousItemID {
|
|
|
|
|
v.ItemIDs = append(v.ItemIDs[:i+1], append([]string{addingItemID}, v.ItemIDs[i+1:]...)...)
|
2025-07-02 17:02:40 +08:00
|
|
|
|
changed = true
|
|
|
|
|
break
|
2023-12-14 11:23:34 +08:00
|
|
|
|
}
|
2023-07-11 22:36:02 +08:00
|
|
|
|
}
|
2025-07-02 17:02:40 +08:00
|
|
|
|
if !changed {
|
2025-08-10 17:25:07 +08:00
|
|
|
|
v.ItemIDs = append(v.ItemIDs, addingItemID)
|
2025-06-09 12:21:18 +08:00
|
|
|
|
}
|
2025-07-02 17:02:40 +08:00
|
|
|
|
} else {
|
2025-08-10 17:25:07 +08:00
|
|
|
|
v.ItemIDs = append([]string{addingItemID}, v.ItemIDs...)
|
2023-07-11 22:36:02 +08:00
|
|
|
|
}
|
2025-07-27 17:08:22 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-08-09 20:39:12 +08:00
|
|
|
|
regenAttrViewGroups(attrView, "force")
|
2023-07-11 22:36:02 +08:00
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-09 13:28:56 +08:00
|
|
|
|
func fillDefaultValue(attrView *av.AttributeView, view, groupView *av.View, previousBlockID, addingBlockID string) {
|
|
|
|
|
defaultValues := getAttrViewAddingBlockDefaultValues(attrView, view, groupView, previousBlockID, addingBlockID)
|
|
|
|
|
for keyID, newValue := range defaultValues {
|
|
|
|
|
newValue.BlockID = addingBlockID
|
|
|
|
|
keyValues, getErr := attrView.GetKeyValues(keyID)
|
|
|
|
|
if nil != getErr {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if av.KeyTypeRollup == newValue.Type {
|
|
|
|
|
// 汇总字段的值是渲染时计算的,不需要添加到数据存储中
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (av.KeyTypeSelect == newValue.Type || av.KeyTypeMSelect == newValue.Type) && 1 > len(newValue.MSelect) {
|
|
|
|
|
// 单选或多选类型的值可能需要从分组条件中获取默认值
|
|
|
|
|
if groupValueDefault != groupView.GetGroupValue() {
|
|
|
|
|
if opt := keyValues.Key.GetOption(groupView.GetGroupValue()); nil != opt {
|
|
|
|
|
newValue.MSelect = append(newValue.MSelect, &av.ValueSelect{Content: opt.Name, Color: opt.Color})
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if av.KeyTypeRelation == newValue.Type && nil != keyValues.Key.Relation && keyValues.Key.Relation.IsTwoWay {
|
|
|
|
|
// 双向关联需要同时更新目标字段的值
|
|
|
|
|
updateTwoWayRelationDestAttrView(attrView, keyValues.Key, newValue, 1, []string{})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
existingVal := keyValues.GetValue(addingBlockID)
|
|
|
|
|
if nil == existingVal {
|
|
|
|
|
keyValues.Values = append(keyValues.Values, newValue)
|
|
|
|
|
} else {
|
|
|
|
|
newValueRaw := newValue.GetValByType(keyValues.Key.Type)
|
|
|
|
|
existingVal.SetValByType(keyValues.Key.Type, newValueRaw)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-02 17:49:30 +08:00
|
|
|
|
func getNewValueByNearItem(nearItem av.Item, key *av.Key, addingBlockID string) (ret *av.Value) {
|
2025-07-28 22:53:04 +08:00
|
|
|
|
if nil != nearItem {
|
|
|
|
|
defaultVal := nearItem.GetValue(key.ID)
|
|
|
|
|
ret = defaultVal.Clone()
|
2025-08-02 17:49:30 +08:00
|
|
|
|
ret.ID = ast.NewNodeID()
|
|
|
|
|
ret.KeyID = key.ID
|
|
|
|
|
ret.BlockID = addingBlockID
|
|
|
|
|
ret.CreatedAt = util.CurrentTimeMillis()
|
|
|
|
|
ret.UpdatedAt = ret.CreatedAt + 1000
|
|
|
|
|
return
|
2025-07-28 22:53:04 +08:00
|
|
|
|
}
|
2025-08-02 17:49:30 +08:00
|
|
|
|
return av.GetAttributeViewDefaultValue(ast.NewNodeID(), key.ID, addingBlockID, key.Type)
|
2025-07-28 22:53:04 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-07-28 22:44:56 +08:00
|
|
|
|
func getNearItem(attrView *av.AttributeView, view, groupView *av.View, previousItemID string) (ret av.Item) {
|
2025-08-01 22:07:11 +08:00
|
|
|
|
viewable := sql.RenderGroupView(attrView, view, groupView, "")
|
2025-07-28 22:44:56 +08:00
|
|
|
|
av.Filter(viewable, attrView)
|
|
|
|
|
av.Sort(viewable, attrView)
|
|
|
|
|
items := viewable.(av.Collection).GetItems()
|
|
|
|
|
if 0 < len(items) {
|
|
|
|
|
if "" != previousItemID {
|
|
|
|
|
for _, row := range items {
|
|
|
|
|
if row.GetID() == previousItemID {
|
|
|
|
|
ret = row
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
if 0 < len(items) {
|
|
|
|
|
ret = items[0]
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-11 22:36:02 +08:00
|
|
|
|
func (tx *Transaction) doRemoveAttrViewBlock(operation *Operation) (ret *TxErr) {
|
2024-01-18 21:01:36 +08:00
|
|
|
|
err := removeAttributeViewBlock(operation.SrcIDs, operation.AvID, tx)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID}
|
2023-07-11 22:37:18 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-18 21:01:36 +08:00
|
|
|
|
func RemoveAttributeViewBlock(srcIDs []string, avID string) (err error) {
|
|
|
|
|
err = removeAttributeViewBlock(srcIDs, avID, nil)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func removeAttributeViewBlock(srcIDs []string, avID string, tx *Transaction) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-11 22:37:18 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
2023-07-11 22:36:02 +08:00
|
|
|
|
|
2023-09-07 18:02:47 +08:00
|
|
|
|
trees := map[string]*parse.Tree{}
|
2023-07-12 19:10:05 +08:00
|
|
|
|
for _, keyValues := range attrView.KeyValues {
|
2023-08-02 00:02:30 +08:00
|
|
|
|
tmp := keyValues.Values[:0]
|
2023-07-12 19:10:05 +08:00
|
|
|
|
for i, values := range keyValues.Values {
|
2024-01-18 21:01:36 +08:00
|
|
|
|
if !gulu.Str.Contains(values.BlockID, srcIDs) {
|
2023-08-02 00:02:30 +08:00
|
|
|
|
tmp = append(tmp, keyValues.Values[i])
|
2023-09-07 18:02:47 +08:00
|
|
|
|
} else {
|
2023-09-07 18:06:10 +08:00
|
|
|
|
// Remove av block also remove node attr https://github.com/siyuan-note/siyuan/issues/9091#issuecomment-1709824006
|
2023-09-07 18:02:47 +08:00
|
|
|
|
if bt := treenode.GetBlockTree(values.BlockID); nil != bt {
|
|
|
|
|
tree := trees[bt.RootID]
|
|
|
|
|
if nil == tree {
|
2024-03-10 23:27:13 +08:00
|
|
|
|
tree, _ = LoadTreeByBlockID(values.BlockID)
|
2023-09-07 18:02:47 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if nil != tree {
|
|
|
|
|
trees[bt.RootID] = tree
|
|
|
|
|
if node := treenode.GetNodeInTree(tree, values.BlockID); nil != node {
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = removeNodeAvID(node, avID, tx, tree); err != nil {
|
2024-02-16 16:02:22 +08:00
|
|
|
|
return
|
2023-09-07 18:02:47 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-07-12 19:10:05 +08:00
|
|
|
|
}
|
2023-07-11 22:37:18 +08:00
|
|
|
|
}
|
2023-08-02 00:02:30 +08:00
|
|
|
|
keyValues.Values = tmp
|
2023-07-11 19:45:27 +08:00
|
|
|
|
}
|
2023-07-11 22:37:18 +08:00
|
|
|
|
|
2023-12-14 11:23:34 +08:00
|
|
|
|
for _, view := range attrView.Views {
|
2024-01-18 21:01:36 +08:00
|
|
|
|
for _, blockID := range srcIDs {
|
2025-07-02 17:02:40 +08:00
|
|
|
|
view.ItemIDs = gulu.Str.RemoveElem(view.ItemIDs, blockID)
|
2023-12-14 11:23:34 +08:00
|
|
|
|
}
|
2023-08-02 00:02:30 +08:00
|
|
|
|
}
|
2023-07-13 09:37:52 +08:00
|
|
|
|
|
2025-08-09 20:39:12 +08:00
|
|
|
|
regenAttrViewGroups(attrView, "force")
|
2025-07-29 14:02:54 +08:00
|
|
|
|
|
2024-04-03 21:33:53 +08:00
|
|
|
|
relatedAvIDs := av.GetSrcAvIDs(avID)
|
|
|
|
|
for _, relatedAvID := range relatedAvIDs {
|
2024-09-05 12:10:42 +08:00
|
|
|
|
ReloadAttrView(relatedAvID)
|
2024-04-03 21:33:53 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-07-11 22:37:18 +08:00
|
|
|
|
err = av.SaveAttributeView(attrView)
|
2025-07-27 19:08:27 +08:00
|
|
|
|
if nil != err {
|
|
|
|
|
return
|
|
|
|
|
}
|
2025-01-04 22:07:27 +08:00
|
|
|
|
|
|
|
|
|
historyDir, err := GetHistoryDir(HistoryOpUpdate)
|
|
|
|
|
if err != nil {
|
|
|
|
|
logging.LogErrorf("get history dir failed: %s", err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
blockIDs := treenode.GetMirrorAttrViewBlockIDs(avID)
|
|
|
|
|
for _, blockID := range blockIDs {
|
|
|
|
|
tree := trees[blockID]
|
|
|
|
|
if nil == tree {
|
|
|
|
|
tree, _ = LoadTreeByBlockID(blockID)
|
|
|
|
|
}
|
|
|
|
|
if nil == tree {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
historyPath := filepath.Join(historyDir, tree.Box, tree.Path)
|
|
|
|
|
absPath := filepath.Join(util.DataDir, tree.Box, tree.Path)
|
|
|
|
|
if err = filelock.Copy(absPath, historyPath); err != nil {
|
|
|
|
|
logging.LogErrorf("backup [path=%s] to history [%s] failed: %s", absPath, historyPath, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
srcAvPath := filepath.Join(util.DataDir, "storage", "av", avID+".json")
|
|
|
|
|
destAvPath := filepath.Join(historyDir, "storage", "av", avID+".json")
|
|
|
|
|
if copyErr := filelock.Copy(srcAvPath, destAvPath); nil != copyErr {
|
|
|
|
|
logging.LogErrorf("copy av [%s] failed: %s", srcAvPath, copyErr)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
indexHistoryDir(filepath.Base(historyDir), util.NewLute())
|
2023-07-03 19:06:54 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-02-16 16:02:22 +08:00
|
|
|
|
func removeNodeAvID(node *ast.Node, avID string, tx *Transaction, tree *parse.Tree) (err error) {
|
|
|
|
|
attrs := parse.IAL2Map(node.KramdownIAL)
|
|
|
|
|
if ast.NodeDocument == node.Type {
|
|
|
|
|
delete(attrs, "custom-hidden")
|
|
|
|
|
node.RemoveIALAttr("custom-hidden")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if avs := attrs[av.NodeAttrNameAvs]; "" != avs {
|
|
|
|
|
avIDs := strings.Split(avs, ",")
|
|
|
|
|
avIDs = gulu.Str.RemoveElem(avIDs, avID)
|
|
|
|
|
var existAvIDs []string
|
|
|
|
|
for _, attributeViewID := range avIDs {
|
|
|
|
|
if av.IsAttributeViewExist(attributeViewID) {
|
|
|
|
|
existAvIDs = append(existAvIDs, attributeViewID)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
avIDs = existAvIDs
|
|
|
|
|
|
|
|
|
|
if 0 == len(avIDs) {
|
2024-03-10 22:57:21 +08:00
|
|
|
|
attrs[av.NodeAttrNameAvs] = ""
|
2024-02-16 16:02:22 +08:00
|
|
|
|
} else {
|
|
|
|
|
attrs[av.NodeAttrNameAvs] = strings.Join(avIDs, ",")
|
|
|
|
|
node.SetIALAttr(av.NodeAttrNameAvs, strings.Join(avIDs, ","))
|
2024-03-10 22:57:21 +08:00
|
|
|
|
avNames := getAvNames(node.IALAttr(av.NodeAttrNameAvs))
|
2024-03-10 23:00:45 +08:00
|
|
|
|
attrs[av.NodeAttrViewNames] = avNames
|
2024-02-16 16:02:22 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if nil != tx {
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = setNodeAttrsWithTx(tx, node, tree, attrs); err != nil {
|
2024-02-16 16:02:22 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
} else {
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = setNodeAttrs(node, tree, attrs); err != nil {
|
2024-02-16 16:02:22 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-06-10 16:25:57 +08:00
|
|
|
|
func (tx *Transaction) doDuplicateAttrViewKey(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := duplicateAttributeViewKey(operation)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2024-06-10 16:25:57 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func duplicateAttributeViewKey(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2024-06-10 16:25:57 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
key, _ := attrView.GetKey(operation.KeyID)
|
|
|
|
|
if nil == key {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-06 16:58:30 +08:00
|
|
|
|
if av.KeyTypeBlock == key.Type || av.KeyTypeRelation == key.Type {
|
2024-06-10 16:25:57 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
copyKey := &av.Key{}
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = copier.Copy(copyKey, key); err != nil {
|
2024-06-10 16:25:57 +08:00
|
|
|
|
logging.LogErrorf("clone key failed: %s", err)
|
|
|
|
|
}
|
|
|
|
|
copyKey.ID = operation.NextID
|
|
|
|
|
copyKey.Name = util.GetDuplicateName(key.Name)
|
|
|
|
|
|
|
|
|
|
attrView.KeyValues = append(attrView.KeyValues, &av.KeyValues{Key: copyKey})
|
|
|
|
|
|
|
|
|
|
for _, view := range attrView.Views {
|
|
|
|
|
switch view.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
for i, column := range view.Table.Columns {
|
|
|
|
|
if column.ID == key.ID {
|
2024-06-10 17:10:33 +08:00
|
|
|
|
view.Table.Columns = append(view.Table.Columns[:i+1], append([]*av.ViewTableColumn{
|
|
|
|
|
{
|
2025-06-30 15:06:43 +08:00
|
|
|
|
BaseField: &av.BaseField{
|
|
|
|
|
ID: copyKey.ID,
|
|
|
|
|
Wrap: column.Wrap,
|
|
|
|
|
Hidden: column.Hidden,
|
|
|
|
|
Desc: column.Desc,
|
|
|
|
|
},
|
|
|
|
|
Pin: column.Pin,
|
|
|
|
|
Width: column.Width,
|
2024-06-10 17:10:33 +08:00
|
|
|
|
},
|
|
|
|
|
}, view.Table.Columns[i+1:]...)...)
|
2024-06-10 16:25:57 +08:00
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-06-09 16:46:50 +08:00
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
for i, field := range view.Gallery.CardFields {
|
|
|
|
|
if field.ID == key.ID {
|
|
|
|
|
view.Gallery.CardFields = append(view.Gallery.CardFields[:i+1], append([]*av.ViewGalleryCardField{
|
|
|
|
|
{
|
2025-06-30 15:06:43 +08:00
|
|
|
|
BaseField: &av.BaseField{
|
|
|
|
|
ID: copyKey.ID,
|
|
|
|
|
Wrap: field.Wrap,
|
|
|
|
|
Hidden: field.Hidden,
|
|
|
|
|
Desc: field.Desc,
|
|
|
|
|
},
|
2025-06-09 16:46:50 +08:00
|
|
|
|
},
|
|
|
|
|
}, view.Gallery.CardFields[i+1:]...)...)
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-06-10 16:25:57 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-11 22:44:31 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewColumnWidth(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := setAttributeViewColWidth(operation)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-07-11 22:44:31 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func setAttributeViewColWidth(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-11 22:44:31 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-04 16:41:41 +08:00
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, operation.BlockID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-11 23:40:05 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-12 21:39:55 +08:00
|
|
|
|
switch view.LayoutType {
|
2023-07-12 19:10:05 +08:00
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
for _, column := range view.Table.Columns {
|
|
|
|
|
if column.ID == operation.ID {
|
|
|
|
|
column.Width = operation.Data.(string)
|
|
|
|
|
break
|
|
|
|
|
}
|
2023-07-11 22:44:31 +08:00
|
|
|
|
}
|
2025-06-09 16:46:50 +08:00
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
return
|
2023-07-11 22:44:31 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (tx *Transaction) doSetAttrViewColumnWrap(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := setAttributeViewColWrap(operation)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-07-11 22:44:31 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func setAttributeViewColWrap(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-11 22:44:31 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-04 16:41:41 +08:00
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, operation.BlockID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-11 23:40:05 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-30 12:50:50 +08:00
|
|
|
|
newWrap := operation.Data.(bool)
|
|
|
|
|
allFieldWrap := true
|
2023-07-12 21:39:55 +08:00
|
|
|
|
switch view.LayoutType {
|
2023-07-12 19:10:05 +08:00
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
for _, column := range view.Table.Columns {
|
|
|
|
|
if column.ID == operation.ID {
|
2025-06-30 12:50:50 +08:00
|
|
|
|
column.Wrap = newWrap
|
2023-07-12 19:10:05 +08:00
|
|
|
|
}
|
2025-06-30 12:50:50 +08:00
|
|
|
|
allFieldWrap = allFieldWrap && column.Wrap
|
2023-07-11 22:44:31 +08:00
|
|
|
|
}
|
2025-06-30 12:50:50 +08:00
|
|
|
|
view.Table.WrapField = allFieldWrap
|
2025-06-09 16:46:50 +08:00
|
|
|
|
case av.LayoutTypeGallery:
|
2025-06-30 12:50:50 +08:00
|
|
|
|
for _, field := range view.Gallery.CardFields {
|
|
|
|
|
if field.ID == operation.ID {
|
|
|
|
|
field.Wrap = newWrap
|
|
|
|
|
}
|
|
|
|
|
allFieldWrap = allFieldWrap && field.Wrap
|
|
|
|
|
}
|
|
|
|
|
view.Gallery.WrapField = allFieldWrap
|
2023-07-11 22:44:31 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (tx *Transaction) doSetAttrViewColumnHidden(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := setAttributeViewColHidden(operation)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-07-11 22:44:31 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func setAttributeViewColHidden(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-11 22:44:31 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-04 16:41:41 +08:00
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, operation.BlockID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-11 23:40:05 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-12 21:39:55 +08:00
|
|
|
|
switch view.LayoutType {
|
2023-07-12 19:10:05 +08:00
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
for _, column := range view.Table.Columns {
|
|
|
|
|
if column.ID == operation.ID {
|
|
|
|
|
column.Hidden = operation.Data.(bool)
|
|
|
|
|
break
|
|
|
|
|
}
|
2023-07-11 22:44:31 +08:00
|
|
|
|
}
|
2025-06-09 16:46:50 +08:00
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
for _, field := range view.Gallery.CardFields {
|
|
|
|
|
if field.ID == operation.ID {
|
|
|
|
|
field.Hidden = operation.Data.(bool)
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-07-11 22:44:31 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-11-10 10:22:19 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewColumnPin(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := setAttributeViewColPin(operation)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-11-10 10:22:19 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func setAttributeViewColPin(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-11-10 10:22:19 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-04 16:41:41 +08:00
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, operation.BlockID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-11-10 10:22:19 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch view.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
for _, column := range view.Table.Columns {
|
|
|
|
|
if column.ID == operation.ID {
|
|
|
|
|
column.Pin = operation.Data.(bool)
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-06-09 16:46:50 +08:00
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
return
|
2023-11-10 10:22:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-05 11:12:57 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewColumnIcon(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := setAttributeViewColIcon(operation)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-10-05 11:12:57 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func setAttributeViewColIcon(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-10-05 11:12:57 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, keyValues := range attrView.KeyValues {
|
|
|
|
|
if keyValues.Key.ID == operation.ID {
|
|
|
|
|
keyValues.Key.Icon = operation.Data.(string)
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-09 14:09:40 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewColumnDesc(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := setAttributeViewColDesc(operation)
|
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2024-11-09 14:09:40 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func setAttributeViewColDesc(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, keyValues := range attrView.KeyValues {
|
|
|
|
|
if keyValues.Key.ID == operation.ID {
|
|
|
|
|
keyValues.Key.Desc = operation.Data.(string)
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-11 22:47:19 +08:00
|
|
|
|
func (tx *Transaction) doSortAttrViewRow(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := sortAttributeViewRow(operation)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-07-11 22:47:19 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func sortAttributeViewRow(operation *Operation) (err error) {
|
2024-04-18 11:15:32 +08:00
|
|
|
|
if operation.ID == operation.PreviousID {
|
|
|
|
|
// 拖拽到自己的下方,不做任何操作 https://github.com/siyuan-note/siyuan/issues/11048
|
2023-07-11 22:47:19 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-18 11:15:32 +08:00
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-12 10:35:17 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-18 11:15:32 +08:00
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, operation.BlockID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2024-04-16 17:29:13 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-18 12:25:40 +08:00
|
|
|
|
var itemID string
|
2024-04-21 19:26:35 +08:00
|
|
|
|
var idx, previousIndex int
|
2025-07-27 19:05:29 +08:00
|
|
|
|
|
|
|
|
|
if nil != view.Group && "" != operation.GroupID {
|
2025-08-09 11:24:44 +08:00
|
|
|
|
if groupView := view.GetGroupByID(operation.GroupID); nil != groupView {
|
2025-07-27 19:05:29 +08:00
|
|
|
|
for i, id := range groupView.GroupItemIDs {
|
|
|
|
|
if id == operation.ID {
|
|
|
|
|
itemID = id
|
|
|
|
|
idx = i
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if "" == itemID {
|
|
|
|
|
itemID = operation.ID
|
|
|
|
|
groupView.GroupItemIDs = append(groupView.GroupItemIDs, itemID)
|
|
|
|
|
idx = len(groupView.GroupItemIDs) - 1
|
|
|
|
|
}
|
|
|
|
|
groupView.GroupItemIDs = append(groupView.GroupItemIDs[:idx], groupView.GroupItemIDs[idx+1:]...)
|
2025-07-28 11:17:27 +08:00
|
|
|
|
|
2025-07-28 22:53:04 +08:00
|
|
|
|
if operation.GroupID != operation.TargetGroupID { // 跨分组排序
|
2025-08-09 11:24:44 +08:00
|
|
|
|
if targetGroupView := view.GetGroupByID(operation.TargetGroupID); nil != targetGroupView {
|
2025-08-09 13:28:56 +08:00
|
|
|
|
fillDefaultValue(attrView, view, targetGroupView, operation.PreviousID, itemID)
|
2025-07-28 22:53:04 +08:00
|
|
|
|
|
|
|
|
|
for i, r := range targetGroupView.GroupItemIDs {
|
|
|
|
|
if r == operation.PreviousID {
|
|
|
|
|
previousIndex = i + 1
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
targetGroupView.GroupItemIDs = util.InsertElem(targetGroupView.GroupItemIDs, previousIndex, itemID)
|
2025-07-28 22:44:56 +08:00
|
|
|
|
}
|
2025-07-28 22:53:04 +08:00
|
|
|
|
} else { // 同分组内排序
|
|
|
|
|
for i, r := range groupView.GroupItemIDs {
|
2025-07-28 11:17:27 +08:00
|
|
|
|
if r == operation.PreviousID {
|
|
|
|
|
previousIndex = i + 1
|
|
|
|
|
break
|
|
|
|
|
}
|
2025-07-27 19:05:29 +08:00
|
|
|
|
}
|
2025-07-28 22:53:04 +08:00
|
|
|
|
groupView.GroupItemIDs = util.InsertElem(groupView.GroupItemIDs, previousIndex, itemID)
|
2025-07-27 19:05:29 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
for i, id := range view.ItemIDs {
|
|
|
|
|
if id == operation.ID {
|
|
|
|
|
itemID = id
|
|
|
|
|
idx = i
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if "" == itemID {
|
|
|
|
|
itemID = operation.ID
|
|
|
|
|
view.ItemIDs = append(view.ItemIDs, itemID)
|
|
|
|
|
idx = len(view.ItemIDs) - 1
|
2025-06-18 12:25:40 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-07-27 19:05:29 +08:00
|
|
|
|
view.ItemIDs = append(view.ItemIDs[:idx], view.ItemIDs[idx+1:]...)
|
|
|
|
|
for i, r := range view.ItemIDs {
|
|
|
|
|
if r == operation.PreviousID {
|
|
|
|
|
previousIndex = i + 1
|
|
|
|
|
break
|
|
|
|
|
}
|
2025-06-09 16:46:50 +08:00
|
|
|
|
}
|
2025-07-27 19:05:29 +08:00
|
|
|
|
view.ItemIDs = util.InsertElem(view.ItemIDs, previousIndex, itemID)
|
2023-07-11 22:47:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (tx *Transaction) doSortAttrViewColumn(operation *Operation) (ret *TxErr) {
|
2024-05-16 22:24:15 +08:00
|
|
|
|
err := SortAttributeViewViewKey(operation.AvID, operation.BlockID, operation.ID, operation.PreviousID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-07-11 22:47:19 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-05-16 22:24:15 +08:00
|
|
|
|
func SortAttributeViewViewKey(avID, blockID, keyID, previousKeyID string) (err error) {
|
2024-04-18 11:15:32 +08:00
|
|
|
|
if keyID == previousKeyID {
|
|
|
|
|
// 拖拽到自己的右侧,不做任何操作 https://github.com/siyuan-note/siyuan/issues/11048
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-12 21:24:55 +08:00
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-11 22:47:19 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-04 16:41:41 +08:00
|
|
|
|
view, err := getAttrViewViewByBlockID(attrView, blockID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-12 10:35:17 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-09 16:46:50 +08:00
|
|
|
|
var curIndex, previousIndex int
|
2023-07-12 21:39:55 +08:00
|
|
|
|
switch view.LayoutType {
|
2023-07-12 19:10:05 +08:00
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
var col *av.ViewTableColumn
|
|
|
|
|
for i, column := range view.Table.Columns {
|
2024-01-12 21:24:55 +08:00
|
|
|
|
if column.ID == keyID {
|
2023-07-12 19:10:05 +08:00
|
|
|
|
col = column
|
2025-06-09 16:46:50 +08:00
|
|
|
|
curIndex = i
|
2023-07-12 19:10:05 +08:00
|
|
|
|
break
|
|
|
|
|
}
|
2023-07-11 22:47:19 +08:00
|
|
|
|
}
|
2023-07-12 19:10:05 +08:00
|
|
|
|
if nil == col {
|
|
|
|
|
return
|
2023-07-11 22:47:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-06-09 16:46:50 +08:00
|
|
|
|
view.Table.Columns = append(view.Table.Columns[:curIndex], view.Table.Columns[curIndex+1:]...)
|
2023-07-12 19:10:05 +08:00
|
|
|
|
for i, column := range view.Table.Columns {
|
2024-01-12 21:24:55 +08:00
|
|
|
|
if column.ID == previousKeyID {
|
2023-07-12 19:10:05 +08:00
|
|
|
|
previousIndex = i + 1
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
view.Table.Columns = util.InsertElem(view.Table.Columns, previousIndex, col)
|
2025-06-09 16:46:50 +08:00
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
var field *av.ViewGalleryCardField
|
|
|
|
|
for i, cardField := range view.Gallery.CardFields {
|
|
|
|
|
if cardField.ID == keyID {
|
|
|
|
|
field = cardField
|
|
|
|
|
curIndex = i
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if nil == field {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view.Gallery.CardFields = append(view.Gallery.CardFields[:curIndex], view.Gallery.CardFields[curIndex+1:]...)
|
|
|
|
|
for i, cardField := range view.Gallery.CardFields {
|
|
|
|
|
if cardField.ID == previousKeyID {
|
|
|
|
|
previousIndex = i + 1
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-06-12 18:19:08 +08:00
|
|
|
|
view.Gallery.CardFields = util.InsertElem(view.Gallery.CardFields, previousIndex, field)
|
2023-07-11 22:47:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-05-16 22:24:15 +08:00
|
|
|
|
func (tx *Transaction) doSortAttrViewKey(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := SortAttributeViewKey(operation.AvID, operation.ID, operation.PreviousID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2024-05-16 22:24:15 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func SortAttributeViewKey(avID, keyID, previousKeyID string) (err error) {
|
|
|
|
|
if keyID == previousKeyID {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2024-05-16 22:24:15 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-09-04 11:49:22 +08:00
|
|
|
|
refreshAttrViewKeyIDs(attrView, false)
|
2024-05-16 22:24:15 +08:00
|
|
|
|
|
|
|
|
|
var currentKeyID string
|
|
|
|
|
var idx, previousIndex int
|
|
|
|
|
for i, k := range attrView.KeyIDs {
|
|
|
|
|
if k == keyID {
|
|
|
|
|
currentKeyID = k
|
|
|
|
|
idx = i
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if "" == currentKeyID {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
attrView.KeyIDs = append(attrView.KeyIDs[:idx], attrView.KeyIDs[idx+1:]...)
|
|
|
|
|
|
|
|
|
|
for i, k := range attrView.KeyIDs {
|
|
|
|
|
if k == previousKeyID {
|
|
|
|
|
previousIndex = i + 1
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
attrView.KeyIDs = util.InsertElem(attrView.KeyIDs, previousIndex, currentKeyID)
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-09-04 11:49:22 +08:00
|
|
|
|
func refreshAttrViewKeyIDs(attrView *av.AttributeView, needSave bool) {
|
2024-06-03 11:35:43 +08:00
|
|
|
|
// 订正 keyIDs 数据
|
|
|
|
|
|
|
|
|
|
existKeyIDs := map[string]bool{}
|
|
|
|
|
for _, keyValues := range attrView.KeyValues {
|
|
|
|
|
existKeyIDs[keyValues.Key.ID] = true
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for k, _ := range existKeyIDs {
|
|
|
|
|
if !gulu.Str.Contains(k, attrView.KeyIDs) {
|
|
|
|
|
attrView.KeyIDs = append(attrView.KeyIDs, k)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var tmp []string
|
|
|
|
|
for _, k := range attrView.KeyIDs {
|
|
|
|
|
if ok := existKeyIDs[k]; ok {
|
|
|
|
|
tmp = append(tmp, k)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
attrView.KeyIDs = tmp
|
2024-09-04 11:49:22 +08:00
|
|
|
|
|
|
|
|
|
if needSave {
|
|
|
|
|
av.SaveAttributeView(attrView)
|
|
|
|
|
}
|
2024-06-03 11:35:43 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-07-11 23:40:05 +08:00
|
|
|
|
func (tx *Transaction) doAddAttrViewColumn(operation *Operation) (ret *TxErr) {
|
2024-01-12 21:24:55 +08:00
|
|
|
|
var icon string
|
|
|
|
|
if nil != operation.Data {
|
|
|
|
|
icon = operation.Data.(string)
|
|
|
|
|
}
|
|
|
|
|
err := AddAttributeViewKey(operation.AvID, operation.ID, operation.Name, operation.Typ, icon, operation.PreviousID)
|
|
|
|
|
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-07-11 23:40:05 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-12 21:24:55 +08:00
|
|
|
|
func AddAttributeViewKey(avID, keyID, keyName, keyType, keyIcon, previousKeyID string) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-11 23:40:05 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-22 22:57:00 +08:00
|
|
|
|
currentView, err := attrView.GetCurrentView(attrView.ViewID)
|
|
|
|
|
if nil != err {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-12 21:24:55 +08:00
|
|
|
|
keyTyp := av.KeyType(keyType)
|
|
|
|
|
switch keyTyp {
|
2023-12-15 20:05:14 +08:00
|
|
|
|
case av.KeyTypeText, av.KeyTypeNumber, av.KeyTypeDate, av.KeyTypeSelect, av.KeyTypeMSelect, av.KeyTypeURL, av.KeyTypeEmail,
|
|
|
|
|
av.KeyTypePhone, av.KeyTypeMAsset, av.KeyTypeTemplate, av.KeyTypeCreated, av.KeyTypeUpdated, av.KeyTypeCheckbox,
|
2024-04-15 00:49:48 +08:00
|
|
|
|
av.KeyTypeRelation, av.KeyTypeRollup, av.KeyTypeLineNumber:
|
2024-01-12 21:24:55 +08:00
|
|
|
|
|
|
|
|
|
key := av.NewKey(keyID, keyName, keyIcon, keyTyp)
|
|
|
|
|
if av.KeyTypeRollup == keyTyp {
|
2024-01-01 21:17:36 +08:00
|
|
|
|
key.Rollup = &av.Rollup{Calc: &av.RollupCalc{Operator: av.CalcOperatorNone}}
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-12 19:10:05 +08:00
|
|
|
|
attrView.KeyValues = append(attrView.KeyValues, &av.KeyValues{Key: key})
|
2023-07-11 23:40:05 +08:00
|
|
|
|
|
2023-12-29 09:11:40 +08:00
|
|
|
|
for _, view := range attrView.Views {
|
2025-06-21 11:28:17 +08:00
|
|
|
|
if nil != view.Table {
|
2024-01-12 21:24:55 +08:00
|
|
|
|
if "" == previousKeyID {
|
2025-06-22 22:57:00 +08:00
|
|
|
|
if av.LayoutTypeGallery == currentView.LayoutType {
|
2025-07-04 12:00:47 +08:00
|
|
|
|
// 如果当前视图是卡片视图则添加到最后
|
2025-06-30 15:06:43 +08:00
|
|
|
|
view.Table.Columns = append(view.Table.Columns, &av.ViewTableColumn{BaseField: &av.BaseField{ID: key.ID}})
|
2025-06-21 11:28:17 +08:00
|
|
|
|
} else {
|
2025-06-30 15:06:43 +08:00
|
|
|
|
view.Table.Columns = append([]*av.ViewTableColumn{{BaseField: &av.BaseField{ID: key.ID}}}, view.Table.Columns...)
|
2025-06-21 11:28:17 +08:00
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
added := false
|
|
|
|
|
for i, column := range view.Table.Columns {
|
|
|
|
|
if column.ID == previousKeyID {
|
2025-06-30 15:06:43 +08:00
|
|
|
|
view.Table.Columns = append(view.Table.Columns[:i+1], append([]*av.ViewTableColumn{{BaseField: &av.BaseField{ID: key.ID}}}, view.Table.Columns[i+1:]...)...)
|
2025-06-21 11:28:17 +08:00
|
|
|
|
added = true
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if !added {
|
2025-06-30 15:06:43 +08:00
|
|
|
|
view.Table.Columns = append(view.Table.Columns, &av.ViewTableColumn{BaseField: &av.BaseField{ID: key.ID}})
|
2023-12-29 09:11:40 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2025-06-21 11:28:17 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if nil != view.Gallery {
|
2025-06-09 12:21:18 +08:00
|
|
|
|
if "" == previousKeyID {
|
2025-06-30 15:06:43 +08:00
|
|
|
|
view.Gallery.CardFields = append(view.Gallery.CardFields, &av.ViewGalleryCardField{BaseField: &av.BaseField{ID: key.ID}})
|
2025-06-21 11:28:17 +08:00
|
|
|
|
} else {
|
|
|
|
|
added := false
|
|
|
|
|
for i, field := range view.Gallery.CardFields {
|
|
|
|
|
if field.ID == previousKeyID {
|
2025-06-30 15:06:43 +08:00
|
|
|
|
view.Gallery.CardFields = append(view.Gallery.CardFields[:i+1], append([]*av.ViewGalleryCardField{{BaseField: &av.BaseField{ID: key.ID}}}, view.Gallery.CardFields[i+1:]...)...)
|
2025-06-21 11:28:17 +08:00
|
|
|
|
added = true
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if !added {
|
2025-06-30 15:06:43 +08:00
|
|
|
|
view.Gallery.CardFields = append(view.Gallery.CardFields, &av.ViewGalleryCardField{BaseField: &av.BaseField{ID: key.ID}})
|
2025-06-09 12:21:18 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2023-12-01 20:15:14 +08:00
|
|
|
|
}
|
2023-07-11 23:40:05 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-01 17:33:53 +08:00
|
|
|
|
func (tx *Transaction) doUpdateAttrViewColTemplate(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := updateAttributeViewColTemplate(operation)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-10-01 17:33:53 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func updateAttributeViewColTemplate(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-10-01 17:33:53 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
colType := av.KeyType(operation.Typ)
|
|
|
|
|
switch colType {
|
|
|
|
|
case av.KeyTypeTemplate:
|
|
|
|
|
for _, keyValues := range attrView.KeyValues {
|
|
|
|
|
if keyValues.Key.ID == operation.ID && av.KeyTypeTemplate == keyValues.Key.Type {
|
|
|
|
|
keyValues.Key.Template = operation.Data.(string)
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-09 20:39:12 +08:00
|
|
|
|
regenAttrViewGroups(attrView, operation.ID)
|
2023-10-01 17:33:53 +08:00
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-03 23:38:45 +08:00
|
|
|
|
func (tx *Transaction) doUpdateAttrViewColNumberFormat(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := updateAttributeViewColNumberFormat(operation)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-08-03 23:38:45 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func updateAttributeViewColNumberFormat(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-08-03 23:38:45 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
colType := av.KeyType(operation.Typ)
|
|
|
|
|
switch colType {
|
|
|
|
|
case av.KeyTypeNumber:
|
|
|
|
|
for _, keyValues := range attrView.KeyValues {
|
|
|
|
|
if keyValues.Key.ID == operation.ID && av.KeyTypeNumber == keyValues.Key.Type {
|
|
|
|
|
keyValues.Key.NumberFormat = av.NumberFormat(operation.Format)
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-11 23:47:17 +08:00
|
|
|
|
func (tx *Transaction) doUpdateAttrViewColumn(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := updateAttributeViewColumn(operation)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-07-11 23:47:17 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func updateAttributeViewColumn(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-11 23:47:17 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-12 19:10:05 +08:00
|
|
|
|
colType := av.KeyType(operation.Typ)
|
2025-07-26 16:42:50 +08:00
|
|
|
|
changeType := false
|
2023-07-11 23:47:17 +08:00
|
|
|
|
switch colType {
|
2023-12-15 20:05:14 +08:00
|
|
|
|
case av.KeyTypeBlock, av.KeyTypeText, av.KeyTypeNumber, av.KeyTypeDate, av.KeyTypeSelect, av.KeyTypeMSelect, av.KeyTypeURL, av.KeyTypeEmail,
|
|
|
|
|
av.KeyTypePhone, av.KeyTypeMAsset, av.KeyTypeTemplate, av.KeyTypeCreated, av.KeyTypeUpdated, av.KeyTypeCheckbox,
|
2024-04-15 00:49:48 +08:00
|
|
|
|
av.KeyTypeRelation, av.KeyTypeRollup, av.KeyTypeLineNumber:
|
2023-07-12 19:10:05 +08:00
|
|
|
|
for _, keyValues := range attrView.KeyValues {
|
|
|
|
|
if keyValues.Key.ID == operation.ID {
|
2023-12-01 11:47:20 +08:00
|
|
|
|
keyValues.Key.Name = strings.TrimSpace(operation.Name)
|
2025-07-26 16:42:50 +08:00
|
|
|
|
|
|
|
|
|
changeType = keyValues.Key.Type != colType
|
2023-07-12 19:10:05 +08:00
|
|
|
|
keyValues.Key.Type = colType
|
2024-05-08 10:04:22 +08:00
|
|
|
|
|
|
|
|
|
for _, value := range keyValues.Values {
|
|
|
|
|
value.Type = colType
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-11 23:47:17 +08:00
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-26 16:42:50 +08:00
|
|
|
|
if changeType {
|
|
|
|
|
for _, view := range attrView.Views {
|
2025-07-28 23:52:36 +08:00
|
|
|
|
if nil != view.Group {
|
|
|
|
|
if groupKey := view.GetGroupKey(attrView); nil != groupKey && groupKey.ID == operation.ID {
|
|
|
|
|
removeAttributeViewGroup0(view)
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-07-26 16:42:50 +08:00
|
|
|
|
}
|
2025-07-26 16:11:52 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-07-11 23:47:17 +08:00
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-12 00:02:40 +08:00
|
|
|
|
func (tx *Transaction) doRemoveAttrViewColumn(operation *Operation) (ret *TxErr) {
|
2024-10-17 16:53:18 +08:00
|
|
|
|
err := RemoveAttributeViewKey(operation.AvID, operation.ID, operation.RemoveDest)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-07-12 00:02:40 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-10-17 16:53:18 +08:00
|
|
|
|
func RemoveAttributeViewKey(avID, keyID string, removeRelationDest bool) (err error) {
|
2024-01-12 21:24:55 +08:00
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-12 00:02:40 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-24 14:19:00 +08:00
|
|
|
|
var removedKey *av.Key
|
2023-07-12 19:10:05 +08:00
|
|
|
|
for i, keyValues := range attrView.KeyValues {
|
2024-01-12 21:24:55 +08:00
|
|
|
|
if keyValues.Key.ID == keyID {
|
2023-07-12 19:10:05 +08:00
|
|
|
|
attrView.KeyValues = append(attrView.KeyValues[:i], attrView.KeyValues[i+1:]...)
|
2023-12-24 14:19:00 +08:00
|
|
|
|
removedKey = keyValues.Key
|
2023-07-12 00:02:40 +08:00
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-31 10:30:39 +08:00
|
|
|
|
if nil != removedKey && av.KeyTypeRelation == removedKey.Type && nil != removedKey.Relation {
|
|
|
|
|
if removedKey.Relation.IsTwoWay {
|
2024-10-18 23:35:12 +08:00
|
|
|
|
var destAv *av.AttributeView
|
|
|
|
|
if avID == removedKey.Relation.AvID {
|
|
|
|
|
destAv = attrView
|
|
|
|
|
} else {
|
|
|
|
|
destAv, _ = av.ParseAttributeView(removedKey.Relation.AvID)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if nil != destAv {
|
|
|
|
|
oldDestKey, _ := destAv.GetKey(removedKey.Relation.BackKeyID)
|
|
|
|
|
if nil != oldDestKey && nil != oldDestKey.Relation && oldDestKey.Relation.AvID == attrView.ID && oldDestKey.Relation.IsTwoWay {
|
|
|
|
|
oldDestKey.Relation.IsTwoWay = false
|
|
|
|
|
oldDestKey.Relation.BackKeyID = ""
|
2024-10-17 16:53:18 +08:00
|
|
|
|
}
|
2024-05-12 11:39:02 +08:00
|
|
|
|
|
2024-10-18 23:35:12 +08:00
|
|
|
|
destAvRelSrcAv := false
|
|
|
|
|
for i, keyValues := range destAv.KeyValues {
|
|
|
|
|
if keyValues.Key.ID == removedKey.Relation.BackKeyID {
|
2025-08-03 11:44:06 +08:00
|
|
|
|
if removeRelationDest { // 删除双向关联的目标字段
|
2024-10-17 16:53:18 +08:00
|
|
|
|
destAv.KeyValues = append(destAv.KeyValues[:i], destAv.KeyValues[i+1:]...)
|
|
|
|
|
}
|
2024-10-18 23:35:12 +08:00
|
|
|
|
continue
|
|
|
|
|
}
|
2023-12-31 10:30:39 +08:00
|
|
|
|
|
2024-10-18 23:35:12 +08:00
|
|
|
|
if av.KeyTypeRelation == keyValues.Key.Type && keyValues.Key.Relation.AvID == attrView.ID {
|
|
|
|
|
destAvRelSrcAv = true
|
2023-12-31 10:30:39 +08:00
|
|
|
|
}
|
2024-10-18 23:35:12 +08:00
|
|
|
|
}
|
2023-12-24 14:19:00 +08:00
|
|
|
|
|
2024-10-18 23:35:12 +08:00
|
|
|
|
if removeRelationDest {
|
2024-10-17 16:53:18 +08:00
|
|
|
|
for _, view := range destAv.Views {
|
|
|
|
|
switch view.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
for i, column := range view.Table.Columns {
|
|
|
|
|
if column.ID == removedKey.Relation.BackKeyID {
|
|
|
|
|
view.Table.Columns = append(view.Table.Columns[:i], view.Table.Columns[i+1:]...)
|
|
|
|
|
break
|
|
|
|
|
}
|
2023-12-31 10:30:39 +08:00
|
|
|
|
}
|
2025-06-09 16:46:50 +08:00
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
for i, field := range view.Gallery.CardFields {
|
|
|
|
|
if field.ID == removedKey.Relation.BackKeyID {
|
|
|
|
|
view.Gallery.CardFields = append(view.Gallery.CardFields[:i], view.Gallery.CardFields[i+1:]...)
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-12-24 14:19:00 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2024-10-18 23:35:12 +08:00
|
|
|
|
}
|
2023-12-31 10:30:39 +08:00
|
|
|
|
|
2024-10-18 23:35:12 +08:00
|
|
|
|
if destAv != attrView {
|
|
|
|
|
av.SaveAttributeView(destAv)
|
|
|
|
|
ReloadAttrView(destAv.ID)
|
|
|
|
|
}
|
2023-12-31 10:30:39 +08:00
|
|
|
|
|
2024-10-18 23:35:12 +08:00
|
|
|
|
if !destAvRelSrcAv {
|
|
|
|
|
av.RemoveAvRel(destAv.ID, attrView.ID)
|
2023-12-31 10:30:39 +08:00
|
|
|
|
}
|
2023-12-24 14:19:00 +08:00
|
|
|
|
}
|
2023-12-24 14:24:42 +08:00
|
|
|
|
|
2023-12-31 10:30:39 +08:00
|
|
|
|
srcAvRelDestAv := false
|
|
|
|
|
for _, keyValues := range attrView.KeyValues {
|
2024-01-09 11:18:44 +08:00
|
|
|
|
if av.KeyTypeRelation == keyValues.Key.Type && nil != keyValues.Key.Relation && keyValues.Key.Relation.AvID == removedKey.Relation.AvID {
|
2023-12-31 10:30:39 +08:00
|
|
|
|
srcAvRelDestAv = true
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if !srcAvRelDestAv {
|
|
|
|
|
av.RemoveAvRel(attrView.ID, removedKey.Relation.AvID)
|
|
|
|
|
}
|
2023-12-24 14:19:00 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-12 00:02:40 +08:00
|
|
|
|
for _, view := range attrView.Views {
|
2025-06-21 17:37:42 +08:00
|
|
|
|
if nil != view.Table {
|
2023-07-12 19:10:05 +08:00
|
|
|
|
for i, column := range view.Table.Columns {
|
2024-01-12 21:24:55 +08:00
|
|
|
|
if column.ID == keyID {
|
2023-07-12 19:10:05 +08:00
|
|
|
|
view.Table.Columns = append(view.Table.Columns[:i], view.Table.Columns[i+1:]...)
|
|
|
|
|
break
|
|
|
|
|
}
|
2023-07-12 00:02:40 +08:00
|
|
|
|
}
|
2025-06-21 17:37:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if nil != view.Gallery {
|
2025-06-09 16:46:50 +08:00
|
|
|
|
for i, field := range view.Gallery.CardFields {
|
|
|
|
|
if field.ID == keyID {
|
|
|
|
|
view.Gallery.CardFields = append(view.Gallery.CardFields[:i], view.Gallery.CardFields[i+1:]...)
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-07-12 00:02:40 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-25 22:31:46 +08:00
|
|
|
|
for _, view := range attrView.Views {
|
2025-07-28 23:52:36 +08:00
|
|
|
|
if nil != view.Group {
|
|
|
|
|
if groupKey := view.GetGroupKey(attrView); nil != groupKey && groupKey.ID == keyID {
|
|
|
|
|
removeAttributeViewGroup0(view)
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-07-25 22:31:46 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-07-12 00:02:40 +08:00
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-28 10:50:35 +08:00
|
|
|
|
func (tx *Transaction) doReplaceAttrViewBlock(operation *Operation) (ret *TxErr) {
|
2025-07-17 00:34:35 +08:00
|
|
|
|
err := replaceAttributeViewBlock(operation.AvID, operation.PreviousID, operation.NextID, operation.IsDetached, tx)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID}
|
2023-09-28 10:50:35 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-17 00:34:35 +08:00
|
|
|
|
func replaceAttributeViewBlock(avID, oldBlockID, newBlockID string, isDetached bool, tx *Transaction) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-09-28 10:50:35 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-17 00:34:35 +08:00
|
|
|
|
if err = replaceAttributeViewBlock0(attrView, oldBlockID, newBlockID, isDetached, tx); nil != err {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err = av.SaveAttributeView(attrView); nil != err {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func replaceAttributeViewBlock0(attrView *av.AttributeView, oldBlockID, newBlockID string, isDetached bool, tx *Transaction) (err error) {
|
|
|
|
|
avID := attrView.ID
|
2025-08-10 16:22:10 +08:00
|
|
|
|
var tree *parse.Tree
|
2023-09-28 12:48:09 +08:00
|
|
|
|
var node *ast.Node
|
2025-07-17 00:34:35 +08:00
|
|
|
|
if !isDetached {
|
2025-08-10 16:22:10 +08:00
|
|
|
|
node, tree, _ = getNodeByBlockID(tx, newBlockID)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
now := util.CurrentTimeMillis()
|
|
|
|
|
// 检查是否已经存在绑定块,如果存在的话则重新绑定
|
|
|
|
|
for _, blockVal := range attrView.GetBlockKeyValues().Values {
|
|
|
|
|
if !isDetached && blockVal.Block.ID == newBlockID && nil != node && nil != tree {
|
|
|
|
|
bindBlockAv0(tx, avID, node, tree)
|
|
|
|
|
blockVal.IsDetached = false
|
|
|
|
|
icon, content := getNodeAvBlockText(node)
|
|
|
|
|
content = util.UnescapeHTML(content)
|
|
|
|
|
blockVal.Block.Icon, blockVal.Block.Content = icon, content
|
|
|
|
|
blockVal.UpdatedAt = now
|
|
|
|
|
regenAttrViewGroups(attrView, "force")
|
|
|
|
|
return
|
|
|
|
|
}
|
2024-04-10 22:58:59 +08:00
|
|
|
|
}
|
|
|
|
|
|
2024-09-02 17:53:34 +08:00
|
|
|
|
var changedAvIDs []string
|
2025-08-10 16:55:36 +08:00
|
|
|
|
for _, blockVal := range attrView.GetBlockKeyValues().Values {
|
|
|
|
|
if blockVal.BlockID != oldBlockID {
|
|
|
|
|
continue
|
|
|
|
|
}
|
2023-12-18 13:11:56 +08:00
|
|
|
|
|
2025-08-10 16:55:36 +08:00
|
|
|
|
if av.KeyTypeBlock == blockVal.Type {
|
|
|
|
|
blockVal.IsDetached = isDetached
|
|
|
|
|
if !isDetached {
|
|
|
|
|
if "" != blockVal.Block.ID && blockVal.Block.ID != newBlockID {
|
|
|
|
|
unbindBlockAv(tx, avID, blockVal.Block.ID)
|
|
|
|
|
}
|
|
|
|
|
bindBlockAv(tx, avID, newBlockID)
|
2025-08-10 15:30:08 +08:00
|
|
|
|
|
2025-08-10 16:55:36 +08:00
|
|
|
|
blockVal.Block.ID = newBlockID
|
|
|
|
|
icon, content := getNodeAvBlockText(node)
|
|
|
|
|
content = util.UnescapeHTML(content)
|
|
|
|
|
blockVal.Block.Icon, blockVal.Block.Content = icon, content
|
2025-08-10 15:30:08 +08:00
|
|
|
|
|
2025-08-10 16:55:36 +08:00
|
|
|
|
avIDs := replaceRelationAvValues(avID, oldBlockID, newBlockID)
|
|
|
|
|
changedAvIDs = append(changedAvIDs, avIDs...)
|
|
|
|
|
} else {
|
|
|
|
|
blockVal.Block.ID = ""
|
2023-09-28 10:50:35 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-09 20:39:12 +08:00
|
|
|
|
regenAttrViewGroups(attrView, "force")
|
|
|
|
|
|
2024-09-02 17:53:34 +08:00
|
|
|
|
changedAvIDs = gulu.Str.RemoveDuplicatedElem(changedAvIDs)
|
2025-07-17 00:34:35 +08:00
|
|
|
|
for _, id := range changedAvIDs {
|
|
|
|
|
ReloadAttrView(id)
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func BatchReplaceAttributeViewBlocks(avID string, isDetached bool, oldNew []map[string]string) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, oldNewMap := range oldNew {
|
|
|
|
|
for oldBlockID, newBlockID := range oldNewMap {
|
|
|
|
|
if err = replaceAttributeViewBlock0(attrView, oldBlockID, newBlockID, isDetached, nil); nil != err {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err = av.SaveAttributeView(attrView); nil != err {
|
|
|
|
|
return
|
2024-09-02 17:53:34 +08:00
|
|
|
|
}
|
2023-09-28 10:50:35 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-06-10 15:00:04 +08:00
|
|
|
|
func (tx *Transaction) doUpdateAttrViewCell(operation *Operation) (ret *TxErr) {
|
2025-08-10 11:54:31 +08:00
|
|
|
|
_, err := UpdateAttributeViewCell(tx, operation.AvID, operation.KeyID, operation.RowID, operation.Data)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-07-03 15:29:54 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-16 23:44:15 +08:00
|
|
|
|
func BatchUpdateAttributeViewCells(tx *Transaction, avID string, values []interface{}) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, value := range values {
|
|
|
|
|
v := value.(map[string]interface{})
|
|
|
|
|
keyID := v["keyID"].(string)
|
|
|
|
|
rowID := v["rowID"].(string)
|
|
|
|
|
valueData := v["value"]
|
|
|
|
|
_, err = updateAttributeViewValue(tx, attrView, keyID, rowID, valueData)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err = av.SaveAttributeView(attrView); err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
relatedAvIDs := av.GetSrcAvIDs(avID)
|
|
|
|
|
for _, relatedAvID := range relatedAvIDs {
|
|
|
|
|
ReloadAttrView(relatedAvID)
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-02 16:07:56 +08:00
|
|
|
|
func UpdateAttributeViewCell(tx *Transaction, avID, keyID, rowID string, valueData interface{}) (val *av.Value, err error) {
|
2023-07-31 23:17:31 +08:00
|
|
|
|
attrView, err := av.ParseAttributeView(avID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-10 10:57:18 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-16 23:44:15 +08:00
|
|
|
|
val, err = updateAttributeViewValue(tx, attrView, keyID, rowID, valueData)
|
|
|
|
|
if nil != err {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err = av.SaveAttributeView(attrView); err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
relatedAvIDs := av.GetSrcAvIDs(avID)
|
|
|
|
|
for _, relatedAvID := range relatedAvIDs {
|
|
|
|
|
ReloadAttrView(relatedAvID)
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-03 11:44:06 +08:00
|
|
|
|
func updateAttributeViewValue(tx *Transaction, attrView *av.AttributeView, keyID, blockID string, valueData interface{}) (val *av.Value, err error) {
|
2025-07-16 23:44:15 +08:00
|
|
|
|
avID := attrView.ID
|
2023-12-18 12:53:53 +08:00
|
|
|
|
var blockVal *av.Value
|
|
|
|
|
for _, kv := range attrView.KeyValues {
|
|
|
|
|
if av.KeyTypeBlock == kv.Key.Type {
|
|
|
|
|
for _, v := range kv.Values {
|
2025-08-10 11:20:53 +08:00
|
|
|
|
if blockID == v.BlockID {
|
2023-12-18 12:53:53 +08:00
|
|
|
|
blockVal = v
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-07 09:45:27 +08:00
|
|
|
|
now := time.Now().UnixMilli()
|
2023-12-18 21:13:53 +08:00
|
|
|
|
oldIsDetached := true
|
|
|
|
|
if nil != blockVal {
|
|
|
|
|
oldIsDetached = blockVal.IsDetached
|
|
|
|
|
}
|
2023-07-12 19:10:05 +08:00
|
|
|
|
for _, keyValues := range attrView.KeyValues {
|
2023-07-31 23:17:31 +08:00
|
|
|
|
if keyID != keyValues.Key.ID {
|
2023-07-13 00:49:09 +08:00
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, value := range keyValues.Values {
|
2025-08-03 11:44:06 +08:00
|
|
|
|
if blockID == value.BlockID {
|
2023-07-13 00:49:09 +08:00
|
|
|
|
val = value
|
2023-07-13 10:44:47 +08:00
|
|
|
|
val.Type = keyValues.Key.Type
|
2023-07-13 00:49:09 +08:00
|
|
|
|
break
|
2023-07-10 10:57:18 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-13 00:49:09 +08:00
|
|
|
|
if nil == val {
|
2025-08-03 11:44:06 +08:00
|
|
|
|
val = &av.Value{ID: ast.NewNodeID(), KeyID: keyID, BlockID: blockID, Type: keyValues.Key.Type, CreatedAt: now, UpdatedAt: now}
|
2023-07-13 00:49:09 +08:00
|
|
|
|
keyValues.Values = append(keyValues.Values, val)
|
|
|
|
|
}
|
|
|
|
|
break
|
2023-07-10 10:39:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-12-18 13:02:54 +08:00
|
|
|
|
isUpdatingBlockKey := av.KeyTypeBlock == val.Type
|
2023-12-18 10:55:10 +08:00
|
|
|
|
oldBoundBlockID := val.BlockID
|
2023-12-24 23:25:06 +08:00
|
|
|
|
var oldRelationBlockIDs []string
|
2023-12-24 23:18:35 +08:00
|
|
|
|
if av.KeyTypeRelation == val.Type {
|
2023-12-25 00:33:40 +08:00
|
|
|
|
if nil != val.Relation {
|
|
|
|
|
for _, bID := range val.Relation.BlockIDs {
|
|
|
|
|
oldRelationBlockIDs = append(oldRelationBlockIDs, bID)
|
|
|
|
|
}
|
2023-12-24 23:34:30 +08:00
|
|
|
|
}
|
2023-12-24 23:18:35 +08:00
|
|
|
|
}
|
2023-07-31 23:18:07 +08:00
|
|
|
|
data, err := gulu.JSON.MarshalJSON(valueData)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-03 10:28:47 +08:00
|
|
|
|
logging.LogErrorf("marshal value [%+v] failed: %s", valueData, err)
|
2023-07-03 15:29:54 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = gulu.JSON.UnmarshalJSON(data, &val); err != nil {
|
2025-07-03 10:28:47 +08:00
|
|
|
|
logging.LogErrorf("unmarshal data [%s] failed: %s", data, err)
|
2023-07-03 15:29:54 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
2024-01-25 19:26:43 +08:00
|
|
|
|
|
2024-05-15 22:10:26 +08:00
|
|
|
|
key, _ := attrView.GetKey(keyID)
|
|
|
|
|
|
2024-01-25 19:26:43 +08:00
|
|
|
|
if av.KeyTypeNumber == val.Type {
|
2024-09-16 17:06:37 +08:00
|
|
|
|
if nil != val.Number {
|
|
|
|
|
if !val.Number.IsNotEmpty {
|
|
|
|
|
val.Number.Content = 0
|
|
|
|
|
val.Number.FormattedContent = ""
|
|
|
|
|
} else {
|
|
|
|
|
val.Number.FormatNumber()
|
|
|
|
|
}
|
2024-01-25 19:26:43 +08:00
|
|
|
|
}
|
2024-03-01 10:33:25 +08:00
|
|
|
|
} else if av.KeyTypeDate == val.Type {
|
|
|
|
|
if nil != val.Date && !val.Date.IsNotEmpty {
|
|
|
|
|
val.Date.Content = 0
|
|
|
|
|
val.Date.FormattedContent = ""
|
|
|
|
|
}
|
2024-05-15 22:10:26 +08:00
|
|
|
|
} else if av.KeyTypeSelect == val.Type || av.KeyTypeMSelect == val.Type {
|
|
|
|
|
if nil != key && 0 < len(val.MSelect) {
|
|
|
|
|
// The selection options are inconsistent after pasting data into the database https://github.com/siyuan-note/siyuan/issues/11409
|
|
|
|
|
for _, valOpt := range val.MSelect {
|
|
|
|
|
if opt := key.GetOption(valOpt.Content); nil == opt {
|
|
|
|
|
// 不存在的选项新建保存
|
2024-12-12 11:22:37 +08:00
|
|
|
|
color := valOpt.Color
|
|
|
|
|
if "" == color {
|
2024-12-12 11:22:58 +08:00
|
|
|
|
color = fmt.Sprintf("%d", 1+rand.Intn(14))
|
2024-12-12 11:22:37 +08:00
|
|
|
|
}
|
|
|
|
|
opt = &av.SelectOption{Name: valOpt.Content, Color: color}
|
2024-05-15 22:10:26 +08:00
|
|
|
|
key.Options = append(key.Options, opt)
|
|
|
|
|
} else {
|
|
|
|
|
// 已经存在的选项颜色需要保持不变
|
|
|
|
|
valOpt.Color = opt.Color
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-01-25 19:26:43 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-12-25 00:30:49 +08:00
|
|
|
|
relationChangeMode := 0 // 0:不变(仅排序),1:增加,2:减少
|
2023-12-24 23:18:35 +08:00
|
|
|
|
if av.KeyTypeRelation == val.Type {
|
2025-08-03 11:44:06 +08:00
|
|
|
|
// 关联字段得 content 是自动渲染的,所以不需要保存
|
2023-12-24 23:18:35 +08:00
|
|
|
|
val.Relation.Contents = nil
|
2023-12-25 00:30:49 +08:00
|
|
|
|
|
2024-05-06 17:55:32 +08:00
|
|
|
|
// 去重
|
|
|
|
|
val.Relation.BlockIDs = gulu.Str.RemoveDuplicatedElem(val.Relation.BlockIDs)
|
|
|
|
|
|
2023-12-31 10:58:40 +08:00
|
|
|
|
// 计算关联变更模式
|
2023-12-25 00:30:49 +08:00
|
|
|
|
if len(oldRelationBlockIDs) == len(val.Relation.BlockIDs) {
|
|
|
|
|
relationChangeMode = 0
|
|
|
|
|
} else {
|
|
|
|
|
if len(oldRelationBlockIDs) > len(val.Relation.BlockIDs) {
|
|
|
|
|
relationChangeMode = 2
|
|
|
|
|
} else {
|
|
|
|
|
relationChangeMode = 1
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-12-24 23:18:35 +08:00
|
|
|
|
}
|
2023-07-03 15:29:54 +08:00
|
|
|
|
|
2023-12-18 13:02:54 +08:00
|
|
|
|
// val.IsDetached 只有更新主键的时候才会传入,所以下面需要结合 isUpdatingBlockKey 来判断
|
|
|
|
|
|
2025-08-09 22:43:23 +08:00
|
|
|
|
if isUpdatingBlockKey {
|
|
|
|
|
if oldIsDetached {
|
2025-08-10 11:54:31 +08:00
|
|
|
|
// 之前是非绑定块
|
2025-08-09 22:43:23 +08:00
|
|
|
|
|
|
|
|
|
if !val.IsDetached { // 现在绑定了块
|
2025-08-10 11:20:53 +08:00
|
|
|
|
bindBlockAv(tx, avID, val.Block.ID)
|
2024-08-31 22:56:11 +08:00
|
|
|
|
}
|
2025-08-09 22:43:23 +08:00
|
|
|
|
} else {
|
|
|
|
|
// 之前绑定了块
|
2024-05-12 17:20:04 +08:00
|
|
|
|
|
2025-08-10 11:54:31 +08:00
|
|
|
|
if val.IsDetached { // 现在是非绑定块
|
2025-08-10 17:45:46 +08:00
|
|
|
|
unbindBlockAv(tx, avID, val.Block.ID)
|
|
|
|
|
val.Block.ID = ""
|
2024-05-12 17:20:04 +08:00
|
|
|
|
} else {
|
2025-08-03 10:20:25 +08:00
|
|
|
|
// 现在也绑定了块
|
2024-05-12 17:20:04 +08:00
|
|
|
|
|
2023-12-18 13:02:54 +08:00
|
|
|
|
if oldBoundBlockID != val.BlockID { // 之前绑定的块和现在绑定的块不一样
|
|
|
|
|
// 换绑块
|
|
|
|
|
unbindBlockAv(tx, avID, oldBoundBlockID)
|
|
|
|
|
bindBlockAv(tx, avID, val.BlockID)
|
2025-01-12 17:43:46 +08:00
|
|
|
|
val.Block.Content = util.UnescapeHTML(val.Block.Content)
|
2023-12-18 13:02:54 +08:00
|
|
|
|
} else { // 之前绑定的块和现在绑定的块一样
|
2024-12-23 18:08:46 +08:00
|
|
|
|
content := strings.TrimSpace(val.Block.Content)
|
2024-12-23 21:04:43 +08:00
|
|
|
|
node, tree, _ := getNodeByBlockID(tx, val.BlockID)
|
2024-12-24 19:47:45 +08:00
|
|
|
|
_, blockText := getNodeAvBlockText(node)
|
2024-12-23 18:08:46 +08:00
|
|
|
|
if "" == content {
|
2025-08-10 11:54:31 +08:00
|
|
|
|
// 使用动态锚文本
|
2025-08-03 10:20:25 +08:00
|
|
|
|
val.Block.Content = util.UnescapeHTML(blockText)
|
2024-12-24 19:47:45 +08:00
|
|
|
|
} else {
|
2025-08-10 16:39:58 +08:00
|
|
|
|
val.Block.Content = content
|
2024-12-24 19:47:45 +08:00
|
|
|
|
// 设置静态锚文本 Database-bound block primary key supports setting static anchor text https://github.com/siyuan-note/siyuan/issues/10049
|
|
|
|
|
updateBlockValueStaticText(tx, node, tree, avID, content)
|
|
|
|
|
}
|
2023-12-18 12:35:48 +08:00
|
|
|
|
}
|
2023-12-18 10:55:10 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2023-09-28 00:38:56 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-12-18 12:53:53 +08:00
|
|
|
|
if nil != blockVal {
|
2024-03-03 16:39:34 +08:00
|
|
|
|
blockVal.Block.Updated = now
|
2024-04-03 10:54:23 +08:00
|
|
|
|
blockVal.SetUpdatedAt(now)
|
2023-12-18 13:02:54 +08:00
|
|
|
|
if isUpdatingBlockKey {
|
|
|
|
|
blockVal.IsDetached = val.IsDetached
|
|
|
|
|
}
|
2023-10-10 21:55:43 +08:00
|
|
|
|
}
|
2024-04-03 10:54:23 +08:00
|
|
|
|
val.SetUpdatedAt(now)
|
2023-10-10 21:55:43 +08:00
|
|
|
|
|
2024-05-06 13:44:37 +08:00
|
|
|
|
if nil != key && av.KeyTypeRelation == key.Type && nil != key.Relation && key.Relation.IsTwoWay {
|
|
|
|
|
// 双向关联需要同时更新目标字段的值
|
2025-08-03 11:44:06 +08:00
|
|
|
|
updateTwoWayRelationDestAttrView(attrView, key, val, relationChangeMode, oldRelationBlockIDs)
|
|
|
|
|
}
|
2023-12-24 23:18:35 +08:00
|
|
|
|
|
2025-08-09 20:39:12 +08:00
|
|
|
|
regenAttrViewGroups(attrView, keyID)
|
2025-08-03 11:44:06 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
2023-12-25 00:30:49 +08:00
|
|
|
|
|
2025-08-03 11:44:06 +08:00
|
|
|
|
// relationChangeMode
|
|
|
|
|
// 0:关联字段值不变(仅排序),不影响目标值
|
|
|
|
|
// 1:关联字段值增加,增加目标值
|
|
|
|
|
// 2:关联字段值减少,减少目标值
|
|
|
|
|
func updateTwoWayRelationDestAttrView(attrView *av.AttributeView, relKey *av.Key, val *av.Value, relationChangeMode int, oldRelationBlockIDs []string) {
|
|
|
|
|
var destAv *av.AttributeView
|
|
|
|
|
if attrView.ID == relKey.Relation.AvID {
|
|
|
|
|
destAv = attrView
|
|
|
|
|
} else {
|
|
|
|
|
destAv, _ = av.ParseAttributeView(relKey.Relation.AvID)
|
|
|
|
|
}
|
2023-12-25 00:30:49 +08:00
|
|
|
|
|
2025-08-03 11:44:06 +08:00
|
|
|
|
if nil == destAv {
|
|
|
|
|
return
|
|
|
|
|
}
|
2024-05-06 13:44:37 +08:00
|
|
|
|
|
2025-08-03 11:44:06 +08:00
|
|
|
|
now := util.CurrentTimeMillis()
|
|
|
|
|
if 1 == relationChangeMode {
|
|
|
|
|
addBlockIDs := val.Relation.BlockIDs
|
|
|
|
|
for _, bID := range oldRelationBlockIDs {
|
|
|
|
|
addBlockIDs = gulu.Str.RemoveElem(addBlockIDs, bID)
|
|
|
|
|
}
|
2024-05-06 13:44:37 +08:00
|
|
|
|
|
2025-08-03 11:44:06 +08:00
|
|
|
|
for _, blockID := range addBlockIDs {
|
|
|
|
|
for _, keyValues := range destAv.KeyValues {
|
|
|
|
|
if keyValues.Key.ID != relKey.Relation.BackKeyID {
|
|
|
|
|
continue
|
2024-05-06 13:44:37 +08:00
|
|
|
|
}
|
2025-08-03 11:44:06 +08:00
|
|
|
|
|
|
|
|
|
destVal := keyValues.GetValue(blockID)
|
|
|
|
|
if nil == destVal {
|
|
|
|
|
destVal = &av.Value{ID: ast.NewNodeID(), KeyID: keyValues.Key.ID, BlockID: blockID, Type: keyValues.Key.Type, Relation: &av.ValueRelation{}, CreatedAt: now, UpdatedAt: now + 1000}
|
|
|
|
|
keyValues.Values = append(keyValues.Values, destVal)
|
2024-05-06 13:44:37 +08:00
|
|
|
|
}
|
2023-12-24 23:18:35 +08:00
|
|
|
|
|
2025-08-03 11:44:06 +08:00
|
|
|
|
destVal.Relation.BlockIDs = append(destVal.Relation.BlockIDs, val.BlockID)
|
|
|
|
|
destVal.Relation.BlockIDs = gulu.Str.RemoveDuplicatedElem(destVal.Relation.BlockIDs)
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else if 2 == relationChangeMode {
|
|
|
|
|
removeBlockIDs := oldRelationBlockIDs
|
|
|
|
|
for _, bID := range val.Relation.BlockIDs {
|
|
|
|
|
removeBlockIDs = gulu.Str.RemoveElem(removeBlockIDs, bID)
|
|
|
|
|
}
|
2023-12-24 23:02:59 +08:00
|
|
|
|
|
2025-08-03 11:44:06 +08:00
|
|
|
|
for _, blockID := range removeBlockIDs {
|
|
|
|
|
for _, keyValues := range destAv.KeyValues {
|
|
|
|
|
if keyValues.Key.ID != relKey.Relation.BackKeyID {
|
|
|
|
|
continue
|
2023-12-24 23:02:59 +08:00
|
|
|
|
}
|
2024-05-06 13:44:37 +08:00
|
|
|
|
|
2025-08-03 11:44:06 +08:00
|
|
|
|
for _, value := range keyValues.Values {
|
|
|
|
|
if value.BlockID == blockID {
|
|
|
|
|
value.Relation.BlockIDs = gulu.Str.RemoveElem(value.Relation.BlockIDs, val.BlockID)
|
|
|
|
|
value.SetUpdatedAt(now)
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-05-12 11:16:31 +08:00
|
|
|
|
}
|
2023-12-24 23:02:59 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2025-08-03 11:02:02 +08:00
|
|
|
|
|
2025-08-03 11:44:06 +08:00
|
|
|
|
if destAv != attrView {
|
2025-08-09 20:39:12 +08:00
|
|
|
|
regenAttrViewGroups(destAv, "force")
|
2025-08-03 11:44:06 +08:00
|
|
|
|
av.SaveAttributeView(destAv)
|
|
|
|
|
}
|
2023-07-03 15:29:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-08-09 20:39:12 +08:00
|
|
|
|
// regenAttrViewGroups 重新生成分组视图。
|
|
|
|
|
// keyID: 如果是 "force" 则强制重新生成所有分组视图,否则只生成 keyID 指定的分组字段的分组视图
|
|
|
|
|
func regenAttrViewGroups(attrView *av.AttributeView, keyID string) {
|
2025-07-25 22:09:03 +08:00
|
|
|
|
for _, view := range attrView.Views {
|
2025-07-27 19:34:17 +08:00
|
|
|
|
groupKey := view.GetGroupKey(attrView)
|
2025-07-27 17:08:22 +08:00
|
|
|
|
if nil == groupKey {
|
|
|
|
|
continue
|
|
|
|
|
}
|
2025-07-26 11:54:19 +08:00
|
|
|
|
|
2025-07-28 23:28:57 +08:00
|
|
|
|
if "force" != keyID {
|
2025-07-29 10:43:28 +08:00
|
|
|
|
if av.KeyTypeTemplate != groupKey.Type && av.KeyTypeCreated != groupKey.Type && av.KeyTypeUpdated != groupKey.Type &&
|
|
|
|
|
view.Group.Field != keyID {
|
2025-07-28 23:28:57 +08:00
|
|
|
|
continue
|
|
|
|
|
}
|
2025-07-27 17:08:22 +08:00
|
|
|
|
}
|
2025-07-25 22:09:03 +08:00
|
|
|
|
|
2025-08-09 20:39:12 +08:00
|
|
|
|
genAttrViewGroups(view, attrView)
|
2025-07-25 22:09:03 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-10 12:16:10 +08:00
|
|
|
|
func unbindBlockAv(tx *Transaction, avID, nodeID string) {
|
|
|
|
|
node, tree, err := getNodeByBlockID(tx, nodeID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-12-18 10:55:10 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
attrs := parse.IAL2Map(node.KramdownIAL)
|
|
|
|
|
if "" == attrs[av.NodeAttrNameAvs] {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
avIDs := strings.Split(attrs[av.NodeAttrNameAvs], ",")
|
|
|
|
|
avIDs = gulu.Str.RemoveElem(avIDs, avID)
|
|
|
|
|
if 0 == len(avIDs) {
|
2024-04-15 17:15:04 +08:00
|
|
|
|
attrs[av.NodeAttrNameAvs] = ""
|
2023-12-18 10:55:10 +08:00
|
|
|
|
} else {
|
|
|
|
|
attrs[av.NodeAttrNameAvs] = strings.Join(avIDs, ",")
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-08 09:25:19 +08:00
|
|
|
|
avNames := getAvNames(attrs[av.NodeAttrNameAvs])
|
|
|
|
|
if "" != avNames {
|
|
|
|
|
attrs[av.NodeAttrViewNames] = avNames
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-18 10:55:10 +08:00
|
|
|
|
if nil != tx {
|
|
|
|
|
err = setNodeAttrsWithTx(tx, node, tree, attrs)
|
|
|
|
|
} else {
|
|
|
|
|
err = setNodeAttrs(node, tree, attrs)
|
|
|
|
|
}
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-08-10 12:16:10 +08:00
|
|
|
|
logging.LogWarnf("set node [%s] attrs failed: %s", nodeID, err)
|
2023-12-18 10:55:10 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-28 12:48:09 +08:00
|
|
|
|
func bindBlockAv(tx *Transaction, avID, blockID string) {
|
|
|
|
|
node, tree, err := getNodeByBlockID(tx, blockID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-09-28 11:17:49 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-08 09:25:19 +08:00
|
|
|
|
bindBlockAv0(tx, avID, node, tree)
|
2024-03-28 21:27:31 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-08 09:25:19 +08:00
|
|
|
|
func bindBlockAv0(tx *Transaction, avID string, node *ast.Node, tree *parse.Tree) {
|
2023-09-28 11:17:49 +08:00
|
|
|
|
attrs := parse.IAL2Map(node.KramdownIAL)
|
2023-10-05 12:02:17 +08:00
|
|
|
|
if "" == attrs[av.NodeAttrNameAvs] {
|
|
|
|
|
attrs[av.NodeAttrNameAvs] = avID
|
2023-09-28 11:17:49 +08:00
|
|
|
|
} else {
|
2023-10-05 12:02:17 +08:00
|
|
|
|
avIDs := strings.Split(attrs[av.NodeAttrNameAvs], ",")
|
2023-09-28 11:17:49 +08:00
|
|
|
|
avIDs = append(avIDs, avID)
|
|
|
|
|
avIDs = gulu.Str.RemoveDuplicatedElem(avIDs)
|
2023-10-05 12:02:17 +08:00
|
|
|
|
attrs[av.NodeAttrNameAvs] = strings.Join(avIDs, ",")
|
2023-09-28 11:17:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
2024-03-28 21:27:31 +08:00
|
|
|
|
avNames := getAvNames(attrs[av.NodeAttrNameAvs])
|
|
|
|
|
if "" != avNames {
|
|
|
|
|
attrs[av.NodeAttrViewNames] = avNames
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var err error
|
2023-09-28 12:48:09 +08:00
|
|
|
|
if nil != tx {
|
|
|
|
|
err = setNodeAttrsWithTx(tx, node, tree, attrs)
|
|
|
|
|
} else {
|
|
|
|
|
err = setNodeAttrs(node, tree, attrs)
|
|
|
|
|
}
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2024-04-08 09:25:19 +08:00
|
|
|
|
logging.LogWarnf("set node [%s] attrs failed: %s", node.ID, err)
|
2023-09-28 12:48:09 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-12-23 17:27:27 +08:00
|
|
|
|
func updateBlockValueStaticText(tx *Transaction, node *ast.Node, tree *parse.Tree, avID, text string) {
|
2024-12-23 21:04:43 +08:00
|
|
|
|
if nil == node {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-12-21 12:00:02 +08:00
|
|
|
|
attrs := parse.IAL2Map(node.KramdownIAL)
|
|
|
|
|
attrs[av.NodeAttrViewStaticText+"-"+avID] = text
|
|
|
|
|
var err error
|
2024-12-23 17:30:35 +08:00
|
|
|
|
if nil != tx {
|
2024-12-23 17:27:27 +08:00
|
|
|
|
err = setNodeAttrsWithTx(tx, node, tree, attrs)
|
2024-12-21 12:00:02 +08:00
|
|
|
|
} else {
|
2024-12-23 17:31:43 +08:00
|
|
|
|
err = setNodeAttrs(node, tree, attrs)
|
2024-12-21 12:00:02 +08:00
|
|
|
|
}
|
|
|
|
|
if err != nil {
|
|
|
|
|
logging.LogWarnf("set node [%s] attrs failed: %s", node.ID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-28 12:48:09 +08:00
|
|
|
|
func getNodeByBlockID(tx *Transaction, blockID string) (node *ast.Node, tree *parse.Tree, err error) {
|
|
|
|
|
if nil != tx {
|
|
|
|
|
tree, err = tx.loadTree(blockID)
|
|
|
|
|
} else {
|
2024-03-10 23:27:13 +08:00
|
|
|
|
tree, err = LoadTreeByBlockID(blockID)
|
2023-09-28 12:48:09 +08:00
|
|
|
|
}
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-09-28 12:48:09 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
node = treenode.GetNodeInTree(tree, blockID)
|
|
|
|
|
if nil == node {
|
|
|
|
|
logging.LogWarnf("node [%s] not found in tree [%s]", blockID, tree.ID)
|
|
|
|
|
return
|
2023-09-28 11:17:49 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-13 00:08:10 +08:00
|
|
|
|
func (tx *Transaction) doUpdateAttrViewColOptions(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := updateAttributeViewColumnOptions(operation)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-07-13 00:08:10 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
2023-07-12 19:10:05 +08:00
|
|
|
|
|
2023-07-13 00:08:10 +08:00
|
|
|
|
func updateAttributeViewColumnOptions(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-13 00:08:10 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
jsonData, err := gulu.JSON.MarshalJSON(operation.Data)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-13 00:08:10 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-23 17:57:45 +08:00
|
|
|
|
options := []*av.SelectOption{}
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = gulu.JSON.UnmarshalJSON(jsonData, &options); err != nil {
|
2023-07-13 00:08:10 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, keyValues := range attrView.KeyValues {
|
|
|
|
|
if keyValues.Key.ID == operation.ID {
|
|
|
|
|
keyValues.Key.Options = options
|
2025-08-08 16:59:26 +08:00
|
|
|
|
break
|
2023-07-13 00:08:10 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2025-08-08 16:59:26 +08:00
|
|
|
|
|
2025-08-09 20:39:12 +08:00
|
|
|
|
regenAttrViewGroups(attrView, operation.ID)
|
2025-08-08 16:59:26 +08:00
|
|
|
|
err = av.SaveAttributeView(attrView)
|
2023-07-13 00:08:10 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-13 00:16:40 +08:00
|
|
|
|
func (tx *Transaction) doRemoveAttrViewColOption(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := removeAttributeViewColumnOption(operation)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-07-13 00:16:40 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
2023-07-13 00:08:10 +08:00
|
|
|
|
|
2023-07-13 00:16:40 +08:00
|
|
|
|
func removeAttributeViewColumnOption(operation *Operation) (err error) {
|
2023-07-13 00:29:22 +08:00
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-13 00:16:40 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
optName := operation.Data.(string)
|
|
|
|
|
|
|
|
|
|
key, err := attrView.GetKey(operation.ID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-13 00:16:40 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for i, opt := range key.Options {
|
|
|
|
|
if optName == opt.Name {
|
|
|
|
|
key.Options = append(key.Options[:i], key.Options[i+1:]...)
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, keyValues := range attrView.KeyValues {
|
|
|
|
|
if keyValues.Key.ID != operation.ID {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, value := range keyValues.Values {
|
|
|
|
|
if nil == value || nil == value.MSelect {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for i, opt := range value.MSelect {
|
|
|
|
|
if optName == opt.Content {
|
|
|
|
|
value.MSelect = append(value.MSelect[:i], value.MSelect[i+1:]...)
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-09 20:39:12 +08:00
|
|
|
|
regenAttrViewGroups(attrView, operation.ID)
|
2023-07-13 00:16:40 +08:00
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (tx *Transaction) doUpdateAttrViewColOption(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := updateAttributeViewColumnOption(operation)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-07-13 00:16:40 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func updateAttributeViewColumnOption(operation *Operation) (err error) {
|
2023-07-13 00:29:22 +08:00
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-13 00:16:40 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
key, err := attrView.GetKey(operation.ID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-13 00:16:40 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
data := operation.Data.(map[string]interface{})
|
|
|
|
|
|
2024-11-19 15:15:37 +08:00
|
|
|
|
rename := false
|
|
|
|
|
oldName := strings.TrimSpace(data["oldName"].(string))
|
|
|
|
|
newName := strings.TrimSpace(data["newName"].(string))
|
|
|
|
|
newDesc := strings.TrimSpace(data["newDesc"].(string))
|
2023-07-13 00:16:40 +08:00
|
|
|
|
newColor := data["newColor"].(string)
|
|
|
|
|
|
2024-11-17 17:43:52 +08:00
|
|
|
|
found := false
|
|
|
|
|
if oldName != newName {
|
2024-11-19 15:15:37 +08:00
|
|
|
|
rename = true
|
|
|
|
|
|
2024-11-17 17:43:52 +08:00
|
|
|
|
for _, opt := range key.Options {
|
2024-11-18 10:24:38 +08:00
|
|
|
|
if newName == opt.Name { // 如果选项已经存在则直接使用
|
2024-11-17 17:43:52 +08:00
|
|
|
|
found = true
|
|
|
|
|
newColor = opt.Color
|
2024-11-18 15:59:50 +08:00
|
|
|
|
newDesc = opt.Desc
|
2024-11-17 17:43:52 +08:00
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if !found {
|
|
|
|
|
for i, opt := range key.Options {
|
|
|
|
|
if oldName == opt.Name {
|
|
|
|
|
key.Options[i].Name = newName
|
|
|
|
|
key.Options[i].Color = newColor
|
2024-11-18 15:59:50 +08:00
|
|
|
|
key.Options[i].Desc = newDesc
|
2024-11-17 17:43:52 +08:00
|
|
|
|
break
|
|
|
|
|
}
|
2023-07-13 00:16:40 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-06 10:11:23 +08:00
|
|
|
|
// 如果存在选项对应的值,需要更新值中的选项
|
2023-07-13 00:16:40 +08:00
|
|
|
|
for _, keyValues := range attrView.KeyValues {
|
|
|
|
|
if keyValues.Key.ID != operation.ID {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, value := range keyValues.Values {
|
|
|
|
|
if nil == value || nil == value.MSelect {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-18 10:24:38 +08:00
|
|
|
|
found = false
|
|
|
|
|
for _, opt := range value.MSelect {
|
|
|
|
|
if newName == opt.Content {
|
|
|
|
|
found = true
|
2023-07-13 00:16:40 +08:00
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-11-19 15:15:37 +08:00
|
|
|
|
if found && rename {
|
2024-11-18 10:24:38 +08:00
|
|
|
|
idx := -1
|
|
|
|
|
for i, opt := range value.MSelect {
|
|
|
|
|
if oldName == opt.Content {
|
|
|
|
|
idx = i
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if 0 <= idx {
|
|
|
|
|
value.MSelect = util.RemoveElem(value.MSelect, idx)
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
for i, opt := range value.MSelect {
|
|
|
|
|
if oldName == opt.Content {
|
|
|
|
|
value.MSelect[i].Content = newName
|
|
|
|
|
value.MSelect[i].Color = newColor
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-07-13 00:16:40 +08:00
|
|
|
|
}
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-06 10:11:23 +08:00
|
|
|
|
// 如果存在选项对应的过滤器,需要更新过滤器中设置的选项值
|
|
|
|
|
// Database select field filters follow option editing changes https://github.com/siyuan-note/siyuan/issues/10881
|
|
|
|
|
for _, view := range attrView.Views {
|
2025-06-29 11:11:20 +08:00
|
|
|
|
for _, filter := range view.Filters {
|
|
|
|
|
if filter.Column != key.ID {
|
|
|
|
|
continue
|
2025-06-09 16:46:50 +08:00
|
|
|
|
}
|
2024-04-06 10:11:23 +08:00
|
|
|
|
|
2025-06-29 11:11:20 +08:00
|
|
|
|
if nil != filter.Value && (av.KeyTypeSelect == filter.Value.Type || av.KeyTypeMSelect == filter.Value.Type) {
|
|
|
|
|
for i, opt := range filter.Value.MSelect {
|
|
|
|
|
if oldName == opt.Content {
|
|
|
|
|
filter.Value.MSelect[i].Content = newName
|
|
|
|
|
filter.Value.MSelect[i].Color = newColor
|
|
|
|
|
break
|
2024-04-06 10:11:23 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-09 20:39:12 +08:00
|
|
|
|
regenAttrViewGroups(attrView, operation.ID)
|
2023-07-13 00:16:40 +08:00
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
2024-03-04 16:41:41 +08:00
|
|
|
|
|
2024-11-09 14:09:40 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewColOptionDesc(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := setAttributeViewColumnOptionDesc(operation)
|
|
|
|
|
if err != nil {
|
2025-07-05 12:33:41 +08:00
|
|
|
|
return &TxErr{code: TxErrHandleAttributeView, id: operation.AvID, msg: err.Error()}
|
2024-11-09 14:09:40 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func setAttributeViewColumnOptionDesc(operation *Operation) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
key, err := attrView.GetKey(operation.ID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
data := operation.Data.(map[string]interface{})
|
|
|
|
|
name := data["name"].(string)
|
|
|
|
|
desc := data["desc"].(string)
|
|
|
|
|
|
|
|
|
|
for i, opt := range key.Options {
|
|
|
|
|
if name == opt.Name {
|
|
|
|
|
key.Options[i].Desc = desc
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-04 16:41:41 +08:00
|
|
|
|
func getAttrViewViewByBlockID(attrView *av.AttributeView, blockID string) (ret *av.View, err error) {
|
|
|
|
|
var viewID string
|
2025-07-28 23:39:00 +08:00
|
|
|
|
var node *ast.Node
|
|
|
|
|
if "" != blockID {
|
|
|
|
|
node, _, _ = getNodeByBlockID(nil, blockID)
|
|
|
|
|
}
|
2024-03-04 16:41:41 +08:00
|
|
|
|
if nil != node {
|
|
|
|
|
viewID = node.IALAttr(av.NodeAttrView)
|
|
|
|
|
}
|
|
|
|
|
return attrView.GetCurrentView(viewID)
|
|
|
|
|
}
|
2024-04-18 11:29:09 +08:00
|
|
|
|
|
|
|
|
|
func getAttrViewName(attrView *av.AttributeView) string {
|
2024-05-20 23:04:03 +08:00
|
|
|
|
ret := strings.TrimSpace(attrView.Name)
|
2024-04-18 11:29:09 +08:00
|
|
|
|
if "" == ret {
|
|
|
|
|
ret = Conf.language(105)
|
|
|
|
|
}
|
|
|
|
|
return ret
|
|
|
|
|
}
|
2024-04-23 18:22:51 +08:00
|
|
|
|
|
2024-09-02 17:53:34 +08:00
|
|
|
|
func replaceRelationAvValues(avID, previousID, nextID string) (changedSrcAvID []string) {
|
2024-04-23 18:22:51 +08:00
|
|
|
|
// The database relation fields follow the change after the primary key field is changed https://github.com/siyuan-note/siyuan/issues/11117
|
|
|
|
|
|
|
|
|
|
srcAvIDs := av.GetSrcAvIDs(avID)
|
|
|
|
|
for _, srcAvID := range srcAvIDs {
|
|
|
|
|
srcAv, parseErr := av.ParseAttributeView(srcAvID)
|
|
|
|
|
changed := false
|
|
|
|
|
if nil != parseErr {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, srcKeyValues := range srcAv.KeyValues {
|
|
|
|
|
if av.KeyTypeRelation != srcKeyValues.Key.Type {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
2024-05-15 11:33:59 +08:00
|
|
|
|
if nil == srcKeyValues.Key.Relation || avID != srcKeyValues.Key.Relation.AvID {
|
2024-04-23 18:22:51 +08:00
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, srcValue := range srcKeyValues.Values {
|
|
|
|
|
if nil == srcValue.Relation {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
srcAvChanged := false
|
|
|
|
|
srcValue.Relation.BlockIDs, srcAvChanged = util.ReplaceStr(srcValue.Relation.BlockIDs, previousID, nextID)
|
|
|
|
|
if srcAvChanged {
|
|
|
|
|
changed = true
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if changed {
|
2025-08-09 20:39:12 +08:00
|
|
|
|
regenAttrViewGroups(srcAv, "force")
|
2024-04-23 18:22:51 +08:00
|
|
|
|
av.SaveAttributeView(srcAv)
|
2024-09-02 17:53:34 +08:00
|
|
|
|
changedSrcAvID = append(changedSrcAvID, srcAvID)
|
2024-04-23 18:22:51 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2024-09-02 17:53:34 +08:00
|
|
|
|
return
|
2024-04-23 18:22:51 +08:00
|
|
|
|
}
|
2024-05-23 23:57:10 +08:00
|
|
|
|
|
|
|
|
|
func updateBoundBlockAvsAttribute(avIDs []string) {
|
|
|
|
|
// 更新指定 avIDs 中绑定块的 avs 属性
|
|
|
|
|
|
|
|
|
|
cachedTrees, saveTrees := map[string]*parse.Tree{}, map[string]*parse.Tree{}
|
|
|
|
|
luteEngine := util.NewLute()
|
|
|
|
|
for _, avID := range avIDs {
|
|
|
|
|
attrView, _ := av.ParseAttributeView(avID)
|
|
|
|
|
if nil == attrView {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
blockKeyValues := attrView.GetBlockKeyValues()
|
|
|
|
|
for _, blockValue := range blockKeyValues.Values {
|
|
|
|
|
if blockValue.IsDetached {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
bt := treenode.GetBlockTree(blockValue.BlockID)
|
|
|
|
|
if nil == bt {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
tree := cachedTrees[bt.RootID]
|
|
|
|
|
if nil == tree {
|
|
|
|
|
tree, _ = filesys.LoadTree(bt.BoxID, bt.Path, luteEngine)
|
|
|
|
|
if nil == tree {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
cachedTrees[bt.RootID] = tree
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
node := treenode.GetNodeInTree(tree, blockValue.BlockID)
|
|
|
|
|
if nil == node {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
attrs := parse.IAL2Map(node.KramdownIAL)
|
|
|
|
|
if "" == attrs[av.NodeAttrNameAvs] {
|
|
|
|
|
attrs[av.NodeAttrNameAvs] = avID
|
|
|
|
|
} else {
|
|
|
|
|
nodeAvIDs := strings.Split(attrs[av.NodeAttrNameAvs], ",")
|
|
|
|
|
nodeAvIDs = append(nodeAvIDs, avID)
|
|
|
|
|
nodeAvIDs = gulu.Str.RemoveDuplicatedElem(nodeAvIDs)
|
|
|
|
|
attrs[av.NodeAttrNameAvs] = strings.Join(nodeAvIDs, ",")
|
|
|
|
|
saveTrees[bt.RootID] = tree
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
avNames := getAvNames(attrs[av.NodeAttrNameAvs])
|
|
|
|
|
if "" != avNames {
|
|
|
|
|
attrs[av.NodeAttrViewNames] = avNames
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
oldAttrs, setErr := setNodeAttrs0(node, attrs)
|
|
|
|
|
if nil != setErr {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
cache.PutBlockIAL(node.ID, parse.IAL2Map(node.KramdownIAL))
|
|
|
|
|
pushBroadcastAttrTransactions(oldAttrs, node)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, saveTree := range saveTrees {
|
|
|
|
|
if treeErr := indexWriteTreeUpsertQueue(saveTree); nil != treeErr {
|
|
|
|
|
logging.LogErrorf("index write tree upsert queue failed: %s", treeErr)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
avNodes := saveTree.Root.ChildrenByType(ast.NodeAttributeView)
|
|
|
|
|
av.BatchUpsertBlockRel(avNodes)
|
|
|
|
|
}
|
|
|
|
|
}
|