From 426e7f789b313326ed8458927dca11a84dd3a414 Mon Sep 17 00:00:00 2001
From: donghao <donghao@supervision.ltd>
Date: Tue, 16 Jan 2024 13:54:43 +0800
Subject: [PATCH] =?UTF-8?q?feat:=20=E5=8A=A8=E6=95=88=E5=88=9D=E5=A7=8B?=
 =?UTF-8?q?=E5=8C=96?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

---
 src/assets/animate/device/01.json             | 1278 ++++-------------
 src/components/Animate/index.ts               |    3 +
 src/components/Animate/src/AnimateDevice.vue  |   38 +
 .../CustomTree/src/collapseTreeItem           |    0
 src/components/CustomTree/src/demo.md         |  472 ++++++
 src/components/CustomTree/src/demoItem.md     |  347 +++++
 src/views/demo/animateDevice.vue              |    4 +-
 src/views/device/components/DeviceCard.vue    |    7 +-
 8 files changed, 1125 insertions(+), 1024 deletions(-)
 create mode 100644 src/components/Animate/index.ts
 create mode 100644 src/components/Animate/src/AnimateDevice.vue
 create mode 100644 src/components/CustomTree/src/collapseTreeItem
 create mode 100644 src/components/CustomTree/src/demo.md
 create mode 100644 src/components/CustomTree/src/demoItem.md

