|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
#
|
|
|
|
|
|
|
|
from rest_framework import serializers
|
|
|
|
|
|
|
|
|
|
|
|
class TreeNode:
|
|
|
|
id = ""
|
|
|
|
name = ""
|
|
|
|
comment = ""
|
|
|
|
title = ""
|
|
|
|
isParent = False
|
|
|
|
pId = ""
|
|
|
|
open = False
|
|
|
|
iconSkin = ""
|
|
|
|
parentInfo = ''
|
|
|
|
meta = {}
|
|
|
|
checked = False
|
|
|
|
|
|
|
|
_tree = None
|
|
|
|
|
|
|
|
def __init__(self, **kwargs):
|
|
|
|
for k, v in kwargs.items():
|
|
|
|
setattr(self, k, v)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def root(cls):
|
|
|
|
return cls(id="#", name='Root', title='Root', isParent=True, open=True)
|
|
|
|
|
|
|
|
def get_parent(self):
|
|
|
|
return self._tree.get_node(self.pId)
|
|
|
|
|
|
|
|
def get_parents(self):
|
|
|
|
parent = self.get_parent()
|
|
|
|
if parent == self._tree.root:
|
|
|
|
return []
|
|
|
|
parents = [parent]
|
|
|
|
parents.extend(parent.get_parents())
|
|
|
|
return parents
|
|
|
|
|
|
|
|
def add_child(self, child):
|
|
|
|
self._tree.add_node(child, self)
|
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
return '<{}: {}>'.format(self.id, self.name)
|
|
|
|
|
|
|
|
__repr__ = __str__
|
|
|
|
|
|
|
|
def __gt__(self, other):
|
|
|
|
if self.isParent and not other.isParent:
|
|
|
|
result = False
|
|
|
|
elif not self.isParent and other.isParent:
|
|
|
|
result = True
|
|
|
|
elif self.pId != other.pId:
|
|
|
|
result = self.pId > other.pId
|
|
|
|
elif str(self.id).startswith('-') and not str(other.id).startswith('-'):
|
|
|
|
result = False
|
|
|
|
else:
|
|
|
|
result = self.name > other.name
|
|
|
|
return result
|
|
|
|
|
|
|
|
def __le__(self, other):
|
|
|
|
return not self.__gt__(other)
|
|
|
|
|
|
|
|
def __eq__(self, other):
|
|
|
|
return self.id == other.id
|
|
|
|
|
|
|
|
|
|
|
|
class Tree:
|
|
|
|
def __init__(self):
|
|
|
|
self.nodes = {}
|
|
|
|
self.root = TreeNode.root()
|
|
|
|
self.root._tree = self
|
|
|
|
|
|
|
|
def add_node(self, node, parent=None):
|
|
|
|
node._tree = self
|
|
|
|
|
|
|
|
if not parent:
|
|
|
|
parent = self.root
|
|
|
|
if parent.id not in self.nodes and parent != self.root:
|
|
|
|
raise ValueError("Parent not in tree")
|
|
|
|
elif node in parent.get_parents():
|
|
|
|
raise ValueError("Parent must not be node parent")
|
|
|
|
node.pId = parent.id
|
|
|
|
parent.isParent = True
|
|
|
|
self.nodes[node.key] = node
|
|
|
|
|
|
|
|
def get_nodes(self):
|
|
|
|
return sorted(self.nodes.values())
|
|
|
|
|
|
|
|
def get_node(self, tid):
|
|
|
|
return self.nodes.get(tid) or TreeNode.root()
|
|
|
|
|
|
|
|
|
|
|
|
class TreeNodeSerializer(serializers.Serializer):
|
|
|
|
id = serializers.CharField(max_length=128)
|
|
|
|
name = serializers.CharField(max_length=128)
|
|
|
|
title = serializers.CharField(max_length=128)
|
|
|
|
pId = serializers.CharField(max_length=128)
|
|
|
|
parentInfo = serializers.CharField(max_length=4096, allow_blank=True)
|
|
|
|
isParent = serializers.BooleanField(default=False)
|
|
|
|
open = serializers.BooleanField(default=False)
|
|
|
|
iconSkin = serializers.CharField(max_length=128, allow_blank=True)
|
|
|
|
nocheck = serializers.BooleanField(default=False)
|
|
|
|
checked = serializers.BooleanField(default=False)
|
|
|
|
halfCheck = serializers.BooleanField(default=False)
|
|
|
|
chkDisabled = serializers.BooleanField(default=False)
|
|
|
|
meta = serializers.JSONField()
|