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-23 22:10:01 +08:00
|
|
|
|
"github.com/siyuan-note/dejavu/entity"
|
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-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 {
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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)
|
2025-06-17 20:18:15 +08:00
|
|
|
|
|
|
|
|
|
ReloadAttrView(operation.AvID)
|
2025-06-17 17:46:32 +08:00
|
|
|
|
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 {
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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 {
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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 {
|
|
|
|
|
view.Table.Columns = append(view.Table.Columns, &av.ViewTableColumn{ID: field.ID})
|
|
|
|
|
}
|
|
|
|
|
for _, cardID := range view.Gallery.CardIDs {
|
|
|
|
|
view.Table.RowIDs = append(view.Table.RowIDs, cardID)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
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 {
|
|
|
|
|
view.Gallery.CardFields = append(view.Gallery.CardFields, &av.ViewGalleryCardField{ID: col.ID})
|
|
|
|
|
}
|
|
|
|
|
for _, rowID := range view.Table.RowIDs {
|
|
|
|
|
view.Gallery.CardIDs = append(view.Gallery.CardIDs, rowID)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-11 15:32:12 +08:00
|
|
|
|
view.LayoutType = newLayout
|
2025-06-10 12:23:07 +08:00
|
|
|
|
err = av.SaveAttributeView(attrView)
|
2025-06-17 11:25:28 +08:00
|
|
|
|
|
|
|
|
|
node, tree, err := getNodeByBlockID(nil, blockID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
node.AttributeViewType = string(view.LayoutType)
|
|
|
|
|
attrs := parse.IAL2Map(node.KramdownIAL)
|
|
|
|
|
attrs[av.NodeAttrView] = view.ID
|
|
|
|
|
err = setNodeAttrs(node, tree, attrs)
|
|
|
|
|
if err != nil {
|
|
|
|
|
logging.LogWarnf("set node [%s] attrs failed: %s", blockID, 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 {
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch view.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
return
|
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
view.Gallery.WrapField = operation.Data.(bool)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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 {
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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:
|
|
|
|
|
return
|
|
|
|
|
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 {
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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-06-09 22:30:00 +08:00
|
|
|
|
func (tx *Transaction) doSetAttrViewCardSize(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := setAttrViewCardSize(operation)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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 {
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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 {
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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.ID = blockID
|
|
|
|
|
v.Block.Created = now
|
|
|
|
|
v.Block.Updated = now
|
|
|
|
|
}
|
|
|
|
|
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 {
|
|
|
|
|
switch v.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
for _, addingBlockID := range blockIDs {
|
|
|
|
|
v.Table.RowIDs = append(v.Table.RowIDs, addingBlockID)
|
|
|
|
|
}
|
2025-06-09 16:46:50 +08:00
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
for _, addingBlockID := range blockIDs {
|
|
|
|
|
v.Gallery.CardIDs = append(v.Gallery.CardIDs, 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
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-07 22:53:05 +08:00
|
|
|
|
filters = []*av.ViewFilter{}
|
|
|
|
|
sorts = []*av.ViewSort{}
|
2024-03-06 00:00:43 +08:00
|
|
|
|
switch view.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
filters = view.Table.Filters
|
|
|
|
|
sorts = view.Table.Sorts
|
2025-06-09 16:46:50 +08:00
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
filters = view.Gallery.Filters
|
|
|
|
|
sorts = view.Gallery.Sorts
|
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 {
|
2024-04-15 00:49:48 +08:00
|
|
|
|
if av.KeyTypeRelation != keyValues.Key.Type && av.KeyTypeRollup != keyValues.Key.Type && av.KeyTypeTemplate != 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
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-23 12:04:54 +08:00
|
|
|
|
type SearchAttributeViewResult struct {
|
|
|
|
|
AvID string `json:"avID"`
|
|
|
|
|
AvName string `json:"avName"`
|
|
|
|
|
BlockID string `json:"blockID"`
|
2023-12-23 21:02:31 +08:00
|
|
|
|
HPath string `json:"hPath"`
|
2023-12-23 12:04:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
2024-04-25 17:33:35 +08:00
|
|
|
|
func SearchAttributeView(keyword string, excludeAvIDs []string) (ret []*SearchAttributeViewResult) {
|
2023-12-23 12:04:54 +08:00
|
|
|
|
waitForSyncingStorages()
|
2024-04-20 23:10:05 +08:00
|
|
|
|
|
2023-12-23 12:04:54 +08:00
|
|
|
|
ret = []*SearchAttributeViewResult{}
|
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
|
|
|
|
|
2024-04-09 17:51:39 +08:00
|
|
|
|
type result struct {
|
2024-04-09 21:26:41 +08:00
|
|
|
|
AvID string
|
|
|
|
|
AvName string
|
|
|
|
|
AvUpdated int64
|
|
|
|
|
Score float64
|
2024-04-09 17:51:39 +08:00
|
|
|
|
}
|
|
|
|
|
var avs []*result
|
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 {
|
2024-04-09 21:26:41 +08:00
|
|
|
|
a := &result{AvID: id, AvName: name, Score: score}
|
|
|
|
|
if nil != info && !info.ModTime().IsZero() {
|
|
|
|
|
a.AvUpdated = info.ModTime().UnixMilli()
|
|
|
|
|
}
|
|
|
|
|
avs = append(avs, a)
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
a := &result{AvID: id, AvName: name}
|
|
|
|
|
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)
|
2024-04-08 21:58:57 +08:00
|
|
|
|
for _, blockID := range blockIDs {
|
2024-06-21 23:37:47 +08:00
|
|
|
|
tree := trees[blockID]
|
2023-12-23 12:04:54 +08:00
|
|
|
|
if nil == tree {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-08 21:58:57 +08:00
|
|
|
|
node := treenode.GetNodeInTree(tree, blockID)
|
2023-12-23 12:04:54 +08:00
|
|
|
|
if nil == node {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if "" == node.AttributeViewID {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
avID := node.AttributeViewID
|
2024-04-09 17:51:39 +08:00
|
|
|
|
var existAv *result
|
|
|
|
|
for _, av := range avs {
|
|
|
|
|
if av.AvID == avID {
|
|
|
|
|
existAv = av
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if nil == existAv {
|
2023-12-23 12:04:54 +08:00
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
exist := false
|
|
|
|
|
for _, result := range ret {
|
|
|
|
|
if result.AvID == avID {
|
|
|
|
|
exist = true
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-04-25 17:33:35 +08:00
|
|
|
|
if exist {
|
|
|
|
|
continue
|
|
|
|
|
}
|
2023-12-23 21:02:31 +08:00
|
|
|
|
|
|
|
|
|
var hPath string
|
|
|
|
|
baseBlock := treenode.GetBlockTreeRootByPath(node.Box, node.Path)
|
|
|
|
|
if nil != baseBlock {
|
|
|
|
|
hPath = baseBlock.HPath
|
|
|
|
|
}
|
2023-12-23 21:05:52 +08:00
|
|
|
|
box := Conf.Box(node.Box)
|
|
|
|
|
if nil != box {
|
|
|
|
|
hPath = box.Name + hPath
|
|
|
|
|
}
|
2023-12-23 21:02:31 +08:00
|
|
|
|
|
2024-04-25 17:33:35 +08:00
|
|
|
|
if !gulu.Str.Contains(avID, excludeAvIDs) {
|
2023-12-23 12:04:54 +08:00
|
|
|
|
ret = append(ret, &SearchAttributeViewResult{
|
|
|
|
|
AvID: avID,
|
2024-04-09 17:51:39 +08:00
|
|
|
|
AvName: existAv.AvName,
|
2024-04-08 21:58:57 +08:00
|
|
|
|
BlockID: blockID,
|
2023-12-23 21:02:31 +08:00
|
|
|
|
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"`
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func GetBlockAttributeViewKeys(blockID string) (ret []*BlockAttributeViewKeys) {
|
|
|
|
|
waitForSyncingStorages()
|
|
|
|
|
|
|
|
|
|
ret = []*BlockAttributeViewKeys{}
|
2024-10-17 23:44:55 +08:00
|
|
|
|
attrs := sql.GetBlockAttrs(blockID)
|
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 {
|
|
|
|
|
unbindBlockAv(nil, avID, blockID)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
attrViewCache[avID] = attrView
|
2023-07-30 20:35:03 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if 1 > len(attrView.Views) {
|
2024-04-08 17:31:10 +08:00
|
|
|
|
unbindBlockAv(nil, avID, blockID)
|
2023-07-30 20:35:03 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-03 21:17:54 +08:00
|
|
|
|
if !attrView.ExistBlock(blockID) {
|
|
|
|
|
// 比如剪切后粘贴,块 ID 会变,但是属性还在块上,这里做一次数据订正
|
|
|
|
|
// Auto verify the database name when clicking the block superscript icon https://github.com/siyuan-note/siyuan/issues/10861
|
|
|
|
|
unbindBlockAv(nil, avID, blockID)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
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 {
|
|
|
|
|
if v.BlockID == blockID {
|
|
|
|
|
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:
|
2024-01-01 15:14:52 +08:00
|
|
|
|
kValues.Values = append(kValues.Values, &av.Value{ID: ast.NewNodeID(), KeyID: kValues.Key.ID, BlockID: blockID, Type: av.KeyTypeRollup, Rollup: &av.ValueRollup{Contents: []*av.Value{}}})
|
2023-10-09 12:14:11 +08:00
|
|
|
|
case av.KeyTypeTemplate:
|
2023-10-03 11:46:25 +08:00
|
|
|
|
kValues.Values = append(kValues.Values, &av.Value{ID: ast.NewNodeID(), KeyID: kValues.Key.ID, BlockID: blockID, Type: av.KeyTypeTemplate, Template: &av.ValueTemplate{Content: ""}})
|
2023-10-09 12:14:11 +08:00
|
|
|
|
case av.KeyTypeCreated:
|
|
|
|
|
kValues.Values = append(kValues.Values, &av.Value{ID: ast.NewNodeID(), KeyID: kValues.Key.ID, BlockID: blockID, Type: av.KeyTypeCreated})
|
|
|
|
|
case av.KeyTypeUpdated:
|
|
|
|
|
kValues.Values = append(kValues.Values, &av.Value{ID: ast.NewNodeID(), KeyID: kValues.Key.ID, BlockID: blockID, 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 {
|
|
|
|
|
// 如果没有值,那么就补一个默认值
|
2024-04-17 19:38:27 +08:00
|
|
|
|
kValues.Values = append(kValues.Values, av.GetAttributeViewDefaultValue(ast.NewNodeID(), kv.Key.ID, blockID, kv.Key.Type))
|
2024-01-01 21:48:23 +08:00
|
|
|
|
keyValues = append(keyValues, kValues)
|
2023-07-30 20:35:03 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-23 20:51:10 +08:00
|
|
|
|
// 渲染自动生成的列值,比如模板列、关联列、汇总列、创建时间列和更新时间列
|
2023-12-24 22:27:38 +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 {
|
|
|
|
|
ial := sql.GetBlockAttrs(blockID)
|
|
|
|
|
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 {
|
2024-04-03 09:37:42 +08:00
|
|
|
|
if destAv.ExistBlock(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:
|
|
|
|
|
createdStr := blockID[: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:
|
2024-10-17 23:44:55 +08:00
|
|
|
|
ial := sql.GetBlockAttrs(blockID)
|
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
|
|
|
|
|
2023-10-10 21:55:43 +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 {
|
|
|
|
|
kv.Values[0] = av.GetAttributeViewDefaultValue(kv.Values[0].ID, kv.Key.ID, blockID, kv.Key.Type)
|
|
|
|
|
}
|
|
|
|
|
|
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) {
|
2024-03-10 23:27:13 +08:00
|
|
|
|
tree, _ := LoadTreeByBlockID(blockID)
|
2024-02-16 16:02:22 +08:00
|
|
|
|
if nil != tree {
|
|
|
|
|
node := treenode.GetNodeInTree(tree, blockID)
|
|
|
|
|
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
|
|
|
|
|
}
|
|
|
|
|
|
2023-11-23 22:10:01 +08:00
|
|
|
|
func RenderRepoSnapshotAttributeView(indexID, avID string) (viewable av.Viewable, attrView *av.AttributeView, err error) {
|
|
|
|
|
repo, err := newRepository()
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-11-23 22:10:01 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
index, err := repo.GetIndex(indexID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-11-23 22:10:01 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
files, err := repo.GetFiles(index)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-11-23 22:10:01 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
var avFile *entity.File
|
|
|
|
|
for _, f := range files {
|
|
|
|
|
if "/storage/av/"+avID+".json" == f.Path {
|
|
|
|
|
avFile = f
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if nil == avFile {
|
2023-11-24 00:07:58 +08:00
|
|
|
|
attrView = av.NewAttributeView(avID)
|
|
|
|
|
} else {
|
|
|
|
|
data, readErr := repo.OpenFile(avFile)
|
|
|
|
|
if nil != readErr {
|
|
|
|
|
logging.LogErrorf("read attribute view [%s] failed: %s", avID, readErr)
|
|
|
|
|
return
|
|
|
|
|
}
|
2023-11-23 22:10:01 +08:00
|
|
|
|
|
2023-11-24 00:07:58 +08:00
|
|
|
|
attrView = &av.AttributeView{}
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = gulu.JSON.UnmarshalJSON(data, attrView); err != nil {
|
2023-11-24 00:07:58 +08:00
|
|
|
|
logging.LogErrorf("unmarshal attribute view [%s] failed: %s", avID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
2023-11-23 22:10:01 +08:00
|
|
|
|
}
|
|
|
|
|
|
2024-03-09 17:31:06 +08:00
|
|
|
|
viewable, err = renderAttributeView(attrView, "", "", 1, -1)
|
2023-11-23 22:10:01 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-11-23 11:22:35 +08:00
|
|
|
|
func RenderHistoryAttributeView(avID, created string) (viewable av.Viewable, attrView *av.AttributeView, err error) {
|
|
|
|
|
createdUnix, parseErr := strconv.ParseInt(created, 10, 64)
|
|
|
|
|
if nil != parseErr {
|
|
|
|
|
logging.LogErrorf("parse created [%s] failed: %s", created, parseErr)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
dirPrefix := time.Unix(createdUnix, 0).Format("2006-01-02-150405")
|
|
|
|
|
globPath := filepath.Join(util.HistoryDir, dirPrefix+"*")
|
|
|
|
|
matches, err := filepath.Glob(globPath)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-11-23 11:22:35 +08:00
|
|
|
|
logging.LogErrorf("glob [%s] failed: %s", globPath, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
if 1 > len(matches) {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
historyDir := matches[0]
|
|
|
|
|
avJSONPath := filepath.Join(historyDir, "storage", "av", avID+".json")
|
|
|
|
|
if !gulu.File.IsExist(avJSONPath) {
|
2023-11-24 00:07:58 +08:00
|
|
|
|
avJSONPath = filepath.Join(util.DataDir, "storage", "av", avID+".json")
|
2023-11-23 11:22:35 +08:00
|
|
|
|
}
|
2023-11-24 00:07:58 +08:00
|
|
|
|
if !gulu.File.IsExist(avJSONPath) {
|
|
|
|
|
attrView = av.NewAttributeView(avID)
|
|
|
|
|
} else {
|
|
|
|
|
data, readErr := os.ReadFile(avJSONPath)
|
|
|
|
|
if nil != readErr {
|
|
|
|
|
logging.LogErrorf("read attribute view [%s] failed: %s", avID, readErr)
|
|
|
|
|
return
|
|
|
|
|
}
|
2023-11-23 11:22:35 +08:00
|
|
|
|
|
2023-11-24 00:07:58 +08:00
|
|
|
|
attrView = &av.AttributeView{}
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = gulu.JSON.UnmarshalJSON(data, attrView); err != nil {
|
2023-11-24 00:07:58 +08:00
|
|
|
|
logging.LogErrorf("unmarshal attribute view [%s] failed: %s", avID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
2023-11-23 11:22:35 +08:00
|
|
|
|
}
|
|
|
|
|
|
2024-03-09 17:31:06 +08:00
|
|
|
|
viewable, err = renderAttributeView(attrView, "", "", 1, -1)
|
2023-11-23 11:22:35 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-08 23:49:27 +08:00
|
|
|
|
func RenderAttributeView(avID, viewID, query string, page, pageSize int) (viewable av.Viewable, attrView *av.AttributeView, err error) {
|
2023-04-02 11:42:04 +08:00
|
|
|
|
waitForSyncingStorages()
|
|
|
|
|
|
2023-11-06 22:13:04 +08:00
|
|
|
|
if avJSONPath := av.GetAttributeViewDataPath(avID); !filelock.IsExist(avJSONPath) {
|
2023-09-26 09:38:50 +08:00
|
|
|
|
attrView = av.NewAttributeView(avID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = av.SaveAttributeView(attrView); err != nil {
|
2023-07-31 11:20:58 +08:00
|
|
|
|
logging.LogErrorf("save attribute view [%s] failed: %s", avID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-11 19:45:27 +08:00
|
|
|
|
attrView, err = av.ParseAttributeView(avID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-03-03 15:01:42 +08:00
|
|
|
|
logging.LogErrorf("parse attribute view [%s] failed: %s", avID, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
2023-07-03 18:45:41 +08:00
|
|
|
|
|
2024-03-09 17:31:06 +08:00
|
|
|
|
viewable, err = renderAttributeView(attrView, viewID, query, page, pageSize)
|
2023-11-23 11:22:35 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-09 17:31:06 +08:00
|
|
|
|
func renderAttributeView(attrView *av.AttributeView, viewID, query string, page, pageSize int) (viewable av.Viewable, err error) {
|
2023-07-11 19:45:27 +08:00
|
|
|
|
if 1 > len(attrView.Views) {
|
2024-03-24 21:26:01 +08:00
|
|
|
|
view, _, _ := av.NewTableViewWithBlockKey(ast.NewNodeID())
|
2023-10-23 11:38:27 +08:00
|
|
|
|
attrView.Views = append(attrView.Views, view)
|
|
|
|
|
attrView.ViewID = view.ID
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = av.SaveAttributeView(attrView); err != nil {
|
2023-11-23 11:22:35 +08:00
|
|
|
|
logging.LogErrorf("save attribute view [%s] failed: %s", attrView.ID, err)
|
2023-10-23 11:38:27 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
2023-07-11 19:45:27 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var view *av.View
|
2023-11-30 20:04:51 +08:00
|
|
|
|
if "" != viewID {
|
2024-03-04 15:57:35 +08:00
|
|
|
|
view, _ = attrView.GetCurrentView(viewID)
|
2024-03-04 11:51:03 +08:00
|
|
|
|
if nil != view && view.ID != attrView.ViewID {
|
|
|
|
|
attrView.ViewID = view.ID
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = av.SaveAttributeView(attrView); err != nil {
|
2023-11-30 20:04:51 +08:00
|
|
|
|
logging.LogErrorf("save attribute view [%s] failed: %s", attrView.ID, err)
|
|
|
|
|
return
|
2023-07-11 19:45:27 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
2024-03-04 15:57:35 +08:00
|
|
|
|
view = attrView.GetView(attrView.ViewID)
|
2023-11-30 20:04:51 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if nil == view {
|
2023-07-11 19:45:27 +08:00
|
|
|
|
view = attrView.Views[0]
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-16 23:57:09 +08:00
|
|
|
|
// 做一些数据兼容和订正处理,保存的时候也会做 av.SaveAttributeView()
|
2024-12-19 23:08:44 +08:00
|
|
|
|
upgradeAttributeViewSpec(attrView)
|
2024-03-06 23:12:01 +08:00
|
|
|
|
|
2023-07-12 21:39:55 +08:00
|
|
|
|
switch view.LayoutType {
|
2023-07-12 10:35:17 +08:00
|
|
|
|
case av.LayoutTypeTable:
|
2023-10-09 17:17:40 +08:00
|
|
|
|
// 列删除以后需要删除设置的过滤和排序
|
|
|
|
|
tmpFilters := []*av.ViewFilter{}
|
|
|
|
|
for _, f := range view.Table.Filters {
|
|
|
|
|
if k, _ := attrView.GetKey(f.Column); nil != k {
|
|
|
|
|
tmpFilters = append(tmpFilters, f)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
view.Table.Filters = tmpFilters
|
|
|
|
|
|
|
|
|
|
tmpSorts := []*av.ViewSort{}
|
|
|
|
|
for _, s := range view.Table.Sorts {
|
|
|
|
|
if k, _ := attrView.GetKey(s.Column); nil != k {
|
|
|
|
|
tmpSorts = append(tmpSorts, s)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
view.Table.Sorts = tmpSorts
|
2025-06-08 16:12:37 +08:00
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
// 字段删除以后需要删除设置的过滤和排序
|
|
|
|
|
tmpFilters := []*av.ViewFilter{}
|
|
|
|
|
for _, f := range view.Gallery.Filters {
|
|
|
|
|
if k, _ := attrView.GetKey(f.Column); nil != k {
|
|
|
|
|
tmpFilters = append(tmpFilters, f)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
view.Gallery.Filters = tmpFilters
|
|
|
|
|
|
|
|
|
|
tmpSorts := []*av.ViewSort{}
|
|
|
|
|
for _, s := range view.Gallery.Sorts {
|
|
|
|
|
if k, _ := attrView.GetKey(s.Column); nil != k {
|
|
|
|
|
tmpSorts = append(tmpSorts, s)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
view.Gallery.Sorts = tmpSorts
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-13 11:09:28 +08:00
|
|
|
|
viewable = sql.RenderView(view, attrView, query)
|
2025-06-08 16:12:37 +08:00
|
|
|
|
if nil == viewable {
|
|
|
|
|
err = av.ErrViewNotFound
|
|
|
|
|
logging.LogErrorf("render attribute view [%s] failed", attrView.ID)
|
|
|
|
|
return
|
2023-07-11 19:45:27 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-06-08 16:12:37 +08:00
|
|
|
|
viewable.Filter(attrView)
|
|
|
|
|
viewable.Sort(attrView)
|
|
|
|
|
viewable.Calc()
|
2023-12-08 22:17:50 +08:00
|
|
|
|
|
|
|
|
|
// 分页
|
|
|
|
|
switch viewable.GetType() {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
table := viewable.(*av.Table)
|
|
|
|
|
table.RowCount = len(table.Rows)
|
|
|
|
|
if 1 > view.Table.PageSize {
|
2025-06-08 16:12:37 +08:00
|
|
|
|
view.Table.PageSize = av.TableViewDefaultPageSize
|
2023-12-08 22:17:50 +08:00
|
|
|
|
}
|
|
|
|
|
table.PageSize = view.Table.PageSize
|
|
|
|
|
if 1 > pageSize {
|
|
|
|
|
pageSize = table.PageSize
|
|
|
|
|
}
|
|
|
|
|
start := (page - 1) * pageSize
|
|
|
|
|
end := start + pageSize
|
|
|
|
|
if len(table.Rows) < end {
|
|
|
|
|
end = len(table.Rows)
|
|
|
|
|
}
|
|
|
|
|
table.Rows = table.Rows[start:end]
|
2025-06-08 16:12:37 +08:00
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
gallery := viewable.(*av.Gallery)
|
|
|
|
|
gallery.CardCount = len(gallery.Cards)
|
|
|
|
|
if 1 > view.Gallery.PageSize {
|
|
|
|
|
view.Gallery.PageSize = av.GalleryViewDefaultPageSize
|
|
|
|
|
}
|
|
|
|
|
gallery.PageSize = view.Gallery.PageSize
|
|
|
|
|
if 1 > pageSize {
|
|
|
|
|
pageSize = gallery.PageSize
|
|
|
|
|
}
|
|
|
|
|
start := (page - 1) * pageSize
|
|
|
|
|
end := start + pageSize
|
|
|
|
|
if len(gallery.Cards) < end {
|
|
|
|
|
end = len(gallery.Cards)
|
|
|
|
|
}
|
|
|
|
|
gallery.Cards = gallery.Cards[start:end]
|
2023-12-08 22:17:50 +08:00
|
|
|
|
}
|
2023-07-11 19:45:27 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
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-06-08 16:12:37 +08:00
|
|
|
|
table.Filter(attrView)
|
|
|
|
|
table.Sort(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-15 17:47:03 +08:00
|
|
|
|
func (tx *Transaction) doUnbindAttrViewBlock(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := unbindAttributeViewBlock(operation, tx)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2024-04-15 17:47:03 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func unbindAttributeViewBlock(operation *Operation, tx *Transaction) (err error) {
|
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2024-04-15 17:47:03 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
node, _, _ := getNodeByBlockID(tx, operation.ID)
|
|
|
|
|
if nil == node {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-09-02 17:53:34 +08:00
|
|
|
|
var changedAvIDs []string
|
2024-04-15 18:00:04 +08:00
|
|
|
|
for _, keyValues := range attrView.KeyValues {
|
|
|
|
|
for _, value := range keyValues.Values {
|
2024-09-02 17:53:34 +08:00
|
|
|
|
if av.KeyTypeRelation == value.Type {
|
|
|
|
|
if nil != value.Relation {
|
|
|
|
|
for i, relBlockID := range value.Relation.BlockIDs {
|
|
|
|
|
if relBlockID == operation.ID {
|
|
|
|
|
value.Relation.BlockIDs[i] = operation.NextID
|
|
|
|
|
changedAvIDs = append(changedAvIDs, attrView.ID)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-15 18:00:04 +08:00
|
|
|
|
if value.BlockID != operation.ID {
|
|
|
|
|
continue
|
|
|
|
|
}
|
2024-04-15 17:47:03 +08:00
|
|
|
|
|
2024-04-15 18:00:04 +08:00
|
|
|
|
if av.KeyTypeBlock == value.Type {
|
|
|
|
|
unbindBlockAv(tx, operation.AvID, value.BlockID)
|
|
|
|
|
}
|
|
|
|
|
value.BlockID = operation.NextID
|
2024-04-21 17:52:08 +08:00
|
|
|
|
value.IsDetached = true
|
2024-04-15 18:00:04 +08:00
|
|
|
|
if nil != value.Block {
|
|
|
|
|
value.Block.ID = operation.NextID
|
|
|
|
|
}
|
2024-04-23 18:22:51 +08:00
|
|
|
|
|
2024-09-02 17:53:34 +08:00
|
|
|
|
avIDs := replaceRelationAvValues(operation.AvID, operation.ID, operation.NextID)
|
|
|
|
|
changedAvIDs = append(changedAvIDs, avIDs...)
|
2024-04-15 17:47:03 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
replacedRowID := false
|
|
|
|
|
for _, v := range attrView.Views {
|
|
|
|
|
switch v.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
for i, rowID := range v.Table.RowIDs {
|
|
|
|
|
if rowID == operation.ID {
|
|
|
|
|
v.Table.RowIDs[i] = operation.NextID
|
|
|
|
|
replacedRowID = true
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if !replacedRowID {
|
|
|
|
|
v.Table.RowIDs = append(v.Table.RowIDs, operation.NextID)
|
|
|
|
|
}
|
2025-06-09 16:46:50 +08:00
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
for i, cardID := range v.Gallery.CardIDs {
|
|
|
|
|
if cardID == operation.ID {
|
|
|
|
|
v.Gallery.CardIDs[i] = operation.NextID
|
|
|
|
|
replacedRowID = true
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if !replacedRowID {
|
|
|
|
|
v.Gallery.CardIDs = append(v.Gallery.CardIDs, operation.NextID)
|
|
|
|
|
}
|
2024-04-15 17:47:03 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
2024-09-02 17:53:34 +08:00
|
|
|
|
|
|
|
|
|
changedAvIDs = gulu.Str.RemoveDuplicatedElem(changedAvIDs)
|
|
|
|
|
for _, avID := range changedAvIDs {
|
2024-09-05 12:10:42 +08:00
|
|
|
|
ReloadAttrView(avID)
|
2024-09-02 17:53:34 +08:00
|
|
|
|
}
|
2024-04-15 17:47:03 +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 {
|
2024-04-04 11:33:51 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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 {
|
2024-03-01 20:38:53 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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 {
|
2023-12-24 21:47:10 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func updateAttributeViewColRollup(operation *Operation) (err error) {
|
|
|
|
|
// operation.AvID 汇总列所在 av
|
|
|
|
|
// operation.ID 汇总列 ID
|
|
|
|
|
// operation.ParentID 汇总列基于的关联列 ID
|
|
|
|
|
// operation.KeyID 目标列 ID
|
|
|
|
|
// 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 {
|
2023-12-23 11:35:53 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func updateAttributeViewColRelation(operation *Operation) (err error) {
|
2023-12-23 17:45:46 +08:00
|
|
|
|
// operation.AvID 源 avID
|
|
|
|
|
// operation.ID 目标 avID
|
|
|
|
|
// operation.KeyID 源 av 关联列 ID
|
|
|
|
|
// operation.IsTwoWay 是否双向关联
|
|
|
|
|
// operation.BackRelationKeyID 双向关联的目标关联列 ID
|
|
|
|
|
// operation.Name 双向关联的目标关联列名称
|
2023-12-25 12:06:53 +08:00
|
|
|
|
// 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:
|
|
|
|
|
v.Table.Columns = append(v.Table.Columns, &av.ViewTableColumn{ID: operation.BackRelationKeyID})
|
2025-06-09 16:46:50 +08:00
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
v.Gallery.CardFields = append(v.Gallery.CardFields, &av.ViewGalleryCardField{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
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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 {
|
|
|
|
|
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)
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
|
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)
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
if nil == view {
|
2023-12-01 20:06:08 +08:00
|
|
|
|
return
|
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)
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: avID}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
masterView := attrView.GetView(operation.PreviousID)
|
|
|
|
|
if nil == masterView {
|
|
|
|
|
logging.LogErrorf("get master view failed: %s", avID)
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: avID}
|
|
|
|
|
}
|
|
|
|
|
|
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)
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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-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{
|
|
|
|
|
ID: col.ID,
|
|
|
|
|
Wrap: col.Wrap,
|
|
|
|
|
Hidden: col.Hidden,
|
|
|
|
|
Pin: col.Pin,
|
|
|
|
|
Width: col.Width,
|
|
|
|
|
Desc: col.Desc,
|
|
|
|
|
Calc: col.Calc,
|
|
|
|
|
})
|
|
|
|
|
}
|
2023-12-01 09:48:20 +08:00
|
|
|
|
|
2025-06-10 11:36:30 +08:00
|
|
|
|
for _, filter := range masterView.Table.Filters {
|
|
|
|
|
view.Table.Filters = append(view.Table.Filters, &av.ViewFilter{
|
|
|
|
|
Column: filter.Column,
|
|
|
|
|
Operator: filter.Operator,
|
|
|
|
|
Value: filter.Value,
|
|
|
|
|
RelativeDate: filter.RelativeDate,
|
|
|
|
|
RelativeDate2: filter.RelativeDate2,
|
|
|
|
|
})
|
|
|
|
|
}
|
2023-12-01 09:48:20 +08:00
|
|
|
|
|
2025-06-10 11:36:30 +08:00
|
|
|
|
for _, s := range masterView.Table.Sorts {
|
|
|
|
|
view.Table.Sorts = append(view.Table.Sorts, &av.ViewSort{
|
|
|
|
|
Column: s.Column,
|
|
|
|
|
Order: s.Order,
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view.Table.PageSize = masterView.Table.PageSize
|
|
|
|
|
view.Table.RowIDs = masterView.Table.RowIDs
|
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
for _, field := range masterView.Gallery.CardFields {
|
|
|
|
|
view.Gallery.CardFields = append(view.Gallery.CardFields, &av.ViewGalleryCardField{
|
|
|
|
|
ID: field.ID,
|
|
|
|
|
Hidden: field.Hidden,
|
|
|
|
|
Desc: field.Desc,
|
|
|
|
|
})
|
|
|
|
|
}
|
2023-12-01 09:48:20 +08:00
|
|
|
|
|
2025-06-10 11:36:30 +08:00
|
|
|
|
for _, filter := range masterView.Gallery.Filters {
|
|
|
|
|
view.Gallery.Filters = append(view.Gallery.Filters, &av.ViewFilter{
|
|
|
|
|
Column: filter.Column,
|
|
|
|
|
Operator: filter.Operator,
|
|
|
|
|
Value: filter.Value,
|
|
|
|
|
RelativeDate: filter.RelativeDate,
|
|
|
|
|
RelativeDate2: filter.RelativeDate2,
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, s := range masterView.Gallery.Sorts {
|
|
|
|
|
view.Gallery.Sorts = append(view.Gallery.Sorts, &av.ViewSort{
|
|
|
|
|
Column: s.Column,
|
|
|
|
|
Order: s.Order,
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view.Gallery.PageSize = masterView.Gallery.PageSize
|
|
|
|
|
view.Gallery.CardIDs = masterView.Gallery.CardIDs
|
|
|
|
|
|
|
|
|
|
view.Gallery.CoverFrom = masterView.Gallery.CoverFrom
|
|
|
|
|
view.Gallery.CoverFromAssetKeyID = masterView.Gallery.CoverFromAssetKeyID
|
|
|
|
|
view.Gallery.CardSize = masterView.Gallery.CardSize
|
|
|
|
|
view.Gallery.FitImage = masterView.Gallery.FitImage
|
|
|
|
|
view.Gallery.ShowIcon = masterView.Gallery.ShowIcon
|
|
|
|
|
view.Gallery.WrapField = masterView.Gallery.WrapField
|
|
|
|
|
}
|
2023-12-08 22:00:23 +08:00
|
|
|
|
|
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)
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, msg: err.Error(), id: avID}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-01 08:55:32 +08:00
|
|
|
|
func (tx *Transaction) doAddAttrViewView(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: TxErrWriteAttributeView, id: avID}
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-04 16:41:41 +08:00
|
|
|
|
if 1 > len(attrView.Views) {
|
|
|
|
|
logging.LogErrorf("no view in attribute view [%s]", avID)
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: avID}
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-01 09:17:44 +08:00
|
|
|
|
firstView := attrView.Views[0]
|
|
|
|
|
if nil == firstView {
|
|
|
|
|
logging.LogErrorf("get first view failed: %s", avID)
|
2023-12-01 08:55:32 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: avID}
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-08 17:22:03 +08:00
|
|
|
|
if "" == operation.Layout {
|
|
|
|
|
operation.Layout = av.LayoutTypeTable
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-08 11:01:21 +08:00
|
|
|
|
var view *av.View
|
|
|
|
|
switch operation.Layout {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
view = av.NewTableView()
|
|
|
|
|
switch firstView.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
for _, col := range firstView.Table.Columns {
|
|
|
|
|
view.Table.Columns = append(view.Table.Columns, &av.ViewTableColumn{ID: col.ID})
|
|
|
|
|
}
|
|
|
|
|
for _, rowID := range firstView.Table.RowIDs {
|
|
|
|
|
view.Table.RowIDs = append(view.Table.RowIDs, rowID)
|
|
|
|
|
}
|
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
for _, field := range firstView.Gallery.CardFields {
|
|
|
|
|
view.Table.Columns = append(view.Table.Columns, &av.ViewTableColumn{ID: field.ID})
|
|
|
|
|
}
|
|
|
|
|
for _, cardID := range firstView.Gallery.CardIDs {
|
|
|
|
|
view.Table.RowIDs = append(view.Table.RowIDs, cardID)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
view = av.NewGalleryView()
|
|
|
|
|
switch firstView.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
for _, col := range firstView.Table.Columns {
|
|
|
|
|
view.Gallery.CardFields = append(view.Gallery.CardFields, &av.ViewGalleryCardField{ID: col.ID})
|
|
|
|
|
}
|
|
|
|
|
for _, rowID := range firstView.Table.RowIDs {
|
|
|
|
|
view.Gallery.CardIDs = append(view.Gallery.CardIDs, rowID)
|
|
|
|
|
}
|
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
for _, field := range firstView.Gallery.CardFields {
|
|
|
|
|
view.Gallery.CardFields = append(view.Gallery.CardFields, &av.ViewGalleryCardField{ID: field.ID})
|
|
|
|
|
}
|
|
|
|
|
for _, cardID := range firstView.Gallery.CardIDs {
|
|
|
|
|
view.Gallery.CardIDs = append(view.Gallery.CardIDs, cardID)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
err = av.ErrWrongLayoutType
|
|
|
|
|
logging.LogErrorf("wrong layout type [%s] for attribute view [%s]", operation.Layout, avID)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-04 16:41:41 +08:00
|
|
|
|
view.ID = operation.ID
|
2023-12-01 08:55:32 +08:00
|
|
|
|
attrView.Views = append(attrView.Views, view)
|
|
|
|
|
attrView.ViewID = view.ID
|
2025-06-09 18:31:38 +08:00
|
|
|
|
|
|
|
|
|
node, tree, _ := getNodeByBlockID(nil, operation.BlockID)
|
|
|
|
|
if nil == node {
|
|
|
|
|
logging.LogErrorf("get node by block ID [%s] failed", operation.BlockID)
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID}
|
|
|
|
|
}
|
|
|
|
|
|
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)
|
|
|
|
|
attrs[av.NodeAttrView] = operation.ID
|
|
|
|
|
err = setNodeAttrs(node, tree, attrs)
|
|
|
|
|
if err != nil {
|
|
|
|
|
logging.LogWarnf("set node [%s] attrs failed: %s", operation.BlockID, err)
|
|
|
|
|
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)
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, msg: err.Error(), id: avID}
|
|
|
|
|
}
|
|
|
|
|
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)
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: avID}
|
|
|
|
|
}
|
|
|
|
|
|
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)
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: viewID}
|
|
|
|
|
}
|
|
|
|
|
|
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)
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, msg: err.Error(), id: avID}
|
|
|
|
|
}
|
|
|
|
|
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)
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: avID}
|
|
|
|
|
}
|
|
|
|
|
|
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)
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: viewID}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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)
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, msg: err.Error(), id: avID}
|
|
|
|
|
}
|
|
|
|
|
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)
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: avID}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
viewID := operation.ID
|
|
|
|
|
view := attrView.GetView(viewID)
|
|
|
|
|
if nil == view {
|
|
|
|
|
logging.LogErrorf("get view [%s] failed: %s", viewID, err)
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: viewID}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view.Desc = strings.TrimSpace(operation.Data.(string))
|
|
|
|
|
if err = av.SaveAttributeView(attrView); err != nil {
|
|
|
|
|
logging.LogErrorf("save attribute view [%s] failed: %s", avID, err)
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, msg: err.Error(), id: avID}
|
|
|
|
|
}
|
|
|
|
|
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 {
|
2023-07-11 22:11:15 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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))
|
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
|
2024-11-05 09:18:11 +08:00
|
|
|
|
tree = trees[blockKeyValue.BlockID]
|
2024-03-08 22:51:28 +08:00
|
|
|
|
if nil == tree {
|
2024-09-26 09:00:35 +08:00
|
|
|
|
if nil == tx {
|
|
|
|
|
tree, _ = LoadTreeByBlockID(blockKeyValue.BlockID)
|
|
|
|
|
} else {
|
|
|
|
|
tree, _ = tx.loadTree(blockKeyValue.BlockID)
|
|
|
|
|
}
|
2024-03-08 22:51:28 +08:00
|
|
|
|
}
|
|
|
|
|
if nil == tree {
|
|
|
|
|
continue
|
|
|
|
|
}
|
2024-11-05 09:18:11 +08:00
|
|
|
|
trees[blockKeyValue.BlockID] = tree
|
2024-03-08 22:51:28 +08:00
|
|
|
|
|
|
|
|
|
node := treenode.GetNodeInTree(tree, blockKeyValue.BlockID)
|
|
|
|
|
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 {
|
2023-07-11 22:36:02 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-12 21:39:55 +08:00
|
|
|
|
switch view.LayoutType {
|
2023-07-12 19:10:05 +08:00
|
|
|
|
case av.LayoutTypeTable:
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = gulu.JSON.UnmarshalJSON(data, &view.Table.Filters); err != nil {
|
2023-07-12 19:10:05 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
2025-06-09 16:46:50 +08:00
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
if err = gulu.JSON.UnmarshalJSON(data, &view.Gallery.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 {
|
2023-07-11 22:36:02 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-12 21:39:55 +08:00
|
|
|
|
switch view.LayoutType {
|
2023-07-12 19:10:05 +08:00
|
|
|
|
case av.LayoutTypeTable:
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = gulu.JSON.UnmarshalJSON(data, &view.Table.Sorts); err != nil {
|
2023-07-12 19:10:05 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
2025-06-09 16:46:50 +08:00
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
if err = gulu.JSON.UnmarshalJSON(data, &view.Gallery.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 {
|
2023-12-08 21:05:21 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch view.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
view.Table.PageSize = int(operation.Data.(float64))
|
2025-06-09 16:46:50 +08:00
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
view.Gallery.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 {
|
2023-07-15 21:36:35 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
calc := &av.ColumnCalc{}
|
|
|
|
|
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) {
|
2024-04-20 22:37:39 +08:00
|
|
|
|
err := AddAttributeViewBlock(tx, operation.Srcs, operation.AvID, operation.BlockID, operation.PreviousID, operation.IgnoreFillFilterVal)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2024-01-18 21:01:36 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-20 22:37:39 +08:00
|
|
|
|
func AddAttributeViewBlock(tx *Transaction, srcs []map[string]interface{}, avID, blockID, previousBlockID string, ignoreFillFilter bool) (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
|
|
|
|
|
}
|
|
|
|
|
|
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)
|
|
|
|
|
}
|
2024-05-12 10:57:36 +08:00
|
|
|
|
if avErr := addAttributeViewBlock(now, avID, blockID, previousBlockID, srcID, srcContent, isDetached, ignoreFillFilter, tree, tx); nil != avErr {
|
2024-01-18 21:01:36 +08:00
|
|
|
|
return avErr
|
2023-07-11 22:36:02 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-05-12 10:57:36 +08:00
|
|
|
|
func addAttributeViewBlock(now int64, avID, blockID, previousBlockID, addingBlockID, addingBlockContent string, isDetached, ignoreFillFilter 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 {
|
2024-03-04 16:41:41 +08:00
|
|
|
|
node = treenode.GetNodeInTree(tree, addingBlockID)
|
2023-09-27 16:23:46 +08:00
|
|
|
|
if nil == node {
|
|
|
|
|
err = ErrBlockNotFound
|
|
|
|
|
return
|
|
|
|
|
}
|
2023-09-27 23:14:42 +08:00
|
|
|
|
} else {
|
2024-03-04 16:41:41 +08:00
|
|
|
|
if "" == addingBlockID {
|
|
|
|
|
addingBlockID = ast.NewNodeID()
|
|
|
|
|
logging.LogWarnf("detached block id is empty, generate a new one [%s]", addingBlockID)
|
2023-11-10 17:10:28 +08:00
|
|
|
|
}
|
2023-07-11 22:36:02 +08:00
|
|
|
|
}
|
|
|
|
|
|
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 {
|
2024-03-04 16:41:41 +08:00
|
|
|
|
if blockValue.Block.ID == addingBlockID {
|
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,
|
2024-03-04 16:41:41 +08:00
|
|
|
|
BlockID: addingBlockID,
|
2024-03-03 16:12:44 +08:00
|
|
|
|
Type: av.KeyTypeBlock,
|
|
|
|
|
IsDetached: isDetached,
|
|
|
|
|
CreatedAt: now,
|
|
|
|
|
UpdatedAt: now,
|
2024-12-16 12:00:38 +08:00
|
|
|
|
Block: &av.ValueBlock{ID: addingBlockID, Icon: blockIcon, Content: addingBlockContent, Created: now, Updated: now}}
|
2023-12-17 12:11:59 +08:00
|
|
|
|
blockValues.Values = append(blockValues.Values, blockValue)
|
2023-07-11 22:36:02 +08:00
|
|
|
|
|
2024-03-03 16:12:44 +08:00
|
|
|
|
// 如果存在过滤条件,则将过滤条件应用到新添加的块上
|
2024-03-04 16:41:41 +08:00
|
|
|
|
view, _ := getAttrViewViewByBlockID(attrView, blockID)
|
2025-06-12 22:21:07 +08:00
|
|
|
|
var filters []*av.ViewFilter
|
|
|
|
|
if nil != view {
|
|
|
|
|
filters = view.GetFilters()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if nil != view && 0 < len(filters) && !ignoreFillFilter {
|
2025-06-13 11:09:28 +08:00
|
|
|
|
viewable := sql.RenderView(view, attrView, "")
|
2025-06-08 16:12:37 +08:00
|
|
|
|
viewable.Filter(attrView)
|
|
|
|
|
viewable.Sort(attrView)
|
2024-01-07 17:30:52 +08:00
|
|
|
|
|
2025-06-12 22:21:07 +08:00
|
|
|
|
collection := viewable.(av.Collection)
|
|
|
|
|
items := collection.GetItems()
|
|
|
|
|
|
|
|
|
|
var nearItem av.Item
|
|
|
|
|
if 0 < len(items) {
|
2024-03-10 22:06:25 +08:00
|
|
|
|
if "" != previousBlockID {
|
2025-06-12 22:21:07 +08:00
|
|
|
|
for _, row := range items {
|
|
|
|
|
if row.GetID() == previousBlockID {
|
|
|
|
|
nearItem = row
|
2024-03-10 22:06:25 +08:00
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
2025-06-12 22:21:07 +08:00
|
|
|
|
if 0 < len(items) {
|
|
|
|
|
nearItem = items[0]
|
2024-03-10 22:06:25 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2024-03-06 23:39:49 +08:00
|
|
|
|
}
|
2024-03-06 11:20:20 +08:00
|
|
|
|
|
2024-03-06 23:39:49 +08:00
|
|
|
|
sameKeyFilterSort := false // 是否在同一个字段上同时存在过滤和排序
|
2025-06-12 22:21:07 +08:00
|
|
|
|
sorts := view.GetSorts()
|
|
|
|
|
if 0 < len(sorts) {
|
2024-03-06 11:20:20 +08:00
|
|
|
|
filterKeys, sortKeys := map[string]bool{}, map[string]bool{}
|
2025-06-12 22:21:07 +08:00
|
|
|
|
for _, f := range filters {
|
2024-03-07 09:37:49 +08:00
|
|
|
|
filterKeys[f.Column] = true
|
2024-03-06 11:20:20 +08:00
|
|
|
|
}
|
2025-06-12 22:21:07 +08:00
|
|
|
|
for _, s := range sorts {
|
2024-03-07 09:37:49 +08:00
|
|
|
|
sortKeys[s.Column] = true
|
2024-03-06 11:20:20 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for key := range filterKeys {
|
|
|
|
|
if sortKeys[key] {
|
|
|
|
|
sameKeyFilterSort = true
|
2024-03-03 16:12:44 +08:00
|
|
|
|
break
|
2023-12-30 16:59:41 +08:00
|
|
|
|
}
|
2023-12-17 12:00:55 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2024-03-06 11:20:20 +08:00
|
|
|
|
|
|
|
|
|
if !sameKeyFilterSort {
|
|
|
|
|
// 如果在同一个字段上仅存在过滤条件,则将过滤条件应用到新添加的块上
|
2025-06-12 22:21:07 +08:00
|
|
|
|
for _, filter := range filters {
|
2024-03-06 11:20:20 +08:00
|
|
|
|
for _, keyValues := range attrView.KeyValues {
|
|
|
|
|
if keyValues.Key.ID == filter.Column {
|
|
|
|
|
var defaultVal *av.Value
|
2025-06-12 22:21:07 +08:00
|
|
|
|
if nil != nearItem {
|
|
|
|
|
defaultVal = nearItem.GetValue(filter.Column)
|
2024-03-06 11:20:20 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
newValue := filter.GetAffectValue(keyValues.Key, defaultVal)
|
|
|
|
|
if nil == newValue {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
2024-05-08 21:26:56 +08:00
|
|
|
|
if av.KeyTypeBlock == newValue.Type {
|
|
|
|
|
// 如果是主键的话前面已经添加过了,这里仅修改内容
|
|
|
|
|
blockValue.Block.Content = newValue.Block.Content
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-06 11:20:20 +08:00
|
|
|
|
newValue.ID = ast.NewNodeID()
|
|
|
|
|
newValue.KeyID = keyValues.Key.ID
|
|
|
|
|
newValue.BlockID = addingBlockID
|
|
|
|
|
newValue.IsDetached = isDetached
|
|
|
|
|
keyValues.Values = append(keyValues.Values, newValue)
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
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{
|
|
|
|
|
ID: ast.NewNodeID(), KeyID: keyValues.Key.ID, BlockID: addingBlockID, Type: av.KeyTypeDate, IsDetached: isDetached, CreatedAt: now, UpdatedAt: now + 1000,
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
|
2024-04-04 11:33:51 +08:00
|
|
|
|
for _, v := range attrView.Views {
|
|
|
|
|
switch v.LayoutType {
|
2023-12-14 11:23:34 +08:00
|
|
|
|
case av.LayoutTypeTable:
|
2024-01-18 21:01:36 +08:00
|
|
|
|
if "" != previousBlockID {
|
2023-12-14 11:23:34 +08:00
|
|
|
|
changed := false
|
2024-04-04 11:33:51 +08:00
|
|
|
|
for i, id := range v.Table.RowIDs {
|
2024-01-18 21:01:36 +08:00
|
|
|
|
if id == previousBlockID {
|
2024-04-04 11:33:51 +08:00
|
|
|
|
v.Table.RowIDs = append(v.Table.RowIDs[:i+1], append([]string{addingBlockID}, v.Table.RowIDs[i+1:]...)...)
|
2023-12-14 11:23:34 +08:00
|
|
|
|
changed = true
|
|
|
|
|
break
|
|
|
|
|
}
|
2023-07-12 19:10:05 +08:00
|
|
|
|
}
|
2023-12-14 11:23:34 +08:00
|
|
|
|
if !changed {
|
2024-04-04 11:33:51 +08:00
|
|
|
|
v.Table.RowIDs = append(v.Table.RowIDs, addingBlockID)
|
2023-12-14 11:23:34 +08:00
|
|
|
|
}
|
|
|
|
|
} else {
|
2024-04-04 11:33:51 +08:00
|
|
|
|
v.Table.RowIDs = append([]string{addingBlockID}, v.Table.RowIDs...)
|
2023-07-11 22:36:02 +08:00
|
|
|
|
}
|
2025-06-09 12:21:18 +08:00
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
if "" != previousBlockID {
|
|
|
|
|
changed := false
|
|
|
|
|
for i, id := range v.Gallery.CardIDs {
|
|
|
|
|
if id == previousBlockID {
|
|
|
|
|
v.Gallery.CardIDs = append(v.Gallery.CardIDs[:i+1], append([]string{addingBlockID}, v.Gallery.CardIDs[i+1:]...)...)
|
|
|
|
|
changed = true
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if !changed {
|
|
|
|
|
v.Gallery.CardIDs = append(v.Gallery.CardIDs, addingBlockID)
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
v.Gallery.CardIDs = append([]string{addingBlockID}, v.Gallery.CardIDs...)
|
|
|
|
|
}
|
2023-07-11 22:36:02 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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 {
|
2023-08-02 00:02:30 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, 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-06-13 12:00:49 +08:00
|
|
|
|
switch view.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
view.Table.RowIDs = gulu.Str.RemoveElem(view.Table.RowIDs, blockID)
|
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
view.Gallery.CardIDs = gulu.Str.RemoveElem(view.Gallery.CardIDs, 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
|
|
|
|
|
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-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 {
|
2024-06-10 16:25:57 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if av.KeyTypeBlock == key.Type || av.KeyTypeRelation == key.Type || av.KeyTypeRollup == key.Type {
|
|
|
|
|
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{
|
|
|
|
|
{
|
|
|
|
|
ID: copyKey.ID,
|
|
|
|
|
Wrap: column.Wrap,
|
|
|
|
|
Hidden: column.Hidden,
|
|
|
|
|
Pin: column.Pin,
|
|
|
|
|
Width: column.Width,
|
2024-11-09 14:09:40 +08:00
|
|
|
|
Desc: column.Desc,
|
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{
|
|
|
|
|
{
|
|
|
|
|
ID: copyKey.ID,
|
|
|
|
|
Desc: field.Desc,
|
|
|
|
|
},
|
|
|
|
|
}, 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 {
|
2023-07-13 00:29:22 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, 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 {
|
2023-07-13 00:29:22 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, 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
|
|
|
|
|
}
|
|
|
|
|
|
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.Wrap = operation.Data.(bool)
|
|
|
|
|
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) doSetAttrViewColumnHidden(operation *Operation) (ret *TxErr) {
|
|
|
|
|
err := setAttributeViewColHidden(operation)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-13 00:29:22 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, 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 {
|
2023-11-10 10:22:19 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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 {
|
2023-10-05 11:12:57 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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 {
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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 {
|
2023-07-13 00:29:22 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, 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
|
2023-07-12 21:39:55 +08:00
|
|
|
|
switch view.LayoutType {
|
2023-07-12 19:10:05 +08:00
|
|
|
|
case av.LayoutTypeTable:
|
2025-06-18 12:25:40 +08:00
|
|
|
|
for i, r := range view.Table.RowIDs {
|
|
|
|
|
if r == operation.ID {
|
|
|
|
|
itemID = r
|
|
|
|
|
idx = i
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if "" == itemID {
|
|
|
|
|
itemID = operation.ID
|
|
|
|
|
view.Table.RowIDs = append(view.Table.RowIDs, itemID)
|
|
|
|
|
idx = len(view.Table.RowIDs) - 1
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-21 19:26:35 +08:00
|
|
|
|
view.Table.RowIDs = append(view.Table.RowIDs[:idx], view.Table.RowIDs[idx+1:]...)
|
2023-07-12 19:10:05 +08:00
|
|
|
|
for i, r := range view.Table.RowIDs {
|
|
|
|
|
if r == operation.PreviousID {
|
|
|
|
|
previousIndex = i + 1
|
|
|
|
|
break
|
|
|
|
|
}
|
2023-07-11 22:47:19 +08:00
|
|
|
|
}
|
2025-06-18 12:25:40 +08:00
|
|
|
|
view.Table.RowIDs = util.InsertElem(view.Table.RowIDs, previousIndex, itemID)
|
2025-06-09 16:46:50 +08:00
|
|
|
|
case av.LayoutTypeGallery:
|
2025-06-18 12:25:40 +08:00
|
|
|
|
for i, c := range view.Gallery.CardIDs {
|
|
|
|
|
if c == operation.ID {
|
|
|
|
|
itemID = c
|
|
|
|
|
idx = i
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if "" == itemID {
|
|
|
|
|
itemID = operation.ID
|
|
|
|
|
view.Gallery.CardIDs = append(view.Gallery.CardIDs, itemID)
|
|
|
|
|
idx = len(view.Gallery.CardIDs) - 1
|
|
|
|
|
}
|
2025-06-09 16:46:50 +08:00
|
|
|
|
view.Gallery.CardIDs = append(view.Gallery.CardIDs[:idx], view.Gallery.CardIDs[idx+1:]...)
|
|
|
|
|
for i, c := range view.Gallery.CardIDs {
|
|
|
|
|
if c == operation.PreviousID {
|
|
|
|
|
previousIndex = i + 1
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-06-18 12:25:40 +08:00
|
|
|
|
view.Gallery.CardIDs = util.InsertElem(view.Gallery.CardIDs, 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 {
|
2023-07-13 00:29:22 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, 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 {
|
2024-05-16 22:24:15 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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 {
|
2023-07-13 00:29:22 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, 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-06-21 11:28:17 +08:00
|
|
|
|
// 如果当前视图是画廊视图则添加到最后
|
|
|
|
|
view.Table.Columns = append(view.Table.Columns, &av.ViewTableColumn{ID: key.ID})
|
|
|
|
|
} else {
|
|
|
|
|
view.Table.Columns = append([]*av.ViewTableColumn{{ID: key.ID}}, view.Table.Columns...)
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
added := false
|
|
|
|
|
for i, column := range view.Table.Columns {
|
|
|
|
|
if column.ID == previousKeyID {
|
|
|
|
|
view.Table.Columns = append(view.Table.Columns[:i+1], append([]*av.ViewTableColumn{{ID: key.ID}}, view.Table.Columns[i+1:]...)...)
|
|
|
|
|
added = true
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if !added {
|
|
|
|
|
view.Table.Columns = append(view.Table.Columns, &av.ViewTableColumn{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-11 16:25:37 +08:00
|
|
|
|
view.Gallery.CardFields = append(view.Gallery.CardFields, &av.ViewGalleryCardField{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 {
|
|
|
|
|
view.Gallery.CardFields = append(view.Gallery.CardFields[:i+1], append([]*av.ViewGalleryCardField{{ID: key.ID}}, view.Gallery.CardFields[i+1:]...)...)
|
|
|
|
|
added = true
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if !added {
|
|
|
|
|
view.Gallery.CardFields = append(view.Gallery.CardFields, &av.ViewGalleryCardField{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 {
|
2023-10-01 17:33:53 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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 {
|
2023-08-03 23:38:45 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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 {
|
2023-07-13 00:29:22 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, 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)
|
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)
|
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
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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 {
|
2023-07-13 00:29:22 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, 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 {
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-28 10:50:35 +08:00
|
|
|
|
func (tx *Transaction) doReplaceAttrViewBlock(operation *Operation) (ret *TxErr) {
|
2023-09-28 12:48:09 +08:00
|
|
|
|
err := replaceAttributeViewBlock(operation, tx)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-09-28 10:50:35 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-28 12:48:09 +08:00
|
|
|
|
func replaceAttributeViewBlock(operation *Operation, tx *Transaction) (err error) {
|
2023-09-28 10:50:35 +08:00
|
|
|
|
attrView, err := av.ParseAttributeView(operation.AvID)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-09-28 10:50:35 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-28 12:48:09 +08:00
|
|
|
|
var node *ast.Node
|
2024-06-23 20:59:48 +08:00
|
|
|
|
var tree *parse.Tree
|
2023-09-28 12:48:09 +08:00
|
|
|
|
if !operation.IsDetached {
|
2024-06-23 20:59:48 +08:00
|
|
|
|
node, tree, _ = getNodeByBlockID(tx, operation.NextID)
|
2023-09-28 12:48:09 +08:00
|
|
|
|
}
|
|
|
|
|
|
2024-06-23 20:59:48 +08:00
|
|
|
|
now := util.CurrentTimeMillis()
|
|
|
|
|
// 检查是否已经存在绑定块,如果存在的话则重新绑定
|
2024-04-10 22:58:59 +08:00
|
|
|
|
for _, keyValues := range attrView.KeyValues {
|
|
|
|
|
for _, value := range keyValues.Values {
|
2024-06-24 07:25:02 +08:00
|
|
|
|
if av.KeyTypeBlock == value.Type && nil != value.Block && value.BlockID == operation.NextID {
|
2024-06-23 20:59:48 +08:00
|
|
|
|
if !operation.IsDetached {
|
|
|
|
|
bindBlockAv0(tx, operation.AvID, node, tree)
|
|
|
|
|
value.IsDetached = false
|
2025-05-08 12:28:09 +08:00
|
|
|
|
icon, content := getNodeAvBlockText(node)
|
|
|
|
|
content = util.UnescapeHTML(content)
|
|
|
|
|
value.Block.Icon, value.Block.Content = icon, content
|
2024-06-23 20:59:48 +08:00
|
|
|
|
value.UpdatedAt = now
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
}
|
2024-04-10 22:58:59 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-09-02 17:53:34 +08:00
|
|
|
|
var changedAvIDs []string
|
2023-09-28 10:50:35 +08:00
|
|
|
|
for _, keyValues := range attrView.KeyValues {
|
|
|
|
|
for _, value := range keyValues.Values {
|
2024-09-02 17:53:34 +08:00
|
|
|
|
if av.KeyTypeRelation == value.Type {
|
|
|
|
|
if nil != value.Relation {
|
|
|
|
|
for i, relBlockID := range value.Relation.BlockIDs {
|
|
|
|
|
if relBlockID == operation.PreviousID {
|
|
|
|
|
value.Relation.BlockIDs[i] = operation.NextID
|
|
|
|
|
changedAvIDs = append(changedAvIDs, attrView.ID)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-15 18:00:04 +08:00
|
|
|
|
if value.BlockID != operation.PreviousID {
|
|
|
|
|
continue
|
|
|
|
|
}
|
2023-12-18 13:11:56 +08:00
|
|
|
|
|
2024-04-15 18:00:04 +08:00
|
|
|
|
if av.KeyTypeBlock == value.Type && value.BlockID != operation.NextID {
|
|
|
|
|
// 换绑
|
|
|
|
|
unbindBlockAv(tx, operation.AvID, value.BlockID)
|
|
|
|
|
}
|
2023-09-28 11:17:49 +08:00
|
|
|
|
|
2024-04-15 18:00:04 +08:00
|
|
|
|
value.BlockID = operation.NextID
|
|
|
|
|
if av.KeyTypeBlock == value.Type && nil != value.Block {
|
|
|
|
|
value.Block.ID = operation.NextID
|
|
|
|
|
value.IsDetached = operation.IsDetached
|
2023-09-28 11:17:49 +08:00
|
|
|
|
if !operation.IsDetached {
|
2025-05-08 12:28:09 +08:00
|
|
|
|
icon, content := getNodeAvBlockText(node)
|
|
|
|
|
content = util.UnescapeHTML(content)
|
|
|
|
|
value.Block.Icon, value.Block.Content = icon, content
|
2023-09-28 11:17:49 +08:00
|
|
|
|
}
|
2023-09-28 10:50:35 +08:00
|
|
|
|
}
|
2024-04-15 18:00:04 +08:00
|
|
|
|
|
|
|
|
|
if av.KeyTypeBlock == value.Type && !operation.IsDetached {
|
|
|
|
|
bindBlockAv(tx, operation.AvID, operation.NextID)
|
2024-04-23 18:22:51 +08:00
|
|
|
|
|
2024-09-02 17:53:34 +08:00
|
|
|
|
avIDs := replaceRelationAvValues(operation.AvID, operation.PreviousID, operation.NextID)
|
|
|
|
|
changedAvIDs = append(changedAvIDs, avIDs...)
|
2024-04-15 18:00:04 +08:00
|
|
|
|
}
|
2023-09-28 10:50:35 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-14 00:14:31 +08:00
|
|
|
|
replacedRowID := false
|
2023-09-28 10:50:35 +08:00
|
|
|
|
for _, v := range attrView.Views {
|
|
|
|
|
switch v.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
|
|
|
|
for i, rowID := range v.Table.RowIDs {
|
|
|
|
|
if rowID == operation.PreviousID {
|
|
|
|
|
v.Table.RowIDs[i] = operation.NextID
|
2023-12-14 00:14:31 +08:00
|
|
|
|
replacedRowID = true
|
|
|
|
|
break
|
2023-09-28 10:50:35 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2023-12-14 00:14:31 +08:00
|
|
|
|
|
|
|
|
|
if !replacedRowID {
|
|
|
|
|
v.Table.RowIDs = append(v.Table.RowIDs, operation.NextID)
|
|
|
|
|
}
|
2025-06-09 16:46:50 +08:00
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
for i, cardID := range v.Gallery.CardIDs {
|
|
|
|
|
if cardID == operation.PreviousID {
|
|
|
|
|
v.Gallery.CardIDs[i] = operation.NextID
|
|
|
|
|
replacedRowID = true
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-09-28 10:50:35 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
2024-09-02 17:53:34 +08:00
|
|
|
|
|
|
|
|
|
changedAvIDs = gulu.Str.RemoveDuplicatedElem(changedAvIDs)
|
|
|
|
|
for _, avID := range changedAvIDs {
|
2024-09-05 12:10:42 +08:00
|
|
|
|
ReloadAttrView(avID)
|
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) {
|
2023-07-12 19:10:05 +08:00
|
|
|
|
err := updateAttributeViewCell(operation, tx)
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err != nil {
|
2023-07-13 00:29:22 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
2023-07-03 15:29:54 +08:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-12 19:10:05 +08:00
|
|
|
|
func updateAttributeViewCell(operation *Operation, tx *Transaction) (err error) {
|
2024-11-02 16:07:56 +08:00
|
|
|
|
_, err = UpdateAttributeViewCell(tx, operation.AvID, operation.KeyID, operation.RowID, operation.Data)
|
2023-07-31 23:17:31 +08:00
|
|
|
|
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
|
|
|
|
|
}
|
|
|
|
|
|
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 {
|
|
|
|
|
if rowID == v.Block.ID {
|
|
|
|
|
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 {
|
2024-11-02 16:07:56 +08:00
|
|
|
|
if rowID == 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 {
|
2024-11-02 16:07:56 +08:00
|
|
|
|
val = &av.Value{ID: ast.NewNodeID(), KeyID: keyID, BlockID: rowID, 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 {
|
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 {
|
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 {
|
|
|
|
|
// 关联列得 content 是自动渲染的,所以不需要保存
|
|
|
|
|
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 来判断
|
|
|
|
|
|
2024-05-12 17:20:04 +08:00
|
|
|
|
if oldIsDetached {
|
|
|
|
|
// 之前是游离行
|
|
|
|
|
|
2023-12-18 10:55:10 +08:00
|
|
|
|
if !val.IsDetached { // 现在绑定了块
|
|
|
|
|
// 将游离行绑定到新建的块上
|
|
|
|
|
bindBlockAv(tx, avID, rowID)
|
2024-08-31 22:56:11 +08:00
|
|
|
|
if nil != val.Block {
|
|
|
|
|
val.BlockID = val.Block.ID
|
|
|
|
|
}
|
2023-12-18 10:55:10 +08:00
|
|
|
|
}
|
2024-05-12 17:20:04 +08:00
|
|
|
|
} else {
|
|
|
|
|
// 之前绑定了块
|
|
|
|
|
|
2023-12-18 13:02:54 +08:00
|
|
|
|
if isUpdatingBlockKey { // 正在更新主键
|
|
|
|
|
if val.IsDetached { // 现在是游离行
|
|
|
|
|
// 将绑定的块从属性视图中移除
|
|
|
|
|
unbindBlockAv(tx, avID, rowID)
|
2024-05-12 17:20:04 +08:00
|
|
|
|
} else {
|
|
|
|
|
// 现在绑定了块
|
|
|
|
|
|
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
|
|
|
|
updateStaticText := true
|
|
|
|
|
_, blockText := getNodeAvBlockText(node)
|
2024-12-23 18:08:46 +08:00
|
|
|
|
if "" == content {
|
2024-12-24 19:47:45 +08:00
|
|
|
|
val.Block.Content = blockText
|
2025-01-12 17:43:46 +08:00
|
|
|
|
val.Block.Content = util.UnescapeHTML(val.Block.Content)
|
2024-12-24 19:47:45 +08:00
|
|
|
|
} else {
|
|
|
|
|
if blockText == content {
|
|
|
|
|
updateStaticText = false
|
|
|
|
|
} else {
|
2025-01-05 00:53:55 +08:00
|
|
|
|
val.Block.Content = content
|
2024-12-24 19:47:45 +08:00
|
|
|
|
}
|
2024-12-21 12:00:02 +08:00
|
|
|
|
}
|
2024-12-23 21:04:43 +08:00
|
|
|
|
|
2024-12-24 19:47:45 +08:00
|
|
|
|
if updateStaticText {
|
|
|
|
|
// 设置静态锚文本 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 {
|
|
|
|
|
// 双向关联需要同时更新目标字段的值
|
2023-12-24 23:18:35 +08:00
|
|
|
|
|
2024-05-12 11:16:31 +08:00
|
|
|
|
var destAv *av.AttributeView
|
|
|
|
|
if avID == key.Relation.AvID {
|
|
|
|
|
destAv = attrView
|
|
|
|
|
} else {
|
|
|
|
|
destAv, _ = av.ParseAttributeView(key.Relation.AvID)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if nil != destAv {
|
2024-05-06 13:44:37 +08:00
|
|
|
|
// relationChangeMode
|
|
|
|
|
// 0:关联列值不变(仅排序),不影响目标值
|
|
|
|
|
// 1:关联列值增加,增加目标值
|
|
|
|
|
// 2:关联列值减少,减少目标值
|
2023-12-25 00:30:49 +08:00
|
|
|
|
|
2024-05-06 13:44:37 +08:00
|
|
|
|
if 1 == relationChangeMode {
|
|
|
|
|
addBlockIDs := val.Relation.BlockIDs
|
|
|
|
|
for _, bID := range oldRelationBlockIDs {
|
|
|
|
|
addBlockIDs = gulu.Str.RemoveElem(addBlockIDs, bID)
|
|
|
|
|
}
|
2023-12-25 00:30:49 +08:00
|
|
|
|
|
2024-05-06 13:44:37 +08:00
|
|
|
|
for _, blockID := range addBlockIDs {
|
|
|
|
|
for _, keyValues := range destAv.KeyValues {
|
|
|
|
|
if keyValues.Key.ID != key.Relation.BackKeyID {
|
|
|
|
|
continue
|
2023-12-31 10:58:40 +08:00
|
|
|
|
}
|
2024-05-06 13:44:37 +08:00
|
|
|
|
|
|
|
|
|
destVal := keyValues.GetValue(blockID)
|
|
|
|
|
if nil == destVal {
|
2024-05-06 13:53:38 +08:00
|
|
|
|
destVal = &av.Value{ID: ast.NewNodeID(), KeyID: keyValues.Key.ID, BlockID: blockID, Type: keyValues.Key.Type, Relation: &av.ValueRelation{}, CreatedAt: now, UpdatedAt: now + 1000}
|
2024-05-06 13:44:37 +08:00
|
|
|
|
keyValues.Values = append(keyValues.Values, destVal)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
destVal.Relation.BlockIDs = append(destVal.Relation.BlockIDs, rowID)
|
|
|
|
|
destVal.Relation.BlockIDs = gulu.Str.RemoveDuplicatedElem(destVal.Relation.BlockIDs)
|
|
|
|
|
break
|
2023-12-31 10:58:40 +08:00
|
|
|
|
}
|
2024-05-06 13:44:37 +08:00
|
|
|
|
}
|
|
|
|
|
} else if 2 == relationChangeMode {
|
|
|
|
|
removeBlockIDs := oldRelationBlockIDs
|
|
|
|
|
for _, bID := range val.Relation.BlockIDs {
|
|
|
|
|
removeBlockIDs = gulu.Str.RemoveElem(removeBlockIDs, bID)
|
|
|
|
|
}
|
2023-12-24 23:18:35 +08:00
|
|
|
|
|
2024-05-06 13:44:37 +08:00
|
|
|
|
for _, blockID := range removeBlockIDs {
|
|
|
|
|
for _, keyValues := range destAv.KeyValues {
|
|
|
|
|
if keyValues.Key.ID != key.Relation.BackKeyID {
|
|
|
|
|
continue
|
|
|
|
|
}
|
2023-12-24 23:02:59 +08:00
|
|
|
|
|
2024-05-06 13:44:37 +08:00
|
|
|
|
for _, value := range keyValues.Values {
|
|
|
|
|
if value.BlockID == blockID {
|
|
|
|
|
value.Relation.BlockIDs = gulu.Str.RemoveElem(value.Relation.BlockIDs, rowID)
|
|
|
|
|
value.SetUpdatedAt(now)
|
|
|
|
|
break
|
2023-12-25 00:30:49 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2023-12-24 23:02:59 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2023-12-31 10:58:40 +08:00
|
|
|
|
}
|
2024-05-06 13:44:37 +08:00
|
|
|
|
|
2024-05-12 11:16:31 +08:00
|
|
|
|
if destAv != attrView {
|
|
|
|
|
av.SaveAttributeView(destAv)
|
|
|
|
|
}
|
2023-12-24 23:02:59 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-31 10:58:40 +08:00
|
|
|
|
relatedAvIDs := av.GetSrcAvIDs(avID)
|
|
|
|
|
for _, relatedAvID := range relatedAvIDs {
|
2024-09-05 12:10:42 +08:00
|
|
|
|
ReloadAttrView(relatedAvID)
|
2023-12-31 10:58:40 +08:00
|
|
|
|
}
|
|
|
|
|
|
2024-09-04 04:40:50 +03:00
|
|
|
|
if err = av.SaveAttributeView(attrView); err != nil {
|
2023-07-03 15:29:54 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-18 10:55:10 +08:00
|
|
|
|
func unbindBlockAv(tx *Transaction, avID, blockID string) {
|
|
|
|
|
node, tree, err := getNodeByBlockID(tx, blockID)
|
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 {
|
2023-12-18 10:55:10 +08:00
|
|
|
|
logging.LogWarnf("set node [%s] attrs failed: %s", blockID, err)
|
|
|
|
|
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 {
|
2023-07-13 00:29:22 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, 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
|
|
|
|
|
err = av.SaveAttributeView(attrView)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
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 {
|
2023-07-13 00:29:22 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, 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
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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 {
|
2023-07-13 00:29:22 +08:00
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, 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 {
|
|
|
|
|
switch view.LayoutType {
|
|
|
|
|
case av.LayoutTypeTable:
|
2025-06-09 16:46:50 +08:00
|
|
|
|
for _, filter := range view.Table.Filters {
|
|
|
|
|
if filter.Column != key.ID {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
case av.LayoutTypeGallery:
|
|
|
|
|
for _, filter := range view.Gallery.Filters {
|
2024-04-06 10:11:23 +08:00
|
|
|
|
if filter.Column != key.ID {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
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 {
|
|
|
|
|
return &TxErr{code: TxErrWriteAttributeView, id: operation.AvID, msg: err.Error()}
|
|
|
|
|
}
|
|
|
|
|
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) {
|
|
|
|
|
node, _, _ := getNodeByBlockID(nil, blockID)
|
|
|
|
|
var viewID string
|
|
|
|
|
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 {
|
|
|
|
|
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)
|
|
|
|
|
}
|
|
|
|
|
}
|