diff --git a/src/assets/animate/device/01.json b/src/assets/animate/device/01.json
index ea93985..7bdeee1 100644
--- a/src/assets/animate/device/01.json
+++ b/src/assets/animate/device/01.json
@@ -1,1062 +1,300 @@
 {
-  "nm": "scene1",
+  "v": "4.8.0",
+  "meta": { "g": "LottieFiles AE 3.5.3", "a": "", "k": "", "d": "", "tc": "" },
+  "fr": 25,
+  "ip": 0,
+  "op": 20,
+  "w": 400,
+  "h": 400,
+  "nm": "合成 1",
   "ddd": 0,
-  "h": 500,
-  "w": 500,
-  "meta": { "g": "@lottiefiles/creator 0.48.0" },
+  "assets": [
+    {
+      "id": "image_0",
+      "w": 400,
+      "h": 272,
+      "u": "",
+      "p": "",
+      "e": 1
+    }
+  ],
   "layers": [
     {
-      "ty": 0,
-      "nm": " scene1",
-      "sr": 1,
-      "st": 0,
-      "op": 210,
-      "ip": 0,
-      "hd": false,
       "ddd": 0,
-      "bm": 0,
-      "hasMask": false,
-      "ao": 0,
-      "ks": {
-        "a": { "a": 0, "k": [226.99806213378906, 251] },
-        "s": { "a": 0, "k": [100, 100] },
-        "sk": { "a": 0, "k": 0 },
-        "p": { "a": 0, "k": [226.99806213378906, 251] },
-        "r": { "a": 0, "k": 0 },
-        "sa": { "a": 0, "k": 0 },
-        "o": { "a": 0, "k": 100 }
-      },
-      "ef": [],
-      "w": 500,
-      "h": 500,
-      "refId": "Scene 1",
-      "ind": 1
-    },
-    {
-      "ty": 0,
-      "nm": " scene1",
+      "ind": 1,
+      "ty": 4,
+      "nm": "形状图层 3",
       "sr": 1,
-      "st": 0,
-      "op": 210,
-      "ip": 0,
-      "hd": false,
-      "ddd": 0,
-      "bm": 0,
-      "hasMask": false,
-      "ao": 0,
       "ks": {
-        "a": { "a": 0, "k": [226.99806213378906, 251] },
-        "s": { "a": 0, "k": [100, 100] },
-        "sk": { "a": 0, "k": 0 },
-        "p": { "a": 0, "k": [226.99806213378906, 251] },
-        "r": { "a": 0, "k": 0 },
-        "sa": { "a": 0, "k": 0 },
-        "o": { "a": 0, "k": 100 }
-      },
-      "ef": [],
-      "w": 500,
-      "h": 500,
-      "refId": "Scene 1",
-      "ind": 2
-    }
-  ],
-  "v": "5.7.0",
-  "fr": 30,
-  "op": 210,
-  "ip": 0,
-  "assets": [
-    {
-      "nm": "scene1",
-      "id": "Scene 1",
-      "layers": [
-        {
-          "ty": 4,
-          "nm": "Shape Layer 1",
-          "sr": 1,
-          "st": 0,
-          "op": 207.89999999999998,
-          "ip": 146.99999999999997,
-          "hd": false,
-          "ddd": 0,
-          "bm": 0,
-          "hasMask": false,
-          "ao": 0,
-          "ks": {
-            "a": { "a": 0, "k": [32.00000000000001, 33] },
-            "s": { "a": 0, "k": [500, 500] },
-            "sk": { "a": 0, "k": 0 },
-            "p": {
-              "a": 1,
-              "k": [
-                {
-                  "o": { "x": 0.167, "y": 0.167 },
-                  "i": { "x": 0.833, "y": 0.833 },
-                  "s": [347, 251],
-                  "t": 148
-                },
-                {
-                  "o": { "x": 0.167, "y": 0.167 },
-                  "i": { "x": 0.833, "y": 0.833 },
-                  "s": [433, 251],
-                  "t": 159
-                },
-                {
-                  "o": { "x": 0.167, "y": 0.167 },
-                  "i": { "x": 0.833, "y": 0.833 },
-                  "s": [433, 251],
-                  "t": 188
-                },
-                {
-                  "o": { "x": 0.167, "y": 0.167 },
-                  "i": { "x": 0.833, "y": 0.833 },
-                  "s": [433, 233],
-                  "t": 191
-                },
-                { "s": [433, 251], "t": 194 }
-              ]
+        "o": {
+          "a": 1,
+          "k": [
+            {
+              "i": { "x": [0.659], "y": [0.843] },
+              "o": { "x": [0.391], "y": [0] },
+              "t": 0,
+              "s": [100]
             },
-            "r": { "a": 0, "k": 0 },
-            "sa": { "a": 0, "k": 0 },
-            "o": { "a": 0, "k": 100 }
-          },
-          "ef": [],
-          "shapes": [
             {
-              "ty": "gr",
-              "bm": 0,
-              "hd": false,
-              "nm": "Group 5",
-              "it": [
-                {
-                  "ty": "sh",
-                  "bm": 0,
-                  "hd": false,
-                  "nm": "Path 5",
-                  "d": 1,
-                  "ks": {
-                    "a": 0,
-                    "k": {
-                      "c": true,
-                      "i": [
-                        [0, 0],
-                        [0, -11.3888407],
-                        [-4.3333333, 0],
-                        [0, 11.388840700000003],
-                        [9.388840700000003, 0],
-                        [3.1184307999999987, -3.696753659999999],
-                        [5.215011799999999, 0]
-                      ],
-                      "o": [
-                        [-9.38884075, 0],
-                        [0, 11.388840700000003],
-                        [4.3333333, 0],
-                        [0, -11.3888407],
-                        [-5.215011799999999, 0],
-                        [-3.1184307999999987, -3.696753659999999],
-                        [0, 0]
-                      ],
-                      "v": [
-                        [19, 6],
-                        [2, 23],
-                        [32, 60],
-                        [62, 23],
-                        [45, 6],
-                        [32, 12.0449665],
-                        [19, 6]
-                      ]
-                    }
-                  }
-                },
-                {
-                  "ty": "fl",
-                  "bm": 0,
-                  "hd": false,
-                  "nm": "Fill",
-                  "c": {
-                    "a": 1,
-                    "k": [
-                      {
-                        "o": { "x": 0.167, "y": 0.167 },
-                        "i": { "x": 0.833, "y": 0.833 },
-                        "s": [0.1137, 0.6392, 0.2235],
-                        "t": 184
-                      },
-                      { "s": [0.6392, 0.1137, 0.6157], "t": 191 }
-                    ]
-                  },
-                  "r": 2,
-                  "o": { "a": 0, "k": 100 }
-                },
-                {
-                  "ty": "tr",
-                  "a": { "a": 0, "k": [32, 33] },
-                  "s": { "a": 0, "k": [25, 25] },
-                  "sk": { "a": 0, "k": 0 },
-                  "p": { "a": 0, "k": [32, 33] },
-                  "r": { "a": 0, "k": 0 },
-                  "sa": { "a": 0, "k": 0 },
-                  "o": { "a": 0, "k": 100 }
-                }
-              ]
-            }
+              "i": { "x": [0.6], "y": [0.927] },
+              "o": { "x": [0.283], "y": [-0.13] },
+              "t": 10,
+              "s": [0]
+            },
+            { "t": 20, "s": [100] }
           ],
-          "ind": 1
+          "ix": 11
         },
+        "r": { "a": 0, "k": 0, "ix": 10 },
+        "p": { "a": 0, "k": [72.25, 240.125, 0], "ix": 2 },
+        "a": { "a": 0, "k": [-178.25, 16.625, 0], "ix": 1 },
+        "s": { "a": 0, "k": [100, 100, 100], "ix": 6 }
+      },
+      "ao": 0,
+      "shapes": [
         {
-          "ty": 4,
-          "nm": "Shape Layer 1",
-          "sr": 1,
-          "st": 0,
-          "op": 207.89999999999998,
-          "ip": 132.3,
-          "hd": false,
-          "ddd": 0,
-          "bm": 0,
-          "hasMask": false,
-          "ao": 0,
-          "ks": {
-            "a": { "a": 0, "k": [32.00000000000001, 33] },
-            "s": { "a": 0, "k": [500, 500] },
-            "sk": { "a": 0, "k": 0 },
-            "p": {
-              "a": 1,
-              "k": [
-                {
-                  "o": { "x": 0.167, "y": 0.167 },
-                  "i": { "x": 0.833, "y": 0.833 },
-                  "s": [261, 251],
-                  "t": 133.3
-                },
-                {
-                  "o": { "x": 0.167, "y": 0.167 },
-                  "i": { "x": 0.833, "y": 0.833 },
-                  "s": [347, 251],
-                  "t": 142.3
-                },
-                {
-                  "o": { "x": 0.167, "y": 0.167 },
-                  "i": { "x": 0.833, "y": 0.833 },
-                  "s": [347, 251],
-                  "t": 184
-                },
-                {
-                  "o": { "x": 0.167, "y": 0.167 },
-                  "i": { "x": 0.833, "y": 0.833 },
-                  "s": [347, 237],
-                  "t": 186
-                },
-                { "s": [347, 251], "t": 189 }
-              ]
+          "ty": "gr",
+          "it": [
+            {
+              "d": 1,
+              "ty": "el",
+              "s": { "a": 0, "k": [16, 16], "ix": 2 },
+              "p": { "a": 0, "k": [0, 0], "ix": 3 },
+              "nm": "椭圆路径 1",
+              "mn": "ADBE Vector Shape - Ellipse",
+              "hd": false
             },
-            "r": { "a": 0, "k": 0 },
-            "sa": { "a": 0, "k": 0 },
-            "o": { "a": 0, "k": 100 }
-          },
-          "ef": [],
-          "shapes": [
             {
-              "ty": "gr",
+              "ty": "fl",
+              "c": {
+                "a": 0,
+                "k": [0.909803921569, 0.050980392157, 0.050980392157, 1],
+                "ix": 4
+              },
+              "o": { "a": 0, "k": 100, "ix": 5 },
+              "r": 1,
               "bm": 0,
-              "hd": false,
-              "nm": "Group 5",
-              "it": [
-                {
-                  "ty": "sh",
-                  "bm": 0,
-                  "hd": false,
-                  "nm": "Path 5",
-                  "d": 1,
-                  "ks": {
-                    "a": 0,
-                    "k": {
-                      "c": true,
-                      "i": [
-                        [0, 0],
-                        [0, -11.3888407],
-                        [-4.3333333, 0],
-                        [0, 11.388840700000003],
-                        [9.388840700000003, 0],
-                        [3.1184307999999987, -3.696753659999999],
-                        [5.215011799999999, 0]
-                      ],
-                      "o": [
-                        [-9.38884075, 0],
-                        [0, 11.388840700000003],
-                        [4.3333333, 0],
-                        [0, -11.3888407],
-                        [-5.215011799999999, 0],
-                        [-3.1184307999999987, -3.696753659999999],
-                        [0, 0]
-                      ],
-                      "v": [
-                        [19, 6],
-                        [2, 23],
-                        [32, 60],
-                        [62, 23],
-                        [45, 6],
-                        [32, 12.0449665],
-                        [19, 6]
-                      ]
-                    }
-                  }
-                },
-                {
-                  "ty": "fl",
-                  "bm": 0,
-                  "hd": false,
-                  "nm": "Fill",
-                  "c": {
-                    "a": 1,
-                    "k": [
-                      {
-                        "o": { "x": 0.167, "y": 0.167 },
-                        "i": { "x": 0.833, "y": 0.833 },
-                        "s": [0.1137, 0.6392, 0.2235],
-                        "t": 178
-                      },
-                      { "s": [0.6392, 0.1137, 0.6157], "t": 185 }
-                    ]
-                  },
-                  "r": 2,
-                  "o": { "a": 0, "k": 100 }
-                },
-                {
-                  "ty": "tr",
-                  "a": { "a": 0, "k": [32, 33] },
-                  "s": { "a": 0, "k": [25, 25] },
-                  "sk": { "a": 0, "k": 0 },
-                  "p": { "a": 0, "k": [32, 33] },
-                  "r": { "a": 0, "k": 0 },
-                  "sa": { "a": 0, "k": 0 },
-                  "o": { "a": 0, "k": 100 }
-                }
-              ]
-            }
-          ],
-          "ind": 2
-        },
-        {
-          "ty": 4,
-          "nm": "Shape Layer 1",
-          "sr": 1,
-          "st": 0,
-          "op": 206.99999999999997,
-          "ip": 117.00000000000001,
-          "hd": false,
-          "ddd": 0,
-          "bm": 0,
-          "hasMask": false,
-          "ao": 0,
-          "ks": {
-            "a": { "a": 0, "k": [32, 33] },
-            "s": { "a": 0, "k": [500, 500] },
-            "sk": { "a": 0, "k": 0 },
-            "p": {
-              "a": 1,
-              "k": [
-                {
-                  "o": { "x": 0.167, "y": 0.167 },
-                  "i": { "x": 0.833, "y": 0.833 },
-                  "s": [174, 251],
-                  "t": 118
-                },
-                {
-                  "o": { "x": 0.167, "y": 0.167 },
-                  "i": { "x": 0.833, "y": 0.833 },
-                  "s": [261, 251],
-                  "t": 128
-                },
-                {
-                  "o": { "x": 0.167, "y": 0.167 },
-                  "i": { "x": 0.833, "y": 0.833 },
-                  "s": [261, 251],
-                  "t": 178
-                },
-                {
-                  "o": { "x": 0.167, "y": 0.167 },
-                  "i": { "x": 0.833, "y": 0.833 },
-                  "s": [261, 233],
-                  "t": 181
-                },
-                { "s": [261, 251], "t": 185 }
-              ]
+              "nm": "填充 1",
+              "mn": "ADBE Vector Graphic - Fill",
+              "hd": false
             },
-            "r": { "a": 0, "k": 0 },
-            "sa": { "a": 0, "k": 0 },
-            "o": { "a": 0, "k": 100 }
-          },
-          "ef": [],
-          "shapes": [
             {
-              "ty": "gr",
-              "bm": 0,
-              "hd": false,
-              "nm": "Group 5",
-              "it": [
-                {
-                  "ty": "sh",
-                  "bm": 0,
-                  "hd": false,
-                  "nm": "Path 5",
-                  "d": 1,
-                  "ks": {
-                    "a": 0,
-                    "k": {
-                      "c": true,
-                      "i": [
-                        [0, 0],
-                        [0, -11.3888407],
-                        [-4.3333333, 0],
-                        [0, 11.388840700000003],
-                        [9.388840700000003, 0],
-                        [3.1184307999999987, -3.696753659999999],
-                        [5.215011799999999, 0]
-                      ],
-                      "o": [
-                        [-9.38884075, 0],
-                        [0, 11.388840700000003],
-                        [4.3333333, 0],
-                        [0, -11.3888407],
-                        [-5.215011799999999, 0],
-                        [-3.1184307999999987, -3.696753659999999],
-                        [0, 0]
-                      ],
-                      "v": [
-                        [19, 6],
-                        [2, 23],
-                        [32, 60],
-                        [62, 23],
-                        [45, 6],
-                        [32, 12.0449665],
-                        [19, 6]
-                      ]
-                    }
-                  }
-                },
-                {
-                  "ty": "fl",
-                  "bm": 0,
-                  "hd": false,
-                  "nm": "Fill",
-                  "c": {
-                    "a": 1,
-                    "k": [
-                      {
-                        "o": { "x": 0.167, "y": 0.167 },
-                        "i": { "x": 0.833, "y": 0.833 },
-                        "s": [0.1137, 0.6392, 0.2235],
-                        "t": 173
-                      },
-                      { "s": [0.6392, 0.1137, 0.6157], "t": 179 }
-                    ]
-                  },
-                  "r": 2,
-                  "o": { "a": 0, "k": 100 }
-                },
-                {
-                  "ty": "tr",
-                  "a": {
-                    "a": 0,
-                    "k": [101.60000000000001, 18.599999999999994]
-                  },
-                  "s": { "a": 0, "k": [25, 25] },
-                  "sk": { "a": 0, "k": 0 },
-                  "p": { "a": 0, "k": [49.400000000000006, 29.4] },
-                  "r": { "a": 0, "k": 0 },
-                  "sa": { "a": 0, "k": 0 },
-                  "o": { "a": 0, "k": 100 }
-                }
-              ]
+              "ty": "tr",
+              "p": { "a": 0, "k": [-178.438, 16.063], "ix": 2 },
+              "a": { "a": 0, "k": [0, 0], "ix": 1 },
+              "s": { "a": 0, "k": [100, 100], "ix": 3 },
+              "r": { "a": 0, "k": 0, "ix": 6 },
+              "o": { "a": 0, "k": 100, "ix": 7 },
+              "sk": { "a": 0, "k": 0, "ix": 4 },
+              "sa": { "a": 0, "k": 0, "ix": 5 },
+              "nm": "变换"
             }
           ],
-          "ind": 3
-        },
-        {
-          "ty": 4,
-          "nm": "Shape Layer 1",
-          "sr": 1,
-          "st": 0,
-          "op": 206.99999999999997,
-          "ip": 105,
-          "hd": false,
-          "ddd": 0,
+          "nm": "椭圆 1",
+          "np": 3,
+          "cix": 2,
           "bm": 0,
-          "hasMask": false,
-          "ao": 0,
-          "ks": {
-            "a": { "a": 0, "k": [32.00000000000001, 33] },
-            "s": { "a": 0, "k": [500, 500] },
-            "sk": { "a": 0, "k": 0 },
-            "p": {
-              "a": 1,
-              "k": [
-                {
-                  "o": { "x": 0.167, "y": 0.167 },
-                  "i": { "x": 0.833, "y": 0.833 },
-                  "s": [91.49612426757818, 251],
-                  "t": 106
-                },
-                {
-                  "o": { "x": 0.167, "y": 0.167 },
-                  "i": { "x": 0.833, "y": 0.833 },
-                  "s": [174.00000000000003, 251],
-                  "t": 116
-                },
-                {
-                  "o": { "x": 0.167, "y": 0.167 },
-                  "i": { "x": 0.833, "y": 0.833 },
-                  "s": [174.00000000000003, 251],
-                  "t": 173
-                },
-                {
-                  "o": { "x": 0.167, "y": 0.167 },
-                  "i": { "x": 0.833, "y": 0.833 },
-                  "s": [174.00000000000003, 233],
-                  "t": 176
-                },
-                { "s": [174.00000000000003, 251], "t": 179 }
-              ]
+          "ix": 1,
+          "mn": "ADBE Vector Group",
+          "hd": false
+        }
+      ],
+      "ip": 5,
+      "op": 20,
+      "st": -10,
+      "bm": 0
+    },
+    {
+      "ddd": 0,
+      "ind": 2,
+      "ty": 4,
+      "nm": "形状图层 2",
+      "sr": 1,
+      "ks": {
+        "o": {
+          "a": 1,
+          "k": [
+            {
+              "i": { "x": [0.659], "y": [0.843] },
+              "o": { "x": [0.391], "y": [0] },
+              "t": 0,
+              "s": [100]
             },
-            "r": { "a": 0, "k": 0 },
-            "sa": { "a": 0, "k": 0 },
-            "o": { "a": 0, "k": 100 }
-          },
-          "ef": [],
-          "shapes": [
             {
-              "ty": "gr",
-              "bm": 0,
-              "hd": false,
-              "nm": "Group 5",
-              "it": [
-                {
-                  "ty": "sh",
-                  "bm": 0,
-                  "hd": false,
-                  "nm": "Path 5",
-                  "d": 1,
-                  "ks": {
-                    "a": 0,
-                    "k": {
-                      "c": true,
-                      "i": [
-                        [0, 0],
-                        [0, -11.3888407],
-                        [-4.3333333, 0],
-                        [0, 11.388840700000003],
-                        [9.388840700000003, 0],
-                        [3.1184307999999987, -3.696753659999999],
-                        [5.215011799999999, 0]
-                      ],
-                      "o": [
-                        [-9.38884075, 0],
-                        [0, 11.388840700000003],
-                        [4.3333333, 0],
-                        [0, -11.3888407],
-                        [-5.215011799999999, 0],
-                        [-3.1184307999999987, -3.696753659999999],
-                        [0, 0]
-                      ],
-                      "v": [
-                        [19, 6],
-                        [2, 23],
-                        [32, 60],
-                        [62, 23],
-                        [45, 6],
-                        [32, 12.0449665],
-                        [19, 6]
-                      ]
-                    }
-                  }
-                },
-                {
-                  "ty": "fl",
-                  "bm": 0,
-                  "hd": false,
-                  "nm": "Fill",
-                  "c": {
-                    "a": 1,
-                    "k": [
-                      {
-                        "o": { "x": 0.167, "y": 0.167 },
-                        "i": { "x": 0.833, "y": 0.833 },
-                        "s": [0.1137, 0.6392, 0.2235],
-                        "t": 168
-                      },
-                      { "s": [0.6392, 0.1137, 0.6157], "t": 174 }
-                    ]
-                  },
-                  "r": 2,
-                  "o": { "a": 0, "k": 100 }
-                },
-                {
-                  "ty": "tr",
-                  "a": { "a": 0, "k": [32, 33] },
-                  "s": { "a": 0, "k": [25, 25] },
-                  "sk": { "a": 0, "k": 0 },
-                  "p": { "a": 0, "k": [32, 33] },
-                  "r": { "a": 0, "k": 0 },
-                  "sa": { "a": 0, "k": 0 },
-                  "o": { "a": 0, "k": 100 }
-                }
-              ]
-            }
+              "i": { "x": [0.6], "y": [0.927] },
+              "o": { "x": [0.283], "y": [-0.13] },
+              "t": 10,
+              "s": [0]
+            },
+            { "t": 20, "s": [100] }
           ],
-          "ind": 4
+          "ix": 11
         },
+        "r": { "a": 0, "k": 0, "ix": 10 },
+        "p": { "a": 0, "k": [46.75, 226.625, 0], "ix": 2 },
+        "a": { "a": 0, "k": [-178.25, 16.625, 0], "ix": 1 },
+        "s": { "a": 0, "k": [100, 100, 100], "ix": 6 }
+      },
+      "ao": 0,
+      "shapes": [
         {
-          "ty": 4,
-          "nm": "Shape Layer 1",
-          "sr": 1,
-          "st": 0,
-          "op": 207.89999999999998,
-          "ip": 86.1,
-          "hd": false,
-          "ddd": 0,
-          "bm": 0,
-          "hasMask": false,
-          "ao": 0,
-          "ks": {
-            "a": { "a": 0, "k": [32, 33] },
-            "s": { "a": 0, "k": [500, 500] },
-            "sk": { "a": 0, "k": 0 },
-            "p": {
-              "a": 1,
-              "k": [
-                {
-                  "o": { "x": 0.167, "y": 0.167 },
-                  "i": { "x": 0.833, "y": 0.833 },
-                  "s": [250, 251],
-                  "t": 98.1
-                },
-                {
-                  "o": { "x": 0.167, "y": 0.167 },
-                  "i": { "x": 0.833, "y": 0.833 },
-                  "s": [91.49612426757815, 251],
-                  "t": 103.1
-                },
-                {
-                  "o": { "x": 0.167, "y": 0.167 },
-                  "i": { "x": 0.833, "y": 0.833 },
-                  "s": [91.49612426757815, 251],
-                  "t": 166
-                },
-                {
-                  "o": { "x": 0.167, "y": 0.167 },
-                  "i": { "x": 0.833, "y": 0.833 },
-                  "s": [91.49612426757815, 232],
-                  "t": 170
-                },
-                { "s": [91.49612426757815, 251], "t": 174 }
-              ]
+          "ty": "gr",
+          "it": [
+            {
+              "d": 1,
+              "ty": "el",
+              "s": { "a": 0, "k": [16, 16], "ix": 2 },
+              "p": { "a": 0, "k": [0, 0], "ix": 3 },
+              "nm": "椭圆路径 1",
+              "mn": "ADBE Vector Shape - Ellipse",
+              "hd": false
             },
-            "r": { "a": 0, "k": 0 },
-            "sa": { "a": 0, "k": 0 },
-            "o": { "a": 0, "k": 100 }
-          },
-          "ef": [],
-          "shapes": [
             {
-              "ty": "gr",
+              "ty": "fl",
+              "c": {
+                "a": 0,
+                "k": [0.909803921569, 0.050980392157, 0.050980392157, 1],
+                "ix": 4
+              },
+              "o": { "a": 0, "k": 100, "ix": 5 },
+              "r": 1,
               "bm": 0,
-              "hd": false,
-              "nm": "Group 5",
-              "it": [
-                {
-                  "ty": "sh",
-                  "bm": 0,
-                  "hd": false,
-                  "nm": "Path 5",
-                  "d": 1,
-                  "ks": {
-                    "a": 0,
-                    "k": {
-                      "c": true,
-                      "i": [
-                        [0, 0],
-                        [0, -11.3888407],
-                        [-4.3333333, 0],
-                        [0, 11.388840700000003],
-                        [9.388840700000003, 0],
-                        [3.1184307999999987, -3.696753659999999],
-                        [5.215011799999999, 0]
-                      ],
-                      "o": [
-                        [-9.38884075, 0],
-                        [0, 11.388840700000003],
-                        [4.3333333, 0],
-                        [0, -11.3888407],
-                        [-5.215011799999999, 0],
-                        [-3.1184307999999987, -3.696753659999999],
-                        [0, 0]
-                      ],
-                      "v": [
-                        [19, 6],
-                        [2, 23],
-                        [32, 60],
-                        [62, 23],
-                        [45, 6],
-                        [32, 12.0449665],
-                        [19, 6]
-                      ]
-                    }
-                  }
-                },
-                {
-                  "ty": "fl",
-                  "bm": 0,
-                  "hd": false,
-                  "nm": "Fill",
-                  "c": {
-                    "a": 1,
-                    "k": [
-                      {
-                        "o": { "x": 0.167, "y": 0.167 },
-                        "i": { "x": 0.833, "y": 0.833 },
-                        "s": [0, 0, 0],
-                        "t": 80
-                      },
-                      {
-                        "o": { "x": 0.167, "y": 0.167 },
-                        "i": { "x": 0.833, "y": 0.833 },
-                        "s": [0.2078, 0.4039, 0.8941],
-                        "t": 86.1
-                      },
-                      {
-                        "o": { "x": 0.167, "y": 0.167 },
-                        "i": { "x": 0.833, "y": 0.833 },
-                        "s": [0.1137, 0.6392, 0.2235],
-                        "t": 96.1
-                      },
-                      {
-                        "o": { "x": 0.167, "y": 0.167 },
-                        "i": { "x": 0.833, "y": 0.833 },
-                        "s": [0.1137, 0.6392, 0.2235],
-                        "t": 164
-                      },
-                      { "s": [0.6392, 0.1137, 0.6157], "t": 169 }
-                    ]
-                  },
-                  "r": 2,
-                  "o": { "a": 0, "k": 100 }
-                },
-                {
-                  "ty": "tr",
-                  "a": { "a": 0, "k": [32, 33] },
-                  "s": {
-                    "a": 1,
-                    "k": [
-                      {
-                        "o": { "x": 0.167, "y": 0.167 },
-                        "i": { "x": 0.833, "y": 0.833 },
-                        "s": [100, 100],
-                        "t": 86.1
-                      },
-                      { "s": [25, 25], "t": 98.1 }
-                    ]
-                  },
-                  "sk": { "a": 0, "k": 0 },
-                  "p": { "a": 0, "k": [32, 33] },
-                  "r": { "a": 0, "k": 0 },
-                  "sa": { "a": 0, "k": 0 },
-                  "o": { "a": 0, "k": 100 }
-                }
-              ]
+              "nm": "填充 1",
+              "mn": "ADBE Vector Graphic - Fill",
+              "hd": false
+            },
+            {
+              "ty": "tr",
+              "p": { "a": 0, "k": [-178.438, 16.063], "ix": 2 },
+              "a": { "a": 0, "k": [0, 0], "ix": 1 },
+              "s": { "a": 0, "k": [100, 100], "ix": 3 },
+              "r": { "a": 0, "k": 0, "ix": 6 },
+              "o": { "a": 0, "k": 100, "ix": 7 },
+              "sk": { "a": 0, "k": 0, "ix": 4 },
+              "sa": { "a": 0, "k": 0, "ix": 5 },
+              "nm": "变换"
             }
           ],
-          "ind": 5
-        },
-        {
-          "ty": 4,
-          "nm": "Shape Layer 1",
-          "sr": 1,
-          "st": 0,
-          "op": 105,
-          "ip": 46.5,
-          "hd": false,
-          "ddd": 0,
+          "nm": "椭圆 1",
+          "np": 3,
+          "cix": 2,
           "bm": 0,
-          "hasMask": false,
-          "ao": 0,
-          "ks": {
-            "a": { "a": 0, "k": [32, 33] },
-            "s": { "a": 0, "k": [500, 500] },
-            "sk": { "a": 0, "k": 0 },
-            "p": { "a": 0, "k": [250, 251] },
-            "r": { "a": 0, "k": 0 },
-            "sa": { "a": 0, "k": 0 },
-            "o": { "a": 0, "k": 100 }
-          },
-          "ef": [],
-          "shapes": [
+          "ix": 1,
+          "mn": "ADBE Vector Group",
+          "hd": false
+        }
+      ],
+      "ip": 2,
+      "op": 20,
+      "st": -5,
+      "bm": 0
+    },
+    {
+      "ddd": 0,
+      "ind": 3,
+      "ty": 4,
+      "nm": "形状图层 1",
+      "sr": 1,
+      "ks": {
+        "o": {
+          "a": 1,
+          "k": [
             {
-              "ty": "gr",
-              "bm": 0,
-              "hd": false,
-              "nm": "Group 5",
-              "it": [
-                {
-                  "ty": "sh",
-                  "bm": 0,
-                  "hd": false,
-                  "nm": "Path 5",
-                  "d": 1,
-                  "ks": {
-                    "a": 0,
-                    "k": {
-                      "c": true,
-                      "i": [
-                        [0, 0],
-                        [0, -11.3888407],
-                        [-4.3333333, 0],
-                        [0, 11.388840700000003],
-                        [9.388840700000003, 0],
-                        [3.1184307999999987, -3.696753659999999],
-                        [5.215011799999999, 0]
-                      ],
-                      "o": [
-                        [-9.38884075, 0],
-                        [0, 11.388840700000003],
-                        [4.3333333, 0],
-                        [0, -11.3888407],
-                        [-5.215011799999999, 0],
-                        [-3.1184307999999987, -3.696753659999999],
-                        [0, 0]
-                      ],
-                      "v": [
-                        [19, 6],
-                        [2, 23],
-                        [32, 60],
-                        [62, 23],
-                        [45, 6],
-                        [32, 12.0449665],
-                        [19, 6]
-                      ]
-                    }
-                  }
-                },
-                {
-                  "ty": "fl",
-                  "bm": 0,
-                  "hd": false,
-                  "nm": "Fill",
-                  "c": {
-                    "a": 1,
-                    "k": [
-                      {
-                        "o": { "x": 0.167, "y": 0.167 },
-                        "i": { "x": 0.833, "y": 0.833 },
-                        "s": [0.8941, 0.2078, 0.2078],
-                        "t": 47
-                      },
-                      { "s": [0.2078, 0.4039, 0.8941], "t": 60 }
-                    ]
-                  },
-                  "r": 2,
-                  "o": {
-                    "a": 1,
-                    "k": [
-                      {
-                        "o": { "x": 0.167, "y": 0.167 },
-                        "i": { "x": 0.833, "y": 0.833 },
-                        "s": [100],
-                        "t": 82
-                      },
-                      { "s": [0], "t": 95 }
-                    ]
-                  }
-                },
-                {
-                  "ty": "tr",
-                  "a": { "a": 0, "k": [32, 33] },
-                  "s": {
-                    "a": 1,
-                    "k": [
-                      {
-                        "o": { "x": 0.167, "y": 0.167 },
-                        "i": { "x": 0.833, "y": 0.833 },
-                        "s": [100, 100],
-                        "t": 46.5
-                      },
-                      {
-                        "o": { "x": 0.167, "y": 0.167 },
-                        "i": { "x": 0.833, "y": 0.833 },
-                        "s": [120, 120],
-                        "t": 65.5
-                      },
-                      {
-                        "o": { "x": 0.167, "y": 0.167 },
-                        "i": { "x": 0.833, "y": 0.833 },
-                        "s": [90, 90],
-                        "t": 67.5
-                      },
-                      {
-                        "o": { "x": 0.167, "y": 0.167 },
-                        "i": { "x": 0.833, "y": 0.833 },
-                        "s": [110, 110],
-                        "t": 73.5
-                      },
-                      {
-                        "o": { "x": 0.167, "y": 0.167 },
-                        "i": { "x": 0.833, "y": 0.833 },
-                        "s": [97, 97],
-                        "t": 78.5
-                      },
-                      {
-                        "o": { "x": 0.167, "y": 0.167 },
-                        "i": { "x": 0.833, "y": 0.833 },
-                        "s": [102, 102],
-                        "t": 82.5
-                      },
-                      { "s": [100, 100], "t": 86.5 }
-                    ]
-                  },
-                  "sk": { "a": 0, "k": 0 },
-                  "p": { "a": 0, "k": [32, 33] },
-                  "r": { "a": 0, "k": 0 },
-                  "sa": { "a": 0, "k": 0 },
-                  "o": { "a": 0, "k": 100 }
-                }
-              ]
-            }
+              "i": { "x": [0.659], "y": [0.843] },
+              "o": { "x": [0.391], "y": [0] },
+              "t": 0,
+              "s": [100]
+            },
+            {
+              "i": { "x": [0.6], "y": [0.927] },
+              "o": { "x": [0.283], "y": [-0.13] },
+              "t": 10,
+              "s": [0]
+            },
+            { "t": 20, "s": [100] }
           ],
-          "ind": 6
+          "ix": 11
         },
+        "r": { "a": 0, "k": 0, "ix": 10 },
+        "p": { "a": 0, "k": [21.75, 211.625, 0], "ix": 2 },
+        "a": { "a": 0, "k": [-178.25, 16.625, 0], "ix": 1 },
+        "s": { "a": 0, "k": [100, 100, 100], "ix": 6 }
+      },
+      "ao": 0,
+      "shapes": [
         {
-          "ty": 4,
-          "nm": "Shape Layer 1",
-          "sr": 1,
-          "st": 0,
-          "op": 58.50000000000001,
-          "ip": 0,
-          "hd": false,
-          "ddd": 0,
-          "bm": 0,
-          "hasMask": false,
-          "ao": 0,
-          "ks": {
-            "a": { "a": 0, "k": [32, 33] },
-            "s": { "a": 0, "k": [500, 500] },
-            "sk": { "a": 0, "k": 0 },
-            "p": { "a": 0, "k": [250, 251] },
-            "r": { "a": 0, "k": 0 },
-            "sa": { "a": 0, "k": 0 },
-            "o": { "a": 0, "k": 100 }
-          },
-          "ef": [],
-          "shapes": [
+          "ty": "gr",
+          "it": [
             {
-              "ty": "gr",
+              "d": 1,
+              "ty": "el",
+              "s": { "a": 0, "k": [16, 16], "ix": 2 },
+              "p": { "a": 0, "k": [0, 0], "ix": 3 },
+              "nm": "椭圆路径 1",
+              "mn": "ADBE Vector Shape - Ellipse",
+              "hd": false
+            },
+            {
+              "ty": "fl",
+              "c": {
+                "a": 0,
+                "k": [0.909803921569, 0.050980392157, 0.050980392157, 1],
+                "ix": 4
+              },
+              "o": { "a": 0, "k": 100, "ix": 5 },
+              "r": 1,
               "bm": 0,
-              "hd": false,
-              "nm": "Group 5",
-              "it": [
-                {
-                  "ty": "sh",
-                  "bm": 0,
-                  "hd": false,
-                  "nm": "Path 5",
-                  "d": 1,
-                  "ks": {
-                    "a": 0,
-                    "k": {
-                      "c": true,
-                      "i": [
-                        [0, 0],
-                        [0, -11.3888407],
-                        [-4.3333333, 0],
-                        [0, 11.388840700000003],
-                        [9.388840700000003, 0],
-                        [3.1184307999999987, -3.696753659999999],
-                        [5.215011799999999, 0]
-                      ],
-                      "o": [
-                        [-9.38884075, 0],
-                        [0, 11.388840700000003],
-                        [4.3333333, 0],
-                        [0, -11.3888407],
-                        [-5.215011799999999, 0],
-                        [-3.1184307999999987, -3.696753659999999],
-                        [0, 0]
-                      ],
-                      "v": [
-                        [19, 6],
-                        [2, 23],
-                        [32, 60],
-                        [62, 23],
-                        [45, 6],
-                        [32, 12.0449665],
-                        [19, 6]
-                      ]
-                    }
-                  }
-                },
-                {
-                  "ty": "fl",
-                  "bm": 0,
-                  "hd": false,
-                  "nm": "Fill",
-                  "c": { "a": 0, "k": [0.8941, 0.2078, 0.2078] },
-                  "r": 2,
-                  "o": { "a": 0, "k": 100 }
-                },
-                {
-                  "ty": "tr",
-                  "a": { "a": 0, "k": [32, 33] },
-                  "s": {
-                    "a": 1,
-                    "k": [
-                      {
-                        "o": { "x": 0.167, "y": 0.167 },
-                        "i": { "x": 0.833, "y": 0.833 },
-                        "s": [100, 100],
-                        "t": 0
-                      },
-                      {
-                        "o": { "x": 0.167, "y": 0.167 },
-                        "i": { "x": 0.833, "y": 0.833 },
-                        "s": [120, 120],
-                        "t": 19
-                      },
-                      {
-                        "o": { "x": 0.167, "y": 0.167 },
-                        "i": { "x": 0.833, "y": 0.833 },
-                        "s": [90, 90],
-                        "t": 27
-                      },
-                      {
-                        "o": { "x": 0.167, "y": 0.167 },
-                        "i": { "x": 0.833, "y": 0.833 },
-                        "s": [110, 110],
-                        "t": 33
-                      },
-                      {
-                        "o": { "x": 0.167, "y": 0.167 },
-                        "i": { "x": 0.833, "y": 0.833 },
-                        "s": [97, 97],
-                        "t": 38
-                      },
-                      {
-                        "o": { "x": 0.167, "y": 0.167 },
-                        "i": { "x": 0.833, "y": 0.833 },
-                        "s": [102, 102],
-                        "t": 42
-                      },
-                      { "s": [100, 100], "t": 46 }
-                    ]
-                  },
-                  "sk": { "a": 0, "k": 0 },
-                  "p": { "a": 0, "k": [32, 33] },
-                  "r": { "a": 0, "k": 0 },
-                  "sa": { "a": 0, "k": 0 },
-                  "o": { "a": 0, "k": 100 }
-                }
-              ]
+              "nm": "填充 1",
+              "mn": "ADBE Vector Graphic - Fill",
+              "hd": false
+            },
+            {
+              "ty": "tr",
+              "p": { "a": 0, "k": [-178.438, 16.063], "ix": 2 },
+              "a": { "a": 0, "k": [0, 0], "ix": 1 },
+              "s": { "a": 0, "k": [100, 100], "ix": 3 },
+              "r": { "a": 0, "k": 0, "ix": 6 },
+              "o": { "a": 0, "k": 100, "ix": 7 },
+              "sk": { "a": 0, "k": 0, "ix": 4 },
+              "sa": { "a": 0, "k": 0, "ix": 5 },
+              "nm": "变换"
             }
           ],
-          "ind": 7
+          "nm": "椭圆 1",
+          "np": 3,
+          "cix": 2,
+          "bm": 0,
+          "ix": 1,
+          "mn": "ADBE Vector Group",
+          "hd": false
         }
-      ]
+      ],
+      "ip": 0,
+      "op": 20,
+      "st": 0,
+      "bm": 0
+    },
+    {
+      "ddd": 0,
+      "ind": 5,
+      "ty": 2,
+      "nm": "Group 427319515.png",
+      "cl": "png",
+      "refId": "image_0",
+      "sr": 1,
+      "ks": {
+        "o": { "a": 0, "k": 100, "ix": 11 },
+        "r": { "a": 0, "k": 0, "ix": 10 },
+        "p": { "a": 0, "k": [200, 200, 0], "ix": 2 },
+        "a": { "a": 0, "k": [200, 136, 0], "ix": 1 },
+        "s": { "a": 0, "k": [100, 100, 100], "ix": 6 }
+      },
+      "ao": 0,
+      "ip": 0,
+      "op": 21,
+      "st": 0,
+      "bm": 0
     }
-  ]
+  ],
+  "markers": []
 }
diff --git a/src/components/Animate/index.ts b/src/components/Animate/index.ts
new file mode 100644
index 0000000..e27357b
--- /dev/null
+++ b/src/components/Animate/index.ts
@@ -0,0 +1,3 @@
+import AnimateDevice from "./src/AnimateDevice.vue";
+
+export { AnimateDevice };
diff --git a/src/components/Animate/src/AnimateDevice.vue b/src/components/Animate/src/AnimateDevice.vue
new file mode 100644
index 0000000..04e08c5
--- /dev/null
+++ b/src/components/Animate/src/AnimateDevice.vue
@@ -0,0 +1,38 @@
+<!--
+ * @Author: donghao donghao@supervision.ltd
+ * @Date: 2024-01-16 13:49:37
+ * @LastEditors: donghao donghao@supervision.ltd
+ * @LastEditTime: 2024-01-16 13:53:47
+ * @FilePath: \General-AI-Platform-Web-Client\src\components\Animate\src\animateDevice.vue
+ * @Description: 这是默认设置,请设置`customMade`, 打开koroFileHeader查看配置 进行设置: https://github.com/OBKoro1/koro1FileHeader/wiki/%E9%85%8D%E7%BD%AE
+-->
+<script setup lang="ts">
+import { ref, onMounted } from "vue"; //引入相关api
+import lottie from "lottie-web"; //引入动效库
+import json01 from "@/assets/animate/device/01.json"; //引入下载的动效json
+
+defineOptions({
+  name: "AnimateDevice"
+});
+
+const animation1 = ref(null); //获取dom
+function initAnimate() {
+  console.log(lottie, "lottie");
+  lottie.loadAnimation({
+    container: animation1.value, //选择渲染dom
+    renderer: "svg", //渲染格式
+    loop: true, //循环播放
+    autoplay: true, //是否i自动播放,
+    animationData: json01 //渲染动效json
+  });
+}
+onMounted(() => {
+  initAnimate();
+});
+</script>
+
+<template>
+  <div style="width: 100px">
+    <div ref="animation1" />
+  </div>
+</template>
diff --git a/src/components/CustomTree/src/collapseTreeItem b/src/components/CustomTree/src/collapseTreeItem
new file mode 100644
index 0000000..e69de29
diff --git a/src/components/CustomTree/src/demo.md b/src/components/CustomTree/src/demo.md
new file mode 100644
index 0000000..fd82da1
--- /dev/null
+++ b/src/components/CustomTree/src/demo.md
@@ -0,0 +1,472 @@
+<template>
+  <div class="courseCat_wrap courseLession_wrap" v-if="pageMode == 'courseCat'">
+    <van-sticky>
+      <div class="bg-white title_box">
+        <p class=" fs18 fw5 text_color title_centent flex-rb flex-ac">
+          {{ detailInfo.name }}
+        </p>
+        <p class="fs12 text-gray progress_box">
+          已学习<span class="theme_color">{{ totalLesson.learnedCount }}</span
+          >节/共{{ totalLesson.totalCount }}节
+        </p>
+      </div>
+
+      <HTab
+        ref="HTabComp"
+        class="bg-white"
+        :options="csTabOptions"
+        :tabList="tabOptions"
+        @change="menuChange"
+      >
+      </HTab>
+    </van-sticky>
+
+    <div class="catalogTree fs14" ref="CSCatTreeRefs">
+      <ul class="bg-white root_cat_box pdx15">
+        <li v-for="(v, index) in state.courseTreeData" :key="v.id">
+          <div
+            class="chapter_box flex-ac flex-rb"
+            v-if="v.type == 0"
+            @click="handleExpend(v)"
+          >
+            <span class="chapter_label fw5 single-line">
+              {{ v.name }}
+            </span>
+            <div class="collapseIcon_box flex-cc">
+              <div class="arrow" :class="v.expended ? '' : 'activeUp'"></div>
+            </div>
+          </div>
+          <div class="courseware_box" v-if="v.type == 2">
+            <!-- 课 -->
+            <CSLessonsBox
+              :lessonDownProgress="lessonDownProgress"
+              :info="v"
+              :courseDetail="detailInfo"
+              :lessonStudyProgress="lessonStudyProgress"
+            />
+          </div>
+          <div class="section_box flex-ac flex-rb" v-if="v.type == 1">
+            <span>{{ v.name }}</span>
+          </div>
+          <transition name="mybox">
+            <div class="box" v-show="v.expended">
+              <ul v-if="isArray(v.childrenList) && v.childrenList.length">
+                <li v-for="(item, index) in [...v.childrenList]" :key="item.id">
+                  <CSLessonsBox
+                    :info="item"
+                    :courseDetail="detailInfo"
+                    :lessonDownProgress="lessonDownProgress"
+                    :lessonStudyProgress="lessonStudyProgress"
+                  />
+                  <hr
+                    class="mdy20"
+                    v-if="index !== [...v.childrenList].length - 1"
+                  />
+                </li>
+              </ul>
+              <CSLessonPracticeBox
+                :info="v.coursePaperVOS"
+                :courseInfo="v"
+                :courseDetail="detailInfo"
+                v-if="
+                  v.type == 0 &&
+                    isArray(v.coursePaperVOS) &&
+                    v.coursePaperVOS.length &&
+                    v.coursePaperVOS != ['']
+                "
+              />
+            </div>
+          </transition>
+        </li>
+        <li v-for="v in practiceList" :key="v.id" class="bg-white exam_box">
+          <p
+            class="fs14 pdy12"
+            v-if="v.type == 0 && detailInfo.joinStatus >= 1"
+          >
+            课后练习
+          </p>
+          <div
+            class="flex pdy12"
+            v-if="v.type == 0 && detailInfo.joinStatus >= 1"
+          >
+            <span class="flex-cc exam_tag fs10">
+              <span class="exam_tag_icon"></span>
+              <span class="fss10">练习</span>
+            </span>
+            <div class="exam_title">
+              <p class="paperName_label fw5 single-line">
+                {{ v.paperName }}
+              </p>
+              <div class="flex-rb exam_status fs12 theme_color">
+                <span class=" single-line" v-if="v.status == 0">
+                  未开始
+                </span>
+                <span class=" single-line paper_name" v-if="v.status == 1">
+                  <span v-if="v.pass">已通过</span>
+                  <span v-else>未通过</span>
+                </span>
+                <span
+                  :class="[`${fetchDetailCatBtn(v).statusType}_tags`]"
+                  @click="doTabCoursePractice(v, {...detailInfo})"
+                >
+                  {{ fetchDetailCatBtn(v).practiceLabel }}
+                </span>
+              </div>
+            </div>
+          </div>
+        </li>
+      </ul>
+      <HEmpty v-if="!state.courseTreeData.length && !isLoading" />
+      <div ref="CSDetailCommentRefs">
+        <CSDetailComment
+          ref="detailCommentRefs"
+          :info="detailInfo"
+          :courseConf="courseConf"
+          @tenantComment="doTenantComment"
+        />
+      </div>
+    </div>
+
+  </div>
+  <!-- {{ pageMode }} -->
+  <div v-if="pageMode == 'courseDot'">
+    <CSLessonPoint
+      ref="CSLessonPointRefs"
+      :stepData="pointData"
+      @closeDot="openCat"
+      @dotTab="doDotTab"
+    />
+  </div>
+  <!-- 评论 -->
+  <CComment
+    ref="commentRefs"
+    :ocInfo="courseConf"
+    :isNeedClose="true"
+    @successComment="successComment"
+  />
+</template>
+
+<script setup>
+import {
+  reactive,
+  nextTick,
+  computed,
+  ref,
+  onMounted,
+  onActivated,
+  getCurrentInstance,
+} from "vue";
+import { useRouter, useRoute } from "vue-router";
+//
+// 组件区
+import {
+  CSLessonsBox,
+  CSLessonPracticeBox,
+  CSDetailComment,
+  CSLessonPoint,
+} from "../components";
+import { HTab, HEmpty } from "@/components/HVant/index";
+import { CComment } from "@/components/common";
+// hooks
+import { clientChangeVideo, clientChangeVideoDot } from "@/hooks/web/useClientFc"
+import { useWebFunc } from "@/hooks/web/useWebFunc";
+import { useCourseCat } from "../hooks/useCourseCat";
+import { useCourseDetail } from "../hooks/useCourseDetail";
+import { useConf, useReqApi, useCourseMain } from "../hooks/useMain";
+import {
+  fetchNextCoursewareNode,
+  fetchCurrentCourseNode,
+} from "../hooks/format";
+// 其它
+import { isArray } from "@/utils/is/isString";
+//数据
+import { testFlag } from "../hooks/test";
+import { testData } from "@/testData/courseDot";
+const { appContext } = getCurrentInstance();
+const route = useRoute();
+const router = useRouter();
+const {
+  getCourseCatalogApi,
+  getCourseInfoApi,
+  updateLastLearnCatalogIdCourseApi,
+  getCourseProgressApi,
+} = useReqApi;
+const { courseVideoDetailTabConf } = useConf;
+
+const {
+  doTabExam,
+  clientCourseCatPracticeAlertView,
+  doLoadCoursePractice,
+  practiceList,
+  doTabCoursePractice
+} = useCourseCat();
+const { setWebFunc } = useWebFunc();
+const { fetchDetailCatBtn } = useCourseDetail();
+
+// 初始值
+const lessonDownProgress = ref({}); // 课件下载进度
+const courseRecordInfo = ref({
+  totalProgress: 0,
+}); // 课程端数据汇总
+const lessonStudyProgress = ref({}); // 课件学习进度
+const totalLesson = ref({
+  learnedCount: 0,
+  totalCount: 0,
+});
+
+const state = reactive({
+  currMenuIndex: 0,
+  activeLesson: [],
+  courseListOptions: {
+    // isCustom: true
+  },
+  courseTreeData: [], //courseCatData
+});
+const detailInfo = ref({});
+const pageMode = ref("courseCat");
+const isLoading = ref(false);
+const courseConf = reactive({
+  operationType: 0,
+});
+const csTabOptions = reactive({
+  defaultValue: 0,
+});
+const tabOptions = computed(() => {
+  return courseVideoDetailTabConf.filter((item) => {
+    if (item.id === "1") {
+      item.name = `评论(${detailInfo.value.commentCount || 0})`;
+    }
+    return item;
+  });
+});
+
+const pointData = ref([]); // 知识点
+const currentCatData = ref({}); // 当前学习目录项
+// 实例
+const CSCatTreeRefs = ref("");
+const CSDetailCommentRefs = ref("");
+const commentRefs = ref("");
+const detailCommentRefs = ref("");
+const CSLessonPointRefs = ref("");
+
+/**step1 目录、评论模块切换 */
+function menuChange(record) {
+  state.currMenuIndex = record;
+  nextTick(() => {
+    switch (record) {
+      case "0":
+         window.scrollTo(
+          0,
+          0
+        );
+        break;
+      case "1":
+        // CSDetailCommentRefs.value.scrollIntoView({
+        //   behavior: "smooth",
+        // });
+        window.scrollTo(
+          0,
+          CSDetailCommentRefs.value.offsetTop - CSCatTreeRefs.value.offsetTop
+        );
+        break;
+    }
+  });
+}
+
+/**step2 目录详情数据 评论数据 推荐课程数据*/
+// 目录数据
+async function loadCatData(nextCat, nextCatId) {
+  isLoading.value = true;
+  const res = await getCourseCatalogApi({ id: route.query.id }).finally(() => {
+    isLoading.value = false;
+  });
+  // console.log(res, "loadCatData");
+  if (isArray(res)) {
+    state.courseTreeData = res.filter((item) => {
+      item.expended = true;
+      return item;
+    });
+    if (nextCatId) {
+      // 下一课已经解锁 广播下一节点id
+      let params = {
+        ...nextCat,
+        courseTrainId: route.query.id,
+      };
+      appContext.config.globalProperties.Bus.emit(
+        "changeCurrentCatId",
+        nextCatId
+      );
+      clientChangeVideo(params, (_) => {});
+    }
+  } else {
+    state.courseTreeData = [];
+  }
+}
+
+/**step3 目录交互完善 展开收起 对接端 */
+//收起/展开
+function handleExpend(record) {
+  record.expended = !record.expended;
+}
+
+/**step4 other */
+// 关闭知识点-打开目录
+function openKnowledge() {
+  if (testFlag.isOpenDot) {
+    pointData.value = testData;
+    openDot();
+  } else {
+    setWebFunc("webOpenKnowledge", (res) => {
+      pointData.value = JSON.parse(res);
+      console.info("pointData", pointData.value);
+      openDot();
+    });
+  }
+}
+// 从端获取下载进度
+function fetchDownloadProgress() {
+  setWebFunc("changeDownloadProgress", (res) => {
+    lessonDownProgress.value = res;
+  });
+}
+// 从端获取学习进度
+function fetchStudyProgress() {
+  setWebFunc("changeStudyProgress", (res) => {
+    // console.log(res, "changeStudyProgress");
+    lessonStudyProgress.value = res.studyCourseware;
+    courseRecordInfo.value = res;
+    // // 播报课程总进度
+    lessonStudyProgress.value.learningProgress>=1 && loadCourseProgress()
+  });
+}
+// 从接口获取学习进度 获取总进度
+async function loadCourseProgress() {
+  const res = await getCourseProgressApi({ id: route.query.id });
+  // console.log(res, "loadCourseProgress_res");
+  totalLesson.value = res;
+}
+
+// 完成本节播放,后续处理 (练习提醒、下一节自动切换) completeStatus 练习完成状态
+function webFnPlayNextVideo() {
+  setWebFunc("playNextVideo", (res) => {
+    const currentCat = fetchCurrentCourseNode(
+      state.courseTreeData,
+      res.catalogId
+    );
+    // 播报课程总进度
+    loadCourseProgress();
+    console.log("currentCat_coursePaperVOS", currentCat);
+    // 确认是练习提醒还是进入下一节
+    if (
+      currentCat.coursePaperVOS &&
+      (![1, 2].includes(currentCat.coursePaperVOS?.status) ||
+        ![1].includes(currentCat?.completeStatus))
+    ) {
+      // 课节练习提醒 练习提醒 课程状态处理 completeStatus 0 未完成 2 音视频完成 1 全部完成含练习
+      currentCatData.value = {
+        ...currentCat,
+        coursePracticeType: "coursewarePractice",
+      };
+      clientCourseCatPracticeAlertView({}, () => {
+        const { coursePaperVOS } = currentCatData.value;
+        doTabExam(coursePaperVOS[0]);
+      });
+    } else {
+      // 直接跳下一节课
+      fetchNextCourse(res);
+    }
+  });
+}
+
+// 判断当前课是否有练习
+function fetchNextCourse(res) {
+  // 弹下一课提醒
+  const nextCat = fetchNextCoursewareNode(state.courseTreeData, res.catalogId);
+  const nextCatId = nextCat.id;
+  //有下一课, 自动播放下一课,没有不处理
+  if (nextCatId != res.catalogId) {
+    updateChangeLessonApi(nextCatId);
+    loadCatData(nextCat, nextCatId);
+  } else {
+    // 广播下一节点id
+    // appContext.config.globalProperties.Bus.emit("changeCurrentCatId", nextCatId);
+  }
+}
+
+// 汇报上次学到的课
+function updateChangeLessonApi(nextCatId) {
+  updateLastLearnCatalogIdCourseApi({
+    courseCatalogId: nextCatId,
+    resourceId: route.query.id,
+  });
+}
+
+// 打开目录
+function openCat() {
+  pageMode.value = "courseCat";
+}
+// 打开知识点
+function openDot() {
+  pageMode.value = "courseDot";
+}
+// 切换知识点
+function doDotTab(params) {
+  clientChangeVideoDot(params, (res) => {
+    if (res == 1) {
+      CSLessonPointRefs.value.initChangePoint();
+    }
+  });
+}
+
+function loadDetail() {
+  nextTick(() => {
+    detailCommentRefs.value.initData();
+  });
+  if (isLoading.value) {
+    return;
+  }
+  doLoadData();
+  loadCatData();
+  doLoadCoursePractice();
+  loadCourseProgress();
+}
+
+// 详情数据
+async function doLoadData() {
+  const res = await getCourseInfoApi({
+    id: Number(route.query.id),
+  });
+  detailInfo.value = res;
+}
+
+function loadWebFunc() {
+  openKnowledge();
+  fetchDownloadProgress();
+  fetchStudyProgress();
+  webFnPlayNextVideo();
+}
+
+// 打开评论
+function doTenantComment() {
+  commentRefs.value.initPopup();
+}
+
+// 评论成功
+function successComment() {
+  loadDetail();
+}
+
+onMounted(() => {
+  loadWebFunc();
+  loadDetail();
+});
+
+onActivated(() => {
+  loadDetail();
+});
+</script>
+
+<style lang="less">
+@import url("../style/courseCat.less");
+@import url("../style/courseDetail.less");
+</style>
diff --git a/src/components/CustomTree/src/demoItem.md b/src/components/CustomTree/src/demoItem.md
new file mode 100644
index 0000000..863b4bb
--- /dev/null
+++ b/src/components/CustomTree/src/demoItem.md
@@ -0,0 +1,347 @@
+<template>
+  <div class="course_lessonsBox fs14 text_1">
+    <!-- 节 -->
+    <div class="section_box flex-ac flex-rb" v-if="catLatestInfo.type == 1">
+      <span>{{ catLatestInfo.name }}</span>
+    </div>
+    <!-- 课 -->
+    <!-- filter_gray -->
+    <div class="lesson_box flex-rb" :class="{'filter_gray': fetchIsNoneEffect(catLatestInfo)}" v-if="catLatestInfo.type == 2" @click="handelTab">
+      <div class="flex-re">
+        <span class="lesson_tag courseCat_tag theme_color flex-cc fs10">
+          <span class="icon_cat" :class="activeClass(catLatestInfo).iconClass"></span>
+          <span class="fss10">{{ activeClass(catLatestInfo).label }}</span>
+        </span>
+        <!-- 课件内容 -->
+        <div class="lesson_content_box">
+          <div class="flex cat_title_box">
+            <p class="lesson_infoName text_2 flex-wr">{{ catLatestInfo.name }}</p>
+            <span class="fss11" v-if="currentCatId == catLatestInfo.id">正在学</span>
+            <span class="fss11" v-if="!currentCatId && catLatestInfo.lastLearn"
+              >上次学到</span
+            >
+          </div>
+
+          <div class="course_status_box text_3">
+            <csLessonStatus
+              :isLock="currentIsLock"
+              :courseInfo="catLatestInfo"
+              :courseDetail="courseDetail"
+              :lessonStudyProgress="lessonStudyProgress"
+            />
+          </div>
+        </div>
+      </div>
+      <div @click.stop="doDownLoadEvent()">
+        <!-- 课件下载 -->
+        <div
+          class="flex-cc lesson_down"
+          v-if="currentIsLock == 0 && [3, 4].includes(catLatestInfo.courseType)"
+
+        >
+          <!-- 下载进度为0 -->
+          <span v-if="courseDetail.joinStatus < 1"></span>
+          <!-- @click.stop="doDownLoad" -->
+          <span
+            class="lesson_down_icon"
+            v-else-if="downloadProgress == 0"
+          ></span>
+          <!-- 正在下载 | 暂停 -->
+          <div
+            class="tc"
+            v-else-if="downloadProgress < 100 && downloadProgress > 0"
+          >
+          <!-- @click.stop="changeDownLoadMode" -->
+            <div>
+              <span
+                class="lesson_down_loading_icon"
+                :class="isDownload ? 'downloadPause' : 'downLoading'"
+              ></span>
+              <p class="w-8 fss11 theme_color">下载 {{ downloadProgress }} %</p>
+            </div>
+            <!-- <div v-if="!isDownload" @click.stop="doDownLoad">
+            <span class="lesson_down_loading_icon"></span>
+            <p class="fss11 theme_color">已暂停</p>
+          </div> -->
+          </div>
+          <!-- 下载完成 -->
+          <span v-else class="theme_color">已下载</span>
+        </div>
+        <div class="flex-cc noeffect_box" v-if="fetchIsNoneEffect(info)">
+          <div class="icon_noEffect"></div>
+        </div>
+      </div>
+    </div>
+    <ul>
+      <li v-for="v in catLatestInfo.childrenList">
+        <courseLessonsBox
+          :info="v"
+          :courseDetail="courseDetail"
+          color="#21CC64"
+          :lessonDownProgress="lessonDownProgress"
+          :lessonStudyProgress="lessonStudyProgress"
+        />
+      </li>
+    </ul>
+    <csLessonPracticeBox
+      :info="catLatestInfo.coursePaperVOS"
+      :courseInfo="info"
+      v-if="
+        catLatestInfo.type == 1 &&
+          isArray(catLatestInfo.coursePaperVOS) &&
+          catLatestInfo.coursePaperVOS.length &&
+          catLatestInfo.coursePaperVOS != ['']
+      "
+    />
+
+  </div>
+</template>
+
+<script setup>
+import {
+  defineComponent,
+  ref,
+  nextTick,
+  getCurrentInstance,
+  onMounted,
+  watch,
+  computed,
+} from "vue";
+import { useRouter, useRoute } from "vue-router";
+// 组件
+import csLessonStatus from "./csLessonStatus.vue";
+import csLessonPracticeBox from "./csLessonPracticeBox.vue";
+// hooks
+import { clientChangeVideo, clientDownloadVideo,  clientMediaDownloadProgress,
+  clientCancelDownloadMedia } from "@/hooks/web/useClientFc"
+  import { formatIsUnLock } from "@/hooks/format/comm";
+import { useCourseMain, useReqApi } from "../hooks/useMain";
+import { useCourseDetail } from "../hooks/useCourseDetail";
+import { debounceFun } from "@/hooks/core/useLodash.js";
+import {globalDebounceTime} from "@/config/settingConfig/core.js"
+
+//方法
+import { isArray } from "@/utils/is/isString";
+const { appContext } = getCurrentInstance();
+const {
+  activeClass
+} = useCourseMain();
+
+const props = defineProps({
+  info: {
+    type: Object,
+    default() {
+      return {
+        id: "",
+        name: "",
+        childrenList: [],
+        isLastLearnCatalog: false,
+      };
+    },
+  },
+
+  courseDetail: {
+    type: Object,
+    default() {
+      return {};
+    },
+  },
+
+  lessonDownProgress: {
+    type: Object,
+    default() {
+      return {};
+    },
+  },
+
+  lessonStudyProgress: {
+    type: Object,
+    default() {
+      return {};
+    },
+  },
+});
+defineComponent({ name: "courseLessonsBox" });
+
+const router = useRouter();
+const route = useRoute();
+const currentCatId = ref(""); // 课件目录id (正在学)
+const currentDownloadCatId = ref(""); // 课件目录id (下载)
+const { fetchCourseTipByRecord , fetchIsNoneEffect} = useCourseDetail();
+const { updateLastLearnCatalogIdCourseApi } = useReqApi;
+
+const isDownload = ref(true); //控制下载课件的暂停
+const currCatFullData = ref({}); //当前目录课节更新所有数据
+
+const catLatestInfo = computed(() => { //实时的课程目录课件数据
+  return {...props.info ,...currCatFullData.value}
+})
+// 下载进度
+const downloadProgress = ref(0);
+// const currentIsLock = ref(1);
+const currentIsLock = computed(()=>{
+  return props.info?.isLock
+})
+
+
+// 跳转播放视频
+const handelTab = 
+  debounceFun(function() {
+   console.log('debounceFun_handelTab')
+   if(fetchIsNoneEffect(props.info)){ // 课节已失效
+    return
+  }
+  if(props.info.id == currentCatId.value){ // 相同目录事件不处理
+    return
+  }
+   updateChangeLessonApi();
+ }, globalDebounceTime);
+ 
+
+  
+
+// 汇报上次学到的课 拿到当前实时的返回进度
+async function updateChangeLessonApi() {
+  const resData = await updateLastLearnCatalogIdCourseApi({
+    courseCatalogId: props.info.id,
+    resourceId: route.query.id,
+  });
+  let params = {
+      ...props.info,
+      courseTrainId: route.query.id,
+  };
+  if(resData){
+    params = {...params, ...resData.data}
+  }
+
+  if (formatIsUnLock({ ...params })) {
+    currCatFullData.value = params // 本课节接口数据更新
+    appContext.config.globalProperties.Bus.emit(
+      "changeCurrentCatId",
+      props.info.id
+  );
+    //判断是否有锁
+    clientChangeVideo(params, (res) => {});
+  } else {
+    // 报名、锁状态提示
+    fetchCourseTipByRecord(props.courseDetail, "课件");
+  }
+  console.log(params, "updateChangeLessonApi");
+}
+
+// 下载相关
+function doDownLoadEvent() {
+  if(currentIsLock.value == 0 && [3, 4].includes(catLatestInfo.value?.courseType)){
+    if(props.courseDetail.joinStatus < 1){
+      return
+    }
+    if(downloadProgress.value == 0){
+      doDownLoad()
+      return
+    }
+    if(downloadProgress.value < 100 && downloadProgress.value > 0){
+      changeDownLoadMode()
+      return
+    }
+  }
+ 
+}
+
+// 下载课件
+const doDownLoad = debounceFun(function() {
+  if(fetchIsNoneEffect(props.info)){
+    return
+  }
+  let params = {
+    ...props.info,
+    courseTrainId: route.query.id,
+  };
+  currentDownloadCatId.value = props.info.id;
+  console.log(params, "clientDownloadVideo");
+  clientDownloadVideo(params, (res) => {});
+}, globalDebounceTime);
+
+//停止下载正在下载的课件
+const changeDownLoadMode = debounceFun(function() {
+  let params = {
+    ...props.info,
+    courseTrainId: route.query.id,
+  };
+  currentDownloadCatId.value = props.info.id;
+  console.log(params, "clientCancelDownloadMedia");
+  if (isDownload.value) {
+    doDownLoad();
+  } else {
+    clientCancelDownloadMedia(params, (res) => {});
+  }
+  changeIsDownload();
+}, globalDebounceTime);
+
+// 切换暂停 | 继续 下载模式
+function changeIsDownload() {
+  isDownload.value = !isDownload.value;
+}
+
+function initChangeDownloadProgress() {
+  console.log(
+    props.info.id,
+    props.lessonDownProgress,
+    "initChangeDownloadProgress"
+  );
+  const { progress } = props.lessonDownProgress;
+  fetchDownloadProgressVal(progress);
+}
+
+function fetchDownloadProgressVal(progress) {
+  let currProgress = Math.floor(100 * progress);
+  if (currProgress <= 0) {
+    currProgress = 0;
+  }
+  if (currProgress >= 100) {
+    currProgress = 100;
+    isDownload.value = false;
+  }
+  downloadProgress.value = currProgress;
+}
+
+nextTick(() => {
+  // if (props.info) {
+  //   currentIsLock.value = props.info?.isLock;
+  // }
+  if (props.info.isLastLearnCatalog) {
+    currentCatId.value = props.info.id;
+  }
+  // console.log(props.info.lessonId, "isLastLearnCatalog");
+  appContext.config.globalProperties.Bus.on("changeCurrentCatId", (data) => {
+    console.log("接收到的内容是:" + data);
+    currentCatId.value = data;
+    // if(){
+    //   currentIsLock.value = 0; //解锁
+    // }
+  });
+  if (props.info.type == 2) {
+    // 只有课才调用获取初始加载进度
+    clientMediaDownloadProgress(
+      { ...props.info, courseTrainId: route.query.id },
+      (res) => {
+        // initChangeDownloadProgress(res)
+        isDownload.value = res?.progress > 0 && res?.progress < 1;
+        fetchDownloadProgressVal(res?.progress);
+      }
+    );
+  }
+});
+watch(
+  () => props.lessonDownProgress,
+  () => {
+    if (props.lessonDownProgress.catalogId == props.info.id) {
+      initChangeDownloadProgress();
+    }
+  },
+  { deep: true }
+);
+</script>
+
+<style lang="less">
+@import url("../style/courseCat.less");
+</style>
diff --git a/src/views/demo/animateDevice.vue b/src/views/demo/animateDevice.vue
index 15d667c..292ca12 100644
--- a/src/views/demo/animateDevice.vue
+++ b/src/views/demo/animateDevice.vue
@@ -2,7 +2,7 @@
  * @Author: donghao donghao@supervision.ltd
  * @Date: 2024-01-15 11:06:34
  * @LastEditors: donghao donghao@supervision.ltd
- * @LastEditTime: 2024-01-15 11:07:03
+ * @LastEditTime: 2024-01-16 13:50:46
  * @FilePath: \general-work-web\General-AI-Platform-Web-Client\src\views\demo\animateDevice.vue
  * @Description: 这是默认设置,请设置`customMade`, 打开koroFileHeader查看配置 进行设置: https://github.com/OBKoro1/koro1FileHeader/wiki/%E9%85%8D%E7%BD%AE
 -->
@@ -32,7 +32,7 @@ onMounted(() => {
 </script>
 
 <template>
-  <div>
+  <div style="width: 100px">
     <div ref="animation1" />
   </div>
 </template>
diff --git a/src/views/device/components/DeviceCard.vue b/src/views/device/components/DeviceCard.vue
index ab0795e..65913e8 100644
--- a/src/views/device/components/DeviceCard.vue
+++ b/src/views/device/components/DeviceCard.vue
@@ -3,6 +3,7 @@ import { computed, PropType } from "vue";
 import control from "@/assets/svg/device/control.svg?component";
 import monitor1 from "@/assets/svg/device/monitor1.svg?component";
 import monitor2 from "@/assets/svg/device/monitor2.svg?component";
+import { AnimateDevice } from "@/components/Animate";
 defineOptions({
   name: "DeviceCard"
 });
@@ -41,7 +42,7 @@ const isEnabledClass = computed(() => [
 <template>
   <div :class="cardClass">
     <div class="device-header">
-      <div class="device--logo mr-2">
+      <div class="mr-2 device--logo">
         <control />
         <monitor1 v-if="false" />
         <monitor2 v-if="false" />
@@ -67,7 +68,9 @@ const isEnabledClass = computed(() => [
         </div>
         <div>创建时间:<span>DEVICE00002</span></div>
       </div>
-      <div class="device-icon"><img src="@/assets/control.png" /></div>
+      <div class="device-icon">
+        <AnimateDevice /><img src="@/assets/control.png" />
+      </div>
     </div>
   </div>
 </template